微信小程序如何实现实现监听相机帧并按需回显于屏幕—wx.createCameraContext()和wx.createCanvasContext()或许可以实现

笔者为大二在读学生,最近做一个小程序植物识别的项目需要实现以下功能,苦苦寻找多方资料未果,故分享在探索路上获得的经验,以作抛砖引玉之用

  1. 进入页面监听自动相机帧数据
  2. 点击按钮推送相机帧数据到小程序AI进行推理
  3. 将推送的相机帧数据以小框显示在屏幕左中部

进入页面监听自动相机帧数据

这是笔者遇到的第一个坑,查询资料可知需先使用小程序的相机组件

html 复制代码
  <camera class="camera" 
  style="  width: 100vw; height: 80vh;" 
  device-position="back" 
  flash="off" 
  frame-size="medium">
  </camera>
js 复制代码
 onReady() {
     const context = wx.createCameraContext(this);
     const listener = context.onCameraFrame(frame => {
        // console.log('持续监听中')
           this.executeClassify(frame);
    });
    this.listener = listener
    
    //开始监听 经测试安卓无法正常监听,ios正常
    this.listener.start
 }

如果和以上代码一样在onReady生命周期里面触发监听,在预览中,我手上两台安卓手机均只能监听前几帧,小程序退到后台再进入又会刷新监听多几帧,ios正常,原因未知

解决办法

bindinitdone属性是可以使得对应方法在相机初始化完成时触发,将开始监听的方法绑定在该属性上可以实现进入页面自动开始监听相机帧

html 复制代码
<camera class="camera" 
   bindinitdone = "startListener"
  style="  width: 100vw; height: 80vh;" 
  device-position="back" 
  flash="off" 
  frame-size="medium">
  </camera>
js 复制代码
startListener(){
    this.listener.start({
      success: ()=> {
        console.log('开始监听成功');
        // 在成功回调函数中执行需要的操作
        wx.showToast({
          title: '开始识别',
          icon: 'loading',
          duration: 1000
        })
        this.setData({
          isStart: true
        })
      },
      fail: ()=> {
        console.log('开始监听失败');
        // 在失败回调函数中执行需要的操作
        wx.showToast({
          title: '请重试',
          icon: 'error',
          duration: 1000
        })
      },
    });
  },

点击按钮推送相机帧数据到小程序AI进行推理

这部分可以参考小程序AI推理的文档 developers.weixin.qq.com/miniprogram...

笔者在其基础上加了一个isClick的判定来实现点击后再推送相机帧数据

js 复制代码
    //原版 前一帧推理成功后自动推理下一帧
    const context = wx.createCameraContext(this); 
     
    const listener = context.onCameraFrame(frame => {

        if (this.classifier && this.classifier.isReady() && !this.predicting) {
           this.executeClassify(frame);
        }
    });
js 复制代码
    //修改后 推理成功且点击按钮后isClick为true才推理下一帧
    const context = wx.createCameraContext(this); 
     
    const listener = context.onCameraFrame(frame => {

        if (this.classifier && this.classifier.isReady() && !this.predicting && this.data.isClick) {
           this.executeClassify(frame);
        }
    });

将推送的相机帧数据以小框显示在屏幕左中部

关于Camera实时帧数据,小程序AI推理文档里面是这样解释的

OnCameraFrame 返回的 frame 包含属性 width、height 和 data,分别表示二维图像数据的宽度,高度,以及图像像素点数据。其中 data 为一个 ArrayBuffer,存储数据类型为 Uint8,存储的数据 format 为 rgba,即每连续的4个值表示一个像素点的 rgba。 详细关于 onCameraFrame 的内容可以参考CameraContext.onCameraFrame.

在Camera组件文档中也有类似的描述developers.weixin.qq.com/miniprogram...

要想将该格式的图像文件渲染到屏幕上,有多种选择,一是使用upng.js进行解码(据说平均耗时6-7s故并未使用),二是使用Canvas绘制,笔者使用的是后者

