zoukankan      html  css  js  c++  java
  • 又折腾到这么晚 , 图片Viewpager PagerIndicator,listview 和侧边栏滑动的事件处理

    代码 思路 根据坐标判断 事件是否拦截 调用 getParent().requestDisallowInterceptTouchEvent(true);方法告诉上层ViewGroup 是否拦截 返回true----- 不拦截,传给自己处理 false拦截 ---我不处理

    具体思路是这样:

       当手指触摸到屏幕时,系统就会调用相应View的onTouchEvent,并传入一系列的action。当有多个层级的View时,在父层级允许的情况下,这个action会一直向下传递直到遇到最深层的View。所以touch事件最先调用的是最底层View的onTouchEent,如果View的onTouchEvent接收到某个touch action并作了相应处理,最后有两种返回方式return true和return false;return true会告诉系统当前的View需要处理这次的touch事件,以后的系统发出的ACTION_MOVE,ACTION_UP还是需要继续监听并接收的,而且这次的action已经被处理掉了,父层的View是不可能出发onTouchEvent了。所以每一个action最多只能有一个onTouchEvent接口返回true。如果return false,便会通知系统,当前View不关心这一次的touch事件,此时这个action会传向父级,调用父级View的onTouchEvent。但是这一次的touch事件之后发出的任何action,该View都不会再接受,onTouchEvent在这一次的touch事件中再也不会触发,也就是说一旦View返回false,那么之后的ACTION_MOVE,ACTION_UP等ACTION就不会在传入这个View,但是下一次touch事件的action还是会传进来的。
        前面说了底层的View能够接收到这次的事件有一个前提条件:在父层级允许的情况下。假设不改变父层级的dispatch方法,在系统调用底层onTouchEvent之前会先调用父View的onInterceptTouchEvent方法判断,父层View是不是要截获本次touch事件之后的action。如果onInterceptTouchEvent返回了true,那么本次touch事件之后的所有action都不会再向深层的View传递,统统都会传给负层View的onTouchEvent,就是说父层已经截获了这次touch事件,之后的action也不必询问onInterceptTouchEvent,在这次的touch事件之后发出的action时onInterceptTouchEvent不会再次调用,知道下一次touch事件的来临。如果onInterceptTouchEvent返回false,那么本次action将发送给更深层的View,并且之后的每一次action都会询问父层的onInterceptTouchEvent需不需要截获本次touch事件。只有ViewGroup才有onInterceptTouchEvent方法,因为一个普通的View肯定是位于最深层的View,touch事件能够传到这里已经是最后一站了,肯定会调用View的onTouchEvent。
    对于底层的View来说,有一种方法可以阻止父层的View截获touch事件,就是调用getParent().requestDisallowInterceptTouchEvent(true);方法。一旦底层View收到touch的action后调用这个方法那么父层View就不会再调用onInterceptTouchEvent了,也无法截获以后的action。
    用例子总结一下onInterceptTouchEvent和onTouchEvent的调用顺序:
    假设最高层View叫OuterLayout,中间层View叫InnerLayout,最底层View叫MyVIew。调用顺序是这样的(假设各个函数返回的都是false)
    OuterLayout.onInterceptTouchEvent->InnerLayout.onInterceptTouchEvent->MyView.onTouchEvent->InnerLayout.onTouchEvent->OuterLayout.onTouchEvent。

     

    public class HorizontalScrollViewPager extends ViewPager {
    
        int startX;
        int startY;
    
        public HorizontalScrollViewPager(Context context, AttributeSet attrs) {
            super(context, attrs);
        }
    
        public HorizontalScrollViewPager(Context context) {
            super(context);
        }
    
        /**
         * 分情况决定父控件是否需要拦截事件
         * 
         * 1. 上下划动需要拦截 2. 向右划&第一个页面,需要拦截 3. 向左划&最后一个页面, 需要拦截
         */
        @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
            switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                startX = (int) ev.getX();
                startY = (int) ev.getY();
                // 请求父控件及祖宗控件不要拦截事件
                getParent().requestDisallowInterceptTouchEvent(true);
                break;
            case MotionEvent.ACTION_MOVE:
                int endX = (int) ev.getX();
                int endY = (int) ev.getY();
    
                int dx = endX - startX;
                int dy = endY - startY;
    
                if (Math.abs(dx) > Math.abs(dy)) {// 左右划
                    if (dx > 0) {// 向右滑动
                        if (this.getCurrentItem() == 0) {
                            // 第一个页面
                            // 请求父控件及祖宗控件拦截事件
                            getParent().requestDisallowInterceptTouchEvent(false);
                        }
                    } else {
                        // 向左滑动
                        if (getCurrentItem() == this.getAdapter().getCount() - 1) {
                            // 最后一个item
                            // 请求父控件及祖宗控件拦截事件
                            getParent().requestDisallowInterceptTouchEvent(false);
                        }
                    }
    
                } else {
                    // 上下滑动
                    // 请求父控件及祖宗控件拦截事件
                    getParent().requestDisallowInterceptTouchEvent(false);
                }
    
                break;
    
            default:
                break;
            }
    
            return super.dispatchTouchEvent(ev);
        }
    
    }

     

     

    1 如果Viewpager图片轮播是第一个页面 那么 我向右滑动 我要划出侧滑菜单,所以我给false 事件交给父层处理,我不处理

    2 如果是上下滑动 这个事件我不处理 交给listview

    3 如果是左滑动 并且Viewpager图片轮播的最后一个图片 我给false 交给他的父层来控制 其他情况我来消费这个touch事件

    模拟器我给关了今天太晚了 我要睡了 明天截图给大家看也许更形象

  • 相关阅读:
    nyoj 202 红黑树
    nyoj 237 游戏高手的烦恼
    nyoj 203 三国志
    nyoj 118 修路方案
    nyoj 714 Card Trick
    nyoj 710 外星人的供给站
    nyoj 712探 寻 宝 藏
    nyoj 709 异 形 卵
    nyoj 711 最舒适的路线
    HTML5表格简单应用案例之[招聘需求表]
  • 原文地址:https://www.cnblogs.com/AceIsSunshineRain/p/5187398.html
Copyright © 2011-2022 走看看