美文网首页
滑动冲突处理(一)

滑动冲突处理(一)

作者: 钉某人 | 来源:发表于2018-08-05 17:04 被阅读0次

滑动冲突场景:外层横向滑动,内层竖向滑动。这里通过外层拦截的方式来处理冲突。


外层拦截处理滑动冲突.gif

注意:例子中得布局是ViewGroupOutside包裹两个RecyclerView和两个MyTextView,横向排列。TextView内部是处理点击事件的,所以自定义MyTextView继承TextView,将onTouchEvent()的返回修改为false,那么ViewGroupOutside就可以接收到move和up事件,可以实现横向滑动。

ViewGroupOutside

/**
 * Created by dingmouren
 * email: naildingmouren@gmail.com
 * github: https://github.com/DingMouRen
 * 外部拦截法
 */

public class ViewGroupOutside extends ViewGroup {
    private static final String TAG = "ViewGroupOutside";

    private int mChildrenSize;
    private int mChildWidth;
    private int mChildIndex;
    /*分别记录上次滑动的坐标*/
    private int mLastX = 0;
    private int mLastY = 0;
    /*分别记录上次滑动的坐标(onInterceptTouchEvent)*/
    private int mLastXIntercept = 0;
    private int mLastYIntercept = 0;

    private Scroller mScroller;
    private VelocityTracker mVelocityTracker;


    public ViewGroupOutside(Context context) {
        this(context,null);
    }

    public ViewGroupOutside(Context context, AttributeSet attrs) {
        this(context, attrs,-1);
    }

    public ViewGroupOutside(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        mScroller = new Scroller(getContext());
        mVelocityTracker = VelocityTracker.obtain();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        /*测量所有子view*/
        measureChildren(widthMeasureSpec,heightMeasureSpec);
        /*测量自己*/
        measureSelf(widthMeasureSpec,heightMeasureSpec);

    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        int childCount = getChildCount();
        mChildrenSize = childCount;
        for (int i = 0; i < childCount; i++) {
            View child = getChildAt(i);
            int childWidth = child.getMeasuredWidth();
            int childHeight = child.getMeasuredHeight();
            mChildWidth = childWidth;
            child.layout(childWidth * i,0,childWidth * (i + 1),childHeight);
        }
    }