而小程序的Canvas也有新版和旧版,但是笔者使用新版Canvas时出现该报错,重复排查多次仍未解决,故使用小程序旧版Canvas

MiniProgramError Cannot read properties of null (reading 'node') TypeError: Cannot read properties of null (reading 'node')

html 复制代码
<camera class="camera" bindinitdone = "startListener" style="  width: 100vw; height: 80vh;" device-position="back" flash="off" frame-size="medium">
    <view class="res-container" wx:if="{{ isResultShow }}">
      <canvas class="res-image" canvas-id="ssd1" ></canvas>
    <view class="res-text"> 分类结果: {{predClass}}</view>
    <view bindtap="close" class="Icn">x</view>  
  </view>
  </camera>

其中需要注意的是Camera实时相机帧数据类型是ArrayBuffer ,如果使用 wx.canvasPutImageData() 绘制在Canvas画布上需要手动将监听的相机帧的Arraybuffer 数据转为uint8

js 复制代码
 onReady() {
     //获取Canvas绘图上下文
    this.ctx = wx.createCanvasContext('ssd1');
    const context = wx.createCameraContext(this);
    this.initClassifier();

    const listener = context.onCameraFrame(frame => {
        // console.log('持续监听中')
        // const fps = this.fpsHelper.getAverageFps();
        // console.log(`fps=${fps}`);

        if (this.classifier && this.classifier.isReady() && !this.predicting && this.data.isClick) {
           this.executeClassify(frame);
           //将监听的相机帧的arraybuffer数据转为uint8
           const data = new Uint8ClampedArray(frame.data); 

           wx.canvasPutImageData({
            canvasId: 'ssd1',
            x: 0, 
            y: 0,
            width: frame.width,
            height: frame.height,
            data: data,
            success (res) {
              console.log('canvas绘制成功', res)
                // after put image data
            },
            fail(error){
              console.log('canvas绘制失败',error)
            },
            complete(res){
              console.log('canvas绘制流程走完了',res)
            }
        })
        }
    });
    this.listener = listener
  },

至此,便实现标题所述基本功能,但仍存在不足

  1. 第一次点击按钮绘制Camera实时帧数据到Cavas画布时,绘制无内容且控制台报错

canvas绘制失败 {"errMsg": "canvasPutImageData:fail fail canvas is empty"}

不知道是不是wx:if控制显隐导致的,回去试一试 2. Canvas绘制的相机帧数据仅为原图左上角部分,设想是将Camera实时帧数据整体缩放为画布大小绘制,如何解决未知..

欢迎有经验的大佬在评论区畅言,希望有更优方案的出现

谨听教诲

相关推荐
咖啡の猫12 小时前
微信小程序案例 - 自定义 tabBar
微信小程序·小程序·notepad++
咖啡の猫13 小时前
微信小程序全局数据共享
微信小程序·小程序
桐溪漂流13 小时前
微信小程序cli脚本预览上传
微信小程序·小程序
咖啡の猫13 小时前
微信小程序使用 npm 包
微信小程序·小程序·npm
东东5161 天前
xxx食堂移动预约点餐系统 (springboot+微信小程序)
spring boot·微信小程序·小程序·毕业设计·个人开发·毕设
韩立学长2 天前
【开题答辩实录分享】以《智慧校园平台微信小程序》为例进行选题答辩实录分享
spring boot·微信小程序·小程序
h_65432102 天前
微信小程序:按顺序一张图片加载完后,再加载另一张
微信小程序·小程序
qq_316837753 天前
uniapp打包微信小程序使用插件
微信小程序·小程序·uni-app
不爱学习小趴菜3 天前
uniapp微信小程序无法屏蔽右上角胶囊按钮(...)问题解决方案
微信小程序·小程序·uni-app
plmm烟酒僧4 天前
《微信小程序demo开发》第一部分-编写页面逻辑
javascript·微信小程序·小程序·html·微信开发者工具·小程序开发