介绍

简介

  HwCameraKit为开发者提供了一套兼容EMUI的相机能力开放接口,开发者可以通过HwCameraKit使自己的应用快速接入华为相机的私有能力,扩展应用的拍摄功能,为用户提供更好拍摄体验。

  人像模式为HwCameraKit开放的多种相机模式其中之一,它将允许您:

  • 获取华为相机的人像拍摄能力,包括背景虚化、环境光、瘦脸,皮肤光滑、调色等能力;
  • 提供相机模式高级编程接口,简化相机应用开发,并借助IDE工具快速接入上述能力。

关于本次CodeLab

你将建立什么

  在本次CodeLab中,您将使用HwCameraKit建立一款Android相机应用程序,使其可以获得华为相机的人像拍摄功能,实现人像的虚化、美肤等效果。



普通拍摄(左) VS 人像模式开启虚化(右)

你会学到什么

  • 使用HwCameraKit IDE高效集成华为相机开放能力
  • 熟悉HwCameraKit基本开发模式

你需要什么

硬件要求

  1. 开发计算机(台式机或笔记本电脑)
  2. 操作系统为EMUI10.0及以上版本的华为手机

软件要求

  1. JAVA JDK安装包
  2. Android SDK包
  3. HwCameraKit IDE工具:DevEco

申请Camera相关权限

Duration: 0:05

  1. 在工程的Manifest文件中添加相关权限:

<uses-permission   android:name="android.permission.CAMERA"/>

<uses-permission android:name="android.permission.RECORD_AUDIO"/>

<uses-permission   android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

  2. 动态申请相关权限:

private static final String [] PERMISSIONS_ARRAY = new String[]   {
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.ACCESS_FINE_LOCATION
}; private static List<String> permissionsList = new ArrayList<>(PERMISSIONS_ARRAY.length); /** * 动态请求WRITE_EXTERNAL_STORAGE CAMERA RECORD_AUDIO权限
*
* @param activity 应用activity */ public static void requestPermission (final Activity activity) {
for (String permission : PERMISSIONS_ARRAY) {
if (ContextCompat.checkSelfPermission(activity, permission) != PackageManager.PERMISSION_GRANTED) {
permissionsList.add(permission);
}
}
ActivityCompat.requestPermissions(activity, permissionsList.toArray(new String [permissionsList.size()]), REQUEST_CODE_ASK_PERMISSIONS);
}

集成HwCameraKit开放能力

Duration: 0:60

  通过IDE提供的功能卡片,可以快速获取示例代码并将其添加到工程中,以人像模式为例,找到人像能力卡片:可通过Tools->EMUI Kits->Kit Assistant->Camera->Portrait Mode 来找到该卡片。

提示:本次Codelabs提供了配套的app开发工程,请使用IDE工具导入,并根据如下步骤,实现提供的配套工程中带有/* TODO */注释的方法,从而快速集成人像模式。

步骤1 模式创建:获取CameraKit实例,创建人像模式

private   @Mode.Type int mCurrentModeType;

private CameraKit mCameraKit;

private ModeCharacteristics mModeCharacteristics;

private void createMode() {

    mCameraKit = CameraKit.getInstance(getApplicationContext());

    if (mCameraKit == null) {
Log.e(TAG, "This device does not support CameraKit!");
} /** Query camera id list*/ String[] cameraLists = mCameraKit . getCameraIdList (); if ((cameraLists != null) && (cameraLists.length > 0)) { Log.i(TAG, "Try to use camera with id " + cameraLists[0]); /** Query supported modes of this device*/ int[] modes = mCameraKit . getSupportedModes (cameraLists[0]); if (!Arrays.stream(modes).anyMatch((i) -> i == mCurrentModeType)) {
Log.w(TAG, "Current mode is not supported in this device!");
return;
}
mCameraKit.createMode(cameraLists[0], mCurrentModeType, mModeStateCallback, mCameraKitHandler);
} }

步骤2 配置模式:配置模式的状态回调,数据回调及处理这些回调的Handler,以及预览、拍照分辨率等参数

  从mModeStateCallback的onCreated回调后,可从入参可获得人像Mode实例

