Flutter学习笔记(33)--GestureDetector手势识别
如需转载,请注明出处:Flutter学习笔记(33)--GestureDetector手势识别
这篇随笔主要记录的学习内容是GestureDetector手势识别,内容包括识别单击、双击、长按、组件拖拽和缩放处理。
- 单击、双击、长按
先看下demo,很简单,GestureDetector本身也是一个组件,GestureDetector识别其内部子组件的手势动作,GestureDetector的构造方法内给我们提供了onTap单击、onDoubleTap双击、onLongPress长按的是回调方法。
识别到用户的手势操作后会执行对应的回调方法,demo的处理就是简单的更新一下text的文案。
import 'package:flutter/material.dart';
class GestureDetectorDemo extends StatefulWidget {
@override
State<StatefulWidget> createState() {
return _GestureDetectorDemo();
}
}
class _GestureDetectorDemo extends State {
String _operation = "No Gesture detected!"; //保存事件名
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'GestureDetectorDemo',
home: new Scaffold(
appBar: AppBar(
title: Text('GestureDetectorDemo'),
leading: Icon(Icons.arrow_back),
),
body: new GestureDetector(
child: Container(
alignment: Alignment.center,
color: Colors.red,
width: ,
height: ,
child: Text(
_operation,
style: TextStyle(color: Colors.teal),
),
),
onTap: () => setState(() => _operation = 'onTap'),//单机回调
onDoubleTap: () => setState(() => _operation = 'onDoubleTap'),//双击回调
onLongPress: () => setState(() => _operation = 'onLongPress'),//长按回调
),
),
);
}
}
注:这里要说明一下,如果同时监听的onTap和onDoubleTap这两个事件的话,onTap会有200ms的延时,因为因为用户在点击一次之后很有可能会再点击一次来触发双击的事件,所以GestureDetector会等一段时间来确定用户是不是要双击,如果只是监听了onTap而没有监听onDoubleTap的话,就不会有延时了。
- 组件拖动
老样子先看demo再讲解:
import 'package:flutter/material.dart';
class GestureDragDemo extends StatefulWidget {
@override
State<StatefulWidget> createState() {
return _GestureDragDemoState();
}
}
class _GestureDragDemoState extends State<GestureDragDemo> {
double _top = 0.0; //距离顶部的偏移量
double _left = 0.0; //距离底部的偏移量
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'GestureDragDemo',
home: Scaffold(
appBar: AppBar(
title: Text('GestureDragDemo'),
leading: Icon(Icons.keyboard_backspace),
),
body: Stack(
children: <Widget>[
Positioned(
top: _top,
left: _left,
child: GestureDetector(
child: CircleAvatar(
backgroundColor: Colors.red,
child: Text(
'A',
style: TextStyle(color: Colors.white),
),
),
onPanDown: (DragDownDetails downDetails) {
//手指按下时会执行此回调
// print('手指按下的位置:$downDetails.globalPosition');
},
onPanUpdate: (DragUpdateDetails dragUpdateDetails) {
//手指滑动时会执行次回调
setState(() {
//手指滑动时会多次触发onPanUpdate回调,更新偏移量并重新绘制
//dragUpdateDetails.delta.dx获取y轴方向的偏移量
_top += dragUpdateDetails.delta.dy;
//dragUpdateDetails.delta.dy获取x轴方向的偏移量
_left += dragUpdateDetails.delta.dx;
});
},
onPanEnd: (DragEndDetails dragEndDetails) {
//打印滑动结束时在x、y轴上的速度
// print(dragEndDetails.velocity);
},
),
),
],
)),
);
}
}
DragDownDetails.globalPosition:当用户按下时,此属性为用户按下的位置相对于屏幕(而非父组件)原点(左上角)的偏移。DragUpdateDetails.delta:当用户在屏幕上滑动时,会触发多次Update事件,delta指一次Update事件的滑动的偏移量。DragEndDetails.velocity:该属性代表用户抬起手指时的滑动速度(包含x、y两个轴的),示例中并没有处理手指抬起时的速度,常见的效果是根据用户抬起手指时的速度做一个减速动画。
也很简单,我们主要看下onPanUpdate这个回调方法,在我们手指滑动的过程中,会多次执行这个回调,在回调中我们处理下x、y方向的偏移量让后重新绘制就ok了。
偏移量打印:

