仿ios侧滑推出当前界面

本文详细介绍如何在安卓应用中实现类似iOS的侧滑退出功能。通过自定义滑动事件处理逻辑,在特定条件下触发界面滑动,当滑动超过半屏时退出当前Activity。文章提供了完整的代码示例和实现步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们都知道在ios手机上面,有一个侧滑退出当前界面的功能,但是在安卓手机上系统没有给我们提供这样的功能,但是这依然阻挡不了强大的安卓的定制功能,我们完全可以自己定制一套这样的功能。首先看下效果图:


分析:

(1)要想模仿ios的这种效果,因为我们通过手指的滑动,所以这里肯定跟我们的滑动事件有关系(onInterceptTouchEvent,onTouchEvent这两个方法的关系,如果不清楚,请直接查阅事件传递机制原理)

(2)我们要想直接拦截我们的所有触摸事件,我们可以在上层父级布局中进行拦截和处理,这里我们想到了DecorView。首先我们应该知道Activity的顶级父View是DecorView,获取我们的DecorView也很简单

getWindow().getDecorView()

我们平时写的那些xml布局文件都是包裹在这个DecorView中的,所以这里我们就有了一个思路:

我们可以在我们的xml布局和DecorView中间添加一个中间布局(SlidingLayout),然后所有的滑动逻辑和滑动冲突全部在这里面处理。

(3)比较关键的是:当我们需要使用侧滑动能的Activity我们需要将它的主题设置成透明,这样滑动的时候就不会遮挡下面的Activity,代码如下:

<style name="AppTheme.Slide" parent="@style/AppTheme">
        <!--Required-->
        <item name="android:windowBackground">@android:color/transparent</item>
        <item name="android:windowIsTranslucent">true</item>
        <item name="android:windowAnimationStyle">@style/AppTheme.Slide.Animation</item>
    </style>


(4)当我们滑动超过半屏的时候,退出当前界面,否则则回退到原始位置。这里使用Scroller


下面直接上代码进行分析:

(1)触摸事件的处理过程,按下的位置大于X轴的十分之一就拦截当前事件,交给SlidingLayout的onTouchEvent处理

 /**
     * 根据手指移动的距离判断是否拦截触摸事件
     *
     * @param ev
     * @return
     */
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {

        int x = (int) ev.getX();
        int y = (int) ev.getY();
        boolean mIntercept = false;
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:

                mInterceptDownX = x;
                mLastInterceptX = x;
                mLastInterceptY = y;
                break;
            case MotionEvent.ACTION_MOVE:

                int moveX = x - mLastInterceptX;
                int moveY = y - mLastInterceptY;
                //按下的位置的X位置小于屏幕的十分之一,并且x移动的距离大于y移动的距离,就拦截
                if (mInterceptDownX < (getWidth() / 10) && Math.abs(moveX) > Math.abs(moveY)) {
                    mIntercept = true;
                } else {
                    mIntercept = false;
                }
                mLastInterceptX = x;
                mLastInterceptY = y;

                break;
            case MotionEvent.ACTION_UP: //抬起的时候重置参数
                mIntercept = false;
                mInterceptDownX = mLastInterceptX = mLastInterceptY = 0;
                break;
        }
        return mIntercept;
    }

    private int mTouchDownX;
    private int mLastTouchX;
    private int mLastTouchY;

    @Override
    public boolean onTouchEvent(MotionEvent event) {

        boolean mConsumed = false;
        int x = (int) event.getX();
        int y = (int) event.getY();
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:

                mTouchDownX = x;
                mLastTouchX = x;
                mLastTouchY = y;
                break;
            case MotionEvent.ACTION_MOVE:

                int moveX = x - mLastTouchX;
                int moveY = y - mLastTouchY;
                if (mTouchDownX < (getWidth() / 10) && Math.abs(moveX) > Math.abs(moveY) && !mConsumed) {
                    mConsumed = true;
                }
                if (mConsumed) {
                    int rightMoveX = (int) (mLastTouchX - event.getX());
                    if ((getScrollX() + rightMoveX) > 0) {  //向左滑动的时候,getScrollX()和rightMoveX都大于0,所以禁止滑动
                        scrollTo(0, 0);
                    } else {
                        scrollBy(rightMoveX, 0);
                    }
                }
                mLastTouchX = x;
                mLastTouchY = y;
                break;
            case MotionEvent.ACTION_UP:

                mConsumed = false;
                mTouchDownX = mLastTouchX = mLastTouchY = 0;
                if(-getScrollX()<getWidth()/2){ //偏移量不到屏幕宽度的一般,就回到最初的位置
                    scrollBack();
                }else{
                    scrollFinish();
                }
                break;
            case MotionEvent.ACTION_CANCEL:
                mConsumed = false;
                mTouchDownX = mLastTouchX = mLastTouchY = 0;
                if(-getScrollX()<getWidth()/2){ //偏移量不到屏幕宽度的一般,就回到最初的位置
                    scrollBack();
                }else{
                    scrollFinish();
                }
                break;
        }
        return true;
    }

(2)滑动的偏移量超出屏幕的一办,就关闭当前界面否则回到初始位置

 /**
     * 滑动到最初的位置
     */
    private void scrollBack() {
        int startX = getScrollX();
        int dx = -getScrollX();
        mScroller.startScroll(startX, 0, dx, 0, 300);
        invalidate();
    }

    /**
     * 向右滑动关闭
     */
    private void scrollFinish(){
        int dx = -getScrollX() - getWidth();
        mScroller.startScroll(getScrollX(),0,dx,0,300);
        invalidate();
    }

(3)我们可以将这些侧滑处理放在我们的BaseActivity当中,需要侧滑的Activity只要继承这个BaseActivity并且主题设置成透明就可以了

@Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if(enableRightSliding()){

            SlidingLayout slidingLayout = new SlidingLayout(this);
            slidingLayout.replaceCurrentLayout(this);
        }

    }

    /**
     * 子类重写这个方法true:允许向右滑动,false:禁止向右滑动
     * @return
     */
    protected boolean enableRightSliding(){
        return false;
    }

源代码


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值