记录--uni-app实现京东canvas拍照识图功能

2023-10-13 09:40

本文主要是介绍记录--uni-app实现京东canvas拍照识图功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助

最近公司出了一个新的功能模块(如下图),大提上可以描述为实现拍照完上传图片,拖动四方框拍照完成上传功能,大体样子如下图。但是我找遍了 dcloud 插件市场,找到的插件都是移动背景图片来实现裁剪的,跟京东的功能是相反的,没办法只能自己来实现这么一个插件。

第一步

首先就需要实现一个四方框的功能了。从上图可知,四方框有一下几个特点

  1. 四个角粘连外框,随着框的大小和移动范围紧缚移动
  2. 四方框可随意四个方向拖动
  3. 方框外区域阴影不影响方框内

那么我们根据这个特性来实现下这个功能,对于 css 规范的话使用 bem 规范

<div class="clip__content"><div v-for="(item, index) in 4" :key="index" class="clip__edge"></div>
</div>

/more

$edge-border-width: 6rpx;
.clip {&__content {position: fixed;width: 400rpx;height: 400rpx;left: 0;top: 0;border: 1px solid red;z-index: 4;overflow: hidden;box-shadow: rgba(0, 0, 0, 0.5) 0 0 0 200vh;}&__edge {position: absolute;width: 34rpx;height: 34rpx;border: 10rpx solid red;pointer-events: auto;z-index: 2;&::before {content: "";position: absolute;z-index: 2;width: 40rpx;height: 40rpx;background-color: transparent;}&:nth-child(1) {left: $edge-border-width;top: $edge-border-width;border-bottom-width: 0 !important;border-right-width: 0 !important;&:before {top: -50%;left: -50%;}}&:nth-child(2) {right: $edge-border-width;top: $edge-border-width;border-bottom-width: 0 !important;border-left-width: 0 !important;&:before {top: -50%;left: 50%;}}&:nth-child(3) {left: $edge-border-width;bottom: $edge-border-width;border-top-width: 0 !important;border-right-width: 0 !important;&:before {bottom: -50%;left: -50%;}}&:nth-child(4) {right: $edge-border-width;bottom: $edge-border-width;border-top-width: 0 !important;border-left-width: 0 !important;&:before {bottom: -50%;left: 50%;}}}

根据上面的 html 和 css 出来的样式大概如下图 外部的阴影效果我们用: box-shadow: rgba(0, 0, 0, 0.5) 0 0 0 200vh 来达成

第二步

第二步的话就要实现移动功能了,这里是一个比较考验耐心的地方,因为涉及到多个方向的变化,需要不断地进行调试,在此之前需要先分析下四个角变化的特性,下面先看 4 个角的移动特性(以 H5 思维)

  1. 第一个角的移动会改变方框的 left,top,width,right4 个值
  2. 第二个角的移动会改变方框的 top,with,height3 个值
  3. 第三个角的移动会改变方框的 left, width,height3 个值
  4. 第四个角的移动会改变方框的 width,height2 个值
  5. 四个角的移动都不能小于 4 个角的宽高,四个角的移动都不能超过屏幕,相应的逻辑需要做一下限制

首先需要获取下屏幕宽度,区域高度(因为头部可能会有导航栏目占位,所以不拿屏幕高度),四方框初始宽高,

uni.getSystemInfo({success: res => {console.log(res)this.systemInfo = res}
})
uni.createSelectorQuery().select('.clip__content').fields({ size: true }, data => {this.width = data.widththis.height = data.height}).exec()
uni.createSelectorQuery().select('.clip').fields({ size: true }, data => {this.screenHeight = data.height}).exec()

后续的话就可以进行四个角拖拽了,这里用到了 touchStart 和 touchMove,动态地为方框绑定样式

<divv-for="(item, index) in 4"class="clip__edge"@touchstart.stop.prevent="edgeTouchStart"@touchmove.stop.prevent="e => edgeTouchMove(e, index)"@touchend.stop.prevent="edgeTouchEnd"
></div>

接下来开始写逻辑

edgeTouchStart(e) {// 记录坐标xy初始位置this.clientX = e.changedTouches[0].clientX;this.clientY = e.changedTouches[0].clientY;
},
edgeTouchMove(e, index) {const currX = e.changedTouches[0].clientX;const currY = e.changedTouches[0].clientY;// 记录坐标差const moveX = currX - this.clientX;const moveY = currY - this.clientY;// 更新坐标位置this.clientX = currX;this.clientY = currY;const { width, height, left, top, screenHeight } = this;const { screenWidth } = this.systemInfo;// 初始化最大宽高let maxWidth = 0,maxHeight = 0,maxTop = top + moveY < 0 ? 0 : top + moveY,maxLeft = left + moveX < 0 ? 0 : left + moveX;// 四个角的宽高限制if (index % 2 === 0) {maxWidth = width - moveX > screenWidth ? screenWidth : width - moveX;} else {maxWidth = width + moveX > screenWidth ? screenWidth : width + moveX;}if (index < 2) {maxHeight =height - moveY > screenHeight ? screenHeight : height - moveY;} else {maxHeight =height + moveY > screenHeight ? screenHeight : height + moveY;}// 四个角的规则计算逻辑 四边方框暂定40 更详细的要用.createSelectorQuery()去拿if (index === 0) {if (width - moveX <= 40 || height - moveY <= 40) return;console.log(maxLeft);this.clipStyle = {width: maxWidth,height: maxHeight,left: maxLeft,top: maxTop,};this.width = maxWidth;this.height = maxHeight;this.top = maxTop;this.left = maxLeft;// 右上角} else if (index === 1) {if (width + moveX <= 40 || height - moveY <= 40) return;this.clipStyle = {width: maxWidth,height: maxHeight,left,top: maxTop,};this.width = maxWidth;this.height = maxHeight;this.top = maxTop;} else if (index === 2) {if (width - moveX <= 40 || height + moveY <= 40) return;this.clipStyle = {width: maxWidth,height: maxHeight,left: maxLeft,top,};this.width = maxWidth;this.height = maxHeight;this.left = maxLeft;} else if (index === 3) {if (width + moveX <= 40 || height + moveY <= 40) return;this.clipStyle = {width: maxWidth,height: maxHeight,left,top,};this.width = maxWidth;this.height = maxHeight;}
}

效果如下图

第三步

四个角拖拽逻辑完善之后,下一步目标就是做四方框的拖拽,这边需要对四方框的拖拽做一次限制

<divclass="clip__content":style="style"@touchstart.stop.prevent="clipTouchStart"@touchmove.stop.prevent="clipTouchMove"
>...
</div>
clipTouchStart(e) {this.touchX = e.changedTouches[0].pageX;this.touchY = e.changedTouches[0].pageY;
},
clipTouchMove(e) {const { screenWidth } = this.systemInfo;const currX = e.changedTouches[0].pageX;const currY = e.changedTouches[0].pageY;const moveX = currX - this.touchX;const moveY = currY - this.touchY;this.touchX = currX;this.touchY = currY;// 边框限制逻辑if (this.left + moveX < 0) {this.left = 0;} else if (this.left + moveX > screenWidth - this.width) {this.left = screenWidth - this.width;} else {this.left = this.left + moveX;}if (this.top + moveY < 0) {this.top = 0;} else if (this.top + moveY > this.screenHeight - this.height) {this.top = this.screenHeight - this.height;} else {this.top = this.top + moveY;}this.clipStyle = {...this.clipStyle,left: this.left,top: this.top,};
},

效果如下图:

第四步就是做我们的截图了,这里用到了 canvas

<div class="clip__content">...<canvas class="clip-canvas" canvas-id="clip-canvas"></canvas>
</div>

逻辑的话目前这个例子是使用了网络的 url 图片 所以要进行 download,如果是不用网络图片,那么这一句可以删除换成其他的获取图片 api

initCanvas() {uni.showLoading({title: "加载中...",});uni.createSelectorQuery().select(".clip__content").fields({size: true,scrollOffset: true,rect: true,context: true,computedStyle: ["transform", "translateX"],scrollOffset: true,},(data) => {uni.downloadFile({url: this.imageUrl,success: (res) => {this.canvasInstance = uni.createCanvasContext("clip-canvas",this);this.canvasInstance.drawImage(res.tempFilePath,-data.left,-data.top,this.systemInfo.screenWidth,this.screenHeight,0,0);this.canvasInstance.draw(false,(() => {setTimeout(() => {uni.canvasToTempFilePath({x: 0,y: 0,width: data.width,height: data.height,dWidth: data.width,dHeight: data.height,fileType: "jpg",canvasId: "clip-canvas",success: (data) => {uni.hideLoading();this.url = data.tempFilePath;// this.canvasInstance.save();},},this);}, 500);})());},});}).exec();
},

效果如图所示:

本文转载于:

https://juejin.cn/post/6971977095652048910

如果对您有所帮助,欢迎您点个关注,我会定时更新技术文档,大家一起讨论学习,一起进步。

 

这篇关于记录--uni-app实现京东canvas拍照识图功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/202340

相关文章

百度/小米/滴滴/京东,中台架构比较

小米中台建设实践 01 小米的三大中台建设:业务+数据+技术 业务中台--从业务说起 在中台建设中,需要规范化的服务接口、一致整合化的数据、容器化的技术组件以及弹性的基础设施。并结合业务情况,判定是否真的需要中台。 小米参考了业界优秀的案例包括移动中台、数据中台、业务中台、技术中台等,再结合其业务发展历程及业务现状,整理了中台架构的核心方法论,一是企业如何共享服务,二是如何为业务提供便利。

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、