http://www.open-open.com/lib/view/open1455584716230.html#articleHeader2
解決方案
1. 給狀態欄設置顏色
思路是:
先設置狀態欄透明屬性;
給根佈局加上一個和狀態欄一樣大小的矩形View(色塊),添加到頂上;
然後設置根佈局的 FitsSystemWindows 屬性爲 true,此時根佈局會延伸到狀態欄,處在狀態欄位置的就是之前添加的色塊,這樣就給狀態欄設置上顏色了。
代碼如下:
/** * 設置狀態欄顏色 * * @param activity 需要設置的activity * @param color 狀態欄顏色值 */
public static void setColor(Activity activity, int color) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
// 設置狀態欄透明
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
// 生成一個狀態欄大小的矩形
View statusView = createStatusView(activity, color);
// 添加 statusView 到佈局中
ViewGroup decorView = (ViewGroup) activity.getWindow().getDecorView();
decorView.addView(statusView);
// 設置根佈局的參數
ViewGroup rootView = (ViewGroup) ((ViewGroup) activity.findViewById(android.R.id.content)).getChildAt(0);
rootView.setFitsSystemWindows(true);
rootView.setClipToPadding(true);
}
}
其中生成狀態欄一樣大小的矩形色塊的代碼如下:
/** * 生成一個和狀態欄大小相同的矩形條 * * @param activity 需要設置的activity * @param color 狀態欄顏色值 * @return 狀態欄矩形條 */
private static View createStatusView(Activity activity, int color) {
// 獲得狀態欄高度
int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");
int statusBarHeight = activity.getResources().getDimensionPixelSize(resourceId);
// 繪製一個和狀態欄一樣高的矩形
View statusView = new View(activity);
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
statusBarHeight);
statusView.setLayoutParams(params);
statusView.setBackgroundColor(color);
return statusView;
}
在 setContentView() 之後調用 setColor(Activity activity, int color) 方法即可。
- 圖片作背景時,狀態欄透明
這個實現比較簡單,根佈局背景設置爲圖片,然後添加狀態欄透明 Flag, 然後設置根佈局的 FitsSystemWindows 屬性爲 true 即可。代碼如下:
/** * 使狀態欄透明 * <p> * 適用於圖片作爲背景的界面,此時需要圖片填充到狀態欄 * * @param activity 需要設置的activity */
public static void setTranslucent(Activity activity) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
// 設置狀態欄透明
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
// 設置根佈局的參數
ViewGroup rootView = (ViewGroup) ((ViewGroup) activity.findViewById(android.R.id.content)).getChildAt(0);
rootView.setFitsSystemWindows(true);
rootView.setClipToPadding(true);
}
}
同樣的,在 setContentView() 之後調用 setTranslucent(Activity activity) 方法即可。
- 使用 DrawerLayout 時的特殊處理
注意點:
使用 DrawerLayout 時,此時不能再對根佈局,即 DrawerLayout 進行設置,而要針對 DrawerLayout 的內容佈局進行設置,即抽屜之外的另一個佈局。
如下是一個典型的 DrawerLayout 的佈局,其內容佈局即 FrameLayout,我們需要對 FrameLayout 進行仿狀態欄色塊的添加、FitsSystemWindows 屬性的設置。
<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.DrawerLayout
android:id="@+id/drawer_layout"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:id="@+id/main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="@color/colorPrimary"
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"
app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"/>
</LinearLayout>
</FrameLayout>
<android.support.design.widget.NavigationView
android:id="@+id/navigation"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="start"
app:headerLayout="@layout/nav_header"
app:menu="@menu/activity_main_drawer"/>
</android.support.v4.widget.DrawerLayout>
還有一個需要注意的設置抽屜佈局(Drawer)的 FitsSystemWindows 屬性爲 false,即上面佈局中的 NavigationView。
解決方案
DrawerLayout 狀態欄變色
代碼如下:
/** * 爲DrawerLayout 佈局設置狀態欄變色 * * @param activity 需要設置的activity * @param drawerLayout DrawerLayout * @param color 狀態欄顏色值 */
public static void setColorForDrawerLayout(Activity activity, DrawerLayout drawerLayout, int color) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
// 生成一個狀態欄大小的矩形
View statusBarView = createStatusBarView(activity, color);
// 添加 statusBarView 到佈局中
ViewGroup contentLayout = (ViewGroup) drawerLayout.getChildAt(0);
contentLayout.addView(statusBarView, 0);
// 內容佈局不是 LinearLayout 時,設置padding top
if (!(contentLayout instanceof LinearLayout) && contentLayout.getChildAt(1) != null) {
contentLayout.getChildAt(1).setPadding(0, getStatusBarHeight(activity), 0, 0);
}
// 設置屬性
ViewGroup drawer = (ViewGroup) drawerLayout.getChildAt(1);
drawerLayout.setFitsSystemWindows(false);
contentLayout.setFitsSystemWindows(false);
contentLayout.setClipToPadding(true);
drawer.setFitsSystemWindows(false);
}
}
需要注意的是,DrawerLayout 的佈局只能包含兩個直接子佈局,一個是內容佈局,一個是抽屜佈局,結構如前面的示例佈局所示,如果內容佈局的根佈局如果不是 LinearLayout 需要對其子佈局設置padding top值,否則仿狀態欄色塊會被遮擋在最下面,佈局內容延伸到狀態欄,如下圖所示:
(ps:就上圖中的問題,目前的解決方案感覺並不是很好,如果你有更好的解決方案,請告訴我~)
DrawerLayout 狀態欄透明
/** * 爲 DrawerLayout 佈局設置狀態欄透明 * * @param activity 需要設置的activity * @param drawerLayout DrawerLayout */
public static void setTranslucentForDrawerLayout(Activity activity, DrawerLayout drawerLayout) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
// 設置狀態欄透明
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
// 設置內容佈局屬性
ViewGroup contentLayout = (ViewGroup) drawerLayout.getChildAt(0);
contentLayout.setFitsSystemWindows(true);
contentLayout.setClipToPadding(true);
// 設置抽屜佈局屬性
ViewGroup vg = (ViewGroup) drawerLayout.getChildAt(1);
vg.setFitsSystemWindows(false);
// 設置 DrawerLayout 屬性
drawerLayout.setFitsSystemWindows(false);
}
}
同樣的,在 setContentView() 之後調用上述解決方案中的方法即可。
在項目中使用
以上代碼我整理成了一個工具類,放在 github 上:StatusBarUtils.java 文件
在項目中推薦這樣使用,在 BaseActivity 中重寫 setContentView(int layoutResID) 方法,新建一個 setStatusBarColor()方法,全局設置狀態欄顏色,因爲一般 App 大部分界面狀態欄都是主題色。
public class BaseActivity extends AppCompatActivity {
@Override
public void setContentView(int layoutResID) {
super.setContentView(layoutResID);
setStatusBarColor();
}
protected void setStatusBar() {
StatusBarUtils.setColor(this, getResources().getColor(R.color.colorPrimary));
}
}
當子類 Activity 的狀態欄需要特殊處理時,比如設置不同的顏色,或者設置圖片爲背景時,重寫父類的 setStatusBarColor() 方法即可,例如:
public class ImageStatusBarActivity extends BaseActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_image_status_bar);
}
@Override
protected void setStatusBar() {
StatusBarUtils.setTranslucent(this);
}
對 DrawerLayout 佈局使用時,需要注意一點,因爲方法是在 setContentView() 之後立即調用的,所以傳進來的 DrawerLayout 要通過 findViewById() 傳進來。如果傳入在 setContentView() 之後通過 findViewById() 得到的 DrawerLayout, 則會造成空指針異常。
StatusBarUtils.setColorForDrawerLayout(this, (DrawerLayout) findViewById(R.id.drawer_layout), getResources()
.getColor(R.color.colorPrimary));
原文:
http://www.open-open.com/lib/view/open1455584716230.html#articleHeader2