手势响应在RN端和Web端对比分析

在移动端应用开发过程中,手势是一个不可忽视的组成部分。ReactNative针对应用中的手势交互方式,提供了一系列解决方案,包括最基本的点击手势、复杂的responder手势以及滚动手势。本文结合原理和示例,分析RN端不同手势系统解决方案,并与Web端手势系统进行对比。文本所指的Web端,仅限于移动Web端,在PC端有另外一套鼠标的交互方式。

简单的点击事件

移动应用中,点击手势是最简单也是最常用的手势。不管是在RN端还是在Web端,点击事件都已经被系统(RN)或者浏览器(Web)进行了封装,可以认为是各自平台下手势响应系统的简单情况。

一、RN端的Touchable系列组件

RN端提供了四个可以添加点击事件的组件,分别是TouchableHighlight、TouchableNativeFeedback、TouchableOpacity以及TouchableWithoutFeedback。
使用这四个组件,我们可以绑定点击事件回调。这四个组件的区别在于点击时,RN封装了不同的点击效果。
Touchable系列组件可以给以下四个方法绑定回调函数:onPressIn、onPressOut、onPress以及onLongPress。通过这四个方法可以实现按下、抬起、点击、长按的交互方式。

<TouchableOpacity
    style={{width: 300, height: 300, backgroundColor: 'red'}}
    onPressIn={()=> {console.log('press in')}}
    onPressOut={()=> {console.log('press  out')}}
    onPress={()=>{console.log('press')}}
    onLongPress={()=>{console.log('long press')}}>
    <View/>
</TouchableOpacity>

问题:内外多层添加点击事件的Touchable系列组件,点击时回调如何触发,为什么?
(1) 只触发内层;(2)只触发外层;
(3)先触发内层再触发外层; (4)先触发外层再触发内层

二、Web端的click事件

在Web端,可以通过DOM0级事件处理函数onclick或者DOM2级事件处理函数addEventListener('click',callback)来绑定点击事件回调。
在Web端,所有DOM元素都支持添加click点击事件。在移动端,click事件只能实现手指点击的交互方式。如果要给按下、抬起、长按手势添加回调函数,则需要借助web端手势响应系统。

var div = document.getElementById('div');
div.onclick = function () {
    console.log('div');
}
div.addEventListener('click', function () {
    console.log('div');
})

问题:当内外多层添加点击事件的DOM元素,点击时回调如何触发,为什么?
(1)只触发内层;(2)只触发外层;
(3) 先触发内层再触发外层; (4)先触发外层再触发内层

三、RN端与Web端对比

在简单点击事件方面,RN端与Web端主要有以下区别:
1. RN端只提供了四个可以添加点击事件的组件;Web端几乎所有DOM元素都支持点击事件。
2. RN端支持按下、抬起、点击、长按四种交互方式;Web端只支持点击交互方式,如果要实现其他交互方式需要借助Web端手势响应系统。
3. RN端四个组件分别实现了不同的“点击态”,手指按下时出现透明度、颜色等视觉变化;Web端需要借助手势响应系统以及CSS样式实现。
4. 当内外多层添加点击事件,RN端只触发内层组件的事件回调函数(onResponderTerminationRequest默认返回false,当有其他组件请求接替响应者时,当前组件不放权), Web端会先触发内层再触发外层的事件回调函数(stopPropagation默认不阻止事件冒泡)。

手势响应系统

RN端和Web端都有各自的手势响应系统,解决用户各种复杂手势交互问题。在RN端,通过Responder或PanResponder实现复杂手势交互方式,其中PanResponder是对Responder的封装。在Web端,通过Touch events实现复杂手势交互方式。

一、RN端的手势响应系统

  1. responder响应者的声明周期
    (1) 冒泡期是否响应
onStartShouldSetResponder: (evt) => true

在用户开始触摸的时候,是否愿意成为响应者。

onMoveShouldSetResponder: (evt) => true

在每一个触摸点开始移动时,是否愿意成为响应者。
(2) 响应过程

onResponderGrant: (evt) => {} 

响应触摸事件回调。

onResponderMove: (evt) => {} 

用户正在屏幕上移动手指时。

onResponderRelease: (evt) => {} 

触摸操作结束时触发。
(3) 响应放权与失败

onResponderReject: (evt) => {} 

响应者现在“另有其人”。

onResponderTerminationRequest: (evt) => true 

有其他组件请求接替响应者,当前的View是否“放权”。

onResponderTerminate: (evt) => {} 

