HarmonyOS 6 轻相机应用开发2:贴纸效果实现
引言
在上一篇《功能介绍与框架搭建》中,我们确立了"预览+推理"的双路流架构。如果说"实时滤镜"是相机的外衣,那么"实时贴纸"就是它的灵魂。
贴纸效果本质上是在相机预览流之上,叠加了一层动态的图像信息。在 HarmonyOS 6 中,为了保证 60fps 的极致流畅度,我们将舍弃开销较大的 ArkTS Canvas 绘制方案,转而深入 Native NDK 层,利用 Native Drawing (OH_Drawing) 与 OpenGL ES 纹理混合技术,构建一条全链路的 C++ 渲染流水线。
贴纸实现基本原理
贴纸的实现流程可以概括为以下三个核心步骤:
- 资源加载与解码 :从应用的
rawfile目录读取 PNG/JPG 图像并解码为像素数据。 - 离屏绘制 :利用
Native Drawing接口在内存中创建一个透明位图,根据 AI 检测到的坐标(如面部描点)将贴纸素材绘制上去。 - 纹理融合 :将位图上传为 OpenGL 2D 纹理,在 Fragment Shader 中与相机的 OES 纹理进行 Alpha 混合(Alpha Blending)。
NDK 实战:从 rawfile 读取并解析 PNG
在 Native 层直接处理图片资源,可以极大减少内存拷贝。我们需要用到 ResourceManager 来访问资源,并配合 ImageSource NDK 接口进行解码。
以下是实现这一链路的关键 C++ 代码(需链接 libimage_source.so 与 librawfile.z.so):
cpp
#include <multimedia/image_framework/image_source_native.h>
#include <rawfile/raw_file_manager.h>
// 从 rawfile 中加载贴纸位图
OH_PixelmapNative* LoadStickerFromRawFile(NativeResourceManager* resMgr, const char* fileName) {
// 1. 打开 RawFile 资源
RawFile* rawFile = OH_ResourceManager_OpenRawFile(resMgr, fileName);
if (rawFile == nullptr) return nullptr;
// 2. 创建 ImageSourceNative
long rawFileSize = OH_ResourceManager_GetRawFileSize(rawFile);
OH_ImageSourceNative* imageSource = nullptr;
// 注意:需先获取 fd 或者直接从内存创建,这里推荐使用 CreateFromRawFile
Image_ErrorCode ret = OH_ImageSourceNative_CreateFromRawFile(resMgr, (char*)fileName, strlen(fileName), &imageSource);
if (ret != IMAGE_SUCCESS) {
OH_ResourceManager_CloseRawFile(rawFile);
return nullptr;
}
// 3. 解码为 Pixelmap
OH_DecodingOptions* opts = nullptr; // 可设置采样率等选项
OH_PixelmapNative* pixelmap = nullptr;
OH_ImageSourceNative_CreatePixelmap(imageSource, opts, &pixelmap);
// 4. 清理资源
OH_ImageSourceNative_Release(imageSource);
OH_ResourceManager_CloseRawFile(rawFile);
return pixelmap;
}
解码后的 OH_PixelmapNative 对象随后可被绑定到 OH_Drawing_Bitmap 中,作为绘制源。
Native Drawing (OH_Drawing) 绘制逻辑
Native Drawing 是 HarmonyOS 提供的一套高性能 2D 绘图接口。在我们的方案中,它承担了"离屏画布"的角色:
- 创建画布 :利用
OH_Drawing_BitmapCreate创建一个与预览分辨率匹配(或比例一致)的透明位图。 - 绑定 Canvas :使用
OH_Drawing_CanvasBind将画布与位图关联。 - 执行绘制 :根据 AI 推理回传的坐标,调用
OH_Drawing_CanvasDrawPixelMapRect将贴纸素材绘制到指定位置。
由于这一过程全部在 C++ 层内存中完成,不涉及跨语言的大数据拷贝,因此即使是复杂的动态贴纸,也能保持极高的帧率。
OpenGL 纹理混合与 ALPHA_BLEND 数学公式
这是本章最核心的理论部分。相机采集的是 OES 纹理(通常是不透明的 YUV 转 RGB 流),而贴纸是带 Alpha 通道的 RGBA 纹理。我们需要在 Fragment Shader 中将二者进行融合。
1. ALPHA_BLEND 数学模型
在图形学中,最常用的混合算法是"源覆盖(Source Over)"模型。其数学表达式如下:
Cout=Csrc×Asrc+Cdst×(1−Asrc)C_{out} = C_{src} \times A_{src} + C_{dst} \times (1 - A_{src})Cout=Csrc×Asrc+Cdst×(1−Asrc)
其中:
- CsrcC_{src}Csrc:贴纸(源)的颜色值。
- AsrcA_{src}Asrc:贴纸的透明度(0.0 到 1.0)。
- CdstC_{dst}Cdst:预览画面(目标/背景)的颜色值。
- CoutC_{out}Cout:最终混合后的颜色输出。
物理意义 :当贴纸某一点全透明(Asrc=0A_{src}=0Asrc=0)时,输出结果完全由背景决定;当其全不透明(Asrc=1A_{src}=1Asrc=1)时,则完全显示贴纸。
2. Shader 混合代码实现
在 render_thread.cpp 的片段着色器中,我们可以直接落实这一公式:
glsl
// 片段着色器片段
#extension GL_OES_EGL_image_external : require
precision highp float;
varying vec2 vTexCoord;
uniform samplerExternalOES cameraTexture; // 相机 OES 纹理
uniform sampler2D stickerTexture; // 贴纸 RGBA 纹理
void main() {
vec4 cameraColor = texture2D(cameraTexture, vTexCoord);
vec4 stickerColor = texture2D(stickerTexture, vTexCoord);
// 应用 Alpha 混合公式
float r = stickerColor.r * stickerColor.a + cameraColor.r * (1.0 - stickerColor.a);
float g = stickerColor.g * stickerColor.a + cameraColor.g * (1.0 - stickerColor.a);
float b = stickerColor.b * stickerColor.a + cameraColor.b * (1.0 - stickerColor.a);
gl_FragColor = vec4(r, g, b, 1.0);
}
渲染流水线深度剖析
为了实现"语义化"的贴纸效果,我们将全链路串联如下:
- AI 检测阶段:第一路 AI 流探测到面部 106 点坐标。
- 坐标转换:将模型输出的归一化坐标转换至视图坐标系。
- 离屏绘制 (Off-screen) :
OH_Drawing在后台 Bitmap 上开启"透明涂层",绘制贴纸。 - 纹理上传 :通过
glTexImage2D将绘图后的内存数据上传至 GPU 纹理。 - 并轨渲染 :OpenGL 执行双采样 Shader,最后将结果刷新至
XComponent的 Surface 屏幕上。
总结
通过本章的实践,我们不仅掌握了 NDK 层级图片处理的硬核技能,还理解了底层图形渲染的数学之美。贴纸不再是简单的 UI 堆栈,而是深度融合进渲染管线的图像算子。在下一篇中,我们将探讨**"智慧物品检测"**,教你如何让相机不仅能"看面子",还能"认里子"。