• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021-2023 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 "frameworks/bridge/declarative_frontend/jsview/js_refresh.h"
17 
18 #include <cstdint>
19 
20 #include "interfaces/inner_api/ui_session/ui_session_manager.h"
21 
22 #include "base/log/ace_scoring_log.h"
23 #include "bridge/declarative_frontend/jsview/js_refresh.h"
24 #include "bridge/declarative_frontend/jsview/js_view_common_def.h"
25 #include "bridge/declarative_frontend/jsview/models/refresh_model_impl.h"
26 #include "core/components/refresh/refresh_theme.h"
27 #include "core/components_ng/base/view_stack_processor.h"
28 #include "core/components_ng/pattern/refresh/refresh_model_ng.h"
29 
30 namespace OHOS::Ace {
31 namespace {
32 constexpr int32_t DEFAULT_FRICTION = 62;
33 constexpr int32_t MAX_FRICTION = 100;
34 } // namespace
35 std::unique_ptr<RefreshModel> RefreshModel::instance_ = nullptr;
36 std::mutex RefreshModel::mutex_;
37 
GetInstance()38 RefreshModel* RefreshModel::GetInstance()
39 {
40     if (!instance_) {
41         std::lock_guard<std::mutex> lock(mutex_);
42         if (!instance_) {
43 #ifdef NG_BUILD
44             instance_.reset(new NG::RefreshModelNG());
45 #else
46             if (Container::IsCurrentUseNewPipeline()) {
47                 instance_.reset(new NG::RefreshModelNG());
48             } else {
49                 instance_.reset(new Framework::RefreshModelImpl());
50             }
51 #endif
52         }
53     }
54     return instance_.get();
55 }
56 
57 } // namespace OHOS::Ace
58 
59 namespace OHOS::Ace::Framework {
60 
ParseRefreshingObject(const JSCallbackInfo & info,const JSRef<JSVal> & changeEventVal)61 void ParseRefreshingObject(const JSCallbackInfo& info, const JSRef<JSVal>& changeEventVal)
62 {
63     CHECK_NULL_VOID(changeEventVal->IsFunction());
64 
65     auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(changeEventVal));
66     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
67     auto changeEvent = [execCtx = info.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
68                            const std::string& param) {
69         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
70         if (param != "true" && param != "false") {
71             return;
72         }
73         bool newValue = StringToBool(param);
74         ACE_SCORING_EVENT("Refresh.ChangeEvent");
75         PipelineContext::SetCallBackNode(node);
76         auto newJSVal = JSRef<JSVal>::Make(ToJSValue(newValue));
77         func->ExecuteJS(1, &newJSVal);
78     };
79     RefreshModel::GetInstance()->SetChangeEvent(std::move(changeEvent));
80 }
81 
SetPullToRefresh(const JSCallbackInfo & info)82 void JSRefresh::SetPullToRefresh(const JSCallbackInfo& info)
83 {
84     bool pullToRefresh = true;
85     if (info[0]->IsBoolean()) {
86         pullToRefresh = info[0]->ToBoolean();
87     }
88     RefreshModel::GetInstance()->SetPullToRefresh(pullToRefresh);
89 }
90 
JSBind(BindingTarget globalObj)91 void JSRefresh::JSBind(BindingTarget globalObj)
92 {
93     JSClass<JSRefresh>::Declare("Refresh");
94     MethodOptions opt = MethodOptions::NONE;
95     JSClass<JSRefresh>::StaticMethod("create", &JSRefresh::Create, opt);
96     JSClass<JSRefresh>::StaticMethod("refreshOffset", &JSRefresh::JsRefreshOffset);
97     JSClass<JSRefresh>::StaticMethod("pullToRefresh", &JSRefresh::SetPullToRefresh, opt);
98     JSClass<JSRefresh>::StaticMethod("onStateChange", &JSRefresh::OnStateChange);
99     JSClass<JSRefresh>::StaticMethod("onRefreshing", &JSRefresh::OnRefreshing);
100     JSClass<JSRefresh>::StaticMethod("onOffsetChange", &JSRefresh::OnOffsetChange);
101     JSClass<JSRefresh>::StaticMethod("pullDownRatio", &JSRefresh::SetPullDownRatio);
102     JSClass<JSRefresh>::StaticMethod("onAttach", &JSInteractableView::JsOnAttach);
103     JSClass<JSRefresh>::StaticMethod("onAppear", &JSInteractableView::JsOnAppear);
104     JSClass<JSRefresh>::StaticMethod("onDetach", &JSInteractableView::JsOnDetach);
105     JSClass<JSRefresh>::StaticMethod("onDisAppear", &JSInteractableView::JsOnDisAppear);
106     JSClass<JSRefresh>::StaticMethod("onTouch", &JSInteractableView::JsOnTouch);
107     JSClass<JSRefresh>::InheritAndBind<JSContainerBase>(globalObj);
108 }
109 
SetPullDownRatio(const JSCallbackInfo & info)110 void JSRefresh::SetPullDownRatio(const JSCallbackInfo& info)
111 {
112     if (info.Length() < 1) {
113         return;
114     }
115 
116     auto args = info[0];
117     std::optional<float> pulldownRatio = std::nullopt;
118     if (!args->IsNumber() || std::isnan(args->ToNumber<float>())) {
119         RefreshModel::GetInstance()->SetPullDownRatio(pulldownRatio);
120         return;
121     }
122     pulldownRatio = std::clamp(args->ToNumber<float>(), 0.f, 1.f);
123     RefreshModel::GetInstance()->SetPullDownRatio(pulldownRatio);
124 }
125 
JsRefreshOffset(const JSCallbackInfo & info)126 void JSRefresh::JsRefreshOffset(const JSCallbackInfo& info)
127 {
128     if (info.Length() < 1) {
129         return;
130     }
131     JsRefreshOffset(info[0]);
132 }
133 
JsRefreshOffset(const JSRef<JSVal> & jsVal)134 void JSRefresh::JsRefreshOffset(const JSRef<JSVal>& jsVal)
135 {
136     CalcDimension value(0.0f);
137     if (!ParseJsDimensionVpNG(jsVal, value)) {
138         value.SetValue(0.0f);
139     }
140     RefreshModel::GetInstance()->SetRefreshOffset(value);
141 }
142 
Create(const JSCallbackInfo & info)143 void JSRefresh::Create(const JSCallbackInfo& info)
144 {
145     if (!info[0]->IsObject()) {
146         return;
147     }
148     RefPtr<RefreshTheme> theme = GetTheme<RefreshTheme>();
149     if (!theme) {
150         return;
151     }
152     auto paramObject = JSRef<JSObject>::Cast(info[0]);
153     auto refreshing = paramObject->GetProperty("refreshing");
154     auto jsOffset = paramObject->GetProperty("offset");
155     auto friction = paramObject->GetProperty("friction");
156     auto promptText = paramObject->GetProperty("promptText");
157     JSRef<JSVal> changeEventVal;
158     RefreshModel::GetInstance()->Create();
159 
160     if (refreshing->IsBoolean()) {
161         RefreshModel::GetInstance()->SetRefreshing(refreshing->ToBoolean());
162         changeEventVal = paramObject->GetProperty("$refreshing");
163         ParseRefreshingObject(info, changeEventVal);
164     } else if (refreshing->IsObject()) {
165         JSRef<JSObject> refreshingObj = JSRef<JSObject>::Cast(refreshing);
166         changeEventVal = refreshingObj->GetProperty("changeEvent");
167         ParseRefreshingObject(info, changeEventVal);
168         RefreshModel::GetInstance()->SetRefreshing(refreshingObj->GetProperty("value")->ToBoolean());
169     } else {
170         RefreshModel::GetInstance()->SetRefreshing(false);
171     }
172     CalcDimension offset;
173     if (ParseJsDimensionVp(jsOffset, offset)) {
174         if (LessNotEqual(offset.Value(), 0.0) || offset.Unit() == DimensionUnit::PERCENT) {
175             RefreshModel::GetInstance()->SetRefreshDistance(theme->GetRefreshDistance());
176         } else {
177             RefreshModel::GetInstance()->SetIndicatorOffset(offset);
178         }
179     }
180     ParsFrictionData(friction);
181     if (!ParseRefreshingContent(paramObject)) {
182         bool isCustomBuilderExist = ParseCustomBuilder(info);
183         RefreshModel::GetInstance()->SetIsCustomBuilderExist(isCustomBuilderExist);
184     }
185 
186     std::string loadingStr = "";
187     if (ParseJsString(promptText, loadingStr)) {
188         RefreshModel::GetInstance()->SetLoadingText(loadingStr);
189     } else {
190         RefreshModel::GetInstance()->ResetLoadingText();
191     }
192 }
193 
ParseRefreshingContent(const JSRef<JSObject> & paramObject)194 bool JSRefresh::ParseRefreshingContent(const JSRef<JSObject>& paramObject)
195 {
196     JSRef<JSVal> contentParam = paramObject->GetProperty("refreshingContent");
197     if (!contentParam->IsObject()) {
198         return false;
199     }
200     JSRef<JSObject> contentObject = JSRef<JSObject>::Cast(contentParam);
201     JSRef<JSVal> builderNodeParam = contentObject->GetProperty("builderNode_");
202     if (!builderNodeParam->IsObject()) {
203         return false;
204     }
205     JSRef<JSObject> builderNodeObject = JSRef<JSObject>::Cast(builderNodeParam);
206     JSRef<JSVal> nodeptr = builderNodeObject->GetProperty("nodePtr_");
207     if (nodeptr.IsEmpty()) {
208         return false;
209     }
210     const auto* vm = nodeptr->GetEcmaVM();
211     CHECK_NULL_RETURN(nodeptr->GetLocalHandle()->IsNativePointer(vm), false);
212     auto* node = nodeptr->GetLocalHandle()->ToNativePointer(vm)->Value();
213     auto* frameNode = reinterpret_cast<NG::FrameNode*>(node);
214     CHECK_NULL_RETURN(frameNode, false);
215     RefPtr<NG::FrameNode> refPtrFrameNode = AceType::Claim(frameNode);
216     RefreshModel::GetInstance()->SetCustomBuilder(refPtrFrameNode);
217     RefreshModel::GetInstance()->SetIsCustomBuilderExist(false);
218     return true;
219 }
220 
ParseCustomBuilder(const JSCallbackInfo & info)221 bool JSRefresh::ParseCustomBuilder(const JSCallbackInfo& info)
222 {
223     if (!info[0]->IsObject()) {
224         return false;
225     }
226     auto paramObject = JSRef<JSObject>::Cast(info[0]);
227     auto builder = paramObject->GetProperty("builder");
228     RefPtr<NG::UINode> customNode;
229     if (builder->IsFunction()) {
230         {
231             NG::ScopedViewStackProcessor builderViewStackProcessor;
232             JsFunction Jsfunc(info.This(), JSRef<JSFunc>::Cast(builder));
233             Jsfunc.Execute();
234             customNode = NG::ViewStackProcessor::GetInstance()->Finish();
235         }
236         RefreshModel::GetInstance()->SetCustomBuilder(customNode);
237         return true;
238     } else {
239         RefreshModel::GetInstance()->SetCustomBuilder(customNode);
240         return false;
241     }
242 }
243 
OnStateChange(const JSCallbackInfo & args)244 void JSRefresh::OnStateChange(const JSCallbackInfo& args)
245 {
246     if (!args[0]->IsFunction()) {
247         return;
248     }
249     auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(args[0]));
250     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
251     auto onStateChange = [execCtx = args.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
252                              const int32_t& value) {
253         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
254         ACE_SCORING_EVENT("Refresh.OnStateChange");
255         PipelineContext::SetCallBackNode(node);
256         auto newJSVal = JSRef<JSVal>::Make(ToJSValue(value));
257         func->ExecuteJS(1, &newJSVal);
258         UiSessionManager::GetInstance()->ReportComponentChangeEvent("event", "Refresh.OnStateChange");
259     };
260     RefreshModel::GetInstance()->SetOnStateChange(std::move(onStateChange));
261 }
262 
OnRefreshing(const JSCallbackInfo & args)263 void JSRefresh::OnRefreshing(const JSCallbackInfo& args)
264 {
265     if (!args[0]->IsFunction()) {
266         return;
267     }
268     auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(args[0]));
269     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
270     auto onRefreshing = [execCtx = args.GetExecutionContext(), func = std::move(jsFunc), node = targetNode]() {
271         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
272         ACE_SCORING_EVENT("Refresh.OnRefreshing");
273         PipelineContext::SetCallBackNode(node);
274         auto newJSVal = JSRef<JSVal>::Make();
275         func->ExecuteJS(1, &newJSVal);
276     };
277     RefreshModel::GetInstance()->SetOnRefreshing(std::move(onRefreshing));
278 }
279 
OnOffsetChange(const JSCallbackInfo & args)280 void JSRefresh::OnOffsetChange(const JSCallbackInfo& args)
281 {
282     if (!args[0]->IsFunction()) {
283         RefreshModel::GetInstance()->ResetOnOffsetChange();
284         return;
285     }
286     auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(args[0]));
287     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
288     auto offsetChange = [execCtx = args.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
289                             const float& value) {
290         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
291         ACE_SCORING_EVENT("Refresh.OnOffsetChange");
292         PipelineContext::SetCallBackNode(node);
293         auto newJSVal = JSRef<JSVal>::Make(ToJSValue(value));
294         func->ExecuteJS(1, &newJSVal);
295     };
296     RefreshModel::GetInstance()->SetOnOffsetChange(std::move(offsetChange));
297 }
298 
ParsFrictionData(const JsiRef<JsiValue> & friction)299 void JSRefresh::ParsFrictionData(const JsiRef<JsiValue>& friction)
300 {
301     int32_t frictionNumber = DEFAULT_FRICTION;
302     if (friction->IsString()) {
303         frictionNumber = StringUtils::StringToInt(friction->ToString());
304         if ((frictionNumber == 0 && friction->ToString() != "0") || frictionNumber < 0 ||
305             frictionNumber > MAX_FRICTION) {
306             frictionNumber = DEFAULT_FRICTION;
307         }
308     } else if (friction->IsNumber()) {
309         frictionNumber = friction->ToNumber<int32_t>();
310         if (frictionNumber < 0 || frictionNumber > MAX_FRICTION) {
311             frictionNumber = DEFAULT_FRICTION;
312         }
313     }
314     RefreshModel::GetInstance()->SetFriction(frictionNumber);
315 }
316 } // namespace OHOS::Ace::Framework
317