    /**
     * 外部拦截法
     * @param ev
     * @return
     */
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
       boolean intercepted = false;
       int x = (int) ev.getX();
       int y = (int) ev.getY();
       switch (ev.getAction()){
           case MotionEvent.ACTION_DOWN:
               Log.e(TAG,"intercept-down");
               intercepted = false;
               if (!mScroller.isFinished()){
                   mScroller.abortAnimation();
                   intercepted = true;
               }
               break;
           case MotionEvent.ACTION_MOVE:
               Log.e(TAG,"intercept-move");
               int deltaX = x - mLastXIntercept;
               int deltaY = y - mLastYIntercept;
               if (Math.abs(deltaX) > Math.abs(deltaY)){/*横向滑动*/
                   intercepted = true;
               }else {/*竖向滑动*/
                   intercepted = false;
               }
               break;
           case MotionEvent.ACTION_UP:
               intercepted = false;
               break;
       }
       mLastX = x;
       mLastY = y;
       mLastXIntercept = x;
       mLastYIntercept = y;
        Log.e(TAG,"intercepted:"+intercepted);
       return intercepted;
    }

    /**
     * 横向滑动得处理
     * @param event
     * @return
     */
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        mVelocityTracker.addMovement(event);
        int x = (int) event.getX();
        int y = (int) event.getY();
        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:
                if (!mScroller.isFinished()){
                    mScroller.abortAnimation();
                }
                break;
            case MotionEvent.ACTION_MOVE:
                int deltax = x - mLastX;
                int deltay = y - mLastY;
                scrollBy(-deltax,0);
                break;
            case MotionEvent.ACTION_UP:
                int scrollX = getScrollX();
                int scrollToChildIndex = scrollX / mChildWidth;
                mVelocityTracker.computeCurrentVelocity(1000);
                float xVelocity = mVelocityTracker.getXVelocity();
                if (Math.abs(xVelocity) >= 50){
                    mChildIndex = xVelocity > 0 ? mChildIndex - 1 : mChildIndex + 1;
                }else {
                    mChildIndex = (scrollX + mChildWidth / 2) / mChildWidth;
                }
                mChildIndex = Math.max(0,Math.min(mChildIndex,mChildrenSize - 1));
                int dx = mChildIndex * mChildWidth - scrollX;
                smoothScrollBy(dx,0);
                mVelocityTracker.clear();
                break;
        }
        mLastX = x;
        mLastY = y;
        return super.onTouchEvent(event);
    }


    @Override
    public void computeScroll() {
        if (mScroller.computeScrollOffset()){
            scrollTo(mScroller.getCurrX(),mScroller.getCurrY());
            postInvalidate();
        }
    }


    /**
     * 滑动
     * @param dx
     * @param dy
     */
    private void smoothScrollBy(int dx,int dy){
        mScroller.startScroll(getScrollX(),0,dx,0,500);
        invalidate();
    }

    private void measureSelf(int widthMeasureSpec,int heightMeausreSpec){
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeausreSpec);
        int heightSize = MeasureSpec.getSize(heightMeausreSpec);
        int width = 0;
        int height = 0;
        switch (widthMode){
            case MeasureSpec.EXACTLY:
                width = widthSize;
                break;
            case MeasureSpec.AT_MOST:
                width = widthSize;
                break;
            case MeasureSpec.UNSPECIFIED:
                break;
        }
        switch (heightMode){
            case MeasureSpec.EXACTLY:
                height = heightSize;
                break;
            case MeasureSpec.AT_MOST:
                 height = heightSize;
                 break;
            case MeasureSpec.UNSPECIFIED:
                break;
        }
        setMeasuredDimension(width,height);
    }

}

相关文章

  • 滑动冲突处理(一)

    滑动冲突场景:外层横向滑动,内层竖向滑动。这里通过外层拦截的方式来处理冲突。 注意:例子中得布局是ViewGrou...

  • 滑动冲突处理

    分2种:内部拦截和外部拦截 外部拦截 viewgroup用onInterceptTouchEvent处理,对需要的...

  • Android 嵌套滑动冲突的处理

    移动开发中滑动冲突很常见,也有多种处理方式,今天谈下我现在的处理策略(基本和系统处理策略一致) 最常见的滑动冲突:...

  • 滑动冲突处理(二)

    滑动冲突场景:外层横向滑动,内层竖向滑动。这里通过内层拦截的方式来处理冲突。 注意:例子中得布局是ViewGrou...

  • 详解View的滑动冲突

    什么是滑动冲突? 滑动冲突较难使用语言进行直接描述,如果非要说,那就是:View体系中的滑动事件的实际处理和期望处...

  • Android滑动冲突处理

    导言 Android中的滑动冲突很常见,例如ScrollView/ListView,ViewPager/ViewP...

  • Android滑动冲突处理

    在我们平时项目开发的过程中,是不是会遇到滑动View之间的相互嵌套,比如外部的Scrollview或SwipeRe...

  • View滑动冲突处理

    http://www.jianshu.com/p/7d2c88ca24fc 滑动冲突处理一般在ACTION_MOV...

  • 滑动冲突的处理

    简单来说,处理滑动冲突,主要还是对view的事件分发的拦截的处理,原则就是 哪个view需要响应对应的滑动事件,就...

  • 处理View滑动冲突

    内部拦截法和外部拦截法 外部拦截法 ViewGroup可以拦截掉Move事件但是不能拦截掉Down事件,因为Vie...

网友评论

      本文标题:滑动冲突处理(一)

      本文链接:https://www.haomeiwen.com/subject/dnibvftx.html