/** 在onCreated回调中获取mode对象,然后以行为的状态回调和数据回调及对应的线程为入参,初始化ModeConfig.Builder,配置预览的surface和拍照的大小,通过configure命令将配置项设置到mMode中,进行模式的激活*/
private void configMode() { Log.i(TAG, "configMode begin"); /** Query supported preview size*/ List<Size> previewSizes = mModeCharacteristics.getSupportedPreviewSizes(SurfaceTexture.class); /** Query supported capture size*/ List<Size> captureSizes = mModeCharacteristics.getSupportedCaptureSizes(ImageFormat.JPEG); Log.d(TAG, "configMode: captureSizes = " + captureSizes.size() + ";previewSizes=" + previewSizes.size()); /** Use the first one or default 4000x3000*/
mCaptureSize = captureSizes.stream().findFirst().orElse(new Size(4000, 3000));
/** Use the same ratio with preview*/ mPreviewSize = previewSizes.stream().filter((size) -> Math.abs((1.0f * size.getHeight() / size.getWidth()) - (1.0f * mCaptureSize.getHeight() / mCaptureSize.getWidth())) < 0.01).findFirst().get();
Log.i(TAG, "configMode: mCaptureSize = " + mCaptureSize + ";mPreviewSize=" + mPreviewSize); /** Update view*/ runOnUiThread(() -> mTextureView.setAspectRatio(mPreviewSize.getHeight(), mPreviewSize.getWidth())); SurfaceTexture texture = mTextureView.getSurfaceTexture(); if (texture == null) {
Log.e(TAG, "configMode: texture=null!");
return;
} /** Set buffer size of view*/ texture.setDefaultBufferSize(mPreviewSize.getWidth(), mPreviewSize.getHeight()); /** Get surface of texture*/ Surface surface = new Surface(texture); /** Add preview and capture parameters to config builder*/ modeConfigBuilder.addPreviewSurface(surface).addCaptureImage(mCaptureSize, ImageFormat.JPEG); /** Set callback for config builder*/ modeConfigBuilder.setDataCallback(actionDataCallback, mCameraKitHandler); modeConfigBuilder.setStateCallback(actionStateCallback, mCameraKitHandler); /** Configure mode*/ mMode.configure();
Log.i(TAG, "configMode end");
}

  使用构造器ModeConfig.Builder配置模式的状态回调及数据回调及执行回调所在的Handler。从状态回调中,开发者可以获取如启动预览后状态变化、拍照结束状态、手动对焦结束、人脸检测结果返回等信息;从数据回调中,开发者可获取模式动作执行过程中产生数据结果(如拍照图像数据等)。还可通过ModeConfig.Builder配置预览Surface及拍照分辨率。

步骤3 模式操作

3.1 开启预览

/* 在接收到到onConfigured回调后,说明mode配置成功,模式进入开启状态,此时调用startPreview命令开启预览。*/

private   void startPreview() {
mMode.startPreview();
}

3.2 参数设置

/* 用户可以通过ModeCharacteristics#getSupportedParameters查询当前模式下支持的参数(以人像模式为例:支持背景虚化,美肤等),通过ModeCharacteristics#getParameterRange查询参数支持的取值范围,通过Mode#setParameter设置对应效果。*/

/* 设置人像虚化 */
mMode.setParameter(RequestKey.HW_PORTRAIT_SPOTS_BOKEH, validValue); /* 设置美肤:皮肤光滑 */
int[] smoothLevels = modeCharacteristics.getSupportedBeauty(Metadata.BeautyType.HW_BEAUTY_SKIN_SMOOTH);
if (smoothLevels != null && smoothLevels.length != 0) {
mMode.setBeauty(Metadata.BeautyType.HW_BEAUTY_SKIN_SMOOTH,smoothLevels[smoothLevels.length - 1]);
}
/* 用户可以通过设计按钮等方式,调用takePicture进行拍照,异步通过onImageAvailable回调,返回照片Image数据。*/
mMode.takePicture();

步骤4 操作Callback

/* 以数据回调为例,在拍照后从回调里获取图片 */
private final ActionDataCallback actionDataCallback = new ActionDataCallback() {
@Override
public void onImageAvailable(Mode mode, int id, Image image) {
super.onImageAvailable(mode, id, image);
Log.d(TAG, "onImageAvailable");
new ImageSaver(image, mFile, CameraKitActivity.this).run();
}
};

提示:除了从ActionDataCallback获取拍照图片等数据以外,从ActionStateCallback中还可获取模式动作执行过程中,执行状态结果的回调处理,如启动预览后状态变化、拍照结束状态、手动对焦结束、人脸检测结果返回等。

步骤5 模式释放

/* 应用切后台,需要将模式释放 */
@Override
protected void onPause() {
if (mBackgroundHandler != null) {
mBackgroundHandler.post(new Runnable() {
@Override
public void run() {
if (mMode != null) {
mMode.release();
mMode = null;
}
}
});
}
super.onPause();
}

恭喜你

  • HwCameraKit IDE插件的安装和使用
  • 利用HwCameraKit将华为相机开放能力接入app

更多HwCameraKit相关信息请关注我们的官网:

https://developer.huawei.com/consumer/cn/CameraKit


往期链接:Android | 教你如何快速集成机器学习能力

内容来源:https://developer.huawei.com/consumer/cn/forum/topicview?tid=0201234569803940158&fid=3467

原作者:艾罗门特