响应者权力已经交出。

  1. 捕获事件处理
    onStartShouldSetResponder与onMoveShouldSetResponder是以冒泡的形式调用的,即嵌套最深的节点最先调用。这意味着当多个View同时返回true时,最底层的View将优先“夺权”。
    有些时候,某个父View会希望能先成为响应者。我们可以利用“捕获期”来解决这一需求。响应系统在从最底层的组件开始冒泡之前,会首先执行一个“捕获期”,在此期间会触发on*ShouldSetResponderCapture系列事件。如果某个父View想要在触摸操作开始时阻止子组件成为响应者,那就应该处理onStartShouldSetResponderCapture事件并返回true值。
    (1) 捕获期是否响应
    onStartShouldSetResponderCapture: (evt) => true,
    onMoveShouldSetResponderCapture: (evt) => true,
  1. PanResponder
    除了responder之外,RN还抽象出了一套PanResponder类。它提供了一个对触摸响应系统响应器的进一步封装。对于每一个处理函数,它在原生事件之外提供了一个新的gestureState对象。这个gestureState包含手势进行过程中更多的信息,例如dx/dy- 从触摸操作开始时的累计横/纵向路程;vx/vy - 当前的横/纵向移动速度;numberActiveTouches-当前在屏幕上的有效触摸点的数量。
    与responder相比,PanResponder的优势在于提供了更详细的触摸对象信息,特别是dx/dy,简化了代码逻辑书写。

  2. RN端手势系统总结
    (1) RN端手势系统分为捕获阶段和冒泡阶段,在两个阶段都对View是否响应手势系统进行判断。
    (2) RN端手势响应系统默认只有一个responder响应者。正因为如此,才存在onResponderReject(响应失败)、onResponderTerminationRequest(是否放权)、onResponderTerminate(放权回调)三个方法。
    (3) 当有多层手势响应系统时,onResponderTerminationRequest方法是解决手势冲突的关键,对responder响应对象是否变化进行协商。
    (4) onResponderTerminationRequest接口的返回值不固定,一般为false,表示当前View不放权;当双指点击时,默认返回true,表示当前View放权。

  3. RN端手势系统实现报表冻结滚动

this.panResponder = PanResponder.create({
    onMoveShouldSetPanResponder: (e, gestureState) => {
        const { bodyTable } = this;
        this.lastScrollLeft = bodyTable.scrollLeft;
        this.lastScrollTop = bodyTable.scrollTop;
        return true;
    },
    onPanResponderTerminationRequest: (e, gestureState) => {
        const {bodyTable} = this;
        const {dx, dy} = gestureState;
        if((Math.abs(dx) > Math.abs(dy) && this.lastScrollLeft === bodyTable.scrollLeft) || (Math.abs(dx) < Math.abs(dy) && this.lastScrollTop === bodyTable.scrollTop)) {
            return true;
        } 
        return false;
    },
    onPanResponderMove: (e, gestureState) => {
        const {dx, dy} = gestureState;
        const target = e.currentTarget;
        const { headTable, bodyTable, leftBodyTable } = this;
        if(target === headTable) {
            this._setScrollLeft(dx);
        } else if(target === bodyTable) {
            this._setScrollLeft(dx);
            this._setScrollTop(dy);
        } else if(target === leftBodyTable) {
            this._setScrollTop(dy);
        }
    }
})

二、Web端的touch手势响应系

  1. Web端四种touch事件
touchstart: (evt) => {}

响应触摸事件回调。

touchmove: (evt) => {}

用户正在屏幕上移动手指时。

touchend: (evt) => {}

触摸操作结束时触发。

touchcancel: (evt) => {}

系统取消touch事件时触发。

  1. 事件捕获与事件冒泡
    Web端手势响应系统可以设置在事件捕获阶段触发或者在事件冒泡阶段触发,通过addEventListener进行设置。默认在冒泡阶段触发。
target.addEventListener(type, listener, useCapture);
  1. 手势冲突解决
    Web端手势响应系统通过阻止事件冒泡的方式解决滚动冲突问题。默认不阻止事件冒泡。
e.stopPropagation();
  1. Web端手势系统实现报表冻结滚动
