Android 11.0 实现侧滑返回效果

Posted 王睿丶

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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);
        
    ;

以上是关于Android 11.0 实现侧滑返回效果的主要内容,如果未能解决你的问题,请参考以下文章

Android实现侧滑recycleView+CardVeiw卡片阴影效果

Android实现侧滑recycleView+CardVeiw卡片阴影效果

Android实现侧滑recycleView+CardVeiw卡片阴影效果

android:QQ多种侧滑菜单的实现

Android使用ToolBar+DrawerLayout+NavigationView实现侧滑抽屉效果

Compose 实现页面侧滑返回