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 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```