Android | 教你如何使用HwCameraKit接入相机人像模式的更多相关文章

  1. Android | 教你如何用华为HMS MLKit 图像分割 SDK开发一个证件照DIY小程序

    Android | 教你如何用华为HMS MLKit 图像分割 SDK开发一个证件照DIY小程序 引子   上期给大家介绍了如何使用如何用华为HMS MLKit SDK 三十分钟在安卓上开发一个微笑抓 ...

  2. Android | 教你如何用代码一键实现银行卡绑定

    前言   小编前面几期文章分别给大家介绍了用代码实现微笑抓拍.证件照DIY.拍照翻译的功能开发(链接见文章末尾),本次小编给大家带来的是用代码一键实现银行卡识别与绑定功能. 银行卡识别的应用场景    ...

  3. Android | 教你如何用代码开发一个拍照翻译小程序

    引子   想必有很多小伙伴喜欢外出旅游,能去海外玩一圈那是更好不过了,旅游前大家一定会对吃.穿.住.行.游玩路线做各种攻略,然后满怀期待的出发- 想象中的旅游   出发前,想象中的旅游目的地可能有漂亮 ...

  4. Android | 教你如何开发扫二维码功能

    前言   最近要做一个停车场扫码收费的app,在网上搜了一圈,首先接触到了ZXing,上手试了下,集成过程不复杂,但是感觉效果欠佳,比如距离稍微远点儿就扫不出来了,另外角度对的不好,反光或者光线比较暗 ...

  5. Android | 教你如何在安卓上实现通用卡证识别,一键各种卡绑定

    目录 前言 通用卡证识别的应用场景 如何使用通用卡证识别服务 集成通用卡证识别服务的关键流程 开发实战 1 开发准备 1.1 在项目级gradle里添加华为maven仓 1.2 在应用级的build. ...

  6. Android 教你打造炫酷的ViewPagerIndicator 不仅仅是高仿MIUI

    1.概述 哈,今天给大家带来一个ViewPagerIndicator的制作,相信大家在做tabIndicator的时候,大多数人都用过 TabPageIndicator,并且很多知名APP都使用过这个 ...

  7. Android | 教你如何快速集成机器学习能力

    背景   继上篇博文说了如何快速集成扫码以后 我又上官网去了解了一下其他的功能,其中机器学习服务是当下比较火的,而且还是免费的.就赶紧点进去学习一下.看看能够快速实现哪些功能. 链接在这里:https ...

  8. Android 支付宝以及微信支付快速接入流程

    简介 随着移动支付的普及,越来越多的App采用第三发支付,在这里我们以支付宝为例,做一个快速集成! 一.Android快速实现支付宝支付 1.首先,我们需要前往支付宝开放平台,申请我们的支付功能:ht ...

  9. Android 学习笔记之如何实现简单相机功能

    PS:看来算法和数据结构还是非常有用的,以后每天都练习两道算法题目...这次忘了对代码进行折叠了..导致篇幅过长... 学习内容: 1.Android如何实现相机功能... 2.如何实现音频的录制.. ...

随机推荐

  1. Azure SDK for Python Url

    Azure SDK for Python URL 明细表 AZURE_PUBLIC_CLOUD = Cloud( 'AzureCloud', endpoints=CloudEndpoints( man ...

  2. PTA数据结构与算法题目集(中文) 7-36 社交网络图中结点的“重要性”计算 (30 分)

    PTA数据结构与算法题目集(中文)  7-36 社交网络图中结点的“重要性”计算 (30 分) 7-36 社交网络图中结点的“重要性”计算 (30 分)   在社交网络中,个人或单位(结点)之间通过某 ...

  3. Spring (五):AOP

    本文是按照狂神说的教学视频学习的笔记,强力推荐,教学深入浅出一遍就懂!b站搜索狂神说或点击下面链接 https://space.bilibili.com/95256449?spm_id_from=33 ...

  4. ubuntu上安装redis和配置远程访问

    ubuntu上安装redis和配置远程访问 安装redis: 下载安装包: wget http://download.redis.io/releases/redis-4.0.1.tar.gz 解压: ...

  5. javascript 入门 之 bootstrap 第一个程序

    <table data-toggle="table"> <thead> <tr> <th>Item ID</th> &l ...

  6. hadoop(七)集群配置同步(hadoop完全分布式四)|9

    前置配置:rsync远程同步|xsync集群分发(hadoop完全分布式准备三)|9 1. 分布式集群分配原则 部署分配原则 说明Namenode和secondarynamenode占用内存较大,建议 ...

  7. buuctf misc wp 01

    buuctf misc wp 01 1.金三胖 2.二维码 3.N种方法解决 4.大白 5.基础破解 6.你竟然赶我走 1.金三胖 root@kali:~/下载/CTF题目# unzip 77edf3 ...

  8. leetcode c++做题思路和题解(1)——常规题总结

    常规题总结 0. 目录 两数之和 1. 两数之和 耗时4ms(98.82%),内存6.2m. 两数之和--寻找中值向两边扩散法 1.1 思路 思路很简单,就是先找数组中target/2的前后两个值,然 ...

  9. 3分钟掌握Quartz.net分布式定时任务的姿势

    引言 长话短说,今天聊一聊分布式定时任务,我的流水账笔记: ASP.NET Core+Quartz.Net实现web定时任务 AspNetCore结合Redis实践消息队列 细心朋友稍一分析,就知道还 ...

  10. vscode连接云服务,搭建Python远程开发

    配置Python远程开发环境前提 配置步骤 1.windows 10 开发机配置 win10 1809后支持ssh ssh-keygen -t rsa -b 4096 #会显示生成到的目录C:\Use ...