this.touchResponder = {
    onTouchStart: (e) => {
        const maxScrollLeft = this.tableManager.tableMaxScrollLeft();
        const maxScrollTop = this.tableManager.tableMaxScrollTop();
        if(e.touches.length === 1) {
            if(maxScrollLeft > 0 || maxScrollTop > 0){
                const { bodyTable } = this;
                startX = e.touches[0].pageX;
                startY = e.touches[0].pageY;
                this.lastScrollLeft = bodyTable.scrollLeft;
                this.lastScrollTop = bodyTable.scrollTop;
            }
        }
    },
    onTouchMove: (e) => {
        const maxScrollLeft = this.tableManager.tableMaxScrollLeft();
        const maxScrollTop = this.tableManager.tableMaxScrollTop();
        if(e.touches.length === 1) {
            const target = e.currentTarget;
            const { headTable, bodyTable, leftBodyTable } = this;
            const scrollLeftFn = (dx) => {
                if(maxScrollLeft > 0 && (target === headTable || target === bodyTable)) {
                    const scrollLeft = this.lastScrollLeft - dx;
                    if(scrollLeft <= 0) {
                        this._setScrollLeft(0);
                    } else if(scrollLeft >= maxScrollLeft) {
                        this._setScrollLeft(maxScrollLeft);
                    } else {
                        this._setScrollLeft(scrollLeft);
                        return true;
                    }
                }
            };
            const scrollTopFn = (dy) => {
                if(maxScrollTop > 0 && (target === leftBodyTable || target === bodyTable)) {
                    const scrollTop = this.lastScrollTop - dy;
                    if(scrollTop <= 0) {
                        this._setScrollTop(0);
                    } else if(scrollTop >= maxScrollTop) {
                        this._setScrollTop(maxScrollTop);
                    } else {
                        this._setScrollTop(scrollTop);
                        return true;
                    }
                }
            };
            const endX = e.touches[0].pageX;
            const dx = endX - startX;
            const endY = e.touches[0].pageY;
            const dy = endY - startY;
            const isScrollLeft = scrollLeftFn(dx);
            const isScrollTop = scrollTopFn(dy);
            let preventOuterEvent = false;
            if(Math.abs(dx) > 2 * Math.abs(dy)) {
                if(isScrollLeft) {
                    preventOuterEvent = true;
                }
            } else if(Math.abs(dy) > 2 * Math.abs(dx)) {
                if(isScrollTop) {
                    preventOuterEvent = true;
                }
            } else {
                if(isScrollLeft || isScrollTop) {
                    preventOuterEvent = true;
                }
            }
            if(preventOuterEvent) {
                e.stopPropagation();
                e.preventDefault();
            }
        }
    },
    onTouchEnd: () => {}

三、RN端与Web端手势系统对比

在手势响应系统方面,RN端与Web端对比如下:
(1) RN端和Web端手势系统都可以指定事件触发时机(捕获阶段/冒泡阶段),RN端是通过响应者的声明方法进行指定,Web端是通过addEventListener事件处理函数的第三个参数进行指定。
(2) 当有多层视图添加手势响应系统时,RN端默认只有一个responder响应者。Web端默认多层全部都是响应者。
(3) RN端通过onResponderTerminationRequest手势系统方法的方式解决内外层手势冲突,比较方便。Web端通过stopPropagation阻止事件冒泡的方式解决内外层手势冲突。
(4) RN端的PanResponder手势系统的封装性更好,提供了从触摸操作开始时的累计横/纵向路程,简化开发者的代码逻辑。
(5) RN端提供了更全面的接口方法,包括响应阶段、响应过程、响应放权与响应失败。与RN端提供的方法相比,Web端相当与只提供了响应过程的接口方法。

滚动事件

一、滚动的两种实现方式

  1. RN端两种滚动实现方式
    方式一:ScrollView组件实现滚动
    方式二:View和responder手势响应系统实现滚动
    问题:在原生端,这两种滚动实现方式原理相同吗?
  2. Web端两种滚动实现方式
    方式一:CSS添加overflow:scroll实现滚动。
    方式二:div和touch手势响应系统实现滚动。

二、Web端两种滚动实现方式对比

  1. 底层实现
    scroll滚动浏览器默认支持;touch手势系统滚动要开发者代码实现。
  2. 滚动性能
    scroll滚动性能较好;touch手势系统滚动消耗较大。
  3. 嵌套滚动
    scroll默认支持多层滚动嵌套;touch手势系统开发者手动解决滚动冲突。
  4. 惯性滚动
    scroll默认支持滚动惯性;touch手势系统要手动实现滚动惯性,性能较差。
  5. 滚动条样式
    scroll滚动浏览器默认添加滚动条;touch手势系统需要开发者用div模拟滚动条,比较复杂。
  6. 冻结实现
    touch手势系统监听手势实现行列冻结,没有延迟;scroll通过onScroll方法实现行列冻结,冻结滚动有延迟。
  7. 回调参数
    touch手势系统在手指滑动时,可以通过touch对象获取一系列触摸点数据信息,可以根据触摸点数据信息进行更加灵活的操作;scroll滚动在onScroll方法中无法获取触摸点对象,只能够通过事件对象获取当前滚动对象的滚动位置,不够灵活。

三、 Web端通过overflow:scroll实现报表冻结滚动

handleBodyScrollLeft = e => {
    if (e.currentTarget !== e.target) {
        return;
    }
    const target = e.target;
    const { headTable, bodyTable } = this;
    if (target.scrollLeft !== this.lastScrollLeft) {
        if (target === bodyTable && headTable) {
            headTable.scrollLeft = target.scrollLeft;
        } else if (target === headTable && bodyTable) {
            bodyTable.scrollLeft = target.scrollLeft;
        }
    }
    this.lastScrollLeft = target.scrollLeft;
};

handleBodyScrollTop = e => {
    const target = e.target;
    const { headTable, bodyTable, leftBodyTable } = this;
    if (target.scrollTop !== this.lastScrollTop && target !== headTable) {
        const scrollTop = target.scrollTop;
        if (leftBodyTable && target !== leftBodyTable) {
            leftBodyTable.scrollTop = scrollTop;
        }
        if (bodyTable && target !== bodyTable) {
            bodyTable.scrollTop = scrollTop;
        }
    }
    this.lastScrollTop = target.scrollTop;
};

四、RN端H5适配滚动实现

  1. 在h5适配RN时,滚动可以由三种实现方式:
    (1) react-native-web在底层适配的RN端responder手势系统实现滚动
    (2) touch手势响应系统实现滚动
    (3) overflow:scroll的CSS样式实现滚动
    当第一种滚动实现方式和第二三种滚动实现方式混合使用实现多层嵌套滚动时,会发生难以解决的滚动冲突问题,阻止事件冒泡对第一种滚动实现方式没有效果。
    如果只使用第一种滚动解决方案,优点在于滚动冲突容易解决并且接口封装性更好,缺点在于滚动条和惯性滚动实现复杂并且性能较差。
    如果只使用第二种滚动解决方案,则滚动冲突可以解决,缺点在于滚动条和惯性滚动实现复杂并且性能不好。
    如果只使用第三种滚动解决方案,则优点在于滚动条和惯性滚动浏览器默认实现且性能较好,缺点在于有冻结行列时存在滚动延迟的情况。
    现在H5端在目录、表单以及未冻结行列的报表是通过第三种方式实现滚动,在冻结行列的报表中是通过第二种方式实现滚动的,并通过阻止事件冒泡解决滚动冲突。
  2. 目前存在的问题
    如果内层滚动通过touch手势系统实现,外层滚动通过overflow:scroll实现,则内层阻止事件冒泡可以阻止外层滚动触发,反之内层滚动通过overflow:scroll实现,外层滚动通过touch手势系统实现,则内层阻止事件冒泡无法阻止外层滚动触发,导致下拉刷新时发生滚动冲突。
    如果要实现下拉刷新,则需要全部使用touch手势系统实现滚动。如果要模拟滚动条和滚动惯性,可能借助第三方库会更方面些。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,099评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,828评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,540评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,848评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,971评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,132评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,193评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,934评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,376评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,687评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,846评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,537评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,175评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,887评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,134评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,674评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,741评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,892评论 25 707
  • 目前手机市场上,全面屏时代已经势不可挡,为了增大屏幕,一个个物理按键已渐渐消失在手机上。那么,手势将成为在移动应用...
    SEATELL海说软件阅读 611评论 0 1
  • 本篇文章是讲述 iOS 无埋点数据收集 SDK 系列的第二篇。在第一篇 中主要介绍了 SDK 整体实现思路以及...
    zerygao阅读 12,181评论 4 64
  • 刷下微博,看到热点是林心如和霍建华的结婚新闻,于是点开看了下他们的在巴厘岛的婚礼场景照片,很美很温馨。有情人终成眷...
    燕燕芭比阅读 338评论 0 0
  • 姓名:吴保林 公司:东莞耀升机电有限公司 4/25--27六项精进 245期 反省一组 AT六项精进感谢组 【日精...
    保林阅读 334评论 0 1