HarmonyOS开发中自定义相机横向拍照,摄像头方向差了90度?

在开发自定义相机拍照功能时,因为需要横向拍照,所以将界面横向设置之后,发现摄像头看到的画面是差了90度,请问要怎么设置这个摄像头方向?

阅读 801
avatarAI BotBETA

在HarmonyOS开发中,如果你遇到了自定义相机横向拍照时摄像头方向差了90度的问题,这通常是因为摄像头的预览方向与设备的屏幕方向不一致所导致的。为了解决这个问题,你需要在设置相机预览时,根据设备的方向来调整摄像头的预览方向。

以下是一些基本的步骤和代码示例,用于在HarmonyOS中调整摄像头预览的方向:

  1. 获取设备当前的方向
    你可以通过监听设备的屏幕方向变化来获取当前的方向。
  2. 根据方向调整摄像头预览
    当设备的方向改变时,相应地调整摄像头的预览角度。通常,这涉及到设置相机的setDisplayOrientation(如果API支持)或者在你的渲染逻辑中调整渲染的角度。

然而,由于HarmonyOS的API可能与Android或iOS有所不同,具体的API调用可能会有所不同。以下是一个假设性的方法,用于说明如何根据设备方向调整摄像头预览的角度(注意,你需要根据HarmonyOS的实际API进行调整):

// 假设有一个方法用于获取设备当前的方向(如屏幕是横向还是纵向)
int getDeviceOrientation() {
    // 这里应该根据HarmonyOS的API来实现
    // 返回例如0(自然方向)、90(顺时针旋转90度)、180或270等
    return 90; // 假设设备现在是横向的,并且需要顺时针旋转90度来匹配预览
}

// 在设置相机预览时,根据设备方向调整预览角度
void setupCameraPreview() {
    Camera camera = ...; // 获取Camera实例
    int orientation = getDeviceOrientation();

    // 假设HarmonyOS API有类似Android的setDisplayOrientation方法
    // camera.setDisplayOrientation(orientation);

    // 如果API不支持直接设置,你可能需要在渲染预览时手动调整渲染矩阵
    // 例如,在绘制预览帧到SurfaceView或TextureView时,调整Canvas或TextureView的矩阵

    // 这里是伪代码,具体实现取决于你的预览显示方式
    // previewSurface.setTransformationMatrix(createRotationMatrix(orientation));
}

// 创建一个旋转矩阵(仅作为示例,非实际代码)
Matrix createRotationMatrix(int orientation) {
    Matrix matrix = new Matrix();
    // 根据orientation设置旋转角度
    // 注意:这里的旋转角度和矩阵设置取决于你的具体实现
    return matrix;
}

注意:上面的代码是一个简化的示例,用于说明如何根据设备方向调整摄像头预览的思路。在HarmonyOS中,你可能需要查找或创建特定的API调用来实现这一功能。

此外,确保在设备的方向改变时(如用户旋转了设备),你的应用能够捕获到这个变化,并相应地更新摄像头预览的方向。这通常涉及到注册并监听屏幕方向变化的监听器。

1 个回答

解决方案
https://developer.huawei.com/consumer/cn/doc/harmonyos-guides...
指定参数:
https://developer.huawei.com/consumer/cn/doc/harmonyos-refere...
https://developer.huawei.com/consumer/cn/doc/harmonyos-refere...
参考代码:

