View事件体系 -- 滑动冲突

一、概念

在界面中只要内外两层同时可以滑动,这个时候就会产生滑动冲突。

二、冲突场景

场景1:外部滑动方向和内部滑动方向不一致
外部左右滑动、内部上下滑动;
外部上下滑动,内部左右滑动。
例子:ViewPager + ListView
将ViewPager和Fragment配合使用组成页面滑动效果,可以通过左右滑动来切换页面,而每个页面内部往往又是一个ListView,可以上下滑动。本来这种情况是有滑动冲突的,但是ViewPager内部处理了这种滑动冲突,因此采用ViewPager时我们无须关注这个问题。如果我们采用的不是ViewPager而是ScrollView等,那就必须手动处理滑动冲突了,否则会造成内外两层只能有一层能够滑动。
场景2:外部滑动方向和内部滑动方向一致
外部左右滑动、内部左右滑动;
外部上下滑动,内部上下滑动。
场景3:场景1和场景2两种情况的嵌套
例子:SlideMenu + ViewPager + ListView
外部有一个SlideMenu左右滑动效果,内部有一个ViewPager左右滑动效果,ViewPager的每一个页面中又是一个ListView上下滑动效果。虽然场景3看起来更复杂,但是它是几个单一的滑动冲突的叠加,因此只需要分别处理内层和中层、中层和外层之间的滑动冲突即可。

三、处理规则

场景1:
根据滑动是水平滑动还是竖直滑动来判断到底由谁来拦截事件。可以根据滑动的角度、距离差以及速度差来确定滑动方向。
例子中,当用户左右滑动的时候,需要让外部的View拦截点击事件,当用户上下滑动的时候,需要让内部View拦截点击事件。
场景2:
在业务上找到突破点。比如业务上有规定:当处于某种状态时需要外部View响应用户的滑动,而处于另外一种状态时则需要内部View来响应View的滑动。
场景3:
同场景2一样,只能在业务上找到突破点。

四、滑动冲突的解决方式

1.外部拦截法(推荐)
外部拦截法是指点击事件先经过父容器的拦截处理,如果父容器需要此事件就拦截,如果不需要此事件就不拦截,这样就可以解决滑动冲突问题,这种方法比较符合点击事件的分发机制。外部拦截法需要重写父容器的onInterceptTouchEvent方法,在内部做相应的拦截即可。
伪代码如下:

//重写父容器的onInterceptTouchEvent方法
public boolean onInterceptTouchEvent(MotionEvent event) {
    boolean intercepted = false;
    int x = (int)event.getX();
    int y = (int)event.getY();
    switch(event.getAction()) {
    case MotionEvent.ACTION_DOWN: //按下事件不拦截
        intercepted = false;
        break;
    case MotionEvent.ACTION_MOVE: //滑动事件根据需要拦截
        if(父容器需要当前点击事件) {
            intercepted = true;
        } else {
            intercepted = false;
        }
        break;
    case MotionEvent.ACTION_UP: //松开事件不拦截
        intercepted = false;
        break;
    default:
        break;
    }
    return intercepted;
}

2.内部拦截法
内部拦截法是指父容器不拦截任何事件,所有的事件都传递给子元素,如果子元素需要此事件就直接消耗掉,否则就交给父容器进行处理,这种方法和事件分发机制不一致,需要配合ViewGroup#requestDisallowInterceptTouchEvent方法才能正常工作。
伪代码如下:

//重写子元素的dispatchTouchEvent方法
public boolean dispatchTouchEvent(MotionEvent event) {
    int x = (int)event.getX();
    int y = (int)event.getY();
    switch(event.getAction()) {
    case MotionEvent.ACTION_DOWN: 
        parent.requestDisallowInterceptTouchEvent(true);
        break;
    case MotionEvent.ACTION_MOVE: 
        int deltaX = x - mLastX;
        int deltaY = y - mLastY;
        if(父容器需要当前点击事件) {
            parent.requestDisallowInterceptTouchEvent(false);
        }
        break;
    case MotionEvent.ACTION_UP: 
        break;
    default:
        break;
    }
    mLastX = x;
    mLastY = y;
    return super.dispatchTouchEvent(event);
}

//重写父容器的onInterceptTouchEvent方法
public boolean onInterceptTouchEvent(MotionEvent event) {
    int action = event.getAction();
    if(action == MotionEvent.ACTION_DOWN) {
        return false;
    } else {
        return true;
    }
}

五、实例

问题:
App使用ViewPager + TabLayout + Fragment框架,其中一个fragment的内容是只有一个自定义的webview控件,进行加载H5页面,H5页面上有一个水平方向上的图片轮播控件,当水平滑动该图片轮播控件时会出现App切换Tab现象,即外层ViewPager控件拦截了该滑动事件,导致内层H5页面的图片轮播控件无法在手动滑动的情况下正常切换图片。

方案:
当手指接触到H5页面的图片轮播控件区域时,由JS调用native的接口设置一个标志位为true,表示当前的触摸事件需要交给webview来处理,在webview的事件处理方法onTouchEvent中,在该标志位为true的前提下,如果当前是ACTION_DOWN事件,则调用父容器的requestDisallowInterceptTouchEvent(true)方法不允许父容器拦截事件,如果当前是ACTION_UP或者ACTION_CANCEL事件,则调用父容器的requestDisallowInterceptTouchEvent(false)方法允许父容器拦截事件,并将该标志位设置为false。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容