判断状态栏是否显示以及获取状态栏高度的方法

       前段时间遇到一个问题,需要在一个popupWindow里计算当前状态栏的高度,百度了下相关的方法,要么是直接读取系统里状态栏高度的值,但是无法判断状态栏是否显示,要么就需要在一个activity里通过getWindow获取window对象来判断,有局限性。自己鼓捣了半天,外加百度来的一些信息,想出一个可以同时判断状态栏是否显示以及获取状态栏高度的方法,还不依赖于activity。

为了方便自己对比和记忆,先总结几种常见的方法

  1. 直接读取系统里状态栏高度的值,方便快捷,但是缺点就是无法判断当前状态栏是否显示
    // 直接读取系统里状态栏高度的值,但是无法判断状态栏是否显示
    private void getStatusBarHeight(){
        int height = 0;
        //获取status_bar_height资源的ID
        int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android");
        if (resourceId > 0) {
            //根据资源ID获取响应的尺寸值
            height = getResources().getDimensionPixelSize(resourceId);
        }
        Log.d(TAG,height+"");
    }

      2. 判断当前window的flag是否是full_screen,缺点是无法获得状态栏的高度和只能在activity中使用

    private void isStatusBarVisible(){
        if ((getWindow().getAttributes().flags & WindowManager.LayoutParams.FLAG_FULLSCREEN) == 0){
            Log.d(TAG,"status bar is visible");
        } else {
            Log.d(TAG,"status bar is not visible");
        }
    }

      3. 像这样的需求,必须判断状态栏是否显示,同时必须获得状态栏的高度,而且能在任意地方使用,不局限于acivity,就必须用到另外一个方法了,如下

package com.justwen.demo.statusbar;

import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.graphics.PixelFormat;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.view.WindowManager;


public class StatusBarService extends Service {

    private static final String TAG = "statusbar";

    private View mStatusBarHelperView;
    
    private class StatusBaroController extends Binder {
        
        public  int getStatusBarHeight(){
            return StatusBarService.this.getStatusBarHeight();
        }
    }

    private void initStatusBarHelperView() {
        WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE);
        mStatusBarHelperView = new View(this);
        WindowManager.LayoutParams lp = new WindowManager.LayoutParams();
        // 不可触摸,不可获得焦点
        lp.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
        //放在左上角
        lp.gravity = Gravity.LEFT | Gravity.TOP;
        // 需要在manifest里申明android.permission.SYSTEM_ALERT_WINDOW权限
        lp.type = WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY;
        lp.format = PixelFormat.TRANSLUCENT;
        wm.addView(mStatusBarHelperView, lp);
    }

    private void removeStatusBarHelperView(){
        WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE);
        wm.removeView(mStatusBarHelperView);
        mStatusBarHelperView = null;
    }

    private int getStatusBarHeight(){
        int[] screenParams = new int[2];
        mStatusBarHelperView.getLocationOnScreen(screenParams);
        // 如果状态栏隐藏,返回0,如果状态栏显示则返回高度
        return screenParams[1];
    }

    @Override
    public IBinder onBind(Intent intent) {
        return new StatusBaroController();
    }

    @Override
    public void onCreate() {
        super.onCreate();
        initStatusBarHelperView();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        removeStatusBarHelperView();
    }
}

原理是使用WindowManager在窗口左上角添加一个透明的View,然后获取这个View在整个屏幕上的位置screenParams,如果状态栏隐藏,那么screenParams[1]应为0, 如果状态栏是显示的,那么screenParams[1]就是状态栏的高度。


注意 :

  1.   screenParams和windowParams不能在wm.addView之后马上获取,不然会得到0。
  2.   如果不使用后记得调用wm.removeView。
  3.   需要在androidManifest添加浮动窗口权限,如果是M以上的手机,还必须获取动态权限。
        private void requestAlertWindowPermission() {
            Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
            intent.setData(Uri.parse("package:" + getPackageName()));
            startActivityForResult(intent, 1);
        }
    
        @Override
        protected void onActivityResult(int requestCode, int resultCode, Intent data) {
            super.onActivityResult(requestCode, resultCode, data);
            if (requestCode == 1) {
                if (Settings.canDrawOverlays(this)) {
                    Log.i(TAG, "onActivityResult granted");
                    startService(new Intent(this,StatusBarService.class));
                }
            }
        }


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 12 上,你可以使用以下步骤来判断是否在下拉状态栏上: 1. 首先,确保你的应用程序具有 SYSTEM_ALERT_WINDOW 权限,因为在 Android 12 中,只有具有此权限的应用程序才能在状态栏上绘制视图。 2. 在你的代码中注册一个监听器来监听状态栏的变化。 3. 在监听器中,通过获取系统服务的方式获得状态栏的实例 StatusBarManager。 4. 使用 StatusBarManager 的 getStatusBarHeight() 方法获取状态栏高度。 5. 在适当的时机,比如用户下拉状态栏时,可以通过比较触摸事件的 Y 坐标和状态栏高度判断是否在下拉状态栏上。 以下是一个示例代码片段,用于判断是否Android 12 上下拉状态栏: ```java WindowManager.LayoutParams layoutParams = getWindow().getAttributes(); layoutParams.layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES; getWindow().setAttributes(layoutParams); // 注册监听器 View decorView = getWindow().getDecorView(); decorView.setOnApplyWindowInsetsListener(new View.OnApplyWindowInsetsListener() { @Override public WindowInsets onApplyWindowInsets(View view, WindowInsets insets) { // 获取状态栏实例 StatusBarManager statusBarManager = getSystemService(StatusBarManager.class); if (statusBarManager != null) { // 获取状态栏高度 int statusBarHeight = statusBarManager.getStatusBarHeight(); // 判断是否在下拉状态栏上 if (insets.getSystemWindowInsetTop() == statusBarHeight) { // 在下拉状态栏上 } else { // 不在下拉状态栏上 } } return insets; } }); ``` 请注意,以上代码仅用于判断是否在下拉状态栏上。具体的实现可能根据你的应用程序结构和需求有所不同。你可以根据需要进行相应的调整和处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值