Android侧滑返回的实现

IOS有侧滑回退的功能,可以侧滑回退到上一个Activity,并且在侧滑时,底层的actvity是透明显示的。下面代码将实现该功能。
主要实现思路是让Activity的整个容器滑动。

package com.tjpld.tonglingsafetyandroid.customview;

import java.util.LinkedList;
import java.util.List;

import android.app.Activity;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.Scroller;

import com.tjpld.tonglingsafetyandroid.R;


public class SwipeBackLayout extends FrameLayout {
    private static final String TAG = SwipeBackLayout.class.getSimpleName();
    //放置Activity的View
    private View mContentView;
    //侧滑最小距离
    private int mTouchSlop;
    //按下的位置
    private int downX;
    private int downY;
    //当前的位置
    private int tempX;
    private Scroller mScroller;
    private int viewWidth;
    //判断是否为滑动
    private boolean isSilding;
    //是否完成
    private boolean isFinish;
    //侧滑的阴影效果
    private Drawable mShadowDrawable;
    private Activity mActivity;
    private List<ViewPager> mViewPagers = new LinkedList<ViewPager>();

    public SwipeBackLayout(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public SwipeBackLayout(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        //得到最小的距离
        mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
        //初始化滑动对象
        mScroller = new Scroller(context);
        //初始化阴影资源
        mShadowDrawable = getResources().getDrawable(R.drawable.shadow_left);
    }

    //对外部提供的接口
    public void attachToActivity(Activity activity) {
        mActivity = activity;
        TypedArray a = activity.getTheme().obtainStyledAttributes(
                new int[]{android.R.attr.windowBackground});
        int background = a.getResourceId(0, 0);
        a.recycle();
  //从外界传入的activity页面的底层容器,setContentView设置的View的父容   器  父容器一般都是ViewGroup
        ViewGroup decor = (ViewGroup) activity.getWindow().getDecorView();
        //得到setContentView设置的View
 ViewGroup decorChild = (ViewGroup) decor.getChildAt(0);

//设置它的背景色       
decorChild.setBackgroundResource(background);
        //移除到之前父容器中的布局view
        decor.removeView(decorChild);
        //为当前页面父容器添加设置过背景色后的子view
        addView(decorChild);
        setContentView(decorChild);
        decor.addView(this);
    }
    //得到从外部传入的activity的布局容器
    private void setContentView(View decorChild) {      
        mContentView = (View) decorChild.getParent();
    }


    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {

        ViewPager mViewPager = getTouchViewPager(mViewPagers, ev);
        Log.i(TAG, "mViewPager = " + mViewPager);
        //判断是否拦截此次事件
        if (mViewPager != null && mViewPager.getCurrentItem() != 0) {
            return super.onInterceptTouchEvent(ev);
        }

        switch (ev.getAction()) {
            //当只是按下去是,既非滑动事件,不对此事件做处理
            case MotionEvent.ACTION_DOWN:
                downX = tempX = (int) ev.getRawX();
                downY = (int) ev.getRawY();
                break;
            //当移动时拦截次事件 调用onTouchEvent
            case MotionEvent.ACTION_MOVE:
                int moveX = (int) ev.getRawX();
 //如果滑动的位置到我按下的位置之间的距离大于系统默认的最小距离 视为一次滑动
                if (moveX - downX > mTouchSlop
                        && Math.abs((int) ev.getRawY() - downY) < mTouchSlop) {
                    return true;
                }
                break;
        }

        return super.onInterceptTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
      case MotionEvent.ACTION_MOVE:
       //点击事件发生的位置
       int moveX = (int) event.getRawX();
       //滑动的距离,tempX当前的位置,在拦截事件里进行了初始化,第一次是你按下的位置
       int deltaX = tempX - moveX;
       //当前位置就是view滑动后的位置
       tempX = moveX;
       //如果滑动后的位置到按下的位置 大于默认滑动距离 视为一次滑动
      if (moveX - downX > mTouchSlop && Math.abs((int) event.getRawY() - downY) < mTouchSlop) {
                isSilding = true;
                }
  //如果是一次滑动的话,就让整个容器滑动
  if (moveX - downX >= 0 && isSilding) {
        mContentView.scrollBy(deltaX, 0);
                }
                break;
            case MotionEvent.ACTION_UP:
                isSilding = false;
      //向右滑动,值是负的 所以是小于等于整个view的宽的1/2时 代表完成了一次滑动 触发事件
                if (mContentView.getScrollX() <= -viewWidth / 2) {
                    isFinish = true;
                    scrollRight();
                } else {
                    scrollOrigin();
                    isFinish = false;
                }
                break;
        }

        return true;
    }


