【React Native】进阶指南之二(手势响应系统)
移动设备上的手势识别要比在 web 上复杂得多。用户的一次触摸操作的真实意图是什么,App 要经过好几个阶段才能判断。比如 App 需要判断用户的触摸到底是在滚动页面,还是滑动一个 widget,或者只是一个单纯的点击。甚至随着持续时间的不同,这些操作还会转化。此外,还有多点同时触控的情况。
手势响应系统可以使组件在不关心父组件或子组件的前提下自行处理触摸交互。
作为与用户交互的第一层,触摸事件直接影响着用户行为体验。在Android 和 iOS 平台设备中,对于触摸机制做了非常完善的封装,能够很方便的帮助开发者处理基本的触摸行为操作,原生平台通过注册Listener的方式可以轻松的实现单击,双击等操作。在RN中同样提供了与Native触摸事件映射一致的处理方式,方便React Native开发者处理触摸行为,定义触摸操作。
RN系统中为我们提供了TouchableHighlight 与 Touchable 系列组件,不懂得自己找度娘就行了。
一、响应者的生命周期

一个View只要实现了正确的协商方法,就可以成为触摸事件的响应者。通过以下两种方法去“询问”一个View是否愿意成为响应者:
View.props.onStartShouldSetResponder: (evt) => true,在用户开始触摸的时候(手指刚刚接触屏幕的瞬间),是否愿意成为响应者;
View.props.onMoveShouldSetResponder: (evt) => true, 如果View不是响应者,那么在每一个触摸点开始移动(没有停下也没有离开屏幕)时再询问一次:是否愿意成为响应者?
如果 View 返回 true,并开始尝试成为响应者,那么会触发下列事件之一:
View.props.onResponderGrant: (evt) => {} View现在要开始响应触摸事件了,这也是需要做高亮的时候,使用户知道他点了哪里。
View.props.onResponderReject: (evt) => {}响应者现在“另有其人”而且暂时不会“放权”,请另作安排。
如果 View 已经开始响应触摸事件了,那么下列这些处理函数会被一一调用:
View.props.onResponderMove: (evt) => {} - 用户正在屏幕上移动手指时(没有停下也没有离开屏幕)。
View.props.onResponderRelease: (evt) => {} - 触摸操作结束时触发,比如"touchUp"(手指抬起离开屏幕)。
View.props.onResponderTerminationRequest: (evt) => true - 有其他组件请求接替响应者,当前的 View 是否“放权”?返回 true 的话则释放响应者权力。
View.props.onResponderTerminate: (evt) => {} - 响应者权力已经交出。这可能是由于其他 View 通过onResponderTerminationRequest请求的,也可能是由操作系统强制夺权(比如 iOS 上的控制中心或是通知中心)。
其中evt是一个合成事件,它包含以下结构:
nativeEvent
- changedTouches - 在上一次事件之后,所有发生变化的触摸事件的数组集合(即上一次事件后,所有移动过的触摸点)
- identifier - 触摸点的 ID
- locationX - 触摸点相对于当前元素的横坐标
- locationY - 触摸点相对于当前元素的纵坐标
- pageX - 触摸点相对于根元素的横坐标
- pageY - 触摸点相对于根元素的纵坐标
- target - 触摸点所在的元素 ID
- timestamp - 触摸事件的时间戳,可用于移动速度的计算
- touches - 当前屏幕上的所有触摸点的集合
二、捕获 ShouldSet 事件处理
onStartShouldSetResponder与onMoveShouldSetResponder是以冒泡的形式调用的,即嵌套最深的节点最先调用。这意味着当多个 View 同时在*ShouldSetResponder中返回 true 时,最底层的 View 将优先“夺权”。在多数情况下这并没有什么问题,因为这样可以确保所有控件和按钮是可用的。
但是有些时候,某个父 View 会希望能先成为响应者。我们可以利用“捕获期”来解决这一需求。响应系统在从最底层的组件开始冒泡之前,会首先执行一个“捕获期”,在此期间会触发on*ShouldSetResponderCapture系列事件。因此,如果某个父 View 想要在触摸操作开始时阻止子组件成为响应者,那就应该处理onStartShouldSetResponderCapture事件并返回 true 值。
View.props.onStartShouldSetResponderCapture: (evt) => true,
View.props.onMoveShouldSetResponderCapture: (evt) => true,
三、高级的手势功能PanResponder
onStartShouldSetPanResponderCapture: (evt, gestureState) => {
// 在触摸事件 开始,RN父布局组件会回调 onStartShouldSetResponderCapture,询问是否要拦截事件,自己接收处理, true 表示拦截。
console.log('onStartShouldSetPanResponderCapture')
console.log(gestureState.dx)
return false;
},
onMoveShouldSetPanResponderCapture: (evt, gestureState) => {
// 在触摸 滑动 事件时,RN父布局组件会回调 onMoveShouldSetResponderCapture,询问是否要拦截事件,自己接收处理, true 表示拦截。
console.log('onMoveShouldSetPanResponderCapture')
console.log(gestureState)
return false;
},
onStartShouldSetPanResponder: (evt, gestureState) => {
/**
* 在手指触摸开始时申请成为响应者
*/
console.log('onStartShouldSetPanResponder')
console.log(gestureState)
return true;
},
onMoveShouldSetPanResponder: (evt, gestureState) => {
/**
* 在手指在屏幕移动时申请成为响应者
*/
console.log('onMoveShouldSetPanResponder')
console.log(gestureState)
return true;
},
onPanResponderGrant: (evt, gestureState) => {
//开始手势操作。给用户一些视觉反馈,让他们知道发生了什么事情!
/**
* 申请成功,组件成为了事件处理响应者,这时组件就开始接收后序的触摸事件输入。
* 一般情况下,这时开始,组件进入了激活状态,并进行一些事件处理或者手势识别的初始化
*/
console.log('onPanResponderGrant')
console.log(gestureState)
},
onPanResponderReject: (evt, gestureState) => {
/**
* 表示申请失败了,这意味者其他组件正在进行事件处理,
* 并且它不想放弃事件处理,所以你的申请被拒绝了,后续输入事件不会传递给本组件进行处理。
*/
console.log('onPanResponderReject')
},
onPanResponderStart:(evt, gestureState) => {
/**
* 表示手指按下时,成功申请为事件响应者的回调
*/
console.log('onPanResponderStart')
console.log(gestureState)
},
onPanResponderMove:(evt, gestureState) => {
//最近一次的移动距离为gestureState.move{X,Y}
// 从成为响应者开始时的累计手势移动距离为gestureState.d{x,y}
/**
* 表示触摸手指移动的事件,这个回调可能非常频繁,所以这个回调函数的内容需要尽量简单
*/
console.log('onPanResponderMove')
console.log(gestureState)
},
onPanResponderRelease:(evt, gestureState) => {
//用户放开了所有的触摸点,且此时视图已经成为了响应者。
//一般来说这个意味着一个手势操作已经完成了。
/**
* 表示触摸完成(touchUp)的时候的回调,表示用户完成了本次的触摸交互,这里应该完成手势识别的处理,
* 这以后,组件不再是事件响应者,组件取消激活
*/
console.log('onPanResponderRelease')
console.log(gestureState)
},
onPanResponderEnd:(evt, gestureState) => {
/**
* 组件结束事件响应的回调
*/
console.log('onPanResponderEnd')
console.log(gestureState)
},
onResponderTerminationRequest: (evt) => {
/**
* 当其他组件申请成为响应者时,询问你是否可以释放响应者角色让给其他组件
*/
console.log('onResponderTerminationRequest');
return true;
},
onResponderTerminate: (evt) => {
/**
* 如果 onResponderTerminationRequest 回调函数返回为 true,
* 则表示同意释放响应者角色,同时会回调如下函数,通知组件事件响应处理被终止
* 这可能是由于其他View通过onResponderTerminationRequest请求的,也可能是由操作系统强制夺权(比如iOS上的控制中心或是通知中心)。
*/
console.log('onResponderTerminate');
}
注释已经说明了,不多做阐述。案例如下:
/**
* PanResponder 触摸事件
* @export
* @class PanResponderView
* @extends {Component}
*/
import React, { Component } from 'react';
import {
View,
Text,
StyleSheet,
PanResponder,
} from 'react-native'; export default class HomeScreen extends Component { constructor(props) {
super(props)
this.panResponder={}
} componentWillMount() {
this.panResponder = PanResponder.create({
onStartShouldSetPanResponderCapture: (evt, gestureState) => {
// 在触摸事件 开始,RN父布局组件会回调 onStartShouldSetResponderCapture,询问是否要拦截事件,自己接收处理, true 表示拦截。
console.log('onStartShouldSetPanResponderCapture')
console.log(gestureState.dx)
return false;
},
onMoveShouldSetPanResponderCapture: (evt, gestureState) => {
// 在触摸 滑动 事件时,RN父布局组件会回调 onMoveShouldSetResponderCapture,询问是否要拦截事件,自己接收处理, true 表示拦截。
console.log('onMoveShouldSetPanResponderCapture')
console.log(gestureState)
return false;
},
onStartShouldSetPanResponder: (evt, gestureState) => {
/**
* 在手指触摸开始时申请成为响应者
*/
console.log('onStartShouldSetPanResponder')
console.log(gestureState)
return true;
},
onMoveShouldSetPanResponder: (evt, gestureState) => {
/**
* 在手指在屏幕移动时申请成为响应者
*/
console.log('onMoveShouldSetPanResponder')
console.log(gestureState)
return true;
},
onPanResponderGrant: (evt, gestureState) => {
//开始手势操作。给用户一些视觉反馈,让他们知道发生了什么事情!
/**
* 申请成功,组件成为了事件处理响应者,这时组件就开始接收后序的触摸事件输入。
* 一般情况下,这时开始,组件进入了激活状态,并进行一些事件处理或者手势识别的初始化
*/
console.log('onPanResponderGrant')
console.log(gestureState)
},
onPanResponderReject: (evt, gestureState) => {
/**
* 表示申请失败了,这意味者其他组件正在进行事件处理,
* 并且它不想放弃事件处理,所以你的申请被拒绝了,后续输入事件不会传递给本组件进行处理。
*/
console.log('onPanResponderReject')
},
onPanResponderStart:(evt, gestureState) => {
/**
* 表示手指按下时,成功申请为事件响应者的回调
*/
console.log('onPanResponderStart')
console.log(gestureState)
},
onPanResponderMove:(evt, gestureState) => {
//最近一次的移动距离为gestureState.move{X,Y} // 从成为响应者开始时的累计手势移动距离为gestureState.d{x,y}
/**
* 表示触摸手指移动的事件,这个回调可能非常频繁,所以这个回调函数的内容需要尽量简单
*/
console.log('onPanResponderMove')
console.log(gestureState)
}, onPanResponderRelease:(evt, gestureState) => {
//用户放开了所有的触摸点,且此时视图已经成为了响应者。
//一般来说这个意味着一个手势操作已经完成了。
/**
* 表示触摸完成(touchUp)的时候的回调,表示用户完成了本次的触摸交互,这里应该完成手势识别的处理,
* 这以后,组件不再是事件响应者,组件取消激活
*/
console.log('onPanResponderRelease')
console.log(gestureState)
},
onPanResponderEnd:(evt, gestureState) => {
/**
* 组件结束事件响应的回调
*/
console.log('onPanResponderEnd')
console.log(gestureState)
}, onResponderTerminationRequest: (evt) => {
/**
* 当其他组件申请成为响应者时,询问你是否可以释放响应者角色让给其他组件
*/
console.log('onResponderTerminationRequest');
return true;
}, onResponderTerminate: (evt) => {
/**
* 如果 onResponderTerminationRequest 回调函数返回为 true,
* 则表示同意释放响应者角色,同时会回调如下函数,通知组件事件响应处理被终止
* 这可能是由于其他View通过onResponderTerminationRequest请求的,也可能是由操作系统强制夺权(比如iOS上的控制中心或是通知中心)。
*/
console.log('onResponderTerminate');
}
});
} render() {
return (
<View {...this.panResponder.panHandlers } style={ styles.container }> </View>
)
}
} const styles = StyleSheet.create({ container: {
width: 100,
height: 100,
borderRadius: 50,
backgroundColor: '#87CEFA'
}, btn: {
width: 100,
height: 60,
alignItems: 'center',
justifyContent: 'center',
backgroundColor: '#ff5511'
}, btnText: {
color: 'white'
}
});
运行效果:

