Android工具欄頂出轉(zhuǎn)場動畫的實(shí)現(xiàn)方法實(shí)例
前言
ndroid5.0之后新增了很多好看的轉(zhuǎn)場動畫,相比于以前的overridePendingTransition()豐富了很多,特別新增了共享元素跳轉(zhuǎn)的方式。本文將給大家詳細(xì)介紹關(guān)于Android工具欄頂出轉(zhuǎn)場動畫實(shí)現(xiàn)的相關(guān)資料,下面話不多說了,來一起看看詳細(xì)的介紹吧
實(shí)現(xiàn)效果
為何做這個(gè)動畫
起初對于這兩個(gè)界面的轉(zhuǎn)場動畫打算簡單使用android原生的共享元素動畫,可是實(shí)現(xiàn)后發(fā)現(xiàn)效果并不是很好,在很多手機(jī)上流暢度太差。
以下在敘述時(shí)把轉(zhuǎn)場前的頁面稱為A頁面, 轉(zhuǎn)場的目前頁面稱為B頁面
實(shí)現(xiàn)原理
在A頁面,把需要頂出的區(qū)域截取出來
val contentView = (context as Activity).window.decorView.findViewById<View>(android.R.id.content) ?: return null var captureImage: Bitmap? = null contentView.isDrawingCacheEnabled = true captureImageFromView(contentView) contentView.destroyDrawingCache()
在B頁面,把A頁面截取的圖片用ImageView展示在頂部,并做上移動畫。同時(shí),取消按鈕做出現(xiàn)的動畫。
ImageView的上移動畫非常簡單:通過不斷改變topMarigin來實(shí)現(xiàn)上移效果
val captureImageUpAnimator = ValueAnimator.ofFloat(0f, 1f).apply { addUpdateListener { val newTopMargin = captureImageStartLocationY * (animatedValue as Float) (captureView.layoutParams as LinearLayout.LayoutParams)?.apply { topMargin = -(newTopMargin.toInt()) } } } val captureImageDownAnimator = ValueAnimator.ofFloat(1f, 0f).apply { addUpdateListener { val newTopMargin = captureImageStartLocationY * (animatedValue as Float) (captureView.layoutParams as LinearLayout.LayoutParams)?.apply { topMargin = -(newTopMargin.toInt()) } } }
取消按鈕的出現(xiàn)動畫,則是通過取消按鈕左側(cè)的框不斷變小來實(shí)現(xiàn)的。
val rightBtnShowAnimator = ValueAnimator.ofFloat(0f, 1f).apply { addUpdateListener { val animateWidth = (it.animatedValue as Float) * RIGHT_BTN_WIDTH flexingView.layoutParams = LinearLayout.LayoutParams(containerWidth - animateWidth.toInt(), UIUtil.dp2px(30f)) } }
下圖大概描述了整個(gè)動畫時(shí)如何實(shí)現(xiàn)的:
動畫實(shí)現(xiàn)需要注意的點(diǎn)
頁面A保存的截圖如何傳到B頁面?
1、截圖應(yīng)該放到內(nèi)存還是本地?
這個(gè)截圖應(yīng)該放在內(nèi)存中,如果截圖保存到本地。那么 I/O 占用的時(shí)間肯定會導(dǎo)致動畫實(shí)現(xiàn)效果不好。
2、對于截取的圖片需要壓縮
對于不同分辨率的手機(jī),截取出的圖片的大小是不一樣的。圖片太大放在內(nèi)存中是不合適的,因此在截取圖片后對圖片的大小做了壓縮。
private fun captureFromView(view: View): Bitmap? { return translateToRgb555(view.drawingCache) } //為了壓縮大小,同時(shí)保證圖片寬高不變,直接將圖片轉(zhuǎn)成 RGB_565。 private fun translateToRgb555(srcBitmap: Bitmap): Bitmap? { try { /*傳入bitmap參數(shù),返回bitmap。*/ val dataByte = ByteArrayOutputStream() srcBitmap.compress(Bitmap.CompressFormat.JPEG, 100, dataByte) val opts = BitmapFactory.Options() opts.inPreferredConfig = Bitmap.Config.RGB_565 return BitmapFactory.decodeByteArray(dataByte.toByteArray(), 0, dataByte.size(), opts) } catch (e: Exception) { e.printStackTrace() } catch (e: OutOfMemoryError) { e.printStackTrace() } return null }
經(jīng)過上面的處理,大部分手機(jī)截圖大小在0.2MB左右
3、B界面如何獲取截取的圖片?
放在intent中傳給B
這樣是有問題的,雖然android官方說intent中可以傳遞小于1MB的圖片,但是國內(nèi)各anroid廠商對framework做了不同的定制,有可能你的圖片0.3MB就會出現(xiàn)崩潰,即TransactionTooLargeException
放在Fresco的緩存中?
嘗試將圖片放入Fresco緩存中,不過Fresco提供的接口十分不友好,(大概就不是給框架外使用的:
CloseableReference<V> cache(K key, CloseableReference<V> value);
放在全局靜態(tài)變量中
不過需要注意的是在頁面finish時(shí),將這個(gè)變量置null,以免占用內(nèi)存
覆蓋系統(tǒng)原生轉(zhuǎn)場動畫
如果對于默認(rèn)轉(zhuǎn)場動畫不做處理的話,效果就不是我們想要的,因此要取消默認(rèn)的轉(zhuǎn)場動畫
需要把B頁面的Theme的動畫相關(guān)屬性置null就可以了
<style name="NullAnimation" parent="@android:style/Animation.Activity"> <item name="android:activityOpenEnterAnimation">@null</item> <item name="android:activityOpenExitAnimation">@null</item> <item name="android:activityCloseEnterAnimation">@null</item> <item name="android:activityCloseExitAnimation">@null</item> </style>
不過這樣后,在一些手機(jī),比如華為,還是會有默認(rèn)的轉(zhuǎn)場動畫,為了保險(xiǎn)起見,在startActivity是最好這樣
context.overridePendingTransition(0, 0)
頁面閃爍的問題
在覆蓋了系統(tǒng)原生動畫后,大部分手機(jī)效果都還是ok的,不過在一些手機(jī)上會出現(xiàn)閃屏的問題,比如華為 Mate10。在網(wǎng)上參考一些大家的解決方法:
<item name="android:windowIsTranslucent">true</item>
即設(shè)置B頁面的背景為透明的。
不過問題到這里還沒有結(jié)束
windowIsTranslucent引發(fā)的崩潰問題
在設(shè)置windowIsTranslucent屬性后,在Mate10進(jìn)行測試,發(fā)現(xiàn)頁面啟動就崩潰:
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.linkedin.android.XXXX.XXXX/com.linkedin.android.XXXX.XXXX.activity.LoginActivity}:
java.lang.IllegalStateException: Only fullscreen activities can request orientation
最后google找到了原因,原來 Android 8.0某SDK要求,如果界面是鎖死方向的,那么是不允許設(shè)置這個(gè)屬性的,不然會引發(fā)崩潰, 如果想知道詳細(xì)細(xì)節(jié)可以看一下這篇文章:http://www.dbjr.com.cn/article/146945.htm
怎么解決呢?
前有山后有虎,腦殼子疼,最后決定取消B頁面的鎖屏屬性,并且頁面旋轉(zhuǎn)時(shí)頁面不做變化。
<activity android:name=".activity.BActivity" android:exported="false" android:configChanges="orientation|keyboardHidden|screenSize" android:theme="@style/AnimationTheme">
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
ANDROID中使用VIEWFLIPPER類實(shí)現(xiàn)屏幕切換(關(guān)于坐標(biāo)軸的問題已補(bǔ)充更改)
本篇文章主要介紹了ANDROID中使用VIEWFLIPPER類實(shí)現(xiàn)屏幕切換,具有一定的參考價(jià)值,有興趣的可以了解一下。2016-11-11Android自定義View圖片按Path運(yùn)動和旋轉(zhuǎn)
這篇文章主要為大家詳細(xì)介紹了Android自定義View圖片按Path運(yùn)動和旋轉(zhuǎn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01Android實(shí)現(xiàn)應(yīng)用內(nèi)置語言切換功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)應(yīng)用內(nèi)置語言切換功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02Android實(shí)現(xiàn)獲取簽名及公鑰的方法
這篇文章主要介紹了Android實(shí)現(xiàn)獲取簽名及公鑰的方法,可實(shí)現(xiàn)Android通過包名獲取相關(guān)簽名及公鑰的功能,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10Android中AsyncTask的入門使用學(xué)習(xí)指南
AsyncTask異步任務(wù),用于執(zhí)行耗時(shí)任務(wù)并在UI線程中更新結(jié)果。下面這篇文章主要給大家介紹了關(guān)于Android中AsyncTask入門使用的相關(guān)資料,需要的朋友可以參考下2019-02-02詳解Android中的沉浸式狀態(tài)欄效果實(shí)例
本篇文章主要介紹了Android中的沉浸式狀態(tài)欄效果,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-12-12基于android樣式與主題(style&theme)的詳解
本篇文章是對android中的樣式與主題(style&theme)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06Android實(shí)現(xiàn)網(wǎng)易新聞客戶端側(cè)滑菜單(1)
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)網(wǎng)易新聞客戶端側(cè)滑菜單第一篇,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11Android Broadcast原理分析之registerReceiver詳解
這篇文章主要介紹了Android Broadcast原理分析之registerReceiver詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08