(转)Recent键及状态栏屏蔽

转自:http://blog.csdn.net/codezjx/article/details/45542137

Back键和Home键的屏蔽可以看我这篇贴:http://blog.csdn.NET/codezjx/article/details/45485975

Home键与Recent键的点击事件是在framework层进行处理的,因此onKeyDown与dispatchKeyEvent都捕获不到点击事件。
查看StatusBarManager.Java源码,目前只能通过其void disable(int what) {…}设置,并可传入值:

public static final int DISABLE_HOME = View.STATUS_BAR_DISABLE_HOME;  
public static final int DISABLE_RECENT = View.STATUS_BAR_DISABLE_RECENT;  
public static final int DISABLE_BACK = View.STATUS_BAR_DISABLE_BACK;  
public static final int DISABLE_NONE = 0x00000000;  

等等… 具体值可以查view的源码,因为都是@hide的
首先想到了反射机制来进行调用:(或者使用移除了@hide的API 库classes.jar)

public static final String STATUS_BAR_SERVICE = “statusbar”;  
public static final String CLASS_STATUS_BAR_MANAGER = “android.app.StatusBarManager”;  
public static final String METHOD_DISABLE = “disable”;  
try {  
    Object service = getSystemService(STATUS_BAR_SERVICE);  
    Class  statusBarManager = Class.forName(CLASS_STATUS_BAR_MANAGER);  
    Method disable = statusBarManager.getMethod(METHOD_DISABLE, int.class);  
    disable.invoke (service, 0x01000000); //为View.STATUS_BAR_DISABLE_RECENT的值  
} catch (Exception e) {  
    e.printStackTrace();  
}  

会报出以下错误提示:
Neither user 10076 nor current process has Android.permission.STATUS_BAR.
提示缺少权限,Manifest添加之,提示:Permission is only granted to system apps
总结:通过这种方法屏蔽状态栏下拉,必须得有系统签名,WTF。。。

后续:参考了一些锁屏软件,如小米百变锁屏、Go锁屏等,他们在锁屏的时候也没有屏蔽掉Recent键和状态栏下拉。而是在点击按键后迅速将弹出的Dialog和Window收回去,其实这也可以看作另一种应急解决方法。继续回去看StatusBarManager.java源码:发现了另外一个方法,public void collapse() {…}和public void expand() {…},其中collapse为收缩的方法,expand为展开方法,而且调用这两个方法不需要系统签名。用上面的反射方法继续调用之:
注意:点击Recent键弹出dialog会回调onWindowFocusChanged(),因此可在该方法中调用:

public void onWindowFocusChanged(boolean hasFocus) {  
    super.onWindowFocusChanged(hasFocus);  
    if(!hasFocus) {  
        ……  
        Method collapse = statusBarManager.getMethod(METHOD_COLLAPSE);  
        collapse.invoke(service);  
        ……  
    }  
}  

加入权限:


后面又发现了另一种方法,更加简便,同样是点击后迅速消失(还可以隐藏长按电源的弹出框):

sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS));

可以通过View焦点丢失的情况下去触发,发送该广播,如:

class MyView extends View {  
  
    public MyView(Context pContext) {  
        super(pContext);  
    }  
  
    @Override  
    public void onWindowFocusChanged(boolean pHasWindowFocus) {  
        super.onWindowFocusChanged(pHasWindowFocus);  
        if (!pHasWindowFocus) {  
            mContext.sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS));  
        }  
    }  
}  

状态栏的隐藏:

方法一:

// 隐藏(但点击menu会恢复,然后又自动隐藏)
getWindow().getDecorView().setSystemUiVisibility(Window.FEATURE_ACTION_BAR_OVERLAY); 
// 显示
getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE);
  
// (用这个type可以将按钮变成小圆点:View.SYSTEM_UI_FLAG_LOW_PROFILE)
WindowManager.LayoutParams params = window.getAttributes();  
params.systemUiVisibility = View.SYSTEM_UI_FLAG_LOW_PROFILE;  
window.setAttributes(params);  

方法二:
隐藏:进入\system\目录,编辑build.prop,在最后一行添加qemu.hw.mainkeys=1,保存重启即可。
显示:改成qemu.hw.mainkeys=0,保存重启即可。

方法三:
参考一个叫GMD Hide Bar的软件。反编译了他的源码,他是用命令行直接干掉SystemUI这个进程来达到隐藏目的的。大家可以尝试直接在DDMS中关闭:com.android.systemui这个进程,状态栏直接消失,不过一段时间后又会自动运行。因此需要运行后台线程不断执行关闭操作,比较耗资源,且需要root权限。

// 隐藏
int k = getPid();  
if (k > 0)  
ProcessUtil.execAsRoot(“kill -s CONT ” + k);  

// 显示
ProcessUtil.execAsApp(“am startservice -n com.android.systemui/.SystemUIService”);  

你可能感兴趣的:((转)Recent键及状态栏屏蔽)