SwiftUI中的手势(DragGesture拖拽手势及Drag动画组件)

上一篇文章我们了解了如何使用.gesture修饰符和@GestureState属性包装器,让我们看看另一种常见的手势:DragGesture拖拽手势。

下面先看个效果图:
在这里插入图片描述
这个效果中,我们实现了一个Text文本,并添加了拖拽手势,可以拖动到屏幕的任意位置,松手后停留在目标位置,而非回到原来的起点位置。

UI组件就不多说了,我们在Text组件上添加了.gesture修饰符,并在该修饰符内部添加了DragGesture手势,然后给DragGesture添加了.updating.onEnded修饰符。

关于.updating修饰符上一篇文章已经介绍过了(SwiftUI中的手势(TapGesture、LongPressGesture、GestureState的使用)),这里创建了一个@GestureState修饰的dragOffset变量,用于绑定到.updating修饰符上的参数上。在.updating修饰符的body中进行了赋值操作,这样在拖拽过程中,state将不断地被赋予新的值。

@GestureState private var dragOffset: CGSize = .zero
.updating($dragOffset, body: { value, state, _ instate = value.translation
})

同时我们对Text组件添加了.offset(dragOffset)修饰符,并传入dragOffset变量,到此就可以拖拽Text组件了,但是松手后,它会自动地回到原来的位置。

.offset(dragOffset)

为了解决这个问题,我们还需要再来一个变量position记录松手时的位置信息。

@State private var position: CGSize = .zero

并且在.onEnded修饰符的闭包中,给position赋值。该闭包中返回了当前的手势信息变量,而移动信息储存在该变量的translation中。

.onEnded({ value inposition.width += value.translation.widthposition.height += value.translation.height
})

这里更新了position信息,这里主要注意用了+=运算符,因为拖动不止一次,上一次的结束位置即是下一次的起点位置。如果说拖拽松手后想回到原点,那就移出关于position的相关代码。

最后在给Text添加一个.offset(position)修饰符,并传入position变量。

.offset(position)

到现在已经添加了两个.offset()修饰符,也可以添加一个,但是要将positiondragOffset两个变量加起来传入进去,比如这样:

.offset(CGSize(width: dragOffset.width + position.width, height: dragOffset.height + position.height))

至此,该动画就全部完成了。另外上面的视图中,在界面的顶部加了两个Text,用来显示dragOffsetposition的数值,可以看出,每次动作结束@GestureState修饰的dragOffset都恢复了初始值。

完整代码如下:

struct DragGestureDemo: View {@GestureState private var dragOffset: CGSize = .zero@State private var position: CGSize = .zerovar body: some View {ZStack {VStack {Text("dragOffset: \(dragOffset)")Text("position: \(position)")Spacer()}Text("Drag me!").font(.title).padding().background(Color.cyan).cornerRadius(10.0).offset(dragOffset).offset(position).gesture(DragGesture().updating($dragOffset, body: { value, state, _ instate = value.translation}).onEnded({ value inposition.width += value.translation.widthposition.height += value.translation.height}))}}
}

可能有人会说,我习惯用了.onChange()修饰符,不习惯用.updating()修饰符,下面这个就是用.onChange()修饰符实现的拖动。
在这里插入图片描述
这里需要两个@State修饰的变量来记录信息,dragOffset记录相对于最初位置的偏移量,position记录松手后相对于最初位置的偏移量。

Text设置一个.offset()修饰符即可,要传入dragOffsetdragOffset是被@State修饰器修饰的,在松手后该值不会重置。

.onChange()修饰符闭包内,计算dragOffset的值,dragOffset = 上一次位置偏移量 + 手势偏移量。

.onEnded()修饰符闭包内,计算position的值,position = dragOffset,因为dragOffset不会重置,且也是手离开的时候的偏移量。给position赋值,是为了在.onChange()修饰符闭包内给dragOffset赋值。

最终代码如下:

struct DragGestureDemo2: View {@State private var dragOffset: CGSize  = .zero@State private var position: CGSize = .zerovar body: some View {ZStack {VStack {Text("dragOffset: \(dragOffset)")Text("position: \(position)")Spacer()}Text("Drag me!").font(.title).padding().background(Color.cyan).cornerRadius(10.0).offset(dragOffset).gesture(DragGesture().onChanged({ value indragOffset.width = position.width + value.translation.widthdragOffset.height = position.height + value.translation.height}).onEnded({ _ inposition = dragOffset}))}}
}

如果想要在松手后回到原点,那就不需要position记录位置了,直接修改成下面代码即可:

.gesture(DragGesture().onChanged({ value indragOffset = value.translation}).onEnded({ _ indragOffset = .zero})
)

