uniapp 微信小程序 canvas 手写板获取书写内容区域并输出


uni.canvasGetImageData

返回一个数组,用来描述 canvas 区域隐含的像素数据,在自定义组件下,第二个参数传入自定义组件实例 this,以操作组件内 组件。

js 复制代码
// 获取目标 canvas 的像素信息 pixelData
let canvas = uni.createSelectorQuery().select('.signature__canvas')
canvas.boundingClientRect().exec(function (data) {
  let canvasw = Math.ceil(data[0].width)
  let canvash = Math.ceil(data[0].height)
  uni.canvasGetImageData({
    width: canvasw,
    height: canvash,
    canvasId: 'canvas_sign',
    x: 0,
    y: 0,
    success: function (pixelData) {
      console.log(pixelData)
    },
    fail: (err) => {
      console.log('canvas 区域隐含的像素数据失败', err)
    },
  })
})

此处获取像素信息,传入 x:0,y:0,width:canvas宽度,height:canvas高度,获取到整个画布所有的像素点信息

如图是 pixelData 的打印

注意:里面的 Uint8ClampedArray 的长度,808752,此值是 width(812)* height(249)* 4 得出来的

为什么是 *4?

cannvas 的 imagedata 数据是一个由 RGBA 构成的数组,每四个值包含一个像素点的信息,RGBA: R - 红色(0-255)、G - 绿色(0-255)、B - 蓝色(0-255)、A - alpha 通道(0-255; 0 是透明的,255 是完全可见的)

所以,imagedata 实际上是 R G B A R G B A ... 的一个数组

判断像素点是否有值

简单来讲,如果是4*3的12个像素点,中间两个像素点有值的情况下

程序判断有值无值,需要先从第一行开始(从第一列开始也可以),看第一行第一个有没有,第一行第二个有没有。。。依此看下去,第一行看完了看第二行。。。

js 复制代码
const pixelDataWidth = pixelData.width
const pixelDataHeight = pixelData.height
for (var x = 0; x <= pixelData.width; x++) {
  for (var y = 0; y <= pixelData.height; y++) {
    var i = (x * pixelData.width + y) * 4
    // 每个像素判断 rgba 是否有值,则认为有数据
    if (pixelData.data[i] != 0 || pixelData.data[i + 1] != 0 || pixelData.data[i + 2] != 0 || pixelData.data[i + 3] != 0) {
      // ...
    }
  }
}

我们通过上面循环能够知道,哪个像素点有值了,之后就能得出一个边界值 startX、startY、endX、endY

startX 为最偏左的点的 X,startY 为最偏上的 Y,endX 为最偏右的 X,endY 为最偏下的 Y

javascript 复制代码
// startX 和 startY 为取小逻辑,即有值坐标,比初始坐标小,则将 startX 和 startY 赋值
var startX = pixelData.width + 1
var startY = pixelData.height + 1
// endX 和 endY 为取大逻辑,即有值坐标,比初始(-1,-1)大,则将 endX 和 endY 赋值
var endX = -1
var endY = -1

if (startX > x) startX = x
if (startY > y) startY = y
if (endX < x) endX = x
if (endY < y) endY = y

这样我们就得到了,startX、startY、endX、endY

uni.canvasToTempFilePath

js 复制代码
uni.canvasToTempFilePath({
  x: 100, // 画布x轴起点(默认0)
  y: 200, // 画布y轴起点(默认0)
  width: 50, // 画布宽度(默认为canvas宽度-x)
  height: 50, // 画布高度(默认为canvas高度-y)
  destWidth: 100, // 输出图片宽度(默认为 width * 屏幕像素密度)
  destHeight: 100, // 输出图片高度(默认为 height * 屏幕像素密度)
  canvasId: 'myCanvas',
  success: function(res) {
    // 在H5平台下,tempFilePath 为 base64
    console.log(res.tempFilePath)
  } 
})
js 复制代码
x: startX,
y: startY,
width: endX - startX,
height: endY - startY,
destWidth: endX - startX,
destHeight: endY - startY,

按照上面的 x、y、width。。。输出即可

相关推荐
V+zmm1013419 分钟前
基于微信小程序的乡村政务服务系统springboot+论文源码调试讲解
java·微信小程序·小程序·毕业设计·ssm
还这么多错误?!23 分钟前
uniapp微信小程序,使用fastadmin完成一个一键获取微信手机号的功能
微信小程序·小程序·uni-app
_院长大人_34 分钟前
微信小程序用户信息解密 AES/CBC/NoPadding 解密失败问题
微信小程序·小程序
IT 前端 张1 小时前
Uniapp 手机基座调试App 打包成Apk文件,并上传到应用商店
uni-app
User_undefined1 小时前
uniapp Native.js原生arr插件服务发送广播到uniapp页面中
android·javascript·uni-app
web135085886351 小时前
uniapp小程序使用webview 嵌套 vue 项目
vue.js·小程序·uni-app
麦兜*1 小时前
轮播图带详情插件、uniApp插件
前端·javascript·uni-app·vue
veminhe1 小时前
uni-app使用组件button遇到的问题
uni-app·vue
m0_748240021 小时前
uniapp跨平台开发---webview调用app方法
uni-app
guanpinkeji1 小时前
废品回收小程序:助力企业转型发展
小程序·小程序开发·小程序制作·回收·废品回收小程序·废品回收