• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 双路预览(ArkTS)
2
3在开发相机应用时,需要先参考开发准备[申请相关权限](camera-preparation.md)。
4
5双路预览,即应用可同时使用两路预览流,一路用于在屏幕上显示,一路用于图像处理等其他操作,提升处理效率。
6
7相机应用通过控制相机,实现图像显示(预览)、照片保存(拍照)、视频录制(录像)等基础操作。相机开发模型为Surface模型,即应用通过Surface进行数据传递,通过ImageReceiver的surface获取拍照流的数据、通过XComponent的surface获取预览流的数据。
8
9如果要实现双路预览,即将拍照流改为预览流,将拍照流中的surface改为预览流的surface,通过ImageReceiver的surface创建previewOutput,其余流程与拍照流和预览流一致。
10
11详细的API说明请参考[Camera API参考](../../reference/apis-camera-kit/js-apis-camera.md)。
12
13## 约束与限制
14
15- 暂不支持动态添加流,即不能在没有调用[session.stop](../../reference/apis-camera-kit/js-apis-camera.md#stop11)的情况下,调用[addOutput](../../reference/apis-camera-kit/js-apis-camera.md#addoutput11)添加流。
16- 对ImageReceiver组件获取到的图像数据处理后,需要将对应的图像Buffer释放,确保Surface的BufferQueue正常轮转。
17
18## 调用流程
19
20双路方案调用流程图建议如下:
21
22![dual-preview-streams-instructions](figures/dual-preview-streams-instructions.png)
23
24## 开发步骤
25
26- 用于处理图像的第一路预览流:创建ImageReceiver对象,获取SurfaceId创建第一路预览流,注册图像监听,按需处理预览流每帧图像。
27- 用于显示画面的第二路预览流:创建XComponent组件,获取SurfaceId创建第二路预览流,预览流画面直接在组件内渲染。
28- 创建预览流获取数据:创建上述两路预览流,配置进相机会话,启动会话后,两路预览流同时获取数据。
29
30### 用于处理图像的第一路预览流
311. 导入依赖,本篇文档需要用到图片和相机框架等相关依赖包。
32    ```ts
33    import { image } from '@kit.ImageKit';
34    import { camera } from '@kit.CameraKit';
35    import { BusinessError } from '@kit.BasicServicesKit';
36    ```
37
382. 获取第一路预览流SurfaceId:创建ImageReceiver对象,通过ImageReceiver对象可获取其SurfaceId。
39
40    ```ts
41    let imageWidth: number = 1920; // 请使用设备支持profile的size的宽。
42    let imageHeight: number = 1080; // 请使用设备支持profile的size的高。
43
44    async function initImageReceiver():Promise<void>{
45      // 创建ImageReceiver对象。
46      let size: image.Size = { width: imageWidth, height: imageHeight };
47      let imageReceiver = image.createImageReceiver(size, image.ImageFormat.JPEG, 8);
48      // 获取取第一路流SurfaceId。
49      let imageReceiverSurfaceId = await imageReceiver.getReceivingSurfaceId();
50      console.info(`initImageReceiver imageReceiverSurfaceId:${imageReceiverSurfaceId}`);
51    }
52    ```
53
543. 注册监听处理预览流每帧图像数据:通过ImageReceiver组件中imageArrival事件监听获取底层返回的图像数据,详细的API说明请参考[Image API参考](../../reference/apis-image-kit/js-apis-image.md)。
55
56    ```ts
57    function onImageArrival(receiver: image.ImageReceiver): void {
58      // 注册imageArrival监听。
59      receiver.on('imageArrival', () => {
60        // 获取图像。
61        receiver.readNextImage((err: BusinessError, nextImage: image.Image) => {
62          if (err || nextImage === undefined) {
63            console.error('readNextImage failed');
64            return;
65          }
66          // 解析图像内容。
67          nextImage.getComponent(image.ComponentType.JPEG, async (err: BusinessError, imgComponent: image.Component) => {
68            if (err || imgComponent === undefined) {
69              console.error('getComponent failed');
70            }
71            if (imgComponent.byteBuffer) {
72              // 详情见下方解析图片buffer数据参考,本示例以方式一为例。
73              let width = nextImage.size.width; // 获取图片的宽。
74              let height = nextImage.size.height; // 获取图片的高。
75              let stride = imgComponent.rowStride; // 获取图片的stride。
76              console.debug(`getComponent with width:${width} height:${height} stride:${stride}`);
77              // stride与width一致。
78              if (stride == width) {
79                let pixelMap = await image.createPixelMap(imgComponent.byteBuffer, {
80                  size: { height: height, width: width },
81                  srcPixelFormat: 8,
82                })
83              } else {
84                // stride与width不一致。
85                const dstBufferSize = width * height * 1.5
86                const dstArr = new Uint8Array(dstBufferSize)
87                for (let j = 0; j < height * 1.5; j++) {
88                  const srcBuf = new Uint8Array(imgComponent.byteBuffer, j * stride, width)
89                  dstArr.set(srcBuf, j * width)
90                }
91                let pixelMap = await image.createPixelMap(dstArr.buffer, {
92                  size: { height: height, width: width },
93                  srcPixelFormat: 8,
94                })
95              }
96            } else {
97              console.error('byteBuffer is null');
98            }
99            // 确保当前buffer没有在使用的情况下,可进行资源释放。
100            // 如果对buffer进行异步操作,需要在异步操作结束后再释放该资源(nextImage.release())。
101            nextImage.release();
102          })
103        })
104      })
105    }
106    ```
107
108    通过 [image.Component](../../reference/apis-image-kit/js-apis-image.md#component9) 解析图片buffer数据参考:
109
110    > **注意:**
111    > 需要确认图像的宽width是否与行距rowStride一致,如果不一致可参考以下方式处理:
112
113    方式一:去除imgComponent.byteBuffer中stride数据,拷贝得到新的buffer,调用不支持stride的接口处理buffer。
114
115    ```ts
116    // 以NV21为例(YUV_420_SP格式的图片)YUV_420_SP内存计算公式:长x宽+(长x宽)/2。
117    const dstBufferSize = width * height * 1.5;
118    const dstArr = new Uint8Array(dstBufferSize);
119    // 逐行读取buffer数据。
120    for (let j = 0; j < height * 1.5; j++) {
121      // imgComponent.byteBuffer的每行数据拷贝前width个字节到dstArr中。
122      const srcBuf = new Uint8Array(imgComponent.byteBuffer, j * stride, width);
123      dstArr.set(srcBuf, j * width);
124    }
125    let pixelMap = await image.createPixelMap(dstArr.buffer, {
126      size: { height: height, width: width }, srcPixelFormat: 8
127    });
128    ```
129
130    方式二:根据stride*height创建pixelMap,然后调用pixelMap的cropSync方法裁剪掉多余的像素。
131
132    ```ts
133    // 创建pixelMap,width宽传行距stride的值。
134    let pixelMap = await image.createPixelMap(imgComponent.byteBuffer, {
135      size:{height: height, width: stride}, srcPixelFormat: 8});
136    // 裁剪多余的像素。
137    pixelMap.cropSync({size:{width:width, height:height}, x:0, y:0});
138    ```
139
140    方式三:将原始imgComponent.byteBuffer和stride信息一起传给支持stride的接口处理。
141
142
143
144### 用于显示画面的第二路预览流
145
146获取第二路预览流SurfaceId:创建XComponent组件用于预览流显示,获取surfaceId请参考XComponent组件提供的[getXcomponentSurfaceId](../../reference/apis-arkui/arkui-ts/ts-basic-components-xcomponent.md#getxcomponentsurfaceid9)方法,而XComponent的能力由UI提供,相关介绍可参考[XComponent组件参考](../../reference/apis-arkui/arkui-ts/ts-basic-components-xcomponent.md)。
147
148```ts
149@Component
150struct example {
151  xComponentCtl: XComponentController = new XComponentController();
152  surfaceId:string = '';
153  imageWidth: number = 1920;
154  imageHeight: number = 1080;
155  private uiContext: UIContext = this.getUIContext();
156
157  build() {
158    XComponent({
159      id: 'componentId',
160      type: XComponentType.SURFACE,
161      controller: this.xComponentCtl
162    })
163      .onLoad(async () => {
164        console.info('onLoad is called');
165        this.surfaceId = this.xComponentCtl.getXComponentSurfaceId(); // 获取组件surfaceId。
166        // 使用surfaceId创建预览流,开启相机,组件实时渲染每帧预览流数据。
167      })
168      // surface的宽、高设置与XComponent组件的宽、高设置相反,或使用.renderFit(RenderFit.RESIZE_CONTAIN)自动填充显示无需设置宽、高。
169      .width(this.uiContext.px2vp(this.imageHeight))
170      .height(this.uiContext.px2vp(this.imageWidth))
171  }
172}
173```
174
175
176
177### 创建预览流获取数据
178
179通过两个SurfaceId分别创建两路预览流输出,加入相机会话,启动相机会话,获取预览流数据。
180
181```ts
182function createDualPreviewOutput(cameraManager: camera.CameraManager, previewProfile: camera.Profile,
183  session: camera.Session, imageReceiverSurfaceId: string, xComponentSurfaceId: string): void {
184  // 使用imageReceiverSurfaceId创建第一路预览。
185  let previewOutput1 = cameraManager.createPreviewOutput(previewProfile, imageReceiverSurfaceId);
186  if (!previewOutput1) {
187  console.error('createPreviewOutput1 error');
188  }
189  // 使用xComponentSurfaceId创建第二路预览。
190  let previewOutput2 = cameraManager.createPreviewOutput(previewProfile, xComponentSurfaceId);
191  if (!previewOutput2) {
192  console.error('createPreviewOutput2 error');
193  }
194  // 添加第一路预览流输出。
195  session.addOutput(previewOutput1);
196  // 添加第二路预览流输出。
197  session.addOutput(previewOutput2);
198}
199```
200
201
202
203## 完整示例
204
205```ts
206import { camera } from '@kit.CameraKit';
207import { image } from '@kit.ImageKit';
208import { BusinessError } from '@kit.BasicServicesKit';
209import { abilityAccessCtrl, Permissions } from '@kit.AbilityKit';
210
211@Entry
212@Component
213struct Index {
214  private imageReceiver: image.ImageReceiver | undefined = undefined;
215  private imageReceiverSurfaceId: string = '';
216  private xComponentCtl: XComponentController = new XComponentController();
217  private xComponentSurfaceId: string = '';
218  @State imageWidth: number = 1920;
219  @State imageHeight: number = 1080;
220  private cameraManager: camera.CameraManager | undefined = undefined;
221  private cameras: Array<camera.CameraDevice> | Array<camera.CameraDevice> = [];
222  private cameraInput: camera.CameraInput | undefined = undefined;
223  private previewOutput1: camera.PreviewOutput | undefined = undefined;
224  private previewOutput2: camera.PreviewOutput | undefined = undefined;
225  private session: camera.VideoSession | undefined = undefined;
226  private uiContext: UIContext = this.getUIContext();
227  private context: Context | undefined = this.uiContext.getHostContext();
228  private cameraPermission: Permissions = 'ohos.permission.CAMERA'; // 申请权限相关问题可参考本篇开头的申请相关权限文档
229  @State isShow: boolean = false;
230
231  async requestPermissionsFn(): Promise<void> {
232    let atManager = abilityAccessCtrl.createAtManager();
233    if (this.context) {
234      let res = await atManager.requestPermissionsFromUser(this.context, [this.cameraPermission]);
235      for (let i =0; i < res.permissions.length; i++) {
236        if (this.cameraPermission.toString() === res.permissions[i] && res.authResults[i] === 0) {
237          this.isShow = true;
238        }
239      }
240    }
241  }
242
243  aboutToAppear(): void {
244    this.requestPermissionsFn();
245  }
246
247  onPageShow(): void {
248    console.info('onPageShow');
249    this.initImageReceiver();
250    if (this.xComponentSurfaceId !== '') {
251      this.initCamera();
252    }
253  }
254
255  onPageHide(): void {
256    console.info('onPageHide');
257    this.releaseCamera();
258  }
259
260  /**
261   * 获取ImageReceiver的SurfaceId
262   * @param receiver
263   * @returns
264   */
265  async initImageReceiver(): Promise<void> {
266    if (!this.imageReceiver) {
267      // 创建ImageReceiver。
268      let size: image.Size = { width: this.imageWidth, height: this.imageHeight };
269      this.imageReceiver = image.createImageReceiver(size, image.ImageFormat.JPEG, 8);
270      // 获取取第一路流SurfaceId。
271      this.imageReceiverSurfaceId = await this.imageReceiver.getReceivingSurfaceId();
272      console.info(`initImageReceiver imageReceiverSurfaceId:${this.imageReceiverSurfaceId}`);
273      // 注册监听处理预览流每帧图像数据。
274      this.onImageArrival(this.imageReceiver);
275    }
276  }
277
278  /**
279   * 注册ImageReceiver图像监听
280   * @param receiver
281   */
282  onImageArrival(receiver: image.ImageReceiver): void {
283    // 注册imageArrival监听。
284    receiver.on('imageArrival', () => {
285      console.info('image arrival');
286      // 获取图像。
287      receiver.readNextImage((err: BusinessError, nextImage: image.Image) => {
288        if (err || nextImage === undefined) {
289          console.error('readNextImage failed');
290          return;
291        }
292        // 解析图像内容。
293        nextImage.getComponent(image.ComponentType.JPEG, async (err: BusinessError, imgComponent: image.Component) => {
294          if (err || imgComponent === undefined) {
295            console.error('getComponent failed');
296          }
297          if (imgComponent.byteBuffer) {
298            // 请参考步骤7解析buffer数据,本示例以方式一为例。
299            let width = nextImage.size.width; // 获取图片的宽。
300            let height = nextImage.size.height; // 获取图片的高。
301            let stride = imgComponent.rowStride; // 获取图片的stride。
302            console.debug(`getComponent with width:${width} height:${height} stride:${stride}`);
303            // stride与width一致。
304            if (stride == width) {
305              let pixelMap = await image.createPixelMap(imgComponent.byteBuffer, {
306                size: { height: height, width: width },
307                srcPixelFormat: 8,
308              })
309            } else {
310              // stride与width不一致。
311              const dstBufferSize = width * height * 1.5 // 以NV21为例(YUV_420_SP格式的图片)YUV_420_SP内存计算公式:长x宽+(长x宽)/2。
312              const dstArr = new Uint8Array(dstBufferSize)
313              for (let j = 0; j < height * 1.5; j++) {
314                const srcBuf = new Uint8Array(imgComponent.byteBuffer, j * stride, width)
315                dstArr.set(srcBuf, j * width)
316              }
317              let pixelMap = await image.createPixelMap(dstArr.buffer, {
318                size: { height: height, width: width },
319                srcPixelFormat: 8,
320              })
321            }
322          } else {
323            console.error('byteBuffer is null');
324          }
325          // 确保当前buffer没有在使用的情况下,可进行资源释放。
326          // 如果对buffer进行异步操作,需要在异步操作结束后再释放该资源(nextImage.release())。
327          nextImage.release();
328          console.info('image process done');
329        })
330      })
331    })
332  }
333
334  build() {
335    Column() {
336      if (this.isShow) {
337        XComponent({
338          id: 'componentId',
339          type: XComponentType.SURFACE,
340          controller: this.xComponentCtl
341        })
342          .onLoad(async () => {
343            console.info('onLoad is called');
344            this.xComponentSurfaceId = this.xComponentCtl.getXComponentSurfaceId(); // 获取组件surfaceId。
345            // 初始化相机,组件实时渲染每帧预览流数据。
346            this.initCamera()
347          })
348          .width(this.uiContext.px2vp(this.imageHeight))
349          .height(this.uiContext.px2vp(this.imageWidth))
350      }
351    }
352    .justifyContent(FlexAlign.Center)
353    .height('100%')
354    .width('100%')
355  }
356
357  // 初始化相机。
358  async initCamera(): Promise<void> {
359    console.info(`initCamera imageReceiverSurfaceId:${this.imageReceiverSurfaceId} xComponentSurfaceId:${this.xComponentSurfaceId}`);
360    try {
361      // 获取相机管理器实例。
362      this.cameraManager = camera.getCameraManager(this.context);
363      if (!this.cameraManager) {
364        console.error('initCamera getCameraManager');
365      }
366      // 获取当前设备支持的相机device列表。
367      this.cameras = this.cameraManager.getSupportedCameras();
368      if (!this.cameras) {
369        console.error('initCamera getSupportedCameras');
370      }
371      // 选择一个相机device,创建cameraInput输出对象。
372      this.cameraInput = this.cameraManager.createCameraInput(this.cameras[0]);
373      if (!this.cameraInput) {
374        console.error('initCamera createCameraInput');
375      }
376      // 打开相机。
377      await this.cameraInput.open().catch((err: BusinessError) => {
378        console.error(`initCamera open fail: ${err}`);
379      })
380      // 获取相机device支持的profile。
381      let capability: camera.CameraOutputCapability =
382        this.cameraManager.getSupportedOutputCapability(this.cameras[0], camera.SceneMode.NORMAL_VIDEO);
383      if (!capability) {
384        console.error('initCamera getSupportedOutputCapability');
385      }
386      // 根据业务需求选择一个支持的预览流profile。
387      let previewProfile: camera.Profile = capability.previewProfiles[0];
388      this.imageWidth = previewProfile.size.width; // 更新xComponent组件的宽。
389      this.imageHeight = previewProfile.size.height; // 更新xComponent组件的高。
390      console.info(`initCamera imageWidth:${this.imageWidth} imageHeight:${this.imageHeight}`);
391      // 使用imageReceiverSurfaceId创建第一路预览。
392      this.previewOutput1 = this.cameraManager.createPreviewOutput(previewProfile, this.imageReceiverSurfaceId);
393      if (!this.previewOutput1) {
394        console.error('initCamera createPreviewOutput1');
395      }
396      // 使用xComponentSurfaceId创建第二路预览。
397      this.previewOutput2 = this.cameraManager.createPreviewOutput(previewProfile, this.xComponentSurfaceId);
398      if (!this.previewOutput2) {
399        console.error('initCamera createPreviewOutput2');
400      }
401      // 创建录像模式相机会话。
402      this.session = this.cameraManager.createSession(camera.SceneMode.NORMAL_VIDEO) as camera.VideoSession;
403      if (!this.session) {
404        console.error('initCamera createSession');
405      }
406      // 开始配置会话。
407      this.session.beginConfig();
408      // 添加相机设备输入。
409      this.session.addInput(this.cameraInput);
410      // 添加第一路预览流输出。
411      this.session.addOutput(this.previewOutput1);
412      // 添加第二路预览流输出。
413      this.session.addOutput(this.previewOutput2);
414      // 提交会话配置。
415      await this.session.commitConfig();
416      // 开始启动已配置的输入输出流。
417      await this.session.start();
418    } catch (error) {
419      console.error(`initCamera fail: ${error}`);
420    }
421  }
422
423  // 释放相机。
424  async releaseCamera(): Promise<void> {
425    console.info('releaseCamera E');
426    try {
427      // 停止当前会话。
428      await this.session?.stop();
429      // 释放相机输入流。
430      await this.cameraInput?.close();
431      // 释放预览输出流。
432      await this.previewOutput1?.release();
433      // 释放拍照输出流。
434      await this.previewOutput2?.release();
435      // 释放会话。
436      await this.session?.release();
437    } catch (error) {
438      console.error(`initCamera fail: ${error}`);
439    }
440  }
441}
442```