以上就是两种方式实现的拖拽动画,下面看一个比较酷的左右滑动卡片的拖拽动画。
在这里插入图片描述
关于这个拖拽动画效果,不做过多的说明了,附上源码,有问题可以留言。

struct DragGestureDemo3: View {@State private var offset: CGSize = .zerovar body: some View {Image("liuyifei").resizable().frame(width: 260).aspectRatio(1.0, contentMode: .fit).offset(offset).scaleEffect(getScale()).rotationEffect(Angle(degrees: getRotation())).gesture(DragGesture().onChanged({ value inwithAnimation(.spring()) {offset = value.translation}}).onEnded({ _ inwithAnimation(.spring()) {offset = .zero}}))}func getScale() -> CGFloat {let max = UIScreen.main.bounds.width / 2let current = abs(offset.width)let percentage = current / maxreturn 1.0 - min(percentage, 0.5) * 0.5}func getRotation() -> Double {let max = UIScreen.main.bounds.width / 2let current = offset.widthlet percentage = Double(current / max)let maxAngle: Double = 10return percentage * maxAngle}
}

最后,希望能够帮助到有需要的朋友,如果觉得有帮助,还望点个赞,添加个关注,笔者也会不断地努力,写出更多更好用的文章。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/14286.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Flutter Text导致A RenderFlex overflowed by xxx pixels on the right.

使用Row用来展示两个Text的时候页面出现如下异常,提示"A RenderFlex overflowed by xxx pixels on the right." The following assertion was thrown during layout: A RenderFlex overflowed by 4.8 pixels on the right.The relevant error-causing widget was:…

【仿RabbitMQ消息队列项目day2】使用muduo库中基于protobuf的应用层协议进行通信

一.什么是muduo? muduo库是⼀个基于非阻塞IO和事件驱动的C高并发TCP网络编程库。 简单来理解,它就是对原生的TCP套接字的封装,是一个比socket编程接口更好用的编程库。 二.使用muduo库完成一个英译汉翻译服务 TranslateServer.hpp: #pragma once #in…

MyBatis中Where标签:揭秘高效SQL构建的秘密

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 理解Where标签的基础概念 在MyBatis中&#xff0c;<where>标签是用于构建SQL查询语句中的一个非常重要的元素。它允许你在一个动态的SQL语句中添加WHERE子句&#xff0c;而不需要担心SQL语法错误或额外的逗号…

如何利用51建模网,实现3D模型线上展示和应用?

按照下面的步骤&#xff0c;在51建模网上传3D模型&#xff0c;并编辑完成后&#xff0c;接下来就是如何让这些3D模型得到更好的展示、传播和应用。 一、3D内容快速分享与传播 3D模型在51建模网上传发布后&#xff0c;即可获得一个可分享的链接和二维码&#xff0c;直接分享给客…

20240520解决在Ubuntu20.04下编译RK3588的Android12的SDK出现C2_GIT_BUILD_VERSION未定义的问题

20240520解决在Ubuntu20.04下编译RK3588的Android12的SDK出现C2_GIT_BUILD_VERSION未定义的问题 2024/5/20 20:19 缘起&#xff1a;通过./repo/repo/repo sync -l得到的SDK正常&#xff0c;但是解压缩之后的SDK却出错了&#xff01; 通过grep很容易发现有三个地方有&#xff0c…

PointCloudLib 点云投影到XOY平面功能实现 C++版本

0.实现效果 左图为原始点云,右图为投影到XOY平面上的点云 将三维的点云投影到二维平面,方便处理一些二维轮廓方面的计算。 可以投影到空间中任意平面上。 1.算法原理 原理 点云投影是将三维空间中的点云数据映射到一个二维平面上的过程。这通常通过以下步骤实现: 确定投…

微服务实践k8sdapr开发部署调用

前置条件 安装docker与dapr: 手把手教你学Dapr - 3. 使用Dapr运行第一个.Net程序安装k8s dapr 自托管模式运行 新建一个webapi无权限项目 launchSettings.json中applicationUrl端口改成5001,如下: "applicationUrl": "http://localhost:5001" //Wea…

BeautifulSoup4通过lxml使用Xpath,以及获取(定位)元素和其文本或者属性

环境&#xff1a;win10&#xff0c;python3.8.10 首先需要安装&#xff1a;beautifulsoup4&#xff0c;lxml 使用命令&#xff1a; pip38 install beautifulsoup4 pip38 install lxml 安装完毕后查看一下&#xff1a; 写代码&#xff1a; from bs4 import BeautifulSoup …

rocketmq 学习二 基本概念

教程&#xff1a;基本概念 | RocketMQ 视频教程 https://www.bilibili.com/video/BV1d5411y7UW?vd_sourcef1bd3b5218c30adf0a002c8c937e0a27 版本&#xff1a;5.0 一 基本概念 1.1 生产者/Producer 1.1.1 定义 消息发布者。是构建并传输消息到服务端的运行实体。…