    private void getAlLViewPager(List<ViewPager> mViewPagers, ViewGroup parent) {
        int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            View child = parent.getChildAt(i);
            if (child instanceof ViewPager) {
                mViewPagers.add((ViewPager) child);
            } else if (child instanceof ViewGroup) {
                getAlLViewPager(mViewPagers, (ViewGroup) child);
            }
        }
    }


    private ViewPager getTouchViewPager(List<ViewPager> mViewPagers, MotionEvent ev) {
        if (mViewPagers == null || mViewPagers.size() == 0) {
            return null;
        }
        Rect mRect = new Rect();
        for (ViewPager v : mViewPagers) {
            v.getHitRect(mRect);

            if (mRect.contains((int) ev.getX(), (int) ev.getY())) {
                return v;
            }
        }
        return null;
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        super.onLayout(changed, l, t, r, b);
        if (changed) {
            viewWidth = this.getWidth();
            getAlLViewPager(mViewPagers, this);
            Log.i(TAG, "ViewPager size = " + mViewPagers.size());
        }
    }

    @Override
    protected void dispatchDraw(Canvas canvas) {
        super.dispatchDraw(canvas);
        if (mShadowDrawable != null && mContentView != null) {

            int left = mContentView.getLeft()
                    - mShadowDrawable.getIntrinsicWidth();
            int right = left + mShadowDrawable.getIntrinsicWidth();
            int top = mContentView.getTop();
            int bottom = mContentView.getBottom();

            mShadowDrawable.setBounds(left, top, right, bottom);
            mShadowDrawable.draw(canvas);
        }

    }
   //当滑动布局超过1/2抬手时,用户抬手,但用户认为这样算是完成了操作,接下来隐藏另一部分contentView的工作由系统完成
    private void scrollRight() {
   //计算出剩余的宽度 向右换滑动mContentView.getScrollX()负值
        final int delta = (viewWidth + mContentView.getScrollX());

//参数,起止X,起止Y,目标X,目标Y,持续时间
//同时会调用computeScoller共同完成滑动       mScroller.startScroll(mContentView.getScrollX(), 0, -delta + 1, 0,
                Math.abs(delta));
        postInvalidate();
    }


    private void scrollOrigin() {
        int delta = mContentView.getScrollX();
        mScroller.startScroll(mContentView.getScrollX(), 0, -delta, 0,
                Math.abs(delta));
        postInvalidate();
    }

    @Override
    public void computeScroll() {

        if (mScroller.computeScrollOffset()) {           mContentView.scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
            postInvalidate();
       //完成滑动时销毁页面     
       if (mScroller.isFinished() && isFinish) {
                mActivity.finish();
            }
        }
    }
}

使用的时候让Activity继承SwipeBackActivity 这是一个空实现的Activity

public class SwipeBackActivity extends AppCompatActivity {
    protected SwipeBackLayout layout;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        layout = (SwipeBackLayout) LayoutInflater.from(this).inflate(
                R.layout.base, null);
        layout.attachToActivity(this);
    }


    @Override
    public void startActivity(Intent intent) {
        super.startActivity(intent);
        overridePendingTransition(R.anim.base_slide_right_in, R.anim.base_slide_remain);
    }




    // Press the back button in mobile phone
    @Override
    public void onBackPressed() {
        super.onBackPressed();
        overridePendingTransition(0, R.anim.base_slide_right_out);
        //第一动画代表下一个即将出来的页的进入动画效果,第二个代表当前页的退出动画效果
//      overridePendingTransition(R.anim.base_slide_right_in,R.anim.base_slide_left_out );
    }
}

