• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "bridge/declarative_frontend/jsview/canvas/js_render_image.h"
17 
18 #include "napi/native_api.h"
19 #include "napi/native_node_api.h"
20 
21 #include "base/memory/ace_type.h"
22 #include "base/utils/utils.h"
23 #include "bridge/declarative_frontend/jsview/canvas/js_rendering_context.h"
24 #include "bridge/declarative_frontend/jsview/js_view_common_def.h"
25 #include "core/common/container.h"
26 
27 #ifdef PIXEL_MAP_SUPPORTED
28 #include "pixel_map.h"
29 #include "pixel_map_napi.h"
30 #endif
31 
32 namespace OHOS::Ace::Framework {
33 
BindNativeFunction(napi_env env,napi_value object,const char * name,napi_callback func)34 void BindNativeFunction(napi_env env, napi_value object, const char* name, napi_callback func)
35 {
36     std::string funcName(name);
37     napi_value result = nullptr;
38     NAPI_CALL_RETURN_VOID(env, napi_create_function(env, funcName.c_str(), funcName.length(), func, nullptr, &result));
39     napi_set_named_property(env, object, name, result);
40 }
41 
GetNapiCallbackInfoAndThis(napi_env env,napi_callback_info info)42 void* GetNapiCallbackInfoAndThis(napi_env env, napi_callback_info info)
43 {
44     napi_value value = nullptr;
45     NAPI_CALL(env, napi_get_cb_info(env, info, nullptr, nullptr, &value, nullptr));
46     void* result = nullptr;
47     NAPI_CALL(env, napi_unwrap(env, value, &result));
48     return result;
49 }
50 
DetachImageBitmap(napi_env env,void * value,void * hint)51 void* DetachImageBitmap(napi_env env, void* value, void* hint)
52 {
53     return value;
54 }
55 
AttachImageBitmap(napi_env env,void * value,void *)56 napi_value AttachImageBitmap(napi_env env, void* value, void*)
57 {
58     if (value == nullptr) {
59         LOGW("Invalid parameter.");
60         return nullptr;
61     }
62     auto* wrapper = (JSRenderImage*)value;
63     if (wrapper == nullptr) {
64         LOGW("Invalid context.");
65         return nullptr;
66     }
67 
68     napi_value imageBitmap = nullptr;
69     napi_create_object(env, &imageBitmap);
70     napi_value isImageBitmap = nullptr;
71     napi_create_int32(env, 1, &isImageBitmap);
72     napi_property_descriptor desc[] = {
73         DECLARE_NAPI_GETTER_SETTER("width", JSRenderImage::JsGetWidth, JSRenderImage::JsSetWidth),
74         DECLARE_NAPI_GETTER_SETTER("height", JSRenderImage::JsGetHeight, JSRenderImage::JsSetHeight),
75         DECLARE_NAPI_FUNCTION("close", JSRenderImage::JsClose),
76         DECLARE_NAPI_PROPERTY("isImageBitmap", isImageBitmap),
77     };
78     napi_define_properties(env, imageBitmap, sizeof(desc) / sizeof(*desc), desc);
79 
80     napi_coerce_to_native_binding_object(env, imageBitmap, DetachImageBitmap, AttachImageBitmap, value, nullptr);
81     napi_wrap_with_size(env, imageBitmap, value, JSRenderImage::Finalizer, nullptr, nullptr, wrapper->GetBindingSize());
82     wrapper->IncRefCount();
83     return imageBitmap;
84 }
85 
JSRenderImage()86 JSRenderImage::JSRenderImage() {}
87 
Finalizer(napi_env env,void * data,void * hint)88 void JSRenderImage::Finalizer(napi_env env, void* data, void* hint)
89 {
90     auto wrapper = reinterpret_cast<JSRenderImage*>(data);
91     if (wrapper) {
92         wrapper->DecRefCount();
93     }
94 }
95 
Constructor(napi_env env,napi_callback_info info)96 napi_value JSRenderImage::Constructor(napi_env env, napi_callback_info info)
97 {
98     ContainerScope scope(Container::CurrentIdSafely());
99     size_t argc = 2; // The number of params is 2.
100     napi_value argv[2] = { nullptr }; // The number of params is 2.
101     napi_value thisVar = nullptr;
102     NAPI_CALL(env, napi_get_cb_info(env, info, &argc, argv, &thisVar, nullptr));
103     auto wrapper = AceType::MakeRefPtr<JSRenderImage>();
104     wrapper->SetInstanceId(OHOS::Ace::Container::CurrentId());
105     if (argc <= 0) {
106         napi_coerce_to_native_binding_object(
107             env, thisVar, DetachImageBitmap, AttachImageBitmap, AceType::RawPtr(wrapper), nullptr);
108         napi_wrap(env, thisVar, AceType::RawPtr(wrapper), Finalizer, nullptr, nullptr);
109         wrapper->IncRefCount();
110         return thisVar;
111     }
112     if (argc == 2) {  // 2: args count
113         int32_t unit = 0;
114         napi_get_value_int32(env, argv[1], &unit);
115         if (static_cast<CanvasUnit>(unit) == CanvasUnit::PX) {
116             wrapper->SetUnit(CanvasUnit::PX);
117         }
118     }
119     size_t textLen = 0;
120     napi_status status = napi_get_value_string_utf8(env, argv[0], nullptr, 0, &textLen);
121     if (status == napi_ok) {
122         auto context = PipelineBase::GetCurrentContext();
123         CHECK_NULL_RETURN(context, nullptr);
124         std::string textString = GetSrcString(env, argv[0], textLen);
125         if (context->IsFormRender() && NotFormSupport(textString)) {
126             LOGE("Not supported src : %{public}s when form render", textString.c_str());
127             return nullptr;
128         }
129         wrapper->LoadImage(textString);
130     } else {
131 #ifdef PIXEL_MAP_SUPPORTED
132         auto pixelMap = GetPixelMap(env, argv[0]);
133         CHECK_NULL_RETURN(pixelMap, nullptr);
134         wrapper->LoadImage(pixelMap);
135 #endif
136     }
137     napi_coerce_to_native_binding_object(
138         env, thisVar, DetachImageBitmap, AttachImageBitmap, AceType::RawPtr(wrapper), nullptr);
139     napi_wrap_with_size(env, thisVar, AceType::RawPtr(wrapper), Finalizer, nullptr, nullptr, wrapper->GetBindingSize());
140     wrapper->IncRefCount();
141     return thisVar;
142 }
143 
NotFormSupport(const std::string & textString)144 bool JSRenderImage::NotFormSupport(const std::string& textString)
145 {
146     SrcType srcType = ImageSourceInfo::ResolveURIType(textString);
147     return (srcType == SrcType::NETWORK || srcType == SrcType::FILE || srcType == SrcType::DATA_ABILITY);
148 }
149 
GetSrcString(napi_env env,napi_value value,size_t textLen)150 std::string JSRenderImage::GetSrcString(napi_env env, napi_value value, size_t textLen)
151 {
152     std::unique_ptr<char[]> text = std::make_unique<char[]>(textLen + 1);
153     auto status = napi_get_value_string_utf8(env, value, text.get(), textLen + 1, &textLen);
154     if ((status == napi_ok) && (text != nullptr)) {
155         return text.get();
156     }
157     return "";
158 }
159 
160 #ifdef PIXEL_MAP_SUPPORTED
GetPixelMap(napi_env env,napi_value value)161 RefPtr<PixelMap> JSRenderImage::GetPixelMap(napi_env env, napi_value value)
162 {
163     Media::PixelMapNapi* napiPixelMap = nullptr;
164     auto status = napi_unwrap(env, value, reinterpret_cast<void**>(&napiPixelMap));
165     if ((status != napi_ok) || (napiPixelMap == nullptr)) {
166         return nullptr;
167     }
168     return PixelMap::CreatePixelMap(napiPixelMap->GetPixelMap());
169 }
170 #endif
171 
InitImageBitmap(napi_env env)172 napi_value JSRenderImage::InitImageBitmap(napi_env env)
173 {
174     napi_value object = nullptr;
175     napi_create_object(env, &object);
176     napi_value isImageBitmap = nullptr;
177     napi_create_int32(env, 1, &isImageBitmap);
178 
179     napi_property_descriptor desc[] = {
180         DECLARE_NAPI_GETTER_SETTER("width", JsGetWidth, JsSetWidth),
181         DECLARE_NAPI_GETTER_SETTER("height", JsGetHeight, JsSetHeight),
182         DECLARE_NAPI_FUNCTION("close", JsClose),
183         DECLARE_NAPI_PROPERTY("isImageBitmap", isImageBitmap),
184     };
185     napi_status status = napi_define_class(
186         env, "ImageBitmap", NAPI_AUTO_LENGTH, Constructor, nullptr, sizeof(desc) / sizeof(*desc), desc, &object);
187     if (status != napi_ok) {
188         LOGW("Initialize image bitmap failed");
189         return nullptr;
190     }
191     return object;
192 }
193 
JSBind(BindingTarget globalObj,void * nativeEngine)194 void JSRenderImage::JSBind(BindingTarget globalObj, void* nativeEngine)
195 {
196     if (!nativeEngine) {
197         return;
198     }
199     napi_env env = reinterpret_cast<napi_env>(nativeEngine);
200 
201     napi_value jsGlobalObj = nullptr;
202     napi_get_global(env, &jsGlobalObj);
203 
204     napi_value result = InitImageBitmap(env);
205     napi_set_named_property(env, jsGlobalObj, "ImageBitmap", result);
206 }
207 
JsGetWidth(napi_env env,napi_callback_info info)208 napi_value JSRenderImage::JsGetWidth(napi_env env, napi_callback_info info)
209 {
210     JSRenderImage* me = static_cast<JSRenderImage*>(GetNapiCallbackInfoAndThis(env, info));
211     return (me != nullptr) ? me->OnGetWidth(env) : nullptr;
212 }
213 
JsGetHeight(napi_env env,napi_callback_info info)214 napi_value JSRenderImage::JsGetHeight(napi_env env, napi_callback_info info)
215 {
216     JSRenderImage* me = static_cast<JSRenderImage*>(GetNapiCallbackInfoAndThis(env, info));
217     return (me != nullptr) ? me->OnGetHeight(env) : nullptr;
218 }
219 
JsClose(napi_env env,napi_callback_info info)220 napi_value JSRenderImage::JsClose(napi_env env, napi_callback_info info)
221 {
222     JSRenderImage* me = static_cast<JSRenderImage*>(GetNapiCallbackInfoAndThis(env, info));
223     return (me != nullptr) ? me->OnClose() : nullptr;
224 }
225 
JsSetWidth(napi_env env,napi_callback_info info)226 napi_value JSRenderImage::JsSetWidth(napi_env env, napi_callback_info info)
227 {
228     JSRenderImage* me = static_cast<JSRenderImage*>(GetNapiCallbackInfoAndThis(env, info));
229     return (me != nullptr) ? me->OnSetWidth() : nullptr;
230 }
231 
JsSetHeight(napi_env env,napi_callback_info info)232 napi_value JSRenderImage::JsSetHeight(napi_env env, napi_callback_info info)
233 {
234     JSRenderImage* me = static_cast<JSRenderImage*>(GetNapiCallbackInfoAndThis(env, info));
235     return (me != nullptr) ? me->OnSetHeight() : nullptr;
236 }
237 
OnGetWidth(napi_env env)238 napi_value JSRenderImage::OnGetWidth(napi_env env)
239 {
240     double width = 0.0;
241     double density = GetDensity();
242     width = width_;
243     width /= density;
244     napi_value jsWidth = nullptr;
245     napi_create_double(env, width, &jsWidth);
246     return jsWidth;
247 }
248 
OnGetHeight(napi_env env)249 napi_value JSRenderImage::OnGetHeight(napi_env env)
250 {
251     double height = 0.0;
252     double density = GetDensity();
253     height = height_;
254     height /= density;
255     napi_value jsHeight = nullptr;
256     napi_create_double(env, height, &jsHeight);
257     return jsHeight;
258 }
259 
OnSetWidth()260 napi_value JSRenderImage::OnSetWidth()
261 {
262     return nullptr;
263 }
264 
OnSetHeight()265 napi_value JSRenderImage::OnSetHeight()
266 {
267     return nullptr;
268 }
269 
OnClose()270 napi_value JSRenderImage::OnClose()
271 {
272     for (const auto& closeCallback : closeCallbacks_) {
273         if (!closeCallback) {
274             continue;
275         }
276         closeCallback();
277     }
278     width_ = 0;
279     height_ = 0;
280     return nullptr;
281 }
282 
OnImageDataReady()283 void JSRenderImage::OnImageDataReady()
284 {
285     CHECK_NULL_VOID(loadingCtx_);
286     width_ = loadingCtx_->GetImageSize().Width();
287     height_ = loadingCtx_->GetImageSize().Height();
288     loadingCtx_->MakeCanvasImageIfNeed(loadingCtx_->GetImageSize(), true, ImageFit::NONE);
289 }
290 
OnImageLoadSuccess()291 void JSRenderImage::OnImageLoadSuccess()
292 {
293     CHECK_NULL_VOID(loadingCtx_);
294     image_ = loadingCtx_->MoveCanvasImage();
295     CHECK_NULL_VOID(image_);
296     imageObj_ = loadingCtx_->MoveImageObject();
297     CHECK_NULL_VOID(imageObj_);
298     pixelMap_ = image_->GetPixelMap();
299     svgDom_ = imageObj_->GetSVGDom();
300     imageFit_ = loadingCtx_->GetImageFit();
301     imageSize_ = loadingCtx_->GetImageSize();
302     bindingSize_ = pixelMap_ ? static_cast<size_t>(pixelMap_->GetByteCount()) : 0;
303 }
304 
OnImageLoadFail(const std::string & errorMsg)305 void JSRenderImage::OnImageLoadFail(const std::string& errorMsg)
306 {
307     width_ = 0;
308     height_ = 0;
309     pixelMap_ = nullptr;
310     svgDom_ = nullptr;
311 }
312 
LoadImage(const std::string & src)313 void JSRenderImage::LoadImage(const std::string& src)
314 {
315     src_ = src;
316     auto sourceInfo = ImageSourceInfo(src);
317     sourceInfo_ = sourceInfo;
318     LoadImage(sourceInfo);
319 }
320 
LoadImage(const RefPtr<PixelMap> & pixmap)321 void JSRenderImage::LoadImage(const RefPtr<PixelMap>& pixmap)
322 {
323     auto sourceInfo = ImageSourceInfo(pixmap);
324     sourceInfo_ = sourceInfo;
325     LoadImage(sourceInfo);
326 }
327 
LoadImage(const ImageSourceInfo & sourceInfo)328 void JSRenderImage::LoadImage(const ImageSourceInfo& sourceInfo)
329 {
330     auto dataReadyCallback = [weak = WeakClaim(this)](const ImageSourceInfo& sourceInfo) {
331         auto jsRenderImage = weak.Upgrade();
332         CHECK_NULL_VOID(jsRenderImage);
333         jsRenderImage->OnImageDataReady();
334     };
335     auto loadSuccessCallback = [weak = WeakClaim(this)](const ImageSourceInfo& sourceInfo) {
336         auto jsRenderImage = weak.Upgrade();
337         CHECK_NULL_VOID(jsRenderImage);
338         jsRenderImage->OnImageLoadSuccess();
339     };
340     auto loadFailCallback = [weak = WeakClaim(this)](const ImageSourceInfo& sourceInfo, const std::string& errorMsg,
341                                 ImageErrorInfo /* errorInfo */) {
342         auto jsRenderImage = weak.Upgrade();
343         CHECK_NULL_VOID(jsRenderImage);
344         jsRenderImage->OnImageLoadFail(errorMsg);
345     };
346     NG::LoadNotifier loadNotifier(dataReadyCallback, loadSuccessCallback, loadFailCallback);
347     loadingCtx_ = AceType::MakeRefPtr<NG::ImageLoadingContext>(sourceInfo, std::move(loadNotifier), true);
348     loadingCtx_->LoadImageData();
349 }
350 
GetSrc()351 std::string JSRenderImage::GetSrc()
352 {
353     return src_;
354 }
355 
GetWidth()356 double JSRenderImage::GetWidth()
357 {
358     return width_;
359 }
360 
SetWidth(double width)361 void JSRenderImage::SetWidth(double width)
362 {
363     width_ = width;
364 }
365 
GetHeight()366 double JSRenderImage::GetHeight()
367 {
368     return height_;
369 }
370 
SetHeight(double height)371 void JSRenderImage::SetHeight(double height)
372 {
373     height_ = height;
374 }
375 
SetCloseCallback(std::function<void ()> && callback)376 void JSRenderImage::SetCloseCallback(std::function<void()>&& callback)
377 {
378     closeCallbacks_.emplace_back(std::move(callback));
379 }
380 
CreateJSRenderImage(napi_env env,RefPtr<PixelMap> pixelMap,napi_value & renderImage)381 bool JSRenderImage::CreateJSRenderImage(napi_env env, RefPtr<PixelMap> pixelMap, napi_value& renderImage)
382 {
383     napi_value global = nullptr;
384     napi_status status = napi_get_global(env, &global);
385     if (status != napi_ok) {
386         return false;
387     }
388     napi_value constructor = nullptr;
389     status = napi_get_named_property(env, global, "ImageBitmap", &constructor);
390     if (status != napi_ok) {
391         return false;
392     }
393 #ifdef PIXEL_MAP_SUPPORTED
394     CHECK_NULL_RETURN(pixelMap, false);
395     auto pixelmapSharedPtr = pixelMap->GetPixelMapSharedPtr();
396     napi_value napiValue = OHOS::Media::PixelMapNapi::CreatePixelMap(env, pixelmapSharedPtr);
397     status = napi_new_instance(env, constructor, 1, &napiValue, &renderImage);
398 #else
399     status = napi_new_instance(env, constructor, 0, nullptr, &renderImage);
400 #endif
401     return status == napi_ok;
402 }
403 } // namespace OHOS::Ace::Framework