• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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