注意:在style中设置activity的容器层的(window层)ViewGroup的背景为透明


    <style name="Theme.AppCompat.Translucent" parent="AppTheme">
        <item name="android:windowBackground">@android:color/transparent</item>
        <item name="android:colorBackgroundCacheHint">@null</item>
        <item name="android:windowIsTranslucent">true</item>
        <item name="android:windowAnimationStyle">@android:style/Animation</item>
    </style>

同时给activity的layout布局View设置背景background.否则会使整个页面为透明

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android返回方案,支持SDK19(Android4.4)及以上。目录示例demo功能介绍实现原理集成方式使用方式注意事项示例demoDemo下载示例效果点此下载 或扫描下面二维码功能介绍支持SDK19(Android4.4)及以上Activity的侧返回,同时实现沉浸式状态栏。实现原理侧时利用反射将窗口转为透明,结束后再利用反射将窗口还原为不透明。详细分析集成方式在module的build.gradle中添加如下代码    dependencies {         implementation 'cn.simonlee.widget:swipeback:1.0.10'     }使用方式STEP.1在Activity的styles中配置如下属性<item name="android:windowBackground">@android:color/transparent</item>STEP.2在Activity的onCreate方法中获取SwipeBackHelper实例@Overridepublic void onCreate(Bundle savedInstanceState) {    super.onCreate(savedInstanceState);     setContentView(R.layout.activity_swipeback);    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {         mSwipeBackHelper = new SwipeBackHelper(this);        //设置窗口背景颜色,以覆盖不可见区域的黑色背景(不可见区域常见为当输入法及导航栏变化时的背景)         mSwipeBackHelper.setWindowBackgroundColor(getResources().getColor(R.color.colorWindowBackground));     } }STEP.3在Activity的dispatchTouchEvent和onTouchEvent中分发触摸事件,如果仅希望侧边触发,可以不用onTouchEvent@Overridepublic boolean dispatchTouchEvent(MotionEvent event) {    if (mSwipeBackHelper != null) {         mSwipeBackHelper.dispatchTouchEvent(event);     }    return super.dispatchTouchEvent(event); }@Overridepublic boolean onTouchEvent(MotionEvent event) {    if (mSwipeBackHelper != null) {         mSwipeBackHelper.onTouchEvent(event);     }    return super.onTouchEvent(event); }注意事项Tips.1如需要适配SDK21(Android5.0)以下,必须在styles中配置如下属性。SDK21及以上无需此属性<item name="android:windowIsTranslucent">true</item>Tips.2因状态栏透明,输入法的adjustPan模式不会生效,建议设置为adjustResizeTips.3因状态栏透明,布局会从屏幕顶端开始绘制,需自行调整paddingTop//获取状态栏的高度public int getStatusBarHeight() {    int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android");    try {        return getResources().getDimensionPixelSize(resourceId);     } catch (Resources.NotFoundException e) {        return 0;     } }Tips.4侧的核心原理是利用反射转换窗口透明,这会影响到下层Activity的生命周期。侧过程中窗口转为透明时,下层Activity会被唤醒,进入onStart状态,如果发生屏幕旋转,下层Activity还将会进行重建。当我们将窗口恢复为不透明,下层Activity会重新进入onStop状态。因此如果你的Activity代码逻辑比较混乱,使用之前务必进行逻辑优化。Tips.5当顶层Activity方向与下层Activity方向不一致时侧会失效(下层方向未锁定除外),请关闭该层Activity侧功能。 示例场景:竖屏界面点击视频,进入横屏播放。Tips.6如需动态支持横竖屏切换(比如APP中有“支持横屏”开关),屏幕方向需指定为behind跟随栈底Activity方向,同时在onCreate中进行判断,若不支持横竖屏切换则锁定屏幕方向(因为经测试SDK21中behind无效)。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值