1 /*
2 * Copyright (c) 2022 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 "js_display.h"
17
18 #include <cinttypes>
19 #include <map>
20
21 #include "cutout_info.h"
22 #include "display.h"
23 #include "display_info.h"
24 #include "window_manager_hilog.h"
25
26 namespace OHOS {
27 namespace Rosen {
28 using namespace AbilityRuntime;
29 constexpr size_t ARGC_ONE = 1;
30 namespace {
31 constexpr HiviewDFX::HiLogLabel LABEL = {LOG_CORE, HILOG_DOMAIN_DISPLAY, "JsDisplay"};
32 const std::map<DisplayState, DisplayStateMode> NATIVE_TO_JS_DISPLAY_STATE_MAP {
33 { DisplayState::UNKNOWN, DisplayStateMode::STATE_UNKNOWN },
34 { DisplayState::OFF, DisplayStateMode::STATE_OFF },
35 { DisplayState::ON, DisplayStateMode::STATE_ON },
36 };
37 }
38
39 static thread_local std::map<DisplayId, std::shared_ptr<NativeReference>> g_JsDisplayMap;
40 std::recursive_mutex g_mutex;
41
JsDisplay(const sptr<Display> & display)42 JsDisplay::JsDisplay(const sptr<Display>& display) : display_(display)
43 {
44 }
45
~JsDisplay()46 JsDisplay::~JsDisplay()
47 {
48 WLOGFI("JsDisplay::~JsDisplay is called");
49 }
50
Finalizer(NativeEngine * engine,void * data,void * hint)51 void JsDisplay::Finalizer(NativeEngine* engine, void* data, void* hint)
52 {
53 WLOGFI("JsDisplay::Finalizer is called");
54 auto jsDisplay = std::unique_ptr<JsDisplay>(static_cast<JsDisplay*>(data));
55 if (jsDisplay == nullptr) {
56 WLOGFE("JsDisplay::Finalizer jsDisplay is null");
57 return;
58 }
59 sptr<Display> display = jsDisplay->display_;
60 if (display == nullptr) {
61 WLOGFE("JsDisplay::Finalizer display is null");
62 return;
63 }
64 DisplayId displayId = display->GetId();
65 WLOGFI("JsDisplay::Finalizer displayId : %{public}" PRIu64"", displayId);
66 std::lock_guard<std::recursive_mutex> lock(g_mutex);
67 if (g_JsDisplayMap.find(displayId) != g_JsDisplayMap.end()) {
68 WLOGFI("JsDisplay::Finalizer Display is destroyed: %{public}" PRIu64"", displayId);
69 g_JsDisplayMap.erase(displayId);
70 }
71 }
72
GetCutoutInfo(NativeEngine * engine,NativeCallbackInfo * info)73 NativeValue* JsDisplay::GetCutoutInfo(NativeEngine* engine, NativeCallbackInfo* info)
74 {
75 WLOGFI("GetCutoutInfo is called");
76 JsDisplay* me = CheckParamsAndGetThis<JsDisplay>(engine, info);
77 return (me != nullptr) ? me->OnGetCutoutInfo(*engine, *info) : nullptr;
78 }
79
OnGetCutoutInfo(NativeEngine & engine,NativeCallbackInfo & info)80 NativeValue* JsDisplay::OnGetCutoutInfo(NativeEngine& engine, NativeCallbackInfo& info)
81 {
82 WLOGFI("OnGetCutoutInfo is called");
83 AsyncTask::CompleteCallback complete =
84 [this](NativeEngine& engine, AsyncTask& task, int32_t status) {
85 sptr<CutoutInfo> cutoutInfo = display_->GetCutoutInfo();
86 if (cutoutInfo != nullptr) {
87 task.Resolve(engine, CreateJsCutoutInfoObject(engine, cutoutInfo));
88 WLOGFI("JsDisplay::OnGetCutoutInfo success");
89 } else {
90 task.Reject(engine, CreateJsError(engine,
91 static_cast<int32_t>(DmErrorCode::DM_ERROR_INVALID_SCREEN), "JsDisplay::OnGetCutoutInfo failed."));
92 }
93 };
94 NativeValue* lastParam = nullptr;
95 if (info.argc >= ARGC_ONE && info.argv[ARGC_ONE - 1] != nullptr &&
96 info.argv[ARGC_ONE - 1]->TypeOf() == NATIVE_FUNCTION) {
97 lastParam = info.argv[ARGC_ONE - 1];
98 }
99 NativeValue* result = nullptr;
100 AsyncTask::Schedule("JsDisplay::OnGetCutoutInfo",
101 engine, CreateAsyncTaskWithLastParam(engine, lastParam, nullptr, std::move(complete), &result));
102 return result;
103 }
104
FindJsDisplayObject(DisplayId displayId)105 std::shared_ptr<NativeReference> FindJsDisplayObject(DisplayId displayId)
106 {
107 WLOGFI("[NAPI]Try to find display %{public}" PRIu64" in g_JsDisplayMap", displayId);
108 std::lock_guard<std::recursive_mutex> lock(g_mutex);
109 if (g_JsDisplayMap.find(displayId) == g_JsDisplayMap.end()) {
110 WLOGFI("[NAPI]Can not find display %{public}" PRIu64" in g_JsDisplayMap", displayId);
111 return nullptr;
112 }
113 return g_JsDisplayMap[displayId];
114 }
115
CreateJsCutoutInfoObject(NativeEngine & engine,sptr<CutoutInfo> cutoutInfo)116 NativeValue* CreateJsCutoutInfoObject(NativeEngine& engine, sptr<CutoutInfo> cutoutInfo)
117 {
118 WLOGFI("JsDisplay::CreateJsCutoutInfoObject is called");
119 NativeValue* objValue = engine.CreateObject();
120 NativeObject* object = ConvertNativeValueTo<NativeObject>(objValue);
121 if (object == nullptr) {
122 WLOGFE("Failed to convert prop to jsObject");
123 return engine.CreateUndefined();
124 }
125 if (cutoutInfo == nullptr) {
126 WLOGFE("Get null cutout info");
127 return engine.CreateUndefined();
128 }
129 std::vector<DMRect> boundingRects = cutoutInfo->GetBoundingRects();
130 WaterfallDisplayAreaRects waterfallDisplayAreaRects = cutoutInfo->GetWaterfallDisplayAreaRects();
131 object->SetProperty("boundingRects", CreateJsBoundingRectsArrayObject(engine, boundingRects));
132 object->SetProperty("waterfallDisplayAreaRects",
133 CreateJsWaterfallDisplayAreaRectsObject(engine, waterfallDisplayAreaRects));
134 return objValue;
135 }
136
CreateJsRectObject(NativeEngine & engine,DMRect rect)137 NativeValue* CreateJsRectObject(NativeEngine& engine, DMRect rect)
138 {
139 NativeValue* objValue = engine.CreateObject();
140 NativeObject* object = ConvertNativeValueTo<NativeObject>(objValue);
141 object->SetProperty("left", CreateJsValue(engine, rect.posX_));
142 object->SetProperty("top", CreateJsValue(engine, rect.posY_));
143 object->SetProperty("width", CreateJsValue(engine, rect.width_));
144 object->SetProperty("height", CreateJsValue(engine, rect.height_));
145 return objValue;
146 }
147
CreateJsWaterfallDisplayAreaRectsObject(NativeEngine & engine,WaterfallDisplayAreaRects waterfallDisplayAreaRects)148 NativeValue* CreateJsWaterfallDisplayAreaRectsObject(NativeEngine& engine,
149 WaterfallDisplayAreaRects waterfallDisplayAreaRects)
150 {
151 NativeValue* objValue = engine.CreateObject();
152 NativeObject* object = ConvertNativeValueTo<NativeObject>(objValue);
153 object->SetProperty("left", CreateJsRectObject(engine, waterfallDisplayAreaRects.left));
154 object->SetProperty("top", CreateJsRectObject(engine, waterfallDisplayAreaRects.top));
155 object->SetProperty("right", CreateJsRectObject(engine, waterfallDisplayAreaRects.right));
156 object->SetProperty("bottom", CreateJsRectObject(engine, waterfallDisplayAreaRects.bottom));
157 return objValue;
158 }
159
CreateJsBoundingRectsArrayObject(NativeEngine & engine,std::vector<DMRect> boundingRects)160 NativeValue* CreateJsBoundingRectsArrayObject(NativeEngine& engine, std::vector<DMRect> boundingRects)
161 {
162 NativeValue* arrayValue = engine.CreateArray(boundingRects.size());
163 NativeArray* array = ConvertNativeValueTo<NativeArray>(arrayValue);
164 size_t i = 0;
165 for (const auto& rect : boundingRects) {
166 array->SetElement(i++, CreateJsRectObject(engine, rect));
167 }
168 return arrayValue;
169 }
170
CreateJsDisplayObject(NativeEngine & engine,sptr<Display> & display)171 NativeValue* CreateJsDisplayObject(NativeEngine& engine, sptr<Display>& display)
172 {
173 WLOGFI("JsDisplay::CreateJsDisplay is called");
174 NativeValue* objValue = nullptr;
175 std::shared_ptr<NativeReference> jsDisplayObj = FindJsDisplayObject(display->GetId());
176 if (jsDisplayObj != nullptr && jsDisplayObj->Get() != nullptr) {
177 WLOGFI("[NAPI]FindJsDisplayObject %{public}" PRIu64"", display->GetId());
178 objValue = jsDisplayObj->Get();
179 }
180 if (objValue == nullptr) {
181 objValue = engine.CreateObject();
182 }
183 NativeObject* object = ConvertNativeValueTo<NativeObject>(objValue);
184 if (object == nullptr) {
185 WLOGFE("Failed to convert prop to jsObject");
186 return engine.CreateUndefined();
187 }
188 auto info = display->GetDisplayInfo();
189 if (info == nullptr) {
190 WLOGFE("Failed to GetDisplayInfo");
191 return engine.CreateUndefined();
192 }
193 object->SetProperty("id", CreateJsValue(engine, static_cast<uint32_t>(info->GetDisplayId())));
194 object->SetProperty("width", CreateJsValue(engine, info->GetWidth()));
195 object->SetProperty("height", CreateJsValue(engine, info->GetHeight()));
196 object->SetProperty("refreshRate", CreateJsValue(engine, info->GetRefreshRate()));
197 object->SetProperty("name", CreateJsValue(engine, info->GetName()));
198 object->SetProperty("alive", CreateJsValue(engine, true));
199 if (NATIVE_TO_JS_DISPLAY_STATE_MAP.count(info->GetDisplayState()) != 0) {
200 object->SetProperty("state", CreateJsValue(engine, NATIVE_TO_JS_DISPLAY_STATE_MAP.at(info->GetDisplayState())));
201 } else {
202 object->SetProperty("state", CreateJsValue(engine, DisplayStateMode::STATE_UNKNOWN));
203 }
204
205 object->SetProperty("rotation", CreateJsValue(engine, info->GetRotation()));
206 object->SetProperty("densityDPI", CreateJsValue(engine, info->GetVirtualPixelRatio() * DOT_PER_INCH));
207 object->SetProperty("densityPixels", CreateJsValue(engine, info->GetVirtualPixelRatio()));
208 object->SetProperty("scaledDensity", CreateJsValue(engine, info->GetVirtualPixelRatio()));
209 object->SetProperty("xDPI", CreateJsValue(engine, 0.0f));
210 object->SetProperty("yDPI", CreateJsValue(engine, 0.0f));
211 if (jsDisplayObj == nullptr || jsDisplayObj->Get() == nullptr) {
212 std::unique_ptr<JsDisplay> jsDisplay = std::make_unique<JsDisplay>(display);
213 object->SetNativePointer(jsDisplay.release(), JsDisplay::Finalizer, nullptr);
214 BindNativeFunction(engine, *object, "getCutoutInfo", "JsDisplay", JsDisplay::GetCutoutInfo);
215 std::shared_ptr<NativeReference> jsDisplayRef;
216 jsDisplayRef.reset(engine.CreateReference(objValue, 1));
217 DisplayId displayId = display->GetId();
218 std::lock_guard<std::recursive_mutex> lock(g_mutex);
219 g_JsDisplayMap[displayId] = jsDisplayRef;
220 }
221 return objValue;
222 }
223 } // namespace Rosen
224 } // namespace OHOS
225