- 单一方向拖动
上面的拖动是可以往任意方向拖动的,在日常的开发中,有可能会遇到只允许水平(拼图验证)或竖直方向上进行拖动,DestureDetector也为我们提供了对应的响应事件:

看下demo示例:
import 'package:flutter/material.dart';
class GestureDragDemo extends StatefulWidget {
@override
State<StatefulWidget> createState() {
return _GestureDragDemoState();
}
}
class _GestureDragDemoState extends State<GestureDragDemo> {
double _top = 0.0; //距离顶部的偏移量
double _left = 0.0; //距离底部的偏移量
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'GestureDragDemo',
home: Scaffold(
appBar: AppBar(
title: Text('GestureDragDemo'),
leading: Icon(Icons.keyboard_backspace),
),
body: Stack(
children: <Widget>[
Positioned(
top: _top,
left: _left,
child: GestureDetector(
child: CircleAvatar(
backgroundColor: Colors.red,
child: Text(
'A',
style: TextStyle(color: Colors.white),
),
),
onPanDown: (DragDownDetails downDetails) {
//手指按下时会执行此回调
// print('手指按下的位置:$downDetails.globalPosition');
},
// onPanUpdate: (DragUpdateDetails dragUpdateDetails) {
// //手指滑动时会执行次回调
// setState(() {
// //手指滑动时会多次触发onPanUpdate回调,更新偏移量并重新绘制
// //dragUpdateDetails.delta.dx获取y轴方向的偏移量
// _top += dragUpdateDetails.delta.dy;
// print('Y:$dragUpdateDetails.delta.dy');
// //dragUpdateDetails.delta.dy获取x轴方向的偏移量
// _left += dragUpdateDetails.delta.dx;
// print('X:$dragUpdateDetails.delta.dx');
// });
// },
onHorizontalDragUpdate: (DragUpdateDetails dragUpdateDetails){
setState(() {
_left += dragUpdateDetails.delta.dx;
});
},
onPanEnd: (DragEndDetails dragEndDetails) {
//打印滑动结束时在x、y轴上的速度
// print(dragEndDetails.velocity);
},
),
),
],
)),
);
}
}
- 缩放
import 'package:flutter/material.dart';
class GestureScaleDemo extends StatefulWidget{
@override
State<StatefulWidget> createState() {
return _GestureScaleDemoState();
}
}
class _GestureScaleDemoState extends State {
double _width = 200.0;
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'GestureScaleDemo',
home: Scaffold(
appBar: AppBar(
title: Text('GestureScaleDemo'),
),
body: Center(
child: GestureDetector(
child: Image.asset('images/banner.png',width: _width),
onScaleUpdate: (ScaleUpdateDetails scaleUpdateDetails){
setState(() {
//缩放倍数在0.8到10倍之间
_width=*scaleUpdateDetails.scale.clamp(., 10.0);
});
},
),
),
),
);
}
}
Flutter学习笔记(33)--GestureDetector手势识别的更多相关文章
- Flutter学习笔记(36)--常用内置动画
如需转载,请注明出处:Flutter学习笔记(36)--常用内置动画 Flutter给我们提供了很多而且很好用的内置动画,这些动画仅仅需要简单的几行代码就可以实现一些不错的效果,Flutter的动画分 ...
- Flutter学习笔记(41)--自定义Dialog实现版本更新弹窗
如需转载,请注明出处:Flutter学习笔记(41)--自定义Dialog实现版本更新弹窗 功能点: 1.更新弹窗UI 2.强更与非强更且别控制 3.屏蔽物理返回键(因为强更的时候点击返回键,弹窗会消 ...
- Flutter学习笔记(3)--Dart变量与基本数据类型
一.变量 在Dart里面,变量的声明使用var.Object或Dynamic关键字,如下所示: var name = ‘张三’: 在Dart语言里一切皆为对象,所以如果没有将变量初始化,那么它的默认值 ...
- Flutter学习笔记(4)--Dart函数
如需转载,请注明出处:Flutter学习笔记(4)--Dart函数 Dart是一个面向对象的语言,所以函数也是对象,函数属于Function对象,函数可以像参数一样传递给其他函数,这样便于做回调处理: ...
- Flutter学习笔记(5)--Dart运算符
如需转载,请注明出处:Flutter学习笔记(5)--Dart运算符 先给出一个Dart运算符表,接下来在逐个解释和使用.如下: 描述 ...
- Flutter学习笔记(6)--Dart异常处理
如需转载,请注明出处:Flutter学习笔记(6)--Dart异常处理 异常是表示发生了意外的错误,如果没有捕获异常,引发异常的隔离程序将被挂起,并且程序将被终止: Dart代码可以抛出并捕获异常,但 ...
- Flutter学习笔记(8)--Dart面向对象
如需转载,请注明出处:Flutter学习笔记(7)--Dart异常处理 Dart作为高级语言,支持面向对象的很多特性,并且支持基于mixin的继承方式,基于mixin的继承方式是指:一个类可以继承自多 ...
- Flutter学习笔记(9)--组件Widget
如需转载,请注明出处:Flutter学习笔记(9)--组件Widget 在Flutter中,所有的显示都是Widget,Widget是一切的基础,我们可以通过修改数据,再用setState设置数据(调 ...
- Flutter学习笔记(10)--容器组件、图片组件
如需转载,请注明出处:Flutter学习笔记(10)--容器组件.图片组件 上一篇Flutter学习笔记(9)--组件Widget我们说到了在Flutter中一个非常重要的理念"一切皆为组件 ...
随机推荐
- redis的安装和简单操作
安装gcc 目地是编译软件 yum install gcc-c++ 1.拷贝并解压 2.编译文件 到解压目录下 执行 make 进行编译依赖项 cd /deps make hiredis lua j ...
- Redis学习笔记(2)
一.配置文件(部分) 1. UNITS(单位) 数据单位 2. INCLUDES(包含) 可以包含其他配置文件,而redis.conf作为总的配置文件 3. NETWORK(网络配置) -网络端口的绑 ...
- RabbitMq和ZeroMq
RabbitMQ和ZeroMQ都是极好的消息中间件,下我会对这两个消息中间件做一个比較,个人理解不喜勿喷. RabbitMQ是AMQP协议率先的一个实现,它实现了代理(Broker)架构,意味着消息在 ...
- Java获取主板序列号、MAC地址、CPU序列号工具类
import java.io.File; import java.io.FileWriter; import java.io.BufferedReader; import java.io.IOExce ...
- 【C++】C++数据类型
注意:以下内容摘自文献[1],修改了部分内容. 计算机处理的对象是数据,而数据是以某种特定的形式存在的(例如整数.浮点数.字符等形式).数据结构指的是数据的组织形式.例如,数组就是一种数据结构. 1. ...
- 使用jQuery实现Ajax
jQuery对Ajax操作进行了封装,在jQuery中最底层的方法是$.ajax(), 第二层是load(), $.get(), $.post() 第三层是$.getScript(), $.g ...
- 居然还有人这样解说mybatis运行原理
目录 Mybatis基本认识 动态代理 JDK实现 CGLIB动态代理 总结 反射 Configuration对象作用 映射器结构 sqlsession执行流程(源码跟踪) Executor Stat ...
- Java实现 蓝桥杯 算法提高 抽卡游戏
试题 算法提高 抽卡游戏 某个抽卡游戏卡池抽出限定卡的概率为p,该游戏有一个"井"的机制,抽满k次卡后直接送这张限定卡.试求获得这张限定卡需要的期望抽卡次数.输入为一行,用空格隔开 ...
- Java实现 LeetCode 543. 二叉树的直径(遍历树)
543. 二叉树的直径 给定一棵二叉树,你需要计算它的直径长度.一棵二叉树的直径长度是任意两个结点路径长度中的最大值.这条路径可能穿过也可能不穿过根结点. 示例 : 给定二叉树 1 / \ 2 3 / ...
- Java实现 蓝桥杯VIP 算法训练 数对
算法训练 数对 时间限制:1.0s 内存限制:512.0MB 问题描述 编写一个程序,该程序从用户读入一个整数,然后列出所有的数对,每个数对的乘积即为该数. 输入格式:输入只有一行,即一个整数. 输出 ...