异众比率(variation ratio)

异众比率&#xff08;variation ratio&#xff09;是指非众数组的频数占总频数的比率&#xff0c;其计算公式为: 其中&#xff0c;为众数组的频数。 异众比率主要用于衡量众数对一组数据的代表程度。异众比率越大&#xff0c;说明非众数组的频数占总频数的比重越大&#xff0…

harbor 认证

Harbor 认证过程 Harbor以 Docker Registry v2认证为基础&#xff0c;添加上一层权限保护。1.v2 集成了一个安全认证的功能&#xff0c;将安全认证暴露给外部服务&#xff0c;让外部服务去实现2.强制用户每次Docker pull/push请求都要带一个合法的Token&#xff0c;Registry会…

【AHK V2】设计模式之命令模式

目录 情景剧场什么是命令模式优缺点优点缺点 使用命令模式的步骤命令模式代码示例合理使用AI工具自动生成代码 情景剧场 我们来设想一个场景&#xff1a; 你进入一家餐馆&#xff0c;餐馆只有老板一个人&#xff08;老板即厨师&#xff09;。 “老板&#xff0c;一份小炒肉&am…

吃透1850道真题和解析备考AMC8和AMC(1020240524持续发布)

多做真题&#xff0c;吃透真题和背后的知识点是备考AMC8、AMC10有效的方法之一&#xff0c;通过做真题&#xff0c;可以帮助孩子找到真实竞赛的感觉&#xff0c;而且更加贴近比赛的内容&#xff0c;可以通过真题查漏补缺&#xff0c;更有针对性的补齐知识的短板。 今天我们继续…

java实现List对象转geojson文本返回前端

1.业务需求 查询带有经纬度数据的list列表&#xff0c;将其转为geojson格式给前端。 2.GeoJson格式说明 GeoJSON是一种对各种地理数据结构进行编码的格式&#xff0c;基于Javascript对象表示法(JavaScript Object Notation, 简称JSON)的地理空间信息数据交换格式。GeoJSON对…

计算机系统的层次结构

操作系统的定义 操作系统&#xff08;Operating System&#xff0c; OS&#xff09;是指控制和管理整个计算机系统的硬件和软件资源&#xff0c;并合理地组织调度计算机的工作和资源的分配&#xff1b;以提供给用户和其他软件方便的接口和环境&#xff1b;它是计算机系统中最基…

GBase 8s 如何查看回滚的事务 和对应的SQL

描述&#xff1a; 如何查看当前数据库中是否有事务在回滚&#xff0c; 如果有&#xff0c; 具体是哪条 SQL 在回滚&#xff1f; 解决办法&#xff1a; 方法1&#xff1a; 通过 onstat -u|grep RP&#xff1b; 可以获取相关的 sessionid。 通过 onstat -g ses sid 获取 SQL&a…

【多模态融合】Cross Modal Transformer: Towards Fast and Robust 3D Object Detection

论文链接&#xff1a;Cross Modal Transformer: Towards Fast and Robust 3D Object Detection 代码链接&#xff1a;https://github.com/junjie18/CMT 作者&#xff1a;Junjie Yan, Yingfei Liu, Jianjian Sun, Fan Jia, Shuailin Li, Tiancai Wang, Xiangyu Zhang 发表单位…

字符串的周期:每一期都有那么几位

【题目描述】 如果一个字符串可以由某个长度为k的字符串重复多次得到,则称该串以k为周期。例 如,abcabcabcabc以3为周期(注意,它也以6和12为周期)。 输入一个长度不超过80的字符串(不含空格),输出其最小周期。 输入第一行表示有T组数据,后续是T行字符串。输出的每组…

性能测试工具

性能测试工具 1.Jmeter 环境搭建1.安装JDK2.安装Jmeter1.下载2.安装3.环境配置 3.Jmeter 文件目录介绍1.bin目录2.docs 目录3.printable_docs目录4.lib目录 4.修改默认配置1.汉化配置2.修改主题 5.元件的基本介绍6.元件的作用域作用域的原则 7.元件的执行顺序 1.Jmeter 环境搭建…

【Pytorch】16.使用ImageFolder加载自定义MNIST数据集训练手写数字识别网络(包含数据集下载)

数据集下载 MINST_PNG_Training在github的项目目录中的datasets中有MNIST的png格式数据集的压缩包 用于训练的神经网络模型 自定义数据集训练 在前文【Pytorch】13.搭建完整的CIFAR10模型我们已经知道了基本搭建神经网络的框架了&#xff0c;但是其中的数据集使用的torchvision…