微信小程序如何实现实现监听相机帧并按需回显于屏幕—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实时帧数据整体缩放为画布大小绘制,如何解决未知..

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

谨听教诲

相关推荐
ForteScarlet39 分钟前
Jetbrains 官方微信小程序插件已上线!
前端·ide·微信小程序·小程序
竣子好逑1 小时前
uniapp 微信小程序 编写文章等 editor 编辑器编写富文本
微信小程序·小程序·uni-app
竣子好逑1 小时前
uniapp 微信小程序 金额展示套餐
微信小程序·小程序·uni-app
厘子 车3 小时前
小程序如何引入腾讯位置服务
前端·微信小程序·小程序
2401_897915654 小时前
基于微信小程序的校园运动场地预约系统设计与实现
微信小程序·小程序
计算机学姐4 小时前
基于微信小程序的健身房预约管理系统
java·vue.js·spring boot·后端·mysql·微信小程序·小程序
既生向阳19 小时前
记录一次微信小程序使用云能力开发的过程
微信小程序·小程序
Kika写代码1 天前
【微信小程序】5|我的页面 | 我的咖啡店-综合实训
微信小程序·小程序
雯0609~1 天前
微信小程序:实现单选,多选,通过变量控制单选/多选
微信小程序·小程序·notepad++
黑马源码库miui520861 天前
智慧城市V4电商商城系统源码独立版全插件全开源
人工智能·微信小程序·小程序·uni-app·智慧城市·微信公众平台