参加比赛的作品,开发周期一个月,使用了 Wafer2 框架,后台采用腾讯云提供的 Node.js SDK 接入对象存储 API ,前端核心代码实现了类似于图片编辑器的功能,支持图片和文字的移动、旋转、缩放、生成预览图以及编辑状态的保存,动画部分采用 CSS 动画实现小程序中的模态输入框部分使用了自己封装的 InputBox 组件代码已移除 AppId 等敏感信息,可自行添加自己的 AppId 和 AppSecret 以配置后台环境,实现登录测试,详细添加方法见下文「使用方法」,若本地运行可通过修改 app.json 文件中 page 字段的顺序来查看不同页面微信小程序定制需求请联系作者微信:aweawds (注明来意)效果展示      使用方法首先点击右上角 Star ʕ •ᴥ•ʔ获取Demo代码执行 git clone https://github.com/goolhanrry/Weapp-Demo-LemonJournal.git或 点击此处 下载最新版本的代码解压后在微信开发者工具中打开 Weapp-Demo-LemonJournal 文件夹即可如需进行登录测试,还要执行以下步骤准备好自己的 AppId 和 AppSecret(可在微信公众平台注册后获取)在 project.config.json 的 appid 字段中填入 AppId在 /client/utils/util.js 中相应位置填入 AppId 和 AppSecret在微信开发者工具中重新导入整个项目,上传后台代码后编译运行即可核心代码组件的移动、旋转和缩放主要思路是把  标签(对应图片)和  标签(对应文字)封装在同一个自定义组件  中,通过对外暴露的 text 变量是否为空来进行条件渲染,然后绑定 onTouchStart() 、onTouchEnd() 和 onTouchMove() 三个事件来对整个组件的位置、角度、大小、层级以及 “旋转” 和 “移除” 两个按钮的行为进行操作onTouchStart: function (e) {     // 若未选中则直接返回     if (!this.data.selected) {         return     }     switch (e.target.id) {         case 'sticker': {             this.touch_target = e.target.id             this.start_x = e.touches[0].clientX * 2             this.start_y = e.touches[0].clientY * 2             break         }         case 'handle': {             // 隐藏移除按钮             this.setData({                 hideRemove: true             })             this.touch_target = e.target.id             this.start_x = e.touches[0].clientX * 2             this.start_y = e.touches[0].clientY * 2             this.sticker_center_x = this.data.stickerCenterX;             this.sticker_center_y = this.data.stickerCenterY;             this.remove_center_x = this.data.removeCenterX;             this.remove_center_y = this.data.removeCenterY;             this.handle_center_x = this.data.handleCenterX;             this.handle_center_y = this.data.handleCenterY;             this.scale = this.data.scale;             this.rotate = this.data.rotate;             break         }     } }, onTouchEnd: function (e) {     this.active()     this.touch_target = ''     // 显示移除按钮     this.setData({         removeCenterX: 2 * this.data.stickerCenterX - this.data.handleCenterX,         removeCenterY: 2 * this.data.stickerCenterY - this.data.handleCenterY,         hideRemove: false     })     // 若点击移除按钮则触发移除事件,否则触发刷新数据事件     if (e.target.id === 'remove') {         this.triggerEvent('removeSticker', this.data.sticker_id)     } else {         this.triggerEvent('refreshData', this.data)     } }, onTouchMove: function (e) {     // 若无选中目标则返回     if (!this.touch_target) {         return     }     var current_x = e.touches[0].clientX * 2     var current_y = e.touches[0].clientY * 2     var diff_x = current_x - this.start_x     var diff_y = current_y - this.start_y     switch (e.target.id) {         case 'sticker': {             // 拖动组件则所有控件同时移动             this.setData({                 stickerCenterX: this.data.stickerCenterX   diff_x,                 stickerCenterY: this.data.stickerCenterY   diff_y,                 removeCenterX: this.data.removeCenterX   diff_x,                 removeCenterY: this.data.removeCenterY   diff_y,                 handleCenterX: this.data.handleCenterX   diff_x,                 handleCenterY: this.data.handleCenterY   diff_y             })             break         }         case 'handle': {             // 拖动操作按钮则原地旋转缩放             this.setData({                 handleCenterX: this.data.handleCenterX   diff_x,                 handleCenterY: this.data.handleCenterY   diff_y             })             var diff_x_before = this.handle_center_x - this.sticker_center_x;             var diff_y_before = this.handle_center_y - this.sticker_center_y;             var diff_x_after = this.data.handleCenterX - this.sticker_center_x;             var diff_y_after = this.data.handleCenterY - this.sticker_center_y;             var distance_before = Math.sqrt(diff_x_before * diff_x_before   diff_y_before * diff_y_before);             var distance_after = Math.sqrt(diff_x_after * diff_x_after   diff_y_after * diff_y_after);             var angle_before = Math.atan2(diff_y_before, diff_x_before) / Math.PI * 180;             var angle_after = Math.atan2(diff_y_after, diff_x_after) / Math.PI * 180;             this.setData({                 scale: distance_after / distance_before * this.scale,                 rotate: angle_after - angle_before   this.rotate             })             break         }     }     this.start_x = current_x;     this.start_y = current_y; }编辑状态的保存一篇手帐包含的组件类型包括 sticker(软件自带的贴纸)、image(用户上传的图片)和 text(自定义文字)三种,全部保存在一个如下格式的 json 对象中,每个独立组件都包含了一个不重复的 id 以及相关的信息,保存时由客户端生成该对象并编码成 json 字符串存储在数据库,恢复编辑状态时通过解析 json 字符串获得对象,再由编辑页面渲染{     "backgroundId": "5",                                        背景图id     "assemblies": [         {             "id": "jhjg",                                       组件id             "component_type": "image",                          组件类型(自定义图片)             "image_url": "https://example.com/jhjg.png",        图片地址             "stickerCenterX": 269,                              中心横坐标             "stickerCenterY": 664,                              中心纵坐标             "scale": 1.7123667831396403,                        缩放比例             "rotate": -3.0127875041833434,                      旋转角度             "wh_scale": 1,                                      图片宽高比             "z_index": 19                                       组件层级         },         {             "id": "gs47",             "component_type": "text",                           组件类型(文字)             "text": "test",                                     文字内容             "stickerCenterX": 479,             "stickerCenterY": 546,             "scale": 1.808535318980528,             "rotate": 29.11614626607893,             "z_index": 10         },         {             "id": "chjn",             "component_type": "sticker",                        组件类型(贴纸)             "sticker_type": "food",                             贴纸类型             "sticker_id": "1",                                  贴纸id             "image_url": "https://example.com/weapp/stickers/food/1.png",             "stickerCenterX": 277,             "stickerCenterY": 260,             "scale": 1.3984276885130673,             "rotate": -16.620756913892055,             "z_index": 5         }     ] }
2025-08-09 09:39:27 6.81MB 开源项目
1
在UE4(Unreal Engine 4)和UE5(Unreal Engine 5)这两个流行的实时3D创作工具中,触摸屏交互是为用户提供直观控制的重要功能。尤其在移动设备和现代多触点设备上,理解如何处理触摸屏上的移动、旋转和缩放事件对于创建优秀的触控体验至关重要。本文将深入探讨UE4和UE5中的触摸屏事件处理机制,以及如何利用这些事件来实现上述操作。 UE4和UE5都提供了内置的输入系统来处理各种类型的用户输入,包括鼠标、键盘、游戏手柄和触摸屏。触摸屏事件通过`FInputEvent`类表示,该类是所有输入事件的基础类。对于触摸事件,主要有以下几种类型: 1. **触摸开始**:当用户首次接触屏幕时,会触发`FTouchEvent`,其中`TouchEventType`为`ETouchType::TouchDown`。 2. **触摸移动**:用户在屏幕上滑动手指时,会产生一系列的`FTouchEvent`,其`TouchEventType`为`ETouchType::TouchMove`。 3. **触摸结束**:当用户抬起手指离开屏幕时,事件类型为`ETouchType::TouchUp`。 在UE4中,可以通过`UUserWidget`类来处理触摸事件。你可以重写`BeginTouchMove`, `BeginTouchDown`, 和`EndTouchUp`等函数来响应相应的触摸事件。同时,可以使用`AddTouchHandler`方法注册特定的触摸事件处理函数。 在UE5中,虽然基本的触摸事件处理方式没有本质改变,但引擎引入了更强大的C++和蓝图API,使得处理触摸事件更加便捷。例如,可以使用新的`TouchEvent`节点在蓝图中处理触摸事件。此外,UE5的`WorldContextObject`提供了一种更统一的方式来处理不同设备的输入,包括触摸屏。 移动、旋转和缩放操作通常涉及多个触摸点。例如,在多点触摸场景下,两个手指的触摸开始和移动事件可以用来实现缩放,而一个手指的移动可以用于平移。以下是如何实现这些功能的一般步骤: - **移动**:计算两个触摸点之间的中心点,然后根据新位置与原始中心点的差值来移动对象。 - **旋转**:计算两个触摸点形成的角度变化,并应用这个角度到对象的旋转。 - **缩放**:测量两个触摸点之间的新距离与原始距离,然后根据比例因子调整对象的大小。 在UE4和UE5的蓝图中,可以通过创建`TouchInputComponent`来处理多点触摸。使用`TouchInputComponent`的`AddTouch`和`RemoveTouch`节点来管理触摸事件,并使用`GetPinchZoomDelta`或`GetTwoFingerPanDelta`等节点获取缩放和平移信息。 为了优化触摸输入,还可以考虑触摸输入的防抖动处理,避免因为快速连续的触摸事件导致的不必要动作。此外,触摸事件的响应速度和流畅性也非常重要,可能需要适当调整引擎的输入刷新率和处理逻辑。 UE4和UE5为开发者提供了丰富的工具和API,以支持在触摸屏设备上实现移动、旋转和缩放等交互操作。通过理解和熟练运用这些功能,可以创建出更加直观和自然的用户体验。在实际项目中,应根据具体需求进行定制化开发,确保触控功能既高效又易于使用。
2024-07-09 17:01:57 610.21MB 触摸屏事件
使用MFC开发的小程序,实现图像平移、旋转、缩放。核心代码纯C编写。
2023-04-05 21:20:46 611KB 平移旋转缩放
1
CS裁剪,二维或三维图形任意平移旋转缩放的复合算法,正等轴测投影,中点分割裁剪的QT,Web,VC,C#,OpenGL,Java
2023-03-18 20:40:50 679.76MB 算法 源码软件
1
Runtime Transform Handles插件 Unity中运行时显示编辑器中的平移旋转缩放的功能,并且支持交互,支持跨平台
2023-01-02 19:26:26 12.29MB Unity插件
1
图形学 - OpenGL实现图形平移旋转缩放源码2个
2022-12-17 18:19:01 13.61MB OpenGL 平移 旋转 缩放
1
VS2005里,c#+opengl绘制的立方体,圆锥体,球体,并对三维实体能进行平移,旋转,缩放,可在控件中输入模型的边长或半径改变模型的大小,控制非常方便,界面非常友好
2022-11-03 16:41:34 165KB c#+opengl绘制 vs2005构建三 http://downl
1
一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能 一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能 一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能 一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能一个图像处理系统 包括图像 旋转 缩放 高斯变换等等功能
2022-10-17 21:39:44 1.87MB 一个图像处理系统 包括图像 旋转 缩放
1
主要为大家详细介绍了Unity3D使用鼠标旋转缩放平移视角,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
1
本篇文章是对Android中利用matrix 控制图片的旋转、缩放、移动进行了详细的分析介绍,需要的朋友参考下
2022-08-17 16:50:37 244KB android matrix
1