5 個處理狀態列的函式

非非白發表於2018-02-04

今天為大家介紹 5 個處理狀態列的函式,這 5 個函式互不依賴,可以自由以任意次序組合以達成自己的需求。支援到 Android 4.4 。

如圖所示:

statusbar-screenshot

沉浸式

public static void setStatusBarTranslucent(Window window, boolean translucent) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        View decorView = window.getDecorView();
        if (translucent) {
            decorView.setOnApplyWindowInsetsListener(new View.OnApplyWindowInsetsListener() {
                @TargetApi(Build.VERSION_CODES.LOLLIPOP)
                @Override
                public WindowInsets onApplyWindowInsets(View v, WindowInsets insets) {
                    WindowInsets defaultInsets = v.onApplyWindowInsets(insets);
                    return defaultInsets.replaceSystemWindowInsets(
                            defaultInsets.getSystemWindowInsetLeft(),
                            0,
                            defaultInsets.getSystemWindowInsetRight(),
                            defaultInsets.getSystemWindowInsetBottom());
                }
            });
        } else {
            decorView.setOnApplyWindowInsetsListener(null);
        }
        ViewCompat.requestApplyInsets(decorView);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        if (translucent) {
            window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        } else {
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
        ViewCompat.requestApplyInsets(window.getDecorView());
    }
}
複製程式碼

狀態列著色

public static void setStatusBarColor(final Window window, int color, boolean animated) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
        if (animated) {
            int curColor = window.getStatusBarColor();
            ValueAnimator colorAnimation = ValueAnimator.ofObject(new ArgbEvaluator(), curColor, color);

            colorAnimation.addUpdateListener(
                    new ValueAnimator.AnimatorUpdateListener() {
                        @TargetApi(21)
                        @Override
                        public void onAnimationUpdate(ValueAnimator animator) {
                            window.setStatusBarColor((Integer) animator.getAnimatedValue());
                        }
                    });
            colorAnimation.setDuration(200).setStartDelay(0);
            colorAnimation.start();
        } else {
            window.setStatusBarColor(color);
        }
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        ViewGroup decorViewGroup = (ViewGroup) window.getDecorView();
        View statusBarView = decorViewGroup.findViewWithTag("custom_status_bar_tag");
        if (statusBarView == null) {
            statusBarView = new View(window.getContext());
            statusBarView.setTag("custom_status_bar_tag");
            FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(
                    FrameLayout.LayoutParams.MATCH_PARENT, getStatusBarHeight(window.getContext()));
            params.gravity = Gravity.TOP;
            statusBarView.setLayoutParams(params);
            decorViewGroup.addView(statusBarView);
        }

        if (animated) {
            Drawable drawable = statusBarView.getBackground();
            int curColor = Integer.MAX_VALUE;
            if (drawable != null && drawable instanceof ColorDrawable) {
                ColorDrawable colorDrawable = (ColorDrawable) drawable;
                curColor = colorDrawable.getColor();
            }
            if (curColor != Integer.MAX_VALUE) {
                final View barView = statusBarView;
                ValueAnimator colorAnimation = ValueAnimator.ofObject(new ArgbEvaluator(), curColor, color);
                colorAnimation.addUpdateListener(
                        new ValueAnimator.AnimatorUpdateListener() {
                            @Override
                            public void onAnimationUpdate(ValueAnimator animator) {
                                barView.setBackground(new ColorDrawable((Integer) animator.getAnimatedValue()));
                            }
                        });
                colorAnimation.setDuration(200).setStartDelay(0);
                colorAnimation.start();
            } else {
                statusBarView.setBackground(new ColorDrawable(color));
            }
        } else {
            statusBarView.setBackground(new ColorDrawable(color));
        }
    }
}
複製程式碼

黑色狀態列

public static void setStatusBarStyle(Window window, boolean dark) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        View decorView = window.getDecorView();
        decorView.setSystemUiVisibility(
                dark ? View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR : 0);
    }
}
複製程式碼

隱藏狀態列

public static void setStatusBarHidden(Window window, boolean hidden) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        if (hidden) {
            window.addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
            window.clearFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
        } else {
            window.addFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
            window.clearFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
        }
    }
    
    if (Build.VERSION.SDK_INT == Build.VERSION_CODES.KITKAT) {
        ViewGroup decorViewGroup = (ViewGroup) window.getDecorView();
        final View statusBarView = decorViewGroup.findViewWithTag("custom_status_bar_tag");

        if (statusBarView != null) {
            boolean hiding = statusBarView.isClickable();
            if (hiding == hidden) {
                return;
            }

            if (hidden) {
                statusBarView.setClickable(true);
                ObjectAnimator animator = ObjectAnimator.ofFloat(statusBarView, "y", -getStatusBarHeight(window.getContext()));
                animator.setDuration(200);
                animator.setStartDelay(200);
                animator.start();
            } else {
                statusBarView.setClickable(false);
                ObjectAnimator animator = ObjectAnimator.ofFloat(statusBarView, "y", 0);
                animator.setDuration(200);
                animator.start();
            }
        }
    }
}
複製程式碼

調整 Toolbar 位置

public static void appendStatusBarPadding(View view, int viewHeight) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        if (view != null) {
            int statusBarHeight = getStatusBarHeight(view.getContext());
            view.setPadding(view.getPaddingLeft(), statusBarHeight, view.getPaddingRight(), view.getPaddingBottom());
            if (viewHeight > 0) {
                view.getLayoutParams().height = statusBarHeight + viewHeight;
            } else {
                view.getLayoutParams().height = viewHeight;
            }
        }
    }
}

public static void removeStatusBarPadding(View view, int viewHeight) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        if (view != null) {
            view.setPadding(view.getPaddingLeft(), 0, view.getPaddingRight(),
                    view.getPaddingBottom());
            view.getLayoutParams().height = viewHeight;
        }
    }
}

public static int getStatusBarHeight(Context context) {
    int statusBarHeight = -1;
    //獲取status_bar_height資源的ID
    int resourceId = context.getResources().getIdentifier("status_bar_height", "dimen", "android");
    if (resourceId > 0) {
        //根據資源ID獲取響應的尺寸值
        statusBarHeight = context.getResources().getDimensionPixelSize(resourceId);
    }
    return statusBarHeight;
}
複製程式碼

在 Fragment 中使用

有時,一個 Activity 中有好幾個 fragment, 每個 fragment 的狀態列樣式可能都不一樣,如何優雅地跟隨 fragment 切換狀態列的樣式呢?就像下圖所示的那樣?

5 個處理狀態列的函式

要達成這樣的效果需要作一定的封裝,這裡有一個封裝好的方案,只需要有選擇地重寫下面幾個方法,簡單返回想要的效果即可。

protected BarStyle preferredStatusBarStyle();
protected boolean preferredStatusBarHidden();
protected int preferredStatusBarColor();
protected boolean preferredStatusBarColorAnimated();
複製程式碼

原始碼在此

相關文章