video-recording.md 10.8 KB
Newer Older
Z
zengyawen 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
# 视频录制

在OpenHarmony系统中,当前仅支持AVRecorder开发视频录制,集成了音频捕获,音频编码,视频编码,音视频封装功能,适用于实现简单视频录制并直接得到视频本地文件的场景。

本开发指导将以“开始录制-暂停录制-恢复录制-停止录制”的一次流程为示例,向开发者讲解如何使用AVRecorder进行视频录制。

在进行应用开发的过程中,开发者可以通过AVRecorder的state属性主动获取当前状态,或使用on('stateChange')方法监听状态变化。开发过程中应该严格遵循状态机要求,例如只能在started状态下调用pause()接口,只能在paused状态下调用resume()接口。

**图1** 录制状态变化示意图  

![Recording status change](figures/video-recording-status-change.png)

状态的详细说明请参考[AVRecorderState](../reference/apis/js-apis-media.md#avrecorderstate9)

## 开发步骤及注意事项

> **说明:**
> 
Z
zengyawen 已提交
19
> AVRecorder只负责视频数据的处理,需要与视频数据采集模块配合才能完成视频录制。视频数据采集模块需要通过Surface将视频数据传递给AVRecorder进行数据处理。当前常用的数据采集模块为相机模块,相机模块目前仅对系统应用开放,具体请参考[相机模块](../reference/apis/js-apis-camera.md)。
Z
zengyawen 已提交
20 21 22 23 24 25 26

AVRecorder详细的API说明请参考[AVRecorder API参考](../reference/apis/js-apis-media.md#avrecorder9)

1. 创建AVRecorder实例,实例创建完成进入idle状态。
     
   ```ts
   import media from '@ohos.multimedia.media'
27 28
   let avRecorder: media.AVRecorder;
   media.createAVRecorder().then((recorder: media.AVRecorder) => {
29
     avRecorder = recorder
30
   }, (error: Error) => {
31
     console.error('createAVRecorder failed')
Z
zengyawen 已提交
32 33 34 35 36 37 38 39 40 41 42
   })
   ```

2. 设置业务需要的监听事件,监听状态变化及错误上报。
   | 事件类型 | 说明 | 
   | -------- | -------- |
   | stateChange | 必要事件,监听播放器的state属性改变 | 
   | error | 必要事件,监听播放器的错误信息 | 

   ```ts
   // 状态上报回调函数
43
   avRecorder.on('stateChange', (state: media.AVRecorderState, reason: media.StateChangeReason) => {
44
     console.info('current state is: ' + state);
Z
zengyawen 已提交
45 46
   })
   // 错误上报回调函数
47
   avRecorder.on('error', (err: BusinessError) => {
48
     console.error('error happened, error message is ' + err);
Z
zengyawen 已提交
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
   })
   ```

3. 配置视频录制参数,调用prepare()接口,此时进入prepared状态。
   > **说明:**
   >
   > 配置参数需要注意:
   > 
   > - prepare接口的入参avConfig中仅设置视频相关的配置参数,如示例代码所示。
   >   如果添加了音频参数,系统将认为是“音频+视频录制”。
   > 
   > - 需要使用支持的[录制规格](avplayer-avrecorder-overview.md#支持的格式),视频比特率、分辨率、帧率以实际硬件设备支持的范围为准。
   > 
   > - 录制输出的url地址(即示例里avConfig中的url),形式为fd://xx (fd number)。需要调用基础文件操作接口([ohos.file.fs](../reference/apis/js-apis-file-fs.md))实现应用文件访问能力,获取方式参考[应用文件访问与管理](../file-management/app-file-access.md)。

   ```ts
65
   class AVProfile {
66 67 68 69 70 71
     fileFormat: media.ContainerFormatType;
     videoBitrate: number;
     videoCodec: media.CodecMimeType;
     videoFrameWidth: number;
     videoFrameHeight: number;
     videoFrameRate: number;
72 73
   }
   class AVConfig {
74 75 76 77
     videoSourceType: media.VideoSourceType;
     profile: AVProfile;
     url: string;
     rotation: number;
78 79
   }
   let avProfile: AVProfile = {
Z
zengyawen 已提交
80 81 82 83 84 85 86
     fileFormat : media.ContainerFormatType.CFT_MPEG_4, // 视频文件封装格式,只支持MP4
     videoBitrate : 200000, // 视频比特率
     videoCodec : media.CodecMimeType.VIDEO_AVC, // 视频文件编码格式,支持mpeg4和avc两种格式
     videoFrameWidth : 640,  // 视频分辨率的宽
     videoFrameHeight : 480, // 视频分辨率的高
     videoFrameRate : 30 // 视频帧率
   }
87
   let avConfig: AVConfig = {
Z
zengyawen 已提交
88 89 90 91 92 93
     videoSourceType : media.VideoSourceType.VIDEO_SOURCE_TYPE_SURFACE_YUV, // 视频源类型,支持YUV和ES两种格式
     profile : this.avProfile,
     url : 'fd://35', // 参考应用文件访问与管理开发示例新建并读写一个文件
     rotation : 0, // 视频旋转角度,默认为0不旋转,支持的值为0、90、180、270
   }
   avRecorder.prepare(avConfig).then(() => {
94
     console.info('avRecorder prepare success')
95
   }, (error: Error) => {
96
     console.error('avRecorder prepare failed')
Z
zengyawen 已提交
97 98 99 100 101 102 103 104 105
   })
   ```

4. 获取视频录制需要的SurfaceID。
   调用getInputSurface()接口,接口的返回值SurfaceID用于传递给视频数据输入源模块。常用的输入源模块为相机,以下示例代码中,采用相机作为视频输入源为例。

     输入源模块通过SurfaceID可以获取到Surface,通过Surface可以将视频数据流传递给AVRecorder,由AVRecorder再进行视频数据的处理。
     
   ```ts
106
   avRecorder.getInputSurface().then((surfaceId: string) => {
Z
zengyawen 已提交
107
     console.info('avRecorder getInputSurface success')
108
   }, (error: Error) => {
Z
zengyawen 已提交
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134
     console.error('avRecorder getInputSurface failed')
   })
   ```

5. 初始化视频数据输入源。该步骤需要在输入源模块完成,以相机为例,需要创建录像输出流,包括创建Camera对象、获取相机列表、创建相机输入流等,相机详细步骤请参考[相机-录像实现方案](camera-recording-case.md)

6. 开始录制,启动输入源输入视频数据,例如相机模块调用camera.VideoOutput.start接口启动相机录制。然后调用AVRecorder.start()接口,此时AVRecorder进入started状态。

7. 暂停录制,调用pause()接口,此时AVRecorder进入paused状态,同时暂停输入源输入数据。例如相机模块调用camera.VideoOutput.stop停止相机视频数据输入。

8. 恢复录制,调用resume()接口,此时再次进入started状态。

9. 停止录制,调用stop()接口,此时进入stopped状态,同时停止相机录制。

10. 重置资源,调用reset()重新进入idle状态,允许重新配置录制参数。

11. 销毁实例,调用release()进入released状态,退出录制,释放视频数据输入源相关资源,例如相机资源。


## 完整示例

参考以下示例,完成“开始录制-暂停录制-恢复录制-停止录制”的完整流程。

  
```ts
import media from '@ohos.multimedia.media'
135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
import { BusinessError } from '@ohos.base';
const TAG = 'VideoRecorderDemo:'
class AVProfile {
  fileFormat: media.ContainerFormatType;
  videoBitrate: number;
  videoCodec: media.CodecMimeType;
  videoFrameWidth: number;
  videoFrameHeight: number;
  videoFrameRate: number;
}
class AVConfig {
  videoSourceType: media.VideoSourceType;
  profile: AVProfile;
  url: string;
  rotation: number;
}
Z
zengyawen 已提交
151 152
const TAG = 'VideoRecorderDemo:'
export class VideoRecorderDemo {
153 154 155
  private avRecorder: media.AVRecorder;
  private videoOutSurfaceId: string;
  private avProfile: AVProfile = {
Z
zengyawen 已提交
156 157 158 159 160 161 162
    fileFormat : media.ContainerFormatType.CFT_MPEG_4, // 视频文件封装格式,只支持MP4
    videoBitrate : 100000, // 视频比特率
    videoCodec : media.CodecMimeType.VIDEO_AVC, // 视频文件编码格式,支持mpeg4和avc两种格式
    videoFrameWidth : 640,  // 视频分辨率的宽
    videoFrameHeight : 480, // 视频分辨率的高
    videoFrameRate : 30 // 视频帧率
  }
163
  private avConfig: AVConfig = {
Z
zengyawen 已提交
164 165 166 167 168 169 170 171 172
    videoSourceType : media.VideoSourceType.VIDEO_SOURCE_TYPE_SURFACE_YUV, // 视频源类型,支持YUV和ES两种格式
    profile : this.avProfile,
    url : 'fd://35', //  参考应用文件访问与管理开发示例新建并读写一个文件
    rotation : 0, // 视频旋转角度,默认为0不旋转,支持的值为0、90、180、270
  }

  // 注册avRecorder回调函数
  setAvRecorderCallback() {
    // 状态机变化回调函数
173
    this.avRecorder.on('stateChange', (state: media.AVRecorderState, reason: media.StateChangeReason) => {
Z
zengyawen 已提交
174 175 176
      console.info(TAG + 'current state is: ' + state);
    })
    // 错误上报回调函数
177
    this.avRecorder.on('error', (err: BusinessError) => {
Z
zengyawen 已提交
178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
      console.error(TAG + 'error ocConstantSourceNode, error message is ' + err);
    })
  }

  // 相机相关准备工作
  async prepareCamera() {
    // 具体实现查看相机资料
  }

  // 启动相机出流
  async startCameraOutput() {
    // 调用VideoOutput的start接口开始录像输出
  }

  // 停止相机出流
  async stopCameraOutput() {
    // 调用VideoOutput的stop接口停止录像输出
  }

  // 释放相机实例
  async releaseCamera() {
    // 释放相机准备阶段创建出的实例
  }

  // 开始录制对应的流程
  async startRecordingProcess() {
    // 1.创建录制实例;
    this.avRecorder = await media.createAVRecorder();
    this.setAvRecorderCallback();
    // 2. 获取录制文件fd;获取到的值传递给avConfig里的url,实现略
    // 3.配置录制参数完成准备工作
    await this.avRecorder.prepare(this.avConfig);
    this.videoOutSurfaceId = await this.avRecorder.getInputSurface(); 
    // 4.完成相机相关准备工作
    await this.prepareCamera();
    // 5.启动相机出流
    await this.startCameraOutput();
    // 6. 启动录制
216
    await this.avRecorder.start();
Z
zengyawen 已提交
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
  }

  // 暂停录制对应的流程
  async pauseRecordingProcess() {
    if (this.avRecorder.state === 'started') { // 仅在started状态下调用pause为合理状态切换
      await this.avRecorder.pause();
      await this.stopCameraOutput(); // 停止相机出流
    }
  }

  // 恢复录制对应的流程
  async resumeRecordingProcess() {
    if (this.avRecorder.state === 'paused') { // 仅在paused状态下调用resume为合理状态切换
      await this.startCameraOutput();  // 启动相机出流
      await this.avRecorder.resume();
    }
  }

  async stopRecordingProcess() {
    // 1. 停止录制
    if (this.avRecorder.state === 'started'
    || this.avRecorder.state === 'paused' ) { // 仅在started或者paused状态下调用stop为合理状态切换
      await this.avRecorder.stop();
      await this.stopCameraOutput();
    }
    // 2.重置
    await this.avRecorder.reset();
    // 3.释放录制实例
    await this.avRecorder.release();
    // 4.文件录制完成后,关闭fd,实现略
    // 5.释放相机相关实例
    await this.releaseCamera();
  }

  // 一个完整的【开始录制-暂停录制-恢复录制-停止录制】示例
  async videoRecorderDemo() {
    await this.startRecordingProcess();         // 开始录制
    // 用户此处可以自行设置录制时长,例如通过设置休眠阻止代码执行
    await this.pauseRecordingProcess();         //暂停录制
    await this.resumeRecordingProcess();        // 恢复录制
    await this.stopRecordingProcess();          // 停止录制
  }
}
```