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 "base/log/ace_scoring_log.h"
21 #include "bridge/declarative_frontend/jsview/js_refresh.h"
22 #include "bridge/declarative_frontend/jsview/js_view_common_def.h"
23 #include "bridge/declarative_frontend/jsview/models/refresh_model_impl.h"
24 #include "core/components/refresh/refresh_theme.h"
25 #include "core/components_ng/base/view_stack_processor.h"
26 #include "core/components_ng/pattern/refresh/refresh_model_ng.h"
27
28 namespace OHOS::Ace {
29 namespace {
30 constexpr int32_t DEFAULT_FRICTION = 62;
31 constexpr int32_t MAX_FRICTION = 100;
32 } // namespace
33 std::unique_ptr<RefreshModel> RefreshModel::instance_ = nullptr;
34 std::mutex RefreshModel::mutex_;
35
GetInstance()36 RefreshModel* RefreshModel::GetInstance()
37 {
38 if (!instance_) {
39 std::lock_guard<std::mutex> lock(mutex_);
40 if (!instance_) {
41 #ifdef NG_BUILD
42 instance_.reset(new NG::RefreshModelNG());
43 #else
44 if (Container::IsCurrentUseNewPipeline()) {
45 instance_.reset(new NG::RefreshModelNG());
46 } else {
47 instance_.reset(new Framework::RefreshModelImpl());
48 }
49 #endif
50 }
51 }
52 return instance_.get();
53 }
54
55 } // namespace OHOS::Ace
56
57 namespace OHOS::Ace::Framework {
58
ParseRefreshingObject(const JSCallbackInfo & info,const JSRef<JSObject> & refreshing)59 void ParseRefreshingObject(const JSCallbackInfo& info, const JSRef<JSObject>& refreshing)
60 {
61 JSRef<JSVal> changeEventVal = refreshing->GetProperty("changeEvent");
62 CHECK_NULL_VOID(changeEventVal->IsFunction());
63
64 auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(changeEventVal));
65 WeakPtr<NG::FrameNode> targetNode = NG::ViewStackProcessor::GetInstance()->GetMainFrameNode();
66 auto changeEvent = [execCtx = info.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
67 const std::string& param) {
68 JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
69 if (param != "true" && param != "false") {
70 return;
71 }
72 bool newValue = StringToBool(param);
73 ACE_SCORING_EVENT("Refresh.ChangeEvent");
74 PipelineContext::SetCallBackNode(node);
75 auto newJSVal = JSRef<JSVal>::Make(ToJSValue(newValue));
76 func->ExecuteJS(1, &newJSVal);
77 };
78 RefreshModel::GetInstance()->SetChangeEvent(std::move(changeEvent));
79 }
80
JSBind(BindingTarget globalObj)81 void JSRefresh::JSBind(BindingTarget globalObj)
82 {
83 JSClass<JSRefresh>::Declare("Refresh");
84 MethodOptions opt = MethodOptions::NONE;
85 JSClass<JSRefresh>::StaticMethod("create", &JSRefresh::Create, opt);
86 JSClass<JSRefresh>::StaticMethod("onStateChange", &JSRefresh::OnStateChange);
87 JSClass<JSRefresh>::StaticMethod("onRefreshing", &JSRefresh::OnRefreshing);
88 JSClass<JSRefresh>::StaticMethod("onAppear", &JSInteractableView::JsOnAppear);
89 JSClass<JSRefresh>::StaticMethod("onDisAppear", &JSInteractableView::JsOnDisAppear);
90 JSClass<JSRefresh>::StaticMethod("onTouch", &JSInteractableView::JsOnTouch);
91 JSClass<JSRefresh>::InheritAndBind<JSContainerBase>(globalObj);
92 }
93
Create(const JSCallbackInfo & info)94 void JSRefresh::Create(const JSCallbackInfo& info)
95 {
96 if (!info[0]->IsObject()) {
97 return;
98 }
99 RefPtr<RefreshTheme> theme = GetTheme<RefreshTheme>();
100 if (!theme) {
101 return;
102 }
103 auto paramObject = JSRef<JSObject>::Cast(info[0]);
104 auto refreshing = paramObject->GetProperty("refreshing");
105 auto jsOffset = paramObject->GetProperty("offset");
106 auto friction = paramObject->GetProperty("friction");
107 RefreshModel::GetInstance()->Create();
108 RefreshModel::GetInstance()->SetProgressColor(theme->GetProgressColor());
109
110 if (refreshing->IsBoolean()) {
111 RefreshModel::GetInstance()->SetRefreshing(refreshing->ToBoolean());
112 } else {
113 JSRef<JSObject> refreshingObj = JSRef<JSObject>::Cast(refreshing);
114 ParseRefreshingObject(info, refreshingObj);
115 RefreshModel::GetInstance()->SetRefreshing(refreshingObj->GetProperty("value")->ToBoolean());
116 }
117 CalcDimension offset;
118 if (ParseJsDimensionVp(jsOffset, offset)) {
119 if (LessNotEqual(offset.Value(), 0.0) || offset.Unit() == DimensionUnit::PERCENT) {
120 RefreshModel::GetInstance()->SetRefreshDistance(theme->GetRefreshDistance());
121 } else {
122 RefreshModel::GetInstance()->SetIndicatorOffset(offset);
123 }
124 }
125 ParsFrictionData(friction);
126 ParseCustomBuilder(info);
127 }
128
ParseCustomBuilder(const JSCallbackInfo & info)129 void JSRefresh::ParseCustomBuilder(const JSCallbackInfo& info)
130 {
131 if (!info[0]->IsObject()) {
132 return;
133 }
134 auto paramObject = JSRef<JSObject>::Cast(info[0]);
135 auto builder = paramObject->GetProperty("builder");
136 RefPtr<NG::UINode> customNode;
137 if (builder->IsFunction()) {
138 {
139 NG::ScopedViewStackProcessor builderViewStackProcessor;
140 JsFunction Jsfunc(info.This(), JSRef<JSObject>::Cast(builder));
141 Jsfunc.Execute();
142 customNode = NG::ViewStackProcessor::GetInstance()->Finish();
143 }
144 RefreshModel::GetInstance()->SetCustomBuilder(customNode);
145 } else {
146 RefreshModel::GetInstance()->SetCustomBuilder(customNode);
147 }
148 }
149
OnStateChange(const JSCallbackInfo & args)150 void JSRefresh::OnStateChange(const JSCallbackInfo& args)
151 {
152 if (!args[0]->IsFunction()) {
153 return;
154 }
155 auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(args[0]));
156 WeakPtr<NG::FrameNode> targetNode = NG::ViewStackProcessor::GetInstance()->GetMainFrameNode();
157 auto onStateChange = [execCtx = args.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
158 const int32_t& value) {
159 JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
160 ACE_SCORING_EVENT("Refresh.OnStateChange");
161 PipelineContext::SetCallBackNode(node);
162 auto newJSVal = JSRef<JSVal>::Make(ToJSValue(value));
163 func->ExecuteJS(1, &newJSVal);
164 };
165 RefreshModel::GetInstance()->SetOnStateChange(std::move(onStateChange));
166 }
167
OnRefreshing(const JSCallbackInfo & args)168 void JSRefresh::OnRefreshing(const JSCallbackInfo& args)
169 {
170 if (!args[0]->IsFunction()) {
171 return;
172 }
173 auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(args[0]));
174 WeakPtr<NG::FrameNode> targetNode = NG::ViewStackProcessor::GetInstance()->GetMainFrameNode();
175 auto onRefreshing = [execCtx = args.GetExecutionContext(), func = std::move(jsFunc), node = targetNode]() {
176 JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
177 ACE_SCORING_EVENT("Refresh.OnRefreshing");
178 PipelineContext::SetCallBackNode(node);
179 auto newJSVal = JSRef<JSVal>::Make();
180 func->ExecuteJS(1, &newJSVal);
181 };
182 RefreshModel::GetInstance()->SetOnRefreshing(std::move(onRefreshing));
183 }
184
ParsFrictionData(const JsiRef<JsiValue> & friction)185 void JSRefresh::ParsFrictionData(const JsiRef<JsiValue>& friction)
186 {
187 int32_t frictionNumber = DEFAULT_FRICTION;
188 if (friction->IsString()) {
189 frictionNumber = StringUtils::StringToInt(friction->ToString());
190 if ((frictionNumber == 0 && friction->ToString() != "0") || frictionNumber < 0 ||
191 frictionNumber > MAX_FRICTION) {
192 frictionNumber = DEFAULT_FRICTION;
193 }
194 } else if (friction->IsNumber()) {
195 frictionNumber = friction->ToNumber<int32_t>();
196 if (frictionNumber < 0 || frictionNumber > MAX_FRICTION) {
197 frictionNumber = DEFAULT_FRICTION;
198 }
199 }
200 RefreshModel::GetInstance()->SetFriction(frictionNumber);
201 }
202 } // namespace OHOS::Ace::Framework
203