Android 11.0 实现侧滑返回效果

【相关文件】
frameworks/base/services/core/java/com/android/server/wm/DisplayPolicy.java
frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager

【实现】
frameworks/base/services/core/java/com/android/server/wm/DisplayPolicy.java

这里我们要动态来开关控制这个功能,所以我们需要申请一个Boolean 类型 的flag 变量作为标识。

    //wangrui Whether to enable the sliding back function
    public static boolean flag = true;

侧滑返回其实就跟我们点击了一下back键是同一个意思,那么我们只需要在用户侧滑时,模拟一下back 键被按下就可以实现这个功能!这里我们先自己写好模拟虚拟按键的方法

//wangrui Simulate back button
    private void sendKeyCode(final int keyCode) {
    
    
        if (flag){
    
    
            new Thread(new Runnable() {
    
    
                @Override
                public void run() {
    
    
                    try {
    
    
                        // 创建一个Instrumentation对象
                        Instrumentation inst = new Instrumentation();
                        // 调用inst对象的按键模拟方法
                        inst.sendKeyDownUpSync(keyCode);
                    } catch (Exception e) {
    
    
                        e.printStackTrace();
                    }
                }
            }).start();
        }
    }

右滑时会回调到 onSwipeFromRight(),左滑时会回调到 onSwipeFromLeft,所以我们只需要在用户每次侧滑时,发送一次模拟back按键就可实现这个效果

DisplayPolicy(WindowManagerService service, DisplayContent displayContent) {
    
    
        mService = service;
        mSystemGestures = new SystemGesturesPointerEventListener(mContext, mHandler,
                new SystemGesturesPointerEventListener.Callbacks() {
    
    
                    @Override
                    public void onSwipeFromTop() {
    
    
                        if (mStatusBar != null) {
    
    
                            requestTransientBars(mStatusBar);
                        }
                    }

                    @Override
                    public void onSwipeFromBottom() {
    
    
                        if (mNavigationBar != null && mNavigationBarPosition == NAV_BAR_BOTTOM) {
    
    
                            requestTransientBars(mNavigationBar);
                        }
                    }

                    @Override
                    public void onSwipeFromRight() {
    
    
+                       //wangrui Simulate back button
+                        sendKeyCode(4);
                        final Region excludedRegion = Region.obtain();
                        synchronized (mLock) {
    
    
                            mDisplayContent.calculateSystemGestureExclusion(
                                    excludedRegion, null /* outUnrestricted */);
                        }
                        final boolean sideAllowed = mNavigationBarAlwaysShowOnSideGesture
                                || mNavigationBarPosition == NAV_BAR_RIGHT;
                        if (mNavigationBar != null && sideAllowed
                                && !mSystemGestures.currentGestureStartedInRegion(excludedRegion)) {
    
    
                            requestTransientBars(mNavigationBar);
                        }
                        excludedRegion.recycle();
                    }

                    @Override
                    public void onSwipeFromLeft() {
    
    
+                        //wangrui Simulate back button
+                        sendKeyCode(4);
                        final Region excludedRegion = Region.obtain();
                        synchronized (mLock) {
    
    
                            mDisplayContent.calculateSystemGestureExclusion(
                                    excludedRegion, null /* outUnrestricted */);
                        }
                        final boolean sideAllowed = mNavigationBarAlwaysShowOnSideGesture
                                || mNavigationBarPosition == NAV_BAR_LEFT;
                        if (mNavigationBar != null && sideAllowed
                                && !mSystemGestures.currentGestureStartedInRegion(excludedRegion)) {
    
    
                            requestTransientBars(mNavigationBar);
                        }
                        excludedRegion.recycle();
                    }

                    @Override
                    public void onFling(int duration) {
    
    
                        if (mService.mPowerManagerInternal != null) {
    
    
                            mService.mPowerManagerInternal.powerHint(
                                    PowerHint.INTERACTION, duration);
                        }
                    }
    }

难度增加
以上步骤虽然已经实现了侧滑返回功能,不过如果想要自由控制禁用启用侧滑功能,可以这么做:
frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager

注册一个广播,目的是为了让系统能够接收到用户想开启还是禁用的消息

@Override
    public void init(Context context, IWindowManager windowManager,
            WindowManagerFuncs windowManagerFuncs) {
    
    
+    // wangrui register for whether to enable the sliding back functiont broadcasts
+        filter = new IntentFilter("com.xxx.my.back");
+        context.registerReceiver(myBackReceiver, filter);
}

然后在你的APP或者系统上只需要发出这么一条广播,就可以实现这个功能啦!这里面的逻辑很简单,就是在你每一次发出广播,就会改变是否禁启用侧滑功能标识的状态 flag!是不是跟我们前面设置的 flag 产生联系啦!

    //wangrui Change sideslip status
    BroadcastReceiver myBackReceiver = new BroadcastReceiver() {
    
    
        @Override
        public void onReceive(Context context, Intent intent) {
    
    
            DisplayPolicy.flag = (!DisplayPolicy.flag);
        }
    };

猜你喜欢

转载自blog.csdn.net/qq_27494201/article/details/125196859