参考文档:https://reactnative.cn/docs/gesture-responder-system/
【React Native】进阶指南之二(手势响应系统)的更多相关文章
- 【独家】React Native 版本升级指南
前言 React Native 作为一款跨端框架,有一个最让人头疼的问题,那就是版本更新.尤其是遇到大版本更新,JavaScript.iOS 和 Android 三端的配置构建文件都有非常大的变动,有 ...
- react native 项目使用 expo 二维码扫描失败
今天学习react native,需使用expo在移动端进行调试. npm start 运行项目后,使用expo扫描二维码,始终没有反应.于是决定采用这个方法: 连上手机打开usb调试后,按下‘a’, ...
- react native进阶
一.前沿||潜心修心,学无止尽.生活如此,coding亦然.本人鸟窝,一只正在求职的鸟.联系我可以直接微信:jkxx123321 二.项目总结 **||**文章参考资料:1. http://blog ...
- React Native入门指南
转载自:http://www.jianshu.com/p/b88944250b25 前言 React Native 诞生于 2015 年,名副其实的富二代,主要使命是为父出征,与 Apple 和 Go ...
- React Native细节知识点总结<二>
1.关于React Native导出组件的export default和export的问题: 一个文件只能有一个export default,可以有多个export export class Temp ...
- React Native 学习笔记--进阶(二)--动画
React Native 进阶(二)–动画 动画 流畅.有意义的动画对于移动应用用户体验来说是非常必要的.我们可以联合使用两个互补的系统:用于全局的布局动画LayoutAnimation,和用于创建更 ...
- React Native指南汇集了各类react-native学习资源、开源App和组件
来自:https://github.com/ele828/react-native-guide React Native指南汇集了各类react-native学习资源.开源App和组件 React-N ...
- React Native初探
前言 很久之前就想研究React Native了,但是一直没有落地的机会,我一直认为一个技术要有落地的场景才有研究的意义,刚好最近迎来了新的APP,在可控的范围内,我们可以在上面做任何想做的事情. P ...
- React Native 之 Touchable 介绍与使用
前言 学习本系列内容需要具备一定 HTML 开发基础,没有基础的朋友可以先转至 HTML快速入门(一) 学习 本人接触 React Native 时间并不是特别长,所以对其中的内容和性质了解可能会有所 ...
随机推荐
- JS---案例:无缝的轮播图
案例:无缝的轮播图 w <!DOCTYPE html> <html> <head lang="en"> <meta charset=&qu ...
- Android 程序分析环境搭建-开发环境搭建
1.1 JDK 安装 JDK 的配置,初学java 开发,那是必须会的. 下载,遇到的问题就是要注册oracle 的账号,还有你要下载特定版本,比如jdk 1.7,jdk 1.6,很难找到在哪里.解 ...
- RF之简介
robot framework 是一个通用型的自动测试框架 - 自动测试用例的实现方式 - 自动测试用例的开发支持 : IDE.库 - 和用例管理系统的集成 - 测试执行:相关测试套件和测试用例的执 ...
- 面试连环炮系列(一):如何保证Redis高可用和高并发
如何保证Redis高可用和高并发? Redis主从架构,一主多从,可以满足高可用和高并发.出现实例宕机自动进行主备切换,配置读写分离缓解Master读写压力. Redis高可用方案具体怎么实施? 使用 ...
- 解决JRebel对myBatis Mapper 失效的问题
解决JRebel对myBatis Mapper 失效的问题 在之前的文章中介绍了JRebel这个插件的使用和优势,虽然它对配置文件的改动的热更新是生效的,但是mybatis的mapper文件的改动却无 ...
- 【STM32H7教程】第33章 STM32H7的定时器应用之TIM1-TIM17的中断实现
完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第33章 STM32H7的定时器应用之TIM1-T ...
- 可编程实验板EPM1270T144C5蜂鸣器音调频率选择
always@(tone) begin case(tone) 'd1 : time_end=10'd1911 ;//L1 'd2 : time_end=10'd1702 ;//L2 'd3 : tim ...
- IDEA去除掉虚线,波浪线,和下划线实线的方法
初次安装使用IDEA,总是能看到导入代码后,出现很多的波浪线,下划线和虚线,这是IDEA给我们的一些提示和警告,但是有时候我们并不需要,反而会让人看着很不爽,这里简单记录一下自己的调整方法,供其他的小 ...
- GPS定位的偏移校正(WGS84与火星坐标互转)
地图坐标系目前包括: 地球坐标 (WGS84) WGS84:World Geodetic System 1984,是为GPS全球定位系统使用而建立的坐标系统. 国际标准,从 GPS 设备中取出的数据的 ...
- APScheduler学习
说明 APScheduler是一个 Python 定时任务框架,使用起来十分方便.提供了基于日期.固定时间间隔以及 crontab 类型的任务,并且可以持久化任务.并以 daemon 方式运行应用. ...