1.緒論
現(xiàn)在基本上所有的應(yīng)用都會(huì)去實(shí)現(xiàn)沉浸式狀態(tài)欄,這個(gè)是應(yīng)用的標(biāo)配,如果你開(kāi)發(fā)的應(yīng)用沒(méi)有,那這個(gè)吐槽點(diǎn)就多了,“這美工有審美觀么”“程序猿這么菜,沉浸式都不會(huì)?”….. 咳咳….. 開(kāi)個(gè)玩笑啊,各有各的設(shè)計(jì)思想,不能怪程序猿。
2.問(wèn)題
那么說(shuō)到沉浸式狀態(tài)欄的問(wèn)題是什么呢?不知道大家有沒(méi)有遇到過(guò),應(yīng)用在android7.0系統(tǒng)以下的手機(jī)上運(yùn)行,沉浸式狀態(tài)欄是正常的,但是在7.0以上的手機(jī)上運(yùn)行就感覺(jué)沒(méi)有沉浸式了,是分層的。無(wú)論怎么修改狀態(tài)欄背景色都沒(méi)用,看下圖:圖1是7.0以下運(yùn)行效果,圖2是7.0以上運(yùn)行效果。


看到上面兩幅圖,明顯感覺(jué)到7.0系統(tǒng)做了處理,在狀態(tài)欄上蒙了一層灰色背景。
3.解決方法
首先,先普及下知識(shí):
DecorView是整個(gè)Window界面的最頂層View,它只有一個(gè)子元素為L(zhǎng)inearLayout。代表整個(gè)Window界面,包含通知欄,標(biāo)題欄,內(nèi)容顯示欄三塊區(qū)域。
DecorView這個(gè)大家應(yīng)該很熟悉,通過(guò)getWindow().getDecorView()就可以得到此對(duì)象,在6.0以上,我們可以通過(guò)以下方式設(shè)置狀態(tài)欄字體變黑:
getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN|View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
那么可以猜想狀態(tài)欄蒙灰肯定跟這個(gè)view有關(guān)。
接下來(lái)我們就去看DecorView的源碼,對(duì)比Android6.0和7.0有什么不同。你會(huì)發(fā)現(xiàn)Android7.0以下,DecorView是PhoneWindow的內(nèi)部類,而在7.0以上,是一個(gè)單獨(dú)的類,并且有新的屬性和方法。新的屬性如:mSemiTransparentStatusBarColor,看字面意思應(yīng)該就是我們要找的,我們對(duì)它進(jìn)行跟蹤,與它相關(guān)的代碼如下:
DecorView(Context context, int featureId, PhoneWindow window,WindowManager.LayoutParams params) { super(context); ......//省略無(wú)關(guān)代碼 mForceWindowDrawsStatusBarBackground = context.getResources().getBoolean( R.bool.config_forceWindowDrawsStatusBarBackground) && context.getApplicationInfo().targetSdkVersion >= N; //設(shè)置默認(rèn)的值,灰色 mSemiTransparentStatusBarColor = context.getResources().getColor( R.color.system_bar_background_semi_transparent, null /* theme */); ......//省略無(wú)關(guān)代碼 }private int calculateStatusBarColor() { int flags = mWindow.getAttributes().flags; return (flags & FLAG_TRANSLUCENT_STATUS) != 0 ? mSemiTransparentStatusBarColor : (flags & FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS) != 0 ? mWindow.mStatusBarColor : Color.BLACK;}calculateStatusBarColor這個(gè)方法就是計(jì)算得到狀態(tài)欄的顏色值,其中FLAG_TRANSLUCENT_STATUS是透明標(biāo)識(shí),如果flags與FLAG_TRANSLUCENT_STATUS相與不等于0的話就選擇默認(rèn)灰色值mSemiTransparentStatusBarColor。
這個(gè)calculateStatusBarColor方法在updateColorViews方法中調(diào)用,而updateColorViews方法又在onWindowDragResizeStart、onWindowDragResizeEnd等方法調(diào)用(相關(guān)代碼就不再帖了,請(qǐng)自行查看源碼),看到這里就知道為啥怎么修改狀態(tài)欄顏色也沒(méi)用的原因了,這個(gè)背景色是動(dòng)態(tài)算出來(lái)的。由此見(jiàn)得,只有將這個(gè)mSemiTransparentStatusBarColor變量值改為透明的就ok了。
那么怎么改呢?
解決思路:首先,我們通過(guò)getWindow().getDecorView()可以獲取到這個(gè)DecorView類的對(duì)象,然后通過(guò)反射修改這個(gè)對(duì)象的成員mSemiTransparentStatusBarColor變量值。代碼如下:
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){ try { Class decorViewClazz = Class.forName("com.android.internal.policy.DecorView"); Field field = decorViewClazz.getDeclaredField("mSemiTransparentStatusBarColor"); field.setAccessible(true); field.setInt(getWindow().getDecorView(), Color.TRANSPARENT); //改為透明 } catch (Exception e) {}}注意:這段代碼需要在setContentView方法前調(diào)用。
核心代碼都寫完了,趕緊試試吧!如果你覺(jué)得這篇文章對(duì)你有用,那么贊一個(gè)或者留個(gè)言吧~
以上這篇快速解決Android7.0下沉浸式狀態(tài)欄變灰的問(wèn)題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持VEVB武林網(wǎng)。
最后推薦給大家一個(gè)能夠在線查看源碼的網(wǎng)站:http://androidxref.com/
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注