• 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_tabs.h"
17 #if !defined(PREVIEW) && defined(OHOS_PLATFORM)
18 #include "interfaces/inner_api/ui_session/ui_session_manager.h"
19 #endif
20 
21 #include "base/log/ace_scoring_log.h"
22 #include "bridge/declarative_frontend/engine/functions/js_swiper_function.h"
23 #include "bridge/declarative_frontend/engine/functions/js_tabs_function.h"
24 #include "bridge/declarative_frontend/jsview/js_scrollable.h"
25 #include "bridge/declarative_frontend/jsview/js_tabs_controller.h"
26 #include "bridge/declarative_frontend/jsview/js_view_common_def.h"
27 #include "bridge/declarative_frontend/jsview/models/tabs_model_impl.h"
28 #include "core/components/common/layout/constants.h"
29 #include "core/components/common/properties/decoration.h"
30 #include "core/components_ng/base/view_stack_processor.h"
31 #include "core/components_ng/pattern/tabs/tab_content_transition_proxy.h"
32 #include "core/components_ng/pattern/tabs/tabs_model_ng.h"
33 
34 namespace OHOS::Ace {
35 
36 std::unique_ptr<TabsModel> TabsModel::instance_ = nullptr;
37 std::mutex TabsModel::mutex_;
38 
GetInstance()39 TabsModel* TabsModel::GetInstance()
40 {
41     if (!instance_) {
42         std::lock_guard<std::mutex> lock(mutex_);
43         if (!instance_) {
44 #ifdef NG_BUILD
45             instance_.reset(new NG::TabsModelNG());
46 #else
47             if (Container::IsCurrentUseNewPipeline()) {
48                 instance_.reset(new NG::TabsModelNG());
49             } else {
50                 instance_.reset(new Framework::TabsModelImpl());
51             }
52 #endif
53         }
54     }
55     return instance_.get();
56 }
57 
58 } // namespace OHOS::Ace
59 
60 namespace OHOS::Ace::Framework {
61 namespace {
62 constexpr int32_t SM_COLUMN_NUM = 4;
63 constexpr int32_t MD_COLUMN_NUM = 8;
64 constexpr int32_t LG_COLUMN_NUM = 12;
65 constexpr int32_t DEFAULT_CUSTOM_ANIMATION_TIMEOUT = 1000;
66 const std::vector<BarPosition> BAR_POSITIONS = { BarPosition::START, BarPosition::END };
67 
68 const std::vector<BlurStyle> BAR_BLURSTYLE = {
69     BlurStyle::NO_MATERIAL,
70     BlurStyle::THIN,
71     BlurStyle::REGULAR,
72     BlurStyle::THICK,
73     BlurStyle::BACKGROUND_THIN,
74     BlurStyle::BACKGROUND_REGULAR,
75     BlurStyle::BACKGROUND_THICK,
76     BlurStyle::BACKGROUND_ULTRA_THICK,
77     BlurStyle::COMPONENT_ULTRA_THIN,
78     BlurStyle::COMPONENT_THIN,
79     BlurStyle::COMPONENT_REGULAR,
80     BlurStyle::COMPONENT_THICK,
81     BlurStyle::COMPONENT_ULTRA_THICK,
82 };
83 
TabContentChangeEventToJSValue(const TabContentChangeEvent & eventInfo)84 JSRef<JSVal> TabContentChangeEventToJSValue(const TabContentChangeEvent& eventInfo)
85 {
86     return JSRef<JSVal>::Make(ToJSValue(eventInfo.GetIndex()));
87 }
88 
89 } // namespace
90 
SetOnChange(const JSCallbackInfo & info)91 void JSTabs::SetOnChange(const JSCallbackInfo& info)
92 {
93     if (!info[0]->IsFunction()) {
94         return;
95     }
96 
97     auto changeHandler = AceType::MakeRefPtr<JsEventFunction<TabContentChangeEvent, 1>>(
98         JSRef<JSFunc>::Cast(info[0]), TabContentChangeEventToJSValue);
99     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
100     auto onChange = [executionContext = info.GetExecutionContext(), func = std::move(changeHandler), node = targetNode](
101                         const BaseEventInfo* info) {
102         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
103         const auto* tabsInfo = TypeInfoHelper::DynamicCast<TabContentChangeEvent>(info);
104         if (!tabsInfo) {
105             TAG_LOGW(AceLogTag::ACE_TABS, "Tabs onChange callback execute failed.");
106             return;
107         }
108         ACE_SCORING_EVENT("Tabs.onChange");
109         ACE_SCOPED_TRACE("Tabs.onChange index %d", tabsInfo->GetIndex());
110         PipelineContext::SetCallBackNode(node);
111         func->Execute(*tabsInfo);
112     };
113     TabsModel::GetInstance()->SetOnChange(std::move(onChange));
114 }
115 
SetOnTabBarClick(const JSCallbackInfo & info)116 void JSTabs::SetOnTabBarClick(const JSCallbackInfo& info)
117 {
118     if (!info[0]->IsFunction()) {
119         return;
120     }
121 
122     auto changeHandler = AceType::MakeRefPtr<JsEventFunction<TabContentChangeEvent, 1>>(
123         JSRef<JSFunc>::Cast(info[0]), TabContentChangeEventToJSValue);
124     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
125     auto onTabBarClick = [executionContext = info.GetExecutionContext(), func = std::move(changeHandler),
126                              node = targetNode](const BaseEventInfo* info) {
127         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
128         const auto* tabsInfo = TypeInfoHelper::DynamicCast<TabContentChangeEvent>(info);
129         if (!tabsInfo) {
130             TAG_LOGW(AceLogTag::ACE_TABS, "Tabs onTabBarClick callback execute failed.");
131             return;
132         }
133         ACE_SCORING_EVENT("Tabs.onTabBarClick");
134         PipelineContext::SetCallBackNode(node);
135         func->Execute(*tabsInfo);
136 #if !defined(PREVIEW) && defined(OHOS_PLATFORM)
137         UiSessionManager::GetInstance().ReportComponentChangeEvent("event", "Tabs.onTabBarClick");
138 #endif
139     };
140     TabsModel::GetInstance()->SetOnTabBarClick(std::move(onTabBarClick));
141 }
142 
SetOnAnimationStart(const JSCallbackInfo & info)143 void JSTabs::SetOnAnimationStart(const JSCallbackInfo& info)
144 {
145     if (!info[0]->IsFunction()) {
146         return;
147     }
148 
149     auto animationStartHandler = AceType::MakeRefPtr<JsSwiperFunction>(JSRef<JSFunc>::Cast(info[0]));
150     auto onAnimationStart = [executionContext = info.GetExecutionContext(),
151                                 func = std::move(animationStartHandler)](
152                                 int32_t index, int32_t targetIndex, const AnimationCallbackInfo& info) {
153         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
154         ACE_SCORING_EVENT("Tabs.onAnimationStart");
155         func->Execute(index, targetIndex, info);
156     };
157     TabsModel::GetInstance()->SetOnAnimationStart(std::move(onAnimationStart));
158 }
159 
SetOnAnimationEnd(const JSCallbackInfo & info)160 void JSTabs::SetOnAnimationEnd(const JSCallbackInfo& info)
161 {
162     if (!info[0]->IsFunction()) {
163         return;
164     }
165 
166     auto animationEndHandler = AceType::MakeRefPtr<JsSwiperFunction>(JSRef<JSFunc>::Cast(info[0]));
167     auto onAnimationEnd = [executionContext = info.GetExecutionContext(), func = std::move(animationEndHandler)](
168                               int32_t index, const AnimationCallbackInfo& info) {
169         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
170         ACE_SCORING_EVENT("Tabs.onAnimationEnd");
171         func->Execute(index, info);
172 #if !defined(PREVIEW) && defined(OHOS_PLATFORM)
173         UiSessionManager::GetInstance().ReportComponentChangeEvent("event", "Tabs.onAnimationEnd");
174 #endif
175     };
176     TabsModel::GetInstance()->SetOnAnimationEnd(std::move(onAnimationEnd));
177 }
178 
SetOnGestureSwipe(const JSCallbackInfo & info)179 void JSTabs::SetOnGestureSwipe(const JSCallbackInfo& info)
180 {
181     if (!info[0]->IsFunction()) {
182         return;
183     }
184 
185     auto gestureSwipeHandler = AceType::MakeRefPtr<JsSwiperFunction>(JSRef<JSFunc>::Cast(info[0]));
186     auto onGestureSwipe = [executionContext = info.GetExecutionContext(), func = std::move(gestureSwipeHandler)](
187                               int32_t index, const AnimationCallbackInfo& info) {
188         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
189         ACE_SCORING_EVENT("Tabs.onGestureSwipe");
190         func->Execute(index, info);
191     };
192     TabsModel::GetInstance()->SetOnGestureSwipe(std::move(onGestureSwipe));
193 }
194 
ParseTabsIndexObject(const JSCallbackInfo & info,const JSRef<JSVal> & changeEventVal)195 void ParseTabsIndexObject(const JSCallbackInfo& info, const JSRef<JSVal>& changeEventVal)
196 {
197     CHECK_NULL_VOID(!changeEventVal->IsUndefined() && changeEventVal->IsFunction());
198 
199     auto jsFunc = AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(changeEventVal));
200     WeakPtr<NG::FrameNode> targetNode = AceType::WeakClaim(NG::ViewStackProcessor::GetInstance()->GetMainFrameNode());
201     auto onChangeEvent = [executionContext = info.GetExecutionContext(), func = std::move(jsFunc), node = targetNode](
202                              const BaseEventInfo* info) {
203         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(executionContext);
204         const auto* tabsInfo = TypeInfoHelper::DynamicCast<TabContentChangeEvent>(info);
205         if (!tabsInfo) {
206             TAG_LOGW(AceLogTag::ACE_TABS, "ParseTabsIndexObject execute onChange event failed.");
207             return;
208         }
209         ACE_SCORING_EVENT("Tabs.onChangeEvent");
210         PipelineContext::SetCallBackNode(node);
211         auto newJSVal = JSRef<JSVal>::Make(ToJSValue(tabsInfo->GetIndex()));
212         func->ExecuteJS(1, &newJSVal);
213     };
214     TabsModel::GetInstance()->SetOnChangeEvent(std::move(onChangeEvent));
215 }
216 
Create(const JSCallbackInfo & info)217 void JSTabs::Create(const JSCallbackInfo& info)
218 {
219     BarPosition barPosition = BarPosition::START;
220     RefPtr<TabController> tabController;
221     RefPtr<NG::TabsControllerNG> tabsController;
222     int32_t index = -1;
223     JSRef<JSVal> changeEventVal;
224     auto jsValue = info[0];
225     if (jsValue->IsObject()) {
226         JSRef<JSObject> obj = JSRef<JSObject>::Cast(jsValue);
227         JSRef<JSVal> val = obj->GetProperty("barPosition");
228         if (val->IsNumber()) {
229             auto barPositionVal = val->ToNumber<int32_t>();
230             if (barPositionVal >= 0 && barPositionVal < static_cast<int32_t>(BAR_POSITIONS.size())) {
231                 barPosition = BAR_POSITIONS[barPositionVal];
232             }
233         }
234         JSRef<JSVal> controller = obj->GetProperty("controller");
235         if (controller->IsObject()) {
236             auto* jsTabsController = JSRef<JSObject>::Cast(controller)->Unwrap<JSTabsController>();
237             if (jsTabsController) {
238                 jsTabsController->SetInstanceId(Container::CurrentId());
239                 tabController = jsTabsController->GetController();
240                 tabsController = jsTabsController->GetTabsController();
241             }
242         }
243         JSRef<JSVal> indexVal = obj->GetProperty("index");
244         if (indexVal->IsNumber()) {
245             index = indexVal->ToNumber<int32_t>();
246             index = index < 0 ? 0 : index;
247             if (!tabController) {
248                 tabController = JSTabsController::CreateController();
249             }
250 #ifndef NG_BUILD
251             tabController->SetInitialIndex(index);
252 #endif
253         } else if (indexVal->IsObject()) {
254             JSRef<JSObject> indexObj = JSRef<JSObject>::Cast(indexVal);
255             auto indexValueProperty = indexObj->GetProperty("value");
256             if (indexValueProperty->IsNumber()) {
257                 index = indexValueProperty->ToNumber<int32_t>();
258                 index = index < 0 ? 0 : index;
259             }
260             changeEventVal = indexObj->GetProperty("changeEvent");
261         }
262     }
263 
264     TabsModel::GetInstance()->Create(barPosition, index, tabController, tabsController);
265     ParseTabsIndexObject(info, changeEventVal);
266     SetBarModifier(info, jsValue);
267 }
268 
Pop()269 void JSTabs::Pop()
270 {
271     TabsModel::GetInstance()->Pop();
272 }
273 
SetBarPosition(const JSCallbackInfo & info)274 void JSTabs::SetBarPosition(const JSCallbackInfo& info)
275 {
276     BarPosition barVal = BarPosition::START;
277     if (info.Length() > 0 && info[0]->IsNumber()) {
278         auto barPositionVal = info[0]->ToNumber<int32_t>();
279         if (barPositionVal >= 0 && barPositionVal < static_cast<int32_t>(BAR_POSITIONS.size())) {
280             barVal = BAR_POSITIONS[barPositionVal];
281         }
282     }
283 
284     TabsModel::GetInstance()->SetTabBarPosition(barVal);
285 }
286 
SetVertical(const std::string & value)287 void JSTabs::SetVertical(const std::string& value)
288 {
289     TabsModel::GetInstance()->SetIsVertical(StringToBool(value));
290 }
291 
SetScrollable(const std::string & value)292 void JSTabs::SetScrollable(const std::string& value)
293 {
294     if (value == "undefined") {
295         TabsModel::GetInstance()->SetScrollable(true);
296         return;
297     }
298     TabsModel::GetInstance()->SetScrollable(StringToBool(value));
299 }
300 
SetBarMode(const JSCallbackInfo & info)301 void JSTabs::SetBarMode(const JSCallbackInfo& info)
302 {
303     TabBarMode barMode = TabBarMode::FIXED;
304     if (info.Length() < 1) {
305         TabsModel::GetInstance()->SetTabBarMode(barMode);
306         return;
307     }
308     auto barModeInfo = info[0];
309     if (barModeInfo->IsString()) {
310         barMode = ConvertStrToTabBarMode(barModeInfo->ToString());
311     }
312     if (barMode == TabBarMode::SCROLLABLE) {
313         if (info.Length() > 1 && info[1]->IsObject()) {
314             SetScrollableBarModeOptions(info[1]);
315         } else {
316             ScrollableBarModeOptions option;
317             TabsModel::GetInstance()->SetScrollableBarModeOptions(option);
318         }
319     }
320     TabsModel::GetInstance()->SetTabBarMode(barMode);
321 }
322 
SetBarWidth(const JSCallbackInfo & info)323 void JSTabs::SetBarWidth(const JSCallbackInfo& info)
324 {
325     if (info.Length() < 1) {
326         return;
327     }
328 
329     CalcDimension width = Dimension(-1.0, DimensionUnit::VP);
330     if (Container::GreatOrEqualAPIVersion(PlatformVersion::VERSION_TEN)) {
331         if (!ParseJsDimensionVpNG(info[0], width)) {
332             width = Dimension(-1.0, DimensionUnit::VP);
333             TabsModel::GetInstance()->SetTabBarWidth(width);
334             return;
335         }
336     } else {
337         ParseJsDimensionVp(info[0], width);
338     }
339 
340     TabsModel::GetInstance()->SetTabBarWidth(width);
341 }
342 
SetBarHeight(const JSCallbackInfo & info)343 void JSTabs::SetBarHeight(const JSCallbackInfo& info)
344 {
345     if (info.Length() < 1) {
346         return;
347     }
348     CalcDimension height = Dimension(-1.0, DimensionUnit::VP);
349     bool adaptiveHeight = false;
350     auto barHeightInfo = info[0];
351     if (barHeightInfo->IsString() && barHeightInfo->ToString() == "auto") {
352         adaptiveHeight = true;
353     } else {
354         if (Container::GreatOrEqualAPIVersion(PlatformVersion::VERSION_TEN)) {
355             if (!ParseJsDimensionVpNG(barHeightInfo, height)) {
356                 height = Dimension(-1.0, DimensionUnit::VP);
357             }
358         } else {
359             ParseJsDimensionVp(barHeightInfo, height);
360         }
361     }
362     TabsModel::GetInstance()->SetBarAdaptiveHeight(adaptiveHeight);
363     TabsModel::GetInstance()->SetTabBarHeight(height);
364 }
365 
SetWidth(const JSCallbackInfo & info)366 void JSTabs::SetWidth(const JSCallbackInfo& info)
367 {
368     JSViewAbstract::JsWidth(info);
369     if (info.Length() < 1) {
370         return;
371     }
372     if (info[0]->IsString() && info[0]->ToString().empty()) {
373         return;
374     }
375     if (info[0]->IsString() && info[0]->ToString() == "auto") {
376         ViewAbstractModel::GetInstance()->ClearWidthOrHeight(true);
377         TabsModel::GetInstance()->SetWidthAuto(true);
378         return;
379     }
380 
381     TabsModel::GetInstance()->SetWidthAuto(false);
382 }
383 
SetHeight(const JSCallbackInfo & info)384 void JSTabs::SetHeight(const JSCallbackInfo& info)
385 {
386     JSViewAbstract::JsHeight(info);
387     if (info.Length() < 1) {
388         return;
389     }
390     if (info[0]->IsString() && info[0]->ToString().empty()) {
391         return;
392     }
393     if (info[0]->IsString() && info[0]->ToString() == "auto") {
394         ViewAbstractModel::GetInstance()->ClearWidthOrHeight(false);
395         TabsModel::GetInstance()->SetHeightAuto(true);
396         return;
397     }
398 
399     TabsModel::GetInstance()->SetHeightAuto(false);
400 }
401 
SetIndex(int32_t index)402 void JSTabs::SetIndex(int32_t index)
403 {
404     TabsModel::GetInstance()->SetIndex(index);
405 }
406 
SetAnimationDuration(const JSCallbackInfo & info)407 void JSTabs::SetAnimationDuration(const JSCallbackInfo& info)
408 {
409     if (info.Length() <= 0) {
410         TabsModel::GetInstance()->SetAnimationDuration(-1);
411         return;
412     }
413     auto animationDurationInfo = info[0];
414     if ((!animationDurationInfo->IsNull() && !animationDurationInfo->IsNumber()) ||
415         (animationDurationInfo->IsNull() && Container::GreatOrEqualAPIVersion(PlatformVersion::VERSION_ELEVEN))) {
416         TabsModel::GetInstance()->SetAnimationDuration(-1);
417         return;
418     }
419     auto value = animationDurationInfo->IsNumber() ? animationDurationInfo->ToNumber<int32_t>() : 0;
420     TabsModel::GetInstance()->SetAnimationDuration(value);
421 }
422 
SetFadingEdge(const JSCallbackInfo & info)423 void JSTabs::SetFadingEdge(const JSCallbackInfo& info)
424 {
425     bool fadingEdge = true;
426     if (info.Length() > 0) {
427         ParseJsBool(info[0], fadingEdge);
428     }
429     TabsModel::GetInstance()->SetFadingEdge(fadingEdge);
430 }
431 
SetBarOverlap(const JSCallbackInfo & info)432 void JSTabs::SetBarOverlap(const JSCallbackInfo& info)
433 {
434     bool barOverlap = false;
435     if (info.Length() > 0) {
436         ParseJsBool(info[0], barOverlap);
437     }
438     TabsModel::GetInstance()->SetBarOverlap(barOverlap);
439 }
440 
SetBarBackgroundColor(const JSCallbackInfo & info)441 void JSTabs::SetBarBackgroundColor(const JSCallbackInfo& info)
442 {
443     Color backgroundColor = Color::BLACK.BlendOpacity(0.0f);
444     if (info.Length() > 0) {
445         ConvertFromJSValue(info[0], backgroundColor);
446     }
447     TabsModel::GetInstance()->SetBarBackgroundColor(backgroundColor);
448 }
449 
SetBarBackgroundBlurStyle(const JSCallbackInfo & info)450 void JSTabs::SetBarBackgroundBlurStyle(const JSCallbackInfo& info)
451 {
452     BlurStyle blurStyle = BlurStyle::NO_MATERIAL;
453     if (info.Length() > 0 && info[0]->IsNumber()) {
454         auto barBlurStyle = info[0]->ToNumber<int32_t>();
455         if (barBlurStyle >= 0 && barBlurStyle < static_cast<int32_t>(BAR_BLURSTYLE.size())) {
456             blurStyle = BAR_BLURSTYLE[barBlurStyle];
457         }
458     }
459     TabsModel::GetInstance()->SetBarBackgroundBlurStyle(blurStyle);
460 }
461 
SetDivider(const JSCallbackInfo & info)462 void JSTabs::SetDivider(const JSCallbackInfo& info)
463 {
464     TabsItemDivider divider;
465     CalcDimension dividerStrokeWidth;
466     CalcDimension dividerStartMargin;
467     CalcDimension dividerEndMargin;
468     RefPtr<TabTheme> tabTheme = GetTheme<TabTheme>();
469     CHECK_NULL_VOID(tabTheme);
470 
471     if (info.Length() > 0) {
472         auto dividerInfo = info[0];
473         JSRef<JSObject> obj = JSRef<JSObject>::New();
474         if (dividerInfo->IsObject()) {
475             obj = JSRef<JSObject>::Cast(dividerInfo);
476         }
477         if (dividerInfo->IsNull()) {
478             divider.isNull = true;
479         } else {
480             if (!dividerInfo->IsObject() || !ParseJsDimensionVp(obj->GetProperty("strokeWidth"), dividerStrokeWidth) ||
481                 dividerStrokeWidth.Value() < 0.0f || dividerStrokeWidth.Unit() == DimensionUnit::PERCENT) {
482                 divider.strokeWidth.Reset();
483             } else {
484                 divider.strokeWidth = dividerStrokeWidth;
485             }
486             if (!dividerInfo->IsObject() || !ConvertFromJSValue(obj->GetProperty("color"), divider.color)) {
487                 divider.color = tabTheme->GetDividerColor();
488             }
489             if (!dividerInfo->IsObject() || !ParseJsDimensionVp(obj->GetProperty("startMargin"), dividerStartMargin) ||
490                 dividerStartMargin.Value() < 0.0f || dividerStartMargin.Unit() == DimensionUnit::PERCENT) {
491                 divider.startMargin.Reset();
492             } else {
493                 divider.startMargin = dividerStartMargin;
494             }
495             if (!dividerInfo->IsObject() || !ParseJsDimensionVp(obj->GetProperty("endMargin"), dividerEndMargin) ||
496                 dividerEndMargin.Value() < 0.0f || dividerEndMargin.Unit() == DimensionUnit::PERCENT) {
497                 divider.endMargin.Reset();
498             } else {
499                 divider.endMargin = dividerEndMargin;
500             }
501         }
502     }
503     TabsModel::GetInstance()->SetDivider(divider);
504 }
505 
SetClip(const JSCallbackInfo & info)506 void JSTabs::SetClip(const JSCallbackInfo& info)
507 {
508     if (info[0]->IsObject() || !Container::IsCurrentUseNewPipeline()) {
509         JSViewAbstract::JsClip(info);
510         return;
511     }
512     if (info[0]->IsBoolean()) {
513         TabsModel::GetInstance()->SetClipEdge(info[0]->ToBoolean());
514     }
515 }
516 
SetScrollableBarModeOptions(const JSRef<JSVal> & info)517 void JSTabs::SetScrollableBarModeOptions(const JSRef<JSVal>& info)
518 {
519     ScrollableBarModeOptions option;
520     auto optionParam = JSRef<JSObject>::Cast(info);
521     CalcDimension margin = Dimension(0.0, DimensionUnit::VP);
522     if (!ParseJsDimensionVp(optionParam->GetProperty("margin"), margin) || Negative(margin.Value()) ||
523         margin.Unit() == DimensionUnit::PERCENT) {
524         option.margin = 0.0_vp;
525     } else {
526         option.margin = margin;
527     }
528 
529     auto nonScrollableLayoutStyle = optionParam->GetProperty("nonScrollableLayoutStyle");
530     int32_t layoutStyle;
531     if (!ConvertFromJSValue(nonScrollableLayoutStyle, layoutStyle) ||
532         layoutStyle < static_cast<int32_t>(LayoutStyle::ALWAYS_CENTER) ||
533         layoutStyle > static_cast<int32_t>(LayoutStyle::SPACE_BETWEEN_OR_CENTER)) {
534         option.nonScrollableLayoutStyle = std::nullopt;
535     } else {
536         option.nonScrollableLayoutStyle = (static_cast<LayoutStyle>(layoutStyle));
537     }
538     TabsModel::GetInstance()->SetScrollableBarModeOptions(option);
539 }
540 
SetBarGridAlign(const JSCallbackInfo & info)541 void JSTabs::SetBarGridAlign(const JSCallbackInfo& info)
542 {
543     BarGridColumnOptions columnOption;
544     if (info.Length() > 0 && info[0]->IsObject()) {
545         auto gridParam = JSRef<JSObject>::Cast(info[0]);
546         auto sm = gridParam->GetProperty("sm");
547         if (sm->IsNumber() && sm->ToNumber<int32_t>() >= 0 && sm->ToNumber<int32_t>() <= SM_COLUMN_NUM &&
548             sm->ToNumber<int32_t>() % 2 == 0) {
549             columnOption.sm = sm->ToNumber<int32_t>();
550         }
551         auto md = gridParam->GetProperty("md");
552         if (md->IsNumber() && md->ToNumber<int32_t>() >= 0 && md->ToNumber<int32_t>() <= MD_COLUMN_NUM &&
553             md->ToNumber<int32_t>() % 2 == 0) {
554             columnOption.md = md->ToNumber<int32_t>();
555         }
556         auto lg = gridParam->GetProperty("lg");
557         if (lg->IsNumber() && lg->ToNumber<int32_t>() >= 0 && lg->ToNumber<int32_t>() <= LG_COLUMN_NUM &&
558             lg->ToNumber<int32_t>() % 2 == 0) {
559             columnOption.lg = lg->ToNumber<int32_t>();
560         }
561         CalcDimension columnGutter;
562         if (ParseJsDimensionVp(gridParam->GetProperty("gutter"), columnGutter) && NonNegative(columnGutter.Value()) &&
563             columnGutter.Unit() != DimensionUnit::PERCENT) {
564             columnOption.gutter = columnGutter;
565         }
566         CalcDimension columnMargin;
567         if (ParseJsDimensionVp(gridParam->GetProperty("margin"), columnMargin) && NonNegative(columnMargin.Value()) &&
568             columnMargin.Unit() != DimensionUnit::PERCENT) {
569             columnOption.margin = columnMargin;
570         }
571     }
572     TabsModel::GetInstance()->SetBarGridAlign(columnOption);
573 }
574 
SetCustomContentTransition(const JSCallbackInfo & info)575 void JSTabs::SetCustomContentTransition(const JSCallbackInfo& info)
576 {
577     if (info.Length() != 1) {
578         return;
579     }
580 
581     auto customContentTransitionInfo = info[0];
582     if (customContentTransitionInfo->IsUndefined() || !customContentTransitionInfo->IsFunction()) {
583         TabsModel::GetInstance()->SetIsCustomAnimation(false);
584         return;
585     }
586 
587     RefPtr<JsTabsFunction> jsCustomAnimationFunc =
588         AceType::MakeRefPtr<JsTabsFunction>(JSRef<JSFunc>::Cast(customContentTransitionInfo));
589     auto onCustomAnimation = [execCtx = info.GetExecutionContext(), func = std::move(jsCustomAnimationFunc)](
590                                  int32_t from, int32_t to) -> TabContentAnimatedTransition {
591         TabContentAnimatedTransition transitionInfo;
592         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx, transitionInfo);
593 
594         auto ret = func->Execute(from, to);
595         if (!ret->IsObject()) {
596             return transitionInfo;
597         }
598 
599         auto transitionObj = JSRef<JSObject>::Cast(ret);
600         JSRef<JSVal> timeoutProperty = transitionObj->GetProperty("timeout");
601         if (timeoutProperty->IsNumber()) {
602             auto timeout = timeoutProperty->ToNumber<int32_t>();
603             transitionInfo.timeout = timeout < 0 ? DEFAULT_CUSTOM_ANIMATION_TIMEOUT : timeout;
604         } else {
605             transitionInfo.timeout = DEFAULT_CUSTOM_ANIMATION_TIMEOUT;
606         }
607 
608         JSRef<JSVal> transition = transitionObj->GetProperty("transition");
609         if (transition->IsFunction()) {
610             RefPtr<JsTabsFunction> jsOnTransition =
611                 AceType::MakeRefPtr<JsTabsFunction>(JSRef<JSFunc>::Cast(transition));
612             auto onTransition = [execCtx, func = std::move(jsOnTransition)](
613                                     const RefPtr<TabContentTransitionProxy>& proxy) {
614                 JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
615                 ACE_SCORING_EVENT("onTransition");
616                 func->Execute(proxy);
617             };
618 
619             transitionInfo.transition = std::move(onTransition);
620         }
621 
622         return transitionInfo;
623     };
624     TabsModel::GetInstance()->SetIsCustomAnimation(true);
625     TabsModel::GetInstance()->SetOnCustomAnimation(std::move(onCustomAnimation));
626 }
627 
SetOnContentWillChange(const JSCallbackInfo & info)628 void JSTabs::SetOnContentWillChange(const JSCallbackInfo& info)
629 {
630     if (!info[0]->IsFunction()) {
631         return;
632     }
633 
634     auto handler = AceType::MakeRefPtr<JsTabsFunction>(JSRef<JSFunc>::Cast(info[0]));
635     auto callback = [execCtx = info.GetExecutionContext(), func = std::move(handler)]
636         (int32_t currentIndex, int32_t comingIndex) -> bool {
637         JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx, true);
638         ACE_SCORING_EVENT("Tabs.onContentWillChange");
639         auto ret = func->Execute(currentIndex, comingIndex);
640         if (!ret->IsBoolean()) {
641             return true;
642         }
643         return ret->ToBoolean();
644     };
645     TabsModel::GetInstance()->SetOnContentWillChange(std::move(callback));
646 }
647 
SetAnimateMode(const JSCallbackInfo & info)648 void JSTabs::SetAnimateMode(const JSCallbackInfo& info)
649 {
650     JSRef<JSVal> args = info[0];
651     if (!args->IsNumber()) {
652         TabsModel::GetInstance()->SetAnimateMode(TabAnimateMode::CONTENT_FIRST);
653         return;
654     }
655     uint32_t value = args->ToNumber<uint32_t>();
656     if (value >= static_cast<uint32_t>(TabAnimateMode::MAX_VALUE)) {
657         TabsModel::GetInstance()->SetAnimateMode(TabAnimateMode::CONTENT_FIRST);
658         return;
659     }
660     TabsModel::GetInstance()->SetAnimateMode(static_cast<TabAnimateMode>(value));
661 }
662 
SetEdgeEffect(const JSCallbackInfo & info)663 void JSTabs::SetEdgeEffect(const JSCallbackInfo& info)
664 {
665     auto edgeEffect = EdgeEffect::SPRING;
666     if (info.Length() > 0) {
667         edgeEffect = JSScrollable::ParseEdgeEffect(info[0], EdgeEffect::SPRING);
668     }
669     TabsModel::GetInstance()->SetEdgeEffect(edgeEffect);
670 }
671 
SetPageFlipMode(const JSCallbackInfo & info)672 void JSTabs::SetPageFlipMode(const JSCallbackInfo& info)
673 {
674     // default value
675     int32_t value = 0;
676     if (info.Length() < 1 || !info[0]->IsNumber()) {
677         TabsModel::GetInstance()->SetPageFlipMode(value);
678         return;
679     }
680     JSViewAbstract::ParseJsInt32(info[0], value);
681     TabsModel::GetInstance()->SetPageFlipMode(value);
682 }
683 
SetBarModifier(const JSCallbackInfo & info,const JsiRef<JsiValue> & jsValue)684 void JSTabs::SetBarModifier(const JSCallbackInfo& info, const JsiRef<JsiValue>& jsValue)
685 {
686     if (!jsValue->IsObject()) {
687         return;
688     }
689     JSRef<JSObject> obj = JSRef<JSObject>::Cast(jsValue);
690     JSRef<JSVal> val = obj->GetProperty("barModifier");
691     if (!val->IsObject()) {
692         return;
693     }
694     JSRef<JSObject> modifierObj = JSRef<JSObject>::Cast(val);
695     auto vm = info.GetVm();
696     auto globalObj = JSNApi::GetGlobalObject(vm);
697     auto globalFunc = globalObj->Get(vm, panda::StringRef::NewFromUtf8(vm, "applyCommonModifierToNode"));
698     JsiValue jsiValue(globalFunc);
699     JsiRef<JsiValue> globalFuncRef = JsiRef<JsiValue>::Make(jsiValue);
700     std::function<void(WeakPtr<NG::FrameNode>)> onApply = nullptr;
701     if (globalFuncRef->IsFunction()) {
702         RefPtr<JsFunction> jsFunc =
703             AceType::MakeRefPtr<JsFunction>(JSRef<JSObject>(), JSRef<JSFunc>::Cast(globalFuncRef));
704         onApply = [execCtx = info.GetExecutionContext(), func = std::move(jsFunc),
705                       modifierParam = std::move(modifierObj)](WeakPtr<NG::FrameNode> frameNode) {
706             JAVASCRIPT_EXECUTION_SCOPE_WITH_CHECK(execCtx);
707             CHECK_NULL_VOID(func);
708             auto node = frameNode.Upgrade();
709             CHECK_NULL_VOID(node);
710             JSRef<JSVal> params[2];
711             params[0] = modifierParam;
712             params[1] = JSRef<JSVal>::Make(panda::NativePointerRef::New(execCtx.vm_, AceType::RawPtr(node)));
713             PipelineContext::SetCallBackNode(node);
714             func->ExecuteJS(2, params);
715         };
716     }
717     TabsModel::GetInstance()->SetBarModifier(std::move(onApply));
718 }
719 
JSBind(BindingTarget globalObj)720 void JSTabs::JSBind(BindingTarget globalObj)
721 {
722     JsTabContentTransitionProxy::JSBind(globalObj);
723     JSClass<JSTabs>::Declare("Tabs");
724     JSClass<JSTabs>::StaticMethod("create", &JSTabs::Create);
725     JSClass<JSTabs>::StaticMethod("pop", &JSTabs::Pop);
726     JSClass<JSTabs>::StaticMethod("vertical", &JSTabs::SetVertical);
727     JSClass<JSTabs>::StaticMethod("barPosition", &JSTabs::SetBarPosition);
728     JSClass<JSTabs>::StaticMethod("barBackgroundBlurStyle", &JSTabs::SetBarBackgroundBlurStyle);
729     JSClass<JSTabs>::StaticMethod("scrollable", &JSTabs::SetScrollable);
730     JSClass<JSTabs>::StaticMethod("barMode", &JSTabs::SetBarMode);
731     JSClass<JSTabs>::StaticMethod("barWidth", &JSTabs::SetBarWidth);
732     JSClass<JSTabs>::StaticMethod("barHeight", &JSTabs::SetBarHeight);
733     JSClass<JSTabs>::StaticMethod("width", &JSTabs::SetWidth);
734     JSClass<JSTabs>::StaticMethod("height", &JSTabs::SetHeight);
735     JSClass<JSTabs>::StaticMethod("index", &JSTabs::SetIndex);
736     JSClass<JSTabs>::StaticMethod("animationDuration", &JSTabs::SetAnimationDuration);
737     JSClass<JSTabs>::StaticMethod("divider", &JSTabs::SetDivider);
738     JSClass<JSTabs>::StaticMethod("onChange", &JSTabs::SetOnChange);
739     JSClass<JSTabs>::StaticMethod("onTabBarClick", &JSTabs::SetOnTabBarClick);
740     JSClass<JSTabs>::StaticMethod("onAnimationStart", &JSTabs::SetOnAnimationStart);
741     JSClass<JSTabs>::StaticMethod("onAnimationEnd", &JSTabs::SetOnAnimationEnd);
742     JSClass<JSTabs>::StaticMethod("onGestureSwipe", &JSTabs::SetOnGestureSwipe);
743     JSClass<JSTabs>::StaticMethod("onAttach", &JSInteractableView::JsOnAttach);
744     JSClass<JSTabs>::StaticMethod("onAppear", &JSInteractableView::JsOnAppear);
745     JSClass<JSTabs>::StaticMethod("onDetach", &JSInteractableView::JsOnDetach);
746     JSClass<JSTabs>::StaticMethod("onDisAppear", &JSInteractableView::JsOnDisAppear);
747     JSClass<JSTabs>::StaticMethod("onTouch", &JSInteractableView::JsOnTouch);
748     JSClass<JSTabs>::StaticMethod("onHover", &JSInteractableView::JsOnHover);
749     JSClass<JSTabs>::StaticMethod("onKeyEvent", &JSInteractableView::JsOnKey);
750     JSClass<JSTabs>::StaticMethod("onDeleteEvent", &JSInteractableView::JsOnDelete);
751     JSClass<JSTabs>::StaticMethod("onClick", &JSInteractableView::JsOnClick);
752     JSClass<JSTabs>::StaticMethod("remoteMessage", &JSInteractableView::JsCommonRemoteMessage);
753     JSClass<JSTabs>::StaticMethod("fadingEdge", &JSTabs::SetFadingEdge);
754     JSClass<JSTabs>::StaticMethod("barOverlap", &JSTabs::SetBarOverlap);
755     JSClass<JSTabs>::StaticMethod("barBackgroundColor", &JSTabs::SetBarBackgroundColor);
756     JSClass<JSTabs>::StaticMethod("clip", &JSTabs::SetClip);
757     JSClass<JSTabs>::StaticMethod("barGridAlign", &JSTabs::SetBarGridAlign);
758     JSClass<JSTabs>::StaticMethod("customContentTransition", &JSTabs::SetCustomContentTransition);
759     JSClass<JSTabs>::StaticMethod("onContentWillChange", &JSTabs::SetOnContentWillChange);
760     JSClass<JSTabs>::StaticMethod("animationMode", &JSTabs::SetAnimateMode);
761     JSClass<JSTabs>::StaticMethod("edgeEffect", &JSTabs::SetEdgeEffect);
762     JSClass<JSTabs>::StaticMethod("pageFlipMode", &JSTabs::SetPageFlipMode);
763 
764     JSClass<JSTabs>::InheritAndBind<JSContainerBase>(globalObj);
765 }
766 
767 } // namespace OHOS::Ace::Framework
768