宁泽林_NiZerin - 互联网技术博客

  • PHP
  • Go
  • Java
  • Rust
  • Python
  • 交流群
  • 关于我
  • 留言版
  1. 首页
  2. Flutter
  3. 正文

Flutter 基础(八)手势相关 Widget:GestureDetector 和 Dismissible

2019年7月16日 2711点热度 0人点赞 3条评论
Flutter基础(八)手势相关Widget:GestureDetector和Dismissible

前言

移动开发中,用户交互是一个重要的环节,在 Android 中的触摸、点击、滑动等事件处理都提供了相关的 Api,在 Flutter 中也是一样的,是由 Widget 来实现的。
Flutter 中的手势系统有两个独立的层。第一层是原始指针事件(pointer events),它描述了屏幕上指针,比如触摸、鼠标、触控笔的位置和移动。 第二层是手势,由一个或多个指针移动组成的动作会被识别为不同的手势。

1. 指针事件

指针表示用户与设备屏幕交互的原始数据。有四种类型的指针事件:

  • PointerDownEvent:指针接触到屏幕的特定位置。
  • PointerMoveEvent: 指针已从屏幕上的一个位置移动到另一个位置。
  • PointerUpEvent: 指针已停止接触屏幕。
  • PointerCancelEvent:此指针的输入不再指向此应用,通俗来讲就是事件取消。

在指针按下时,Flutter 框架会对当前应用程序执行命中测试,以确定指针与屏幕接触的位置存在哪个 Widget 上,然后将 PointerDownEvent 事件(以及该指针的后续事件)调度到命中测试找到的最内部的 Widget,事件的分配路径为:从最里面的 Widget 到树的根路径上的所有 Widget。

2. 手势

手势表示由一个或多个指针移动组成的动作。主要有以下几种:
点击
onTapDown:指针已经在特定位置与屏幕接触。
onTapUp:指针停止在特定位置与屏幕接触。
onTap :点击事件触发。
onTapCancel: 先前指针触发的 onTapDown 不会再触发点击事件。
双击
onDoubleTap:用户快速连续两次在同一位置轻敲屏幕。
长按
onLongPress:指针在相同位置长时间保持与屏幕接触。
垂直拖动
onVerticalDragStart:指针已经与屏幕接触并可能开始垂直移动。
onVerticalDragUpdate 指针与屏幕接触并已沿垂直方向移动。
onVerticalDragEnd 先前与屏幕接触并垂直移动的指针不再与屏幕接触,并且在停止接触屏幕时以特定速度移动。
水平拖动
onHorizontalDragStart:指针已经接触到屏幕并可能开始水平移动
onHorizontalDragUpdate:指针与屏幕接触并已沿水平方向移动
onHorizontalDragEnd:先前与屏幕接触并水平移动的指针不再与屏幕接触,并在停止接触屏幕时以特定速度移动。
如何对这些手势进行检测呢?可以使用 GestureDetector。

3. 使用 GestureDetector

要想检测单击、双击、垂直拖动等手势,只要用 GestureDetector 嵌套要检测手势 Widget 并实现想要监听的手势的方法就行。


import 'package:flutter/material.dart';
void main() => runApp(GestureDetectorWidget());
class GestureDetectorWidget extends StatelessWidget {
 @override
 Widget build(BuildContext context) {
 return MaterialApp(
 title: "Flutter",
 home: Scaffold(
 appBar: AppBar(
 title: Text("GestureDetector示例"),
 ),
 body: Center(
 child: GestureDetector(
 child: Text('手势识别'),
 onTap: () {
 print('点击');
 },
 onDoubleTap: () {
 print('双击');
 },
 onLongPress: () {
 print('长按');
 },
 onHorizontalDragStart: (DragStartDetails details) {
 print('水平拖动');
 },
 ),
 ),
 ),
 );
 }
}

只需要在手势识别这个文字上进行操作,那么对应的手势就会被打印出来。

V5ddxA.png

4. 使用 Dismissible

滑动删除这个操作很常见,比如在一个列表中,我们向左滑动,就会直接删除一个条目或者给出删除提示选项。Flutter 提供了 Dismissible 来帮助我们实现滑动删除。


import 'package:flutter/material.dart';
void main() => runApp(DismissibleWidget(
 items: new List<String>.generate(300, (i) => "第$i行"),
 ));
class DismissibleWidget extends StatelessWidget {
 final List<String> items;
 DismissibleWidget({@required this.items});
 @override
 Widget build(BuildContext context) {
 return MaterialApp(
 title: 'Flutter',
 home: Scaffold(
 appBar: AppBar(
 title: Text('Dismissible示例'),
 ),
 body: ListView.builder(
 itemCount: items.length,
 itemBuilder: (context, index) {
 final item = items[index];
 return Dismissible(
 key: Key(item),
 onDismissed: (direction) {
 items.removeAt(index);
 print(index);
 },
 child: ListTile(
 leading: Icon(Icons.access_time),
 title: Text('${items[index]}'),
 ),
 );
 },
 ),
 ),
 );
 }
}

这个例子和 ListView 的例子类似,主要的变化就是用 Dismissible 来嵌套 ListTile。当执行删除操作时,ListView 中的 onDismissed 方法会被回调,我们可以直接在 onDismissed 方法中将被删除的 item 从 List 中移除。

V5dBrt.png

我们向左滑动第一个和第二个 item,会出现删除的动画,结果如下图所示。

V5wpdK.png
本作品采用 知识共享署名 4.0 国际许可协议 进行许可
标签: flutter
最后更新:2019年7月16日

NiZerin

博主已经躺平了,后面很少会更新博客。

打赏 点赞
< 上一篇
下一篇 >

文章评论

  • 137博客

    写什么都不重要,因为写什么都有人看。现在是9012年了 哈哈

    2019年7月25日
    回复
  • yuanchuming

    hjjj

    2019年12月23日
    回复
  • 阿鲍Abble

    阿

    2020年3月19日
    回复
  • razz evil exclaim smile redface biggrin eek confused idea lol mad twisted rolleyes wink cool arrow neutral cry mrgreen drooling persevering
    回复 阿鲍Abble 取消回复

    分类
    • Angular
    • CSS3
    • docker
    • Flutter
    • git
    • Go
    • H5
    • Java
    • JavaScript
    • Laravel
    • linux
    • Node.js
    • PHP
    • Python
    • React
    • redis
    • Vue.js
    • windows
    • WordPress
    • 交流
    • 小程序
    • 工具
    • 网站公告
    标签聚合
    php flutter javascript vue wordpress laravel go translations
    友情链接
    • PHP函数字典
    • 宝塔运维特惠
    • 科学上网
    • 阿里云特惠

    COPYRIGHT © 2021 nizer.in. ALL RIGHTS RESERVED.

    Theme Kratos Made By Seaton Jiang