import { common, Want } from '@kit.AbilityKit'; 
import { BusinessError } from '@kit.BasicServicesKit'; 
import { fileIo, fileUri } from '@kit.CoreFileKit'; 
import { promptAction } from '@kit.ArkUI'; 
import { camera, cameraPicker } from '@kit.CameraKit'; 
const TAG = '[SystemCameraDemo]'; 
@Entry 
@Component 
struct Index { 
  @State path: string = ''; 
  @State isVideo: boolean = false; 
  private context = getContext(this) as common.UIAbilityContext; 
  build() { 
    Row() { 
      Column({ space: 20 }) { 
        Button('拉起系统相机拍照--startAbility方式') 
          .width(300) 
          .height(50) 
          .onClick(async () => { 
            this.startAbility('ohos.want.action.imageCapture', 'com.xxx.xxx.myapplication'); 
          }) 
        Button('拉起系统相机录像--startAbility方式') 
          .width(300) 
          .height(50) 
          .onClick(async () => { 
            this.startAbility('ohos.want.action.videoCapture', 'com.xxx.xxx.myapplication'); 
          }) 
        Button('拉起系统相机拍照--cameraPicker方式') 
          .width(300) 
          .height(50) 
          .onClick(async () => { 
            this.cameraPicker(cameraPicker.PickerMediaType.PHOTO); 
          }) 
        Button('拉起系统相机录像--cameraPicker方式') 
          .width(300) 
          .height(50) 
          .onClick(async () => { 
            this.cameraPicker(cameraPicker.PickerMediaType.Video); 
          }) 
        if (this.isVideo && this.path) { 
          Video({ src: this.path }) 
            .objectFit(ImageFit.Contain) 
            .width('100%') 
            .height(300) 
        } 
        if (!this.isVideo && this.path) { 
          Image(this.path) 
            .objectFit(ImageFit.Contain) 
            .width('100%') 
            .height(300) 
        } 
      } 
      .width('100%') 
    } 
    .height('100%') 
  } 
  async cameraPicker(type: cameraPicker.PickerMediaType) { 
    try { 
      let pickerProfile: cameraPicker.PickerProfile = { 
        // 相机的位置  后置摄像头 
        cameraPosition: camera.CameraPosition.CAMERA_POSITION_BACK, 
        //saveUri 保存配置信息的uri 
        //videoDuration 录制的最大时长 
      }; 
      cameraPicker.pick(getContext(this), [type], pickerProfile) 
        .then((result: cameraPicker.PickerResult) => { 
          if (result.resultCode != 0) { 
            // 处理业务逻辑错误 
            console.error(TAG, `cameraPicker.pick failed, result is ${JSON.stringify(result)}`); 
            promptAction.showToast({ message: '拍摄失败' }) 
            return; 
          } 
          //若saveUri为空,resultUri为公共媒体路径。若saveUri不为空且具备写权限,resultUri与saveUri相同。若saveUri不为空且不具备写权限,则无法获取到resultUri 
          let uri: string = result.resultUri; 
          // 执行正常业务 
          console.info(TAG, `Succeeded in cameraPicker.pick. Data is ${JSON.stringify(result)}, uri is ${uri}`); 
          if (uri) { 
            // 保存到本地 
            this.save2Local(uri); 
          } else { 
            promptAction.showToast({ message: '拍摄失败' }) 
          } 
        }) 
    } catch (error) { 
      let err = error as BusinessError; 
      console.error(TAG, `the pick call failed. error code: ${err.code}`); 
      promptAction.showToast({ message: '拍摄失败' }) 
    } 
  } 
  startAbility(action: string, bundleName: string) { 
    let want: Want = { 
      action: action, 
      parameters: { 
        // 拍照完成后返回的应用BundleName 
        callBundleName: bundleName, 
        supportMultiMode: false 
      } 
    }; 
    try { 
      this.context.startAbilityForResult(want, (err: BusinessError, result: common.AbilityResult) => { 
        if (err.code) { 
          // 处理业务逻辑错误 
          console.error(TAG, `startAbilityForResult failed, code is ${err.code}, message is ${err.message}`); 
          return; 
        } 
        let uri: string = result?.want?.parameters?.resourceUri as string; 
        // 执行正常业务 
        console.info(TAG, `Succeeded in starting ability for result. Data is ${JSON.stringify(result)}, uri is ${uri}`); 
        if (uri) { 
          // 保存到本地 
          this.save2Local(uri); 
        } else { 
          promptAction.showToast({ message: '拍摄失败' }) 
        } 
      }); 
    } catch (err) { 
      let error = err as BusinessError; 
      console.error(TAG, `startAbilityForResult failed, err is ${JSON.stringify(error)}`); 
      promptAction.showToast({ message: '拍摄失败' }) 
    } 
  } 
  save2Local(uri: string) { 
    try { 
      let file = fileIo.openSync(uri, fileIo.OpenMode.READ_ONLY); 
      let prefix = uri.substring(uri.lastIndexOf('.') + 1); 
      let tempFileName = getContext(this).filesDir + '/' + new Date().getTime() + '.' + prefix; 
      let tempFile = fileIo.openSync(tempFileName, fileIo.OpenMode.READ_WRITE | fileIo.OpenMode.CREATE); 
      fileIo.copyFileSync(file.fd, tempFile.fd); 
      this.path = fileUri.getUriFromPath(tempFileName); 
      this.isVideo = (prefix == 'mp4' || prefix == 'MP4'); 
      console.info(TAG, `resolve2Sandbox successful.`); 
      promptAction.showToast({ message: '拍摄成功' }) 
    } catch (err) { 
      let error = err as BusinessError; 
      console.error(TAG, `resolve2Sandbox failed, err is ${JSON.stringify(error)}`); 
    } 
  } 
}
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题