# 使用ImageEffect编辑图片 ## åœºæ™¯ä»‹ç» ImageEffectæä¾›äº†ä¸€ç³»åˆ—接å£ç”¨äºŽå›¾åƒçš„编辑。开å‘者å¯ä»¥é€šè¿‡`ImageEffect`接å£å¤„ç†ä¸åŒå›¾åƒè¾“入类型`Pixelmap`ã€`NativeWindow`ã€`NativeBuffer`或`Uri`ï¼ŒèŽ·å¾—æ»¤é•œå¤„ç†æ•ˆæžœã€‚ 针对ImageEffect,常è§çš„å¼€å‘场景如下: - 通过`ImageEffect`æä¾›çš„Native APIæŽ¥å£æ·»åŠ æ»¤é•œæˆ–æ»¤é•œé“¾ï¼Œè®¾ç½®è¾“å…¥å›¾åƒï¼Œæœ€ç»ˆç”Ÿæ•ˆæ»¤é•œæ•ˆæžœã€‚ - 通过注册自定义滤镜,实现开å‘者的定制化滤镜效果。 - 通过`EffectFilter`æä¾›çš„Native API接å£å¿«é€Ÿå®žçްå•ä¸ªæ»¤é•œçš„å¤„ç†æ•ˆæžœã€‚ ## 接å£è¯´æ˜Ž 详细的接å£è¯´æ˜Žè¯·å‚考[ImageEffect](../../reference/apis-image-kit/_image_effect.md)。 ## 开呿¥éª¤ **æ·»åŠ åŠ¨æ€é“¾æŽ¥åº“** CMakeLists.txt䏿·»åР以䏋lib。 ```txt target_link_libraries(entry PUBLIC libace_ndk.z.so libimage_effect.so libpixelmap.so libnative_window.so libnative_buffer.so ) ``` æ ¹æ®å¤„ç†çš„图åƒç±»åž‹æ·»åŠ å¯¹åº”åŠ¨æ€é“¾æŽ¥åº“:`Pixelmap`(libpixelmap.so)ã€`NativeWindow`(libnative_window.so)ã€`NativeBuffer`(libnative_buffer.so) **æ·»åŠ å¤´æ–‡ä»¶** ```c++ #include <multimedia/image_effect/image_effect.h> #include <multimedia/image_effect/image_effect_filter.h> #include <multimedia/image_effect/image_effect_errors.h> ``` ### 通过ImageEffectæä¾›çš„æŽ¥å£ç”Ÿæ•ˆå›¾åƒæ•ˆæžœ 1. 创建ImageEffect实例。 ```c++ // 创建imageEffect实例,“ImageEditâ€æ˜¯imageEffect实例别å。 OH_ImageEffect *imageEffect = OH_ImageEffect_Create("ImageEdit"); ``` 2. æ·»åŠ EffectFilter滤镜。 ```c++ // æ·»åŠ æ»¤é•œï¼ŒèŽ·å– OH_EffectFilter 实例。多次调用该接å£å¯ä»¥æ·»åŠ å¤šä¸ªæ»¤é•œï¼Œç»„æˆæ»¤é•œé“¾ã€‚ OH_EffectFilter *filter = OH_ImageEffect_AddFilter(imageEffect, OH_EFFECT_BRIGHTNESS_FILTER); CHECK_AND_RETURN_LOG(filter != nullptr, "OH_ImageEffect_AddFilter fail!"); // è®¾ç½®æ»¤é•œå‚æ•°, 例如:滤镜强度设置为50。 ImageEffect_Any value = { .dataType = ImageEffect_DataType::EFFECT_DATA_TYPE_FLOAT, .dataValue.floatValue = 50.f }; ImageEffect_ErrorCode errorCode = OH_EffectFilter_SetValue(filter, OH_EFFECT_FILTER_INTENSITY_KEY, &value); ``` 3. è®¾ç½®å¤„ç†æ•°æ®ã€‚ **场景一:设置 OH_PixelmapNative 输入类型。** OH_PixelmapNative的具体使用方法请å‚考[Pixelmap开呿Œ‡å¯¼](image-pixelmap-operation-native.md)。 ```c++ // 设置输入的Pixelmap。 errorCode = OH_ImageEffect_SetInputPixelmap(imageEffect, inputPixelmap); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetInputPixelmap fail!"); // 设置输出的Pixelmap(å¯é€‰ï¼‰ï¼Œä¸è°ƒç”¨è¯¥æŽ¥å£æ—¶ä¼šåœ¨è¾“å…¥Pixelmap上直接生效滤镜效果。 errorCode = OH_ImageEffect_SetOutputPixelmap(imageEffect, outputPixelmap); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetOutputPixelmap fail!"); ``` **场景二:设置 OH_NativeBuffer 输入类型。** OH_NativeBuffer的具体使用方法请å‚考[NativeBuffer开呿Œ‡å¯¼](../../graphics/native-buffer-guidelines.md)。 ```c++ // 设置输入的NativeBuffer。 errorCode = OH_ImageEffect_SetInputNativeBuffer(imageEffect, inputNativeBuffer); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetInputNativeBuffer fail!"); // 设置输出的NativeBuffer(å¯é€‰ï¼‰ï¼Œä¸è°ƒç”¨è¯¥æŽ¥å£æ—¶ä¼šåœ¨è¾“å…¥NativeBuffer上直接生效滤镜效果。 errorCode = OH_ImageEffect_SetOutputNativeBuffer(imageEffect, outputNativeBuffer); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetOutputNativeBuffer fail!"); ``` **场景三:设置 URI 输入类型。** ```c++ // 设置输入的URI。 errorCode = OH_ImageEffect_SetInputUri(imageEffect, inputUri); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetInputUri fail!"); // 设置输出的URI(å¯é€‰ï¼‰ï¼Œä¸è°ƒç”¨è¯¥æŽ¥å£æ—¶ä¼šåœ¨è¾“å…¥URI上直接生效滤镜效果。 errorCode = OH_ImageEffect_SetOutputUri(imageEffect, outputUri); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetOutputUri fail!"); ``` **场景四:设置 OHNativeWindow 输入类型。** 以相机预览场景为例æ¥è¯´æ˜ŽOHNativeWindow输入场景。XComponentç»„ä»¶ä¸ºç›¸æœºé¢„è§ˆæµæä¾›çš„SurfaceId,å¯åœ¨native c++层将SurfaceIdè½¬æ¢æˆOHNativeWindowï¼Œä¸‹é¢æä¾›ä¸€ä»½ä»£ç 示例。 XComponent模å—的具体使用方法请å‚考[XComponent组件å‚考](../../reference/apis-arkui/arkui-ts/ts-basic-components-xcomponent.md)。 NativeWindow模å—的具体使用方法请å‚考[OHNativeWindow](../../reference/apis-arkgraphics2d/_native_window.md)。 Camera的具体使用方法请å‚考[Camera预览å‚考](../camera/native-camera-preview.md)。 (1) 在xxx.ets䏿·»åŠ ä¸€ä¸ªXComponent组件。 ```ts XComponent({ id: 'xcomponentId', type: 'surface', controller: this.mXComponentController, libraryname: 'entry' }) .onLoad(() => { // 获å–XComponentçš„SurfaceId。 this.mSurfaceId = this.mXComponentController.getXComponentSurfaceId() // 调用native接å£èŽ·å–输入SurfaceId。 this.mSurfaceId = imageEffect.getSurfaceId(this.mSurfaceId) // 调用相机接å£å¯åŠ¨é¢„è§ˆï¼Œå°†èŽ·å–到的输入SurfaceIdä¼ é€’ç»™ç›¸æœºæ¡†æž¶ // ... }) .width('100%') .height('100%') ``` (2) imageEffect.getSurfaceIdçš„native c++层具体实现。 ```c++ // æ ¹æ®SurfaceId创建NativeWindow,注æ„创建出æ¥çš„NativeWindow在使用结æŸåŽéœ€è¦ä¸»åŠ¨è°ƒç”¨OH_NativeWindow_DestoryNativeWindow进行释放。 uint64_t outputSurfaceId; std::istrstream iss(outputSurfaceIdStr); issue >> outputSurfaceId; OHNativeWindow *outputNativeWindow = nullptr; int32_t res = OH_NativeWindow_CreateNativeWindowFromSurfaceId(outputSurfaceId, &outputNativeWindow); CHECK_AND_RETURN_LOG(res == 0, "OH_NativeWindow_CreateNativeWindowFromSurfaceId fail!"); // 设置输出显示的Surface。 ImageEffect_ErrorCode errorCode = OH_ImageEffect_SetOutputSurface(imageEffect, outputNativeWindow); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_SetOutputSurface fail!"); // 获å–输入的Surface。注æ„获å–çš„inputNativeWindow在使用结æŸåŽéœ€è¦ä¸»åŠ¨è°ƒç”¨OH_NativeWindow_DestoryNativeWindow进行释放。 OHNativeWindow *inputNativeWindow = nullptr; errorCode = OH_ImageEffect_GetInputSurface(imageEffect, &inputNativeWindow); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_GetInputSurface fail!"); // 从获å–到输入的NativeWindowä¸èŽ·å–SurfaceId。 uint64_t inputSurfaceId = 0; res = OH_NativeWindow_GetSurfaceId(inputNativeWindow, &inputSurfaceId); CHECK_AND_RETURN_LOG(res == 0, "OH_NativeWindow_GetSurfaceId fail!"); // å°†SurfaceId转æˆå—符串进行返回。 std::string inputSurfaceIdStr = std::to_string(inputSurfaceId); ``` 4. å¯åŠ¨æ•ˆæžœå™¨ã€‚ ```c++ // 执行生效滤镜效果。 errorCode = OH_ImageEffect_Start(imageEffect); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_Start fail!"); ``` 5. åœæ¢ç”Ÿæ•ˆæ•ˆæžœï¼ˆå¯é€‰ï¼Œä»…在输入Surfaceåœºæ™¯ä¸‹æ‰æœ‰æ•ˆï¼‰ã€‚ ```c++ // åœæ¢ç”Ÿæ•ˆæ»¤é•œæ•ˆæžœã€‚ errorCode = OH_ImageEffect_Stop(imageEffect); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_Stop fail!"); ``` 6. åºåˆ—化效果器(å¯é€‰ï¼‰ã€‚ ```c++ char *info = nullptr; errorCode = OH_ImageEffect_Save(imageEffect, &info); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_ImageEffect_Save fail!"); ``` 7. é”€æ¯æ•ˆæžœå™¨å®žä¾‹ã€‚ ```c++ // 释放imageEffect实例资æºã€‚ errorCode = OH_ImageEffect_Release(imageEffect); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, result, "OH_ImageEffect_Release fail!"); ``` ### 自定义滤镜 以下æ¥éª¤æè¿°äº†å¦‚何实现并注册自定义滤镜接å£ï¼š 1. 定义 ImageEffect_FilterDelegate。 ```c++ // 图åƒä¿¡æ¯ç»“构体。 struct EffectBufferInfo { void *addr = nullptr; int32_t width = 0; int32_t height = 0; int32_t rowSize = 0; ImageEffect_Format format = ImageEffect_Format::EFFECT_PIXEL_FORMAT_UNKNOWN; }; // 自定义滤镜具体实现。 ImageEffect_FilterDelegate filterDelegate = { .setValue = [](OH_EffectFilter *filter, const char *key, const ImageEffect_Any *value) { // 傿•°æ ¡éªŒï¼Œæ ¡éªŒæˆåŠŸæ—¶è¿”å›žtrue,å¦åˆ™è¿”回false。 // ... return true; }, .render = [](OH_EffectFilter *filter, OH_EffectBufferInfo *info, OH_EffectFilterDelegate_PushData pushData) { return Render(filter, info, pushData); }, .save = [](OH_EffectFilter *filter, char **info) { // 获å–è‡ªå®šä¹‰æ‰€è®¾ç½®çš„æ»¤é•œå‚æ•°ï¼Œå…¶ä¸"Brightness"为自定义滤镜的Key,由开å‘è€…è‡ªå·±ä»»æ„æŒ‡å®šã€‚ ImageEffect_Any value; ImageEffect_ErrorCode errorCode = OH_EffectFilter_GetValue(filter, "Brightness", &value); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, false, "OH_EffectFilter_GetValue fail!"); // 生æˆé”®å€¼å¯¹ä¿¡æ¯ã€‚ json values; values["Brightness"] = value.dataValue.floatValue; json root; root["name"] = "CustomBrightness"; root["values"] = values;    // å°†json对象转æˆå—符串infoStr // ... // 对*info赋值åºåˆ—化å—符串地å€ã€‚ *info = infoStr; return true; }, .restore = [](const char *info) { // 创建 OH_EffectFilter 实例,其ä¸"CustomBrightness"为自定义滤镜的滤镜å。 OH_EffectFilter *filter = OH_EffectFilter_Create("CustomBrightness"); // è§£æžjsonå—符串info获å–keyå’Œvalue。 // ... // è®¾ç½®æ»¤é•œå‚æ•°, value为info䏿Œ‰jsonè§£æžå‡ºæ¥çš„傿•°ã€‚ ImageEffect_ErrorCode errorCode = OH_EffectFilter_SetValue(filter, "Brightness", &value); // ... return filter; } }; ``` å…¶ä¸Render接å£çš„实现分两ç§åœºæ™¯ã€‚ **场景一:自定义算法å¯ä»¥ç›´æŽ¥ä¿®æ”¹infoä¸çš„åƒç´ æ•°æ®ï¼ˆæ¯”如:亮度调节滤镜)。** ```c++ bool Render(OH_EffectFilter *filter, OH_EffectBufferInfo *info, OH_EffectFilterDelegate_PushData pushData) { // 获å–图åƒä¿¡æ¯å…·ä½“傿•°ã€‚ EffectBufferInfo inputBufferInfo; OH_EffectBufferInfo_GetAddr(info, &inputBufferInfo.addr); OH_EffectBufferInfo_GetWidth(info, &inputBufferInfo.width); OH_EffectBufferInfo_GetHeight(info, &inputBufferInfo.height); OH_EffectBufferInfo_GetRowSize(info, &inputBufferInfo.rowSize); OH_EffectBufferInfo_GetEffectFormat(info, &inputBufferInfo.format); // 调用自定义滤镜算法。 ApplyCustomAlgo(inputBufferInfo); // 编辑完æˆåŽè°ƒç”¨pushDataç›´æŽ¥ä¼ é€’åŽŸå›¾ã€‚ pushData(filter, info); return true; } ``` **场景二:自定义算法ä¸èƒ½ç›´æŽ¥ä¿®æ”¹infoä¸çš„åƒç´ æ•°æ®ï¼ˆæ¯”如:è£å‰ªæ»¤é•œï¼‰ã€‚** ```c++ bool Render(OH_EffectFilter *filter, OH_EffectBufferInfo *info, OH_EffectFilterDelegate_PushData pushData) { // 获å–图åƒä¿¡æ¯å…·ä½“傿•°ã€‚ EffectBufferInfo inputBufferInfo; OH_EffectBufferInfo_GetAddr(info, &inputBufferInfo.addr); OH_EffectBufferInfo_GetWidth(info, &inputBufferInfo.width); OH_EffectBufferInfo_GetHeight(info, &inputBufferInfo.height); OH_EffectBufferInfo_GetRowSize(info, &inputBufferInfo.rowSize); OH_EffectBufferInfo_GetEffectFormat(info, &inputBufferInfo.format); // 创建输出åƒç´ ä¿¡æ¯ã€‚ EffectBufferInfo outputBufferInfo = CreateOutputBufferInfo(inputBufferInfo); // 调用自定义滤镜算法。 ApplyCustomAlgo(inputBufferInfo, outputBufferInfo); // 生æˆoutputOhInfo。 OH_EffectBufferInfo *outputOhInfo = OH_EffectBufferInfo_Create(); OH_EffectBufferInfo_SetAddr(outputOhInfo, outputBufferInfo.addr); OH_EffectBufferInfo_SetWidth(outputOhInfo, outputBufferInfo.width); OH_EffectBufferInfo_SetHeight(outputOhInfo, outputBufferInfo.height); OH_EffectBufferInfo_SetRowSize(outputOhInfo, outputBufferInfo.rowSize); OH_EffectBufferInfo_SetEffectFormat(outputOhInfo, outputBufferInfo.format); // 编辑完æˆåŽè°ƒç”¨pushDataä¼ é€’outputOhInfo。 pushData(filter, outputOhInfo); // 释放资æºã€‚ OH_EffectBufferInfo_Release(outputOhInfo); ReleaseOutputBuffer(outputBufferInfo.addr); return true; } ``` 2. 生æˆè‡ªå®šä¹‰æ»¤é•œä¿¡æ¯ã€‚ ```c++ // 创建 OH_EffectFilterInfo 实例。 OH_EffectFilterInfo *customFilterInfo = OH_EffectFilterInfo_Create(); CHECK_AND_RETURN_LOG(customFilterInfo != nullptr, "OH_EffectFilter_GetValue fail!"); // 设置自定义滤镜滤镜å。 OH_EffectFilterInfo_SetFilterName(customFilterInfo, "CustomBrightness"); // 设置自定义滤镜所支æŒçš„内å˜ç±»åž‹ã€‚ ImageEffect_BufferType bufferTypeArray[] = { ImageEffect_BufferType::EFFECT_BUFFER_TYPE_PIXEL }; OH_EffectFilterInfo_SetSupportedBufferTypes(customFilterInfo, sizeof(bufferTypeArray) / sizeof(ImageEffect_BufferType), bufferTypeArray); // 设置自定义滤镜所支æŒçš„åƒç´ æ ¼å¼ã€‚ ImageEffect_Format formatArray[] = { ImageEffect_Format::EFFECT_PIXEL_FORMAT_RGBA8888 }; OH_EffectFilterInfo_SetSupportedFormats(customFilterInfo, sizeof(formatArray) / sizeof(ImageEffect_Format), formatArray); ``` 3. å°† ImageEffect_FilterDelegate 注册到效果器。 ```c++ // 注册自定义滤镜。 ImageEffect_ErrorCode errorCode = OH_EffectFilter_Register(customFilterInfo, &filterDelegate); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_EffectFilter_Register fail!"); ``` ### EffectFilter快速实现å•ä¸ªæ»¤é•œçš„å¤„ç†æ•ˆæžœ 1. 创建滤镜。 ```c++ // 创建滤镜。比如:创建对比度效果器。 OH_EffectFilter *filter = OH_EffectFilter_Create(OH_EFFECT_CONTRAST_FILTER); ``` 2. è®¾ç½®æ»¤é•œå‚æ•°ã€‚ ```c++ // è®¾ç½®æ»¤é•œå‚æ•°, 滤镜强度设置为50。 ImageEffect_Any value = {.dataType = ImageEffect_DataType::EFFECT_DATA_TYPE_FLOAT, .dataValue.floatValue = 50.f}; ImageEffect_ErrorCode errorCode = OH_EffectFilter_SetValue(filter, OH_EFFECT_FILTER_INTENSITY_KEY, &value); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_EffectFilter_SetValue fail!"); ``` 3. 生效滤镜。 ```c++ // 生效滤镜效果。 errorCode = OH_EffectFilter_Render(filter, inputPixelmap, outputPixelmap); ``` 4. é”€æ¯æ»¤é•œå®žä¾‹ã€‚ ```c++ // é”€æ¯æ»¤é•œå®žä¾‹ã€‚ errorCode = OH_EffectFilter_Release(filter); ``` ### 查询能力 - æ ¹æ®æ»¤é•œå查询滤镜信æ¯ã€‚ ```c++ // 创建 OH_EffectFilterInfo 实例。 OH_EffectFilterInfo *filterInfo = OH_EffectFilterInfo_Create(); CHECK_AND_RETURN_LOG(filterInfo != nullptr, "OH_EffectFilterInfo_Create fail!"); // æ ¹æ®æ»¤é•œå查询滤镜能力信æ¯ã€‚ ImageEffect_ErrorCode errorCode = OH_EffectFilter_LookupFilterInfo(OH_EFFECT_BRIGHTNESS_FILTER, filterInfo); CHECK_AND_RETURN_LOG(errorCode == ImageEffect_ErrorCode::EFFECT_SUCCESS, "OH_EffectFilter_LookupFilterInfo fail!"); // 从滤镜能力信æ¯ä¸èŽ·å–æ»¤é•œå。 char *name = nullptr; OH_EffectFilterInfo_GetFilterName(filterInfo, &name); // èŽ·å–æ”¯æŒçš„内å˜ç±»åž‹ã€‚ uint32_t supportedBufferTypesCnt = 0; ImageEffect_BufferType *bufferTypeArray = nullptr; OH_EffectFilterInfo_GetSupportedBufferTypes(filterInfo, &supportedBufferTypesCnt, &bufferTypeArray); // èŽ·å–æ”¯æŒçš„åƒç´ 类型信æ¯ã€‚ uint32_t supportedFormatsCnt = 0; ImageEffect_Format *formatArray = nullptr; OH_EffectFilterInfo_GetSupportedFormats(filterInfo, supportedFormatsCnt, &formatArray); // é”€æ¯ OH_EffectFilterInfo 实例。 OH_EffectFilterInfo_Release(filterInfo); ``` - æ ¹æ®æ¡ä»¶æŸ¥è¯¢æ»¡è¶³æ¡ä»¶çš„æ»¤é•œã€‚ ```c++ // 查询所有的Filter,需è¦ä¸»åŠ¨è¿›è¡Œèµ„æºé‡Šæ”¾ã€‚ ImageEffect_FilterNames *filterNames = OH_EffectFilter_LookupFilters("Default"); // ... // 释放FilterNames虚拟内å˜èµ„æºã€‚ OH_EffectFilter_ReleaseFilterNames(); ``` ## 相关实例 针对ImageEffect的开å‘,有以下相关实例å¯ä¾›å‚考: - [ImageEffect(API12)](https://gitee.com/openharmony/applications_app_samples/tree/master/code/BasicFeature/Media/ImageEffect)