diff --git a/scenario/arkui/MyNews/.gitignore b/scenario/arkui/MyNews/.gitignore
new file mode 100644
index 0000000000000000000000000000000000000000..fbabf771011fe78f9919db0b1195ab6cadffc2b0
--- /dev/null
+++ b/scenario/arkui/MyNews/.gitignore
@@ -0,0 +1,11 @@
+/node_modules
+/oh_modules
+/local.properties
+/.idea
+**/build
+/.hvigor
+.cxx
+/.clangd
+/.clang-format
+/.clang-tidy
+**/.test
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/AppScope/app.json5 b/scenario/arkui/MyNews/AppScope/app.json5
new file mode 100644
index 0000000000000000000000000000000000000000..0e60244f181c2aeee23a7a0dbab44af59e66efa0
--- /dev/null
+++ b/scenario/arkui/MyNews/AppScope/app.json5
@@ -0,0 +1,10 @@
+{
+ "app": {
+ "bundleName": "com.samples.mynews",
+ "vendor": "samples",
+ "versionCode": 1000000,
+ "versionName": "1.0.0",
+ "icon": "$media:app_icon",
+ "label": "$string:app_name"
+ }
+}
diff --git a/scenario/arkui/MyNews/AppScope/resources/base/element/string.json b/scenario/arkui/MyNews/AppScope/resources/base/element/string.json
new file mode 100644
index 0000000000000000000000000000000000000000..1b539c04d79430480e1fb04c4b75180299ed6361
--- /dev/null
+++ b/scenario/arkui/MyNews/AppScope/resources/base/element/string.json
@@ -0,0 +1,8 @@
+{
+ "string": [
+ {
+ "name": "app_name",
+ "value": "MyNews"
+ }
+ ]
+}
diff --git a/scenario/arkui/MyNews/AppScope/resources/base/media/app_icon.png b/scenario/arkui/MyNews/AppScope/resources/base/media/app_icon.png
new file mode 100644
index 0000000000000000000000000000000000000000..cd45accb1dfd2fd0da16c732c72faa6e46b26521
Binary files /dev/null and b/scenario/arkui/MyNews/AppScope/resources/base/media/app_icon.png differ
diff --git a/scenario/arkui/MyNews/README.md b/scenario/arkui/MyNews/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..398b0b5e5a97ca2925e0ab148f1f033d373f3e25
--- /dev/null
+++ b/scenario/arkui/MyNews/README.md
@@ -0,0 +1,12 @@
+# 我的新闻
+
+## 介绍
+
+本示例主要模拟主流新闻资讯应用,使用 ArkUI 的组件实现应用的布局、动效等,复制应用的界面及交互,以此测试 ArkUI
+是否足够支持主流新闻资讯应用的 UX 实现,以及是否存在问题;
+
+## 效果预览
+
+
+
+
diff --git a/scenario/arkui/MyNews/build-profile.json5 b/scenario/arkui/MyNews/build-profile.json5
new file mode 100644
index 0000000000000000000000000000000000000000..d7ddeff76abfc278dac43de1845bfd0a194dce80
--- /dev/null
+++ b/scenario/arkui/MyNews/build-profile.json5
@@ -0,0 +1,36 @@
+{
+ "app": {
+ "signingConfigs": [],
+ "products": [
+ {
+ "name": "default",
+ "signingConfig": "default",
+ "compileSdkVersion": 10,
+ "compatibleSdkVersion": 10,
+ "runtimeOS": "OpenHarmony",
+ }
+ ],
+ "buildModeSet": [
+ {
+ "name": "debug",
+ },
+ {
+ "name": "release"
+ }
+ ]
+ },
+ "modules": [
+ {
+ "name": "entry",
+ "srcPath": "./entry",
+ "targets": [
+ {
+ "name": "default",
+ "applyToProducts": [
+ "default"
+ ]
+ }
+ ]
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/.gitignore b/scenario/arkui/MyNews/entry/.gitignore
new file mode 100644
index 0000000000000000000000000000000000000000..e2713a2779c5a3e0eb879efe6115455592caeea5
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/.gitignore
@@ -0,0 +1,6 @@
+/node_modules
+/oh_modules
+/.preview
+/build
+/.cxx
+/.test
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/build-profile.json5 b/scenario/arkui/MyNews/entry/build-profile.json5
new file mode 100644
index 0000000000000000000000000000000000000000..197e83c1b9504b12a3837e0189185479662cad4b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/build-profile.json5
@@ -0,0 +1,31 @@
+{
+ "apiType": "stageMode",
+ "buildOption": {
+ "arkOptions": {
+ // "apPath": "./modules.ap" /* Profile used for profile-guided optimization (PGO), a compiler optimization technique to improve app runtime performance. */
+ }
+ },
+ "buildOptionSet": [
+ {
+ "name": "release",
+ "arkOptions": {
+ "obfuscation": {
+ "ruleOptions": {
+ "enable": true,
+ "files": [
+ "./obfuscation-rules.txt"
+ ]
+ }
+ }
+ }
+ },
+ ],
+ "targets": [
+ {
+ "name": "default"
+ },
+ {
+ "name": "ohosTest",
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/hvigorfile.ts b/scenario/arkui/MyNews/entry/hvigorfile.ts
new file mode 100644
index 0000000000000000000000000000000000000000..60e47e178e9702631eaf537cd83a6c400fbbb2da
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/hvigorfile.ts
@@ -0,0 +1,6 @@
+import { hapTasks } from '@ohos/hvigor-ohos-plugin';
+
+export default {
+ system: hapTasks, /* Built-in plugin of Hvigor. It cannot be modified. */
+ plugins: [] /* Custom plugin to extend the functionality of Hvigor. */
+}
diff --git a/scenario/arkui/MyNews/entry/obfuscation-rules.txt b/scenario/arkui/MyNews/entry/obfuscation-rules.txt
new file mode 100644
index 0000000000000000000000000000000000000000..985b2aeb7658286b17bd26eab8f217c3fe75ea8b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/obfuscation-rules.txt
@@ -0,0 +1,18 @@
+# Define project specific obfuscation rules here.
+# You can include the obfuscation configuration files in the current module's build-profile.json5.
+#
+# For more details, see
+# https://gitee.com/openharmony/arkcompiler_ets_frontend/blob/master/arkguard/README.md
+
+# Obfuscation options:
+# -disable-obfuscation: disable all obfuscations
+# -enable-property-obfuscation: obfuscate the property names
+# -enable-toplevel-obfuscation: obfuscate the names in the global scope
+# -compact: remove unnecessary blank spaces and all line feeds
+# -remove-log: remove all console.* statements
+# -print-namecache: print the name cache that contains the mapping from the old names to new names
+# -apply-namecache: reuse the given cache file
+
+# Keep options:
+# -keep-property-name: specifies property names that you want to keep
+# -keep-global-name: specifies names that you want to keep in the global scope
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/oh-package.json5 b/scenario/arkui/MyNews/entry/oh-package.json5
new file mode 100644
index 0000000000000000000000000000000000000000..225946cb11a2c405c8dc81eea89c22f923556638
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/oh-package.json5
@@ -0,0 +1,10 @@
+{
+ "license": "",
+ "devDependencies": {},
+ "author": "",
+ "name": "entry",
+ "description": "Please describe the basic information.",
+ "main": "",
+ "version": "1.0.0",
+ "dependencies": {}
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/entryability/EntryAbility.ets b/scenario/arkui/MyNews/entry/src/main/ets/entryability/EntryAbility.ets
new file mode 100644
index 0000000000000000000000000000000000000000..1c7aa63a9f838b1585caac34ae70a88054489797
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/entryability/EntryAbility.ets
@@ -0,0 +1,43 @@
+import AbilityConstant from '@ohos.app.ability.AbilityConstant';
+import hilog from '@ohos.hilog';
+import UIAbility from '@ohos.app.ability.UIAbility';
+import Want from '@ohos.app.ability.Want';
+import window from '@ohos.window';
+
+export default class EntryAbility extends UIAbility {
+ onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onCreate');
+ }
+
+ onDestroy(): void {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onDestroy');
+ }
+
+ onWindowStageCreate(windowStage: window.WindowStage): void {
+ // Main window is created, set main page for this ability
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate');
+
+ windowStage.loadContent('pages/Index', (err, data) => {
+ if (err.code) {
+ hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
+ return;
+ }
+ hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s', JSON.stringify(data) ?? '');
+ });
+ }
+
+ onWindowStageDestroy(): void {
+ // Main window is destroyed, release UI related resources
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageDestroy');
+ }
+
+ onForeground(): void {
+ // Ability has brought to foreground
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onForeground');
+ }
+
+ onBackground(): void {
+ // Ability has back to background
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onBackground');
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/pages/Index.ets b/scenario/arkui/MyNews/entry/src/main/ets/pages/Index.ets
new file mode 100644
index 0000000000000000000000000000000000000000..c5e123b5b8a4109a282dcc7b144d6a864a090b96
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/pages/Index.ets
@@ -0,0 +1,69 @@
+import { HomePage } from '../view/HomePage';
+import { ProfilePageBuilder } from '../view/profilePage';
+import { TaskPageBuilder } from '../view/TaskPage';
+import { VideoPageBuilder } from '../view/VideoPage';
+import { IndexBottomNavItem, indexBottomNavItemList } from '../viewmodel/IndexBottomNavItem';
+
+@Entry
+@Component
+struct Index {
+ @State currentIndex: number = 0
+ private tabsController: TabsController = new TabsController()
+
+ build() {
+ Tabs({ barPosition: BarPosition.End, controller: this.tabsController }) {
+ ForEach(indexBottomNavItemList, (item: IndexBottomNavItem, itemIndex: number) => {
+ TabContent() {
+ /* Provide different content according to the tab index. */
+ Column() {
+ if (itemIndex === 1) {
+ VideoPageBuilder()
+ }
+ else if (itemIndex === 2) {
+ TaskPageBuilder()
+ }
+ else if (itemIndex === 3) {
+ ProfilePageBuilder()
+ }
+ else {
+ HomePage()
+ }
+ }
+ .width('100%')
+ .height('100%')
+ .backgroundColor($r("app.color.light_bg"))
+ }
+ .width('100%')
+ .height('100%')
+ .align(Alignment.Top)
+ .tabBar(this.TabBuilder(item.title, itemIndex, item.normalIcon, item.selectedIcon))
+ })
+ }
+ .width('100%')
+ .height('100%')
+ .scrollable(false)
+ .backgroundColor($r('app.color.index_tab_bg'))
+ }
+
+ /**
+ * Custom buttons for the bottom navigation bar (icon + text).
+ */
+ @Builder
+ TabBuilder(title: string, targetIndex: number, normalIcon: Resource, selectedIcon: Resource) {
+ Column() {
+ Image(this.currentIndex === targetIndex ? selectedIcon : normalIcon)
+ .size({ width: 28, height: 28 })
+ .fillColor(this.currentIndex === targetIndex ? $r("app.color.primary") : $r('app.color.light_fg'))
+ Text(title)
+ .fontSize(11)
+ .fontColor(this.currentIndex === targetIndex ? $r("app.color.primary") : $r('app.color.light_fg'))
+ }
+ .width('100%')
+ .height('100%')
+ .justifyContent(FlexAlign.Center)
+ .onClick(() => {
+ this.tabsController.changeIndex(targetIndex)
+ this.currentIndex = targetIndex
+ })
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/HomePage.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePage.ets
new file mode 100644
index 0000000000000000000000000000000000000000..348abf1c8a177c505487b918788abce6fc6e8414
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePage.ets
@@ -0,0 +1,172 @@
+import router from '@ohos.router';
+import { HomePageTabItemList } from '../viewmodel/HomePageTabItem';
+import { FollowingTabContent } from './HomePageFollowingTab';
+import { RecommendationTabContent } from './HomePageRecommendationTab';
+
+
+/**
+ * Content in the `HomePage`.
+ * `首页` 整个页面的内容。
+ */
+@Component
+export struct HomePage {
+ @State currentTabIndex: number = 1
+
+ build() {
+ Column() {
+ /* Search box + publish button. */
+ TopBar()
+
+ /* Scrollable tabs + filter button */
+ TabSelector({ currentIndex: this.currentTabIndex })
+ Divider().color($r('app.color.light_border'))
+
+ /* Provide different content according to the tab index. */
+ Column() {
+ // TODO: add other tab contents.
+ if (this.currentTabIndex === 0) {
+ FollowingTabContent()
+ }
+ else if (this.currentTabIndex === 1) {
+ RecommendationTabContent()
+ }
+ else {
+ Text(this.currentTabIndex.toString())
+ }
+ }
+ .width('100%')
+ .height(0)
+ .flexGrow(1)
+ }
+ .width('100%')
+ .height('100%')
+ }
+}
+
+
+/**
+ * Top bar in the `HomePage`, with a search box and a publish button.
+ * `首页` 中的顶栏,包含一个搜索框和一个发布按钮。
+ */
+@Component
+struct TopBar {
+ build() {
+ Row({ space: 12 }) {
+ /* Search box */
+ Search()
+ .width(0)
+ .height('100%')
+ .margin(0)
+ .flexGrow(1)
+ .borderRadius(22)
+ .backgroundColor($r('app.color.primary_fg'))
+
+ /* Publish button */
+ Column() {
+ Image($r("app.media.ic_public_add_norm_filled"))
+ .size({ width: 28, height: 28 })
+ .fillColor($r("app.color.primary_fg"))
+ Text('发布').fontSize(12).fontColor($r("app.color.primary_fg"))
+ }
+ .height('100%')
+ .justifyContent(FlexAlign.SpaceBetween)
+ .onClick(() => {
+ /* Push the `PublishPage` into the page stack. */
+ router.pushUrl({ url: 'view/PublishPage' })
+ })
+ }
+ .width('100%')
+ .height(84)
+ .padding({ top: 33, bottom: 6, left: 17, right: 17 })
+ .alignItems(VerticalAlign.Bottom)
+ .justifyContent(FlexAlign.SpaceBetween)
+ .backgroundColor($r("app.color.primary"))
+ }
+}
+
+
+/**
+ * Tab selector in `HomePage`, for switching between tabs.
+ * `首页` 中的选项卡栏,用于切换当前 tab 内容。
+ */
+@Component
+struct TabSelector {
+ @Link currentIndex: number
+ private scroller: Scroller = new Scroller()
+
+ build() {
+ Row() {
+ Stack() {
+ Scroll(this.scroller) {
+ /* Scrollable tab selector. */
+ Row() {
+ ForEach(HomePageTabItemList, (item: string, itemIndex: number) => {
+ Column({ space: 6 }) {
+ Text(item)
+ .fontSize(19)
+ .fontWeight(this.currentIndex === itemIndex ? FontWeight.Bold : FontWeight.Normal)
+ .fontColor(this.currentIndex === itemIndex ? $r('app.color.primary') : $r('app.color.light_fg'))
+
+ /* The line under the selected tab. */
+ Line()
+ .startPoint([0, 0])
+ .endPoint([20, 0])
+ .stroke(this.currentIndex === itemIndex ? $r('app.color.primary') : Color.Transparent)
+ .strokeLineCap(LineCapStyle.Round)
+ .strokeWidth(3)
+ }
+ .width(58)
+ .height('100%')
+ .padding({ bottom: 1 })
+ .justifyContent(FlexAlign.End)
+ .onClick(() => {
+ this.currentIndex = itemIndex
+ })
+ })
+ }
+ .height('100%')
+ }
+ .width('100%')
+ .height('100%')
+ .padding(10)
+ .scrollable(ScrollDirection.Horizontal)
+ .scrollBar(BarState.Off)
+
+ /* Text fade-out effect. */
+ Row()
+ .width(30)
+ .height('100%')
+ .linearGradient({
+ angle: 90,
+ colors: [['#00FFFFFF', 0.0], ['#FFFFFFFF', 1.0]]
+ })
+ }
+ .width(0)
+ .height('100%')
+ .flexGrow(1)
+ .alignContent(Alignment.End)
+
+
+ /* Buttons on the right side. */
+ Row({ space: 8 }) {
+ Image($r("app.media.ic_public_view_list"))
+ .size({ width: 22, height: 22 })
+ .fillColor($r('app.color.light_fg'))
+
+ Divider()
+ .strokeWidth(2)
+ .height(14)
+ .vertical(true)
+ .color($r('app.color.light_border'))
+
+ Image($r("app.media.ic_device_earphone"))
+ .size({ width: 22, height: 22 })
+ .fillColor($r('app.color.light_fg'))
+ }
+ .height('100%')
+ .padding({ left: 4, right: 18 })
+ }
+ .width('100%')
+ .height(42)
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageFollowingTab.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageFollowingTab.ets
new file mode 100644
index 0000000000000000000000000000000000000000..3fddb9c7e126a3a701cf695160c636187d8d07bb
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageFollowingTab.ets
@@ -0,0 +1,166 @@
+import { recommendedUserList } from '../../mock/RecommendedUserList'
+import { UserInfo } from '../viewmodel/UserInfo'
+
+
+/**
+ * Content in `HomePage - FollowingTab`.
+ * `首页-关注` 中的所有内容。
+ */
+@Component
+export struct FollowingTabContent {
+ build() {
+ Column() {
+ RecommendedUsers()
+ }
+ }
+}
+
+
+/**
+ * Recommended users.
+ * 你可能感兴趣的人。
+ */
+@Component
+struct RecommendedUsers {
+ @State isVisible: boolean = true
+ @State localRecommendedUserList: UserInfo[] = recommendedUserList
+ private scroller: Scroller = new Scroller()
+
+ build() {
+ if (this.isVisible) {
+ Column({ space: 14 }) {
+ Row() {
+ // Subtitle
+ Text('你可能感兴趣的人')
+ .fontSize(18)
+ .fontWeight(FontWeight.Bold)
+ .fontColor($r('app.color.light_fg'))
+
+ // Close button
+ Image($r("app.media.ic_public_cancel"))
+ .size({ width: 18, height: 18 })
+ .fillColor($r('app.color.light_border'))
+ .onClick(() => {
+ /* Hide this component. */
+ // TODO: transition
+ this.isVisible = false
+ })
+ }
+ .width('100%')
+ .padding({ left: 18, right: 16 })
+ .justifyContent(FlexAlign.SpaceBetween)
+
+ // Scrollable user list
+ Scroll(this.scroller) {
+ Row({ space: 8 }) {
+ ForEach(this.localRecommendedUserList, (item: UserInfo) => {
+ UserInfoCard({ userInfo: item })
+ }, (item: UserInfo) => item.userId)
+ }
+ }
+ .width('100%')
+ .padding({ left: 22 })
+ .scrollable(ScrollDirection.Horizontal)
+ .scrollBar(BarState.Off)
+ .edgeEffect(EdgeEffect.Spring)
+ }
+ .width('100%')
+ .padding({ top: 18, bottom: 10 })
+ }
+ }
+}
+
+
+/**
+ * Recommended users information card.
+ * 推荐关注的用户展示卡片。
+ */
+@Component
+struct UserInfoCard {
+ @ObjectLink userInfo: UserInfo
+ @State isVisible: boolean = true
+
+ build() {
+ if (this.isVisible) {
+ Column() {
+ Column({ space: 6 }) {
+ Stack() {
+ // Close button
+ Row() {
+ Image($r("app.media.ic_public_cancel"))
+ .size({
+ width: 15,
+ height: 15
+ })
+ .fillColor($r('app.color.light_border'))
+ .onClick(() => {
+ /* Hide this user. */
+ // TODO: transition
+ this.isVisible = false
+ })
+ }
+ .width('100%')
+ .height('100%')
+ .margin({ top: 14, right: 14 })
+ .padding(0)
+ .justifyContent(FlexAlign.End)
+ .alignItems(VerticalAlign.Top)
+
+ // Avatar
+ Image(this.userInfo.avatar)
+ .margin({ top: 10 })
+ .size({ width: 60, height: 60 })
+ .borderRadius(30)
+ }
+ .width('100%')
+ .height('40%')
+
+ Column({ space: 5 }) {
+ // Username
+ Text(this.userInfo.username)
+ .fontSize(16)
+ .fontWeight(FontWeight.Medium)
+ .fontColor($r('app.color.light_fg'))
+ .textAlign(TextAlign.Center)
+
+ // Brief introduction
+ Text(this.userInfo.brief)
+ .fontSize(14)
+ .fontColor($r('app.color.light_fg_shallow'))
+ .textAlign(TextAlign.Center)
+ }
+ .width('90%')
+ .height('35%')
+ }
+
+ // Follow button
+ Button({ type: ButtonType.Normal }) {
+ Row() {
+ Text(this.userInfo.isFollowing ? '已关注' : '关注')
+ .fontSize(16)
+ .fontColor(this.userInfo.isFollowing
+ ? $r('app.color.light_fg_shallow')
+ : $r('app.color.primary_fg'))
+ }
+ }
+ .width(62)
+ .padding(6)
+ .backgroundColor(this.userInfo.isFollowing
+ ? Color.Transparent
+ : $r('app.color.primary'))
+ .borderWidth(this.userInfo.isFollowing ? 1 : 0)
+ .borderColor($r('app.color.light_border'))
+ .borderRadius(2)
+ .onClick(() => {
+ /* Follow the user. */
+ this.userInfo.isFollowing = !this.userInfo.isFollowing
+ })
+ }
+ .width(160)
+ .height(190)
+ .borderWidth(1)
+ .borderColor($r('app.color.light_border'))
+ .borderRadius(2)
+ }
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageRecommendationTab.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageRecommendationTab.ets
new file mode 100644
index 0000000000000000000000000000000000000000..b13bc169d31582c4bb148e9e12f0a2b35b252782
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/HomePageRecommendationTab.ets
@@ -0,0 +1,140 @@
+import { recommendedArticleList } from '../../mock/RecommendedArticleList'
+import { recommendedVideoList } from '../../mock/RecommendedVideoList'
+import { RecommendedArticle, RecommendedArticleOverview } from '../viewmodel/RecommendedArticle'
+import { RecommendedVideo, RecommendedVideoOverview } from '../viewmodel/RecommendedVideo'
+
+/**
+ * Content in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中的所有内容。
+ */
+@Component
+export struct RecommendationTabContent {
+ private scroller: Scroller = new Scroller()
+
+ build() {
+ List({ space: 14, scroller: this.scroller }) {
+ ListItemGroup({ space: 6 }) {
+ ForEach(recommendedArticleList, (item: RecommendedArticle) => {
+ ListItem() {
+ ArticleOverview({
+ isPinned: item.isPinned,
+ title: item.shortTitle,
+ author: item.author,
+ cmtCnt: item.cmtCnt
+ })
+ }.width('90%')
+ })
+ }
+
+ ListItem() {
+ Divider()
+ .width('90%')
+ .strokeWidth(1)
+ .color($r('app.color.light_border'))
+ }
+
+ ListItemGroup({ space: 14 }) {
+ ForEach(recommendedVideoList, (item: RecommendedVideo) => {
+ ListItem() {
+ VideoOverview({
+ title: item.shortTitle,
+ cover: item.cover,
+ duration: item.duration,
+ author: item.author,
+ cmtCnt: item.cmtCnt
+ })
+ }.width('90%')
+ })
+ }
+ }
+ .width('100%')
+ .height('100%')
+ .padding({ top: 12 })
+ .alignListItem(ListItemAlign.Center)
+ .backgroundColor($r("app.color.light_bg"))
+ }
+}
+
+
+/**
+ * Article overview in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中的文章概览。
+ */
+@Builder
+function ArticleOverview($$: RecommendedArticleOverview) {
+ Column() {
+ /* Article title. */
+ Text($$.title)
+ .fontSize(18)
+ .fontColor($r('app.color.light_fg'))
+
+ Row({ space: 5 }) {
+ /* Display '置顶' if the article is pinned. */
+ if ($$.isPinned) {
+ Text('置顶')
+ .fontSize(12)
+ .fontColor($r('app.color.primary'))
+ }
+
+ /* Article author. */
+ Text($$.author)
+ .fontSize(12)
+ .fontColor($r('app.color.light_fg_shallow'))
+
+ /* Article comment count. */
+ if ($$.cmtCnt > 0) {
+ Text(`${$$.cmtCnt}评论`)
+ .fontSize(12)
+ .fontColor($r('app.color.light_fg_shallow'))
+ }
+ }.width('100%')
+ }
+ .width('100%')
+ .alignItems(HorizontalAlign.Start)
+}
+
+
+/**
+ * Video overview in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中的视频概览。
+ */
+@Builder
+function VideoOverview($$: RecommendedVideoOverview) {
+
+ Column() {
+ Column({ space: 5 }) {
+ /* Video title. */
+ Text($$.title).fontSize(18)
+
+ /* Video cover. */
+ Stack() {
+ Image($$.cover)
+ .width('100%')
+ .objectFit(ImageFit.Contain)
+ .borderRadius(5)
+ Image($r("app.media.ic_public_play"))
+ .size({ width: 45, height: 45 })
+ .fillColor($r('app.color.primary_fg'))
+ .opacity(0.7)
+ }.alignContent(Alignment.Center)
+
+ Row({ space: 5 }) {
+ /* Video author. */
+ Text($$.author)
+ .fontSize(12)
+ .fontColor($r('app.color.light_fg_shallow'))
+
+ /* Video comment count. */
+ if ($$.cmtCnt > 0) {
+ Text(`${$$.cmtCnt}评论`)
+ .fontSize(12)
+ .fontColor($r('app.color.light_fg_shallow'))
+ }
+ }.width('100%')
+ }
+ .width('100%')
+ .alignItems(HorizontalAlign.Start)
+ }
+ .width('100%')
+ .alignItems(HorizontalAlign.Start)
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/ProfilePage.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/ProfilePage.ets
new file mode 100644
index 0000000000000000000000000000000000000000..d0b879c09e6d28a84766f7e8dd7a27af611b90eb
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/ProfilePage.ets
@@ -0,0 +1,7 @@
+// TODO
+@Builder
+export function ProfilePageBuilder() {
+ Column() {
+ Text('个人页面内容')
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/PublishPage.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/PublishPage.ets
new file mode 100644
index 0000000000000000000000000000000000000000..5d4c40722062150ff126aea3e7e016aeb27c244a
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/PublishPage.ets
@@ -0,0 +1,135 @@
+import router from '@ohos.router';
+
+
+/**
+ * All the items in the bottom tab bar.
+ * 页面底部所有选项卡。
+ */
+const BottomTabItemList: string[] = [
+ '灵感',
+ '文章',
+ '微头条',
+ '模板',
+ '视频',
+ '直播'
+]
+
+@Entry
+@Component
+struct PublishPage {
+ @State currentIndex: number = 2
+ private tabsController: TabsController = new TabsController()
+
+ build() {
+ Tabs({ barPosition: BarPosition.End, controller: this.tabsController }) {
+ ForEach(BottomTabItemList, (item: string, itemIndex: number) => {
+ TabContent() {
+ /* Provide different content according to the tab index. */
+ Column() {
+ MicroToutiaoTab()
+ }
+ .width('100%')
+ .height('100%')
+ .backgroundColor($r("app.color.light_bg"))
+ }
+ .width('100%')
+ .height('100%')
+ .align(Alignment.Top)
+ .tabBar(this.TabBuilder(item, itemIndex))
+ })
+ }
+ .width('100%')
+ .height('100%')
+ .barHeight(64)
+ .scrollable(false)
+ .barMode(BarMode.Scrollable)
+ .backgroundColor($r('app.color.light_fg'))
+ }
+
+ @Builder
+ TabBuilder(item: string, itemIndex: number) {
+ Column({ space: 9 }) {
+ Text(item)
+ .fontSize(18)
+ .fontColor($r('app.color.primary_fg'))
+ .opacity(0.6)
+
+ /* The line under the selected tab. */
+ Line()
+ .startPoint([0, 0])
+ .endPoint([20, 0])
+ .stroke(this.currentIndex === itemIndex ? $r('app.color.primary_fg') : Color.Transparent)
+ .strokeLineCap(LineCapStyle.Round)
+ .strokeWidth(3)
+ }
+ .height('100%')
+ .padding({ bottom: 13, left: 18, right: 18 })
+ .justifyContent(FlexAlign.End)
+ .onClick(() => {
+ this.tabsController.changeIndex(itemIndex)
+ this.currentIndex = itemIndex
+ })
+ }
+}
+
+
+@Component
+struct MicroToutiaoTab {
+ build() {
+ Column() {
+ /* Tool bar. */
+ Row() {
+ Text('取消')
+ .margin({ left: 20 })
+ .fontSize(18)
+ .fontColor($r('app.color.light_fg'))
+ .onClick(() => {
+ router.back()
+ })
+ Row({ space: 20 }) {
+ Text('预览')
+ .fontSize(18)
+ .fontColor($r('app.color.light_fg_shallow'))
+ Text('发布')
+ .fontSize(18)
+ .fontColor($r('app.color.light_fg_shallow'))
+ }
+ .margin({ right: 20 })
+ }
+ .width('100%')
+ .height(48)
+ .justifyContent(FlexAlign.SpaceBetween)
+
+ Divider().color($r('app.color.light_border'))
+
+ Column() {
+ TextArea({ placeholder: '说点什么' })
+ .width('100%')
+ .height(90)
+ .padding(0)
+ .borderRadius(0)
+ .backgroundColor(Color.Transparent)
+
+ /* Insert images. */
+ Row() {
+ Row() {
+ Image($r("app.media.ic_public_add"))
+ .size({ width: 60, height: 60 })
+ .fillColor('#D8D8D8')
+ }
+ .width(100)
+ .height(100)
+ .justifyContent(FlexAlign.Center)
+ .backgroundColor('#F2F2F2')
+ }
+ .width('100%')
+ .margin({ top: 15 })
+ }
+ .width('100%')
+ .padding(18)
+ }
+ .width('100%')
+ .height('100%')
+ .margin({ top: 28 })
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/TaskPage.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/TaskPage.ets
new file mode 100644
index 0000000000000000000000000000000000000000..2a2807ff9bed23a14f695fc00bf0c1a16237f284
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/TaskPage.ets
@@ -0,0 +1,7 @@
+// TODO
+@Builder
+export function TaskPageBuilder() {
+ Column() {
+ Text('任务页面内容')
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/view/VideoPage.ets b/scenario/arkui/MyNews/entry/src/main/ets/view/VideoPage.ets
new file mode 100644
index 0000000000000000000000000000000000000000..7fece423b1a52baf44acd9afb042c5ead00116e9
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/view/VideoPage.ets
@@ -0,0 +1,7 @@
+// TODO
+@Builder
+export function VideoPageBuilder() {
+ Column() {
+ Text('视频页面内容')
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/HomePageTabItem.ets b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/HomePageTabItem.ets
new file mode 100644
index 0000000000000000000000000000000000000000..22612007c53eae30232a97734cdb04a36ab6e1f7
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/HomePageTabItem.ets
@@ -0,0 +1,19 @@
+/**
+ * All the items of the tab bar in the `HomePage`.
+ * `首页` 中的所有选项卡。
+ */
+export const HomePageTabItemList: string[] = [
+ '关注',
+ '推荐',
+ '热榜',
+ '发现',
+ '视频',
+ '图片',
+ '娱乐',
+ '畅听',
+ '科技',
+ '懂车',
+ '财经',
+ '军事',
+ '体育'
+]
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/IndexBottomNavItem.ets b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/IndexBottomNavItem.ets
new file mode 100644
index 0000000000000000000000000000000000000000..5a2017c29d286ba5f5f046b773faf55c74bfcae0
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/IndexBottomNavItem.ets
@@ -0,0 +1,34 @@
+/**
+ * Type definition for the items in the bottom navigation bar.
+ * 应用底部导航栏中项的类型定义。
+ */
+export interface IndexBottomNavItem {
+ /* Text under the icon. */
+ title: string
+ /* Display outlined icons when not clicked. */
+ normalIcon: Resource
+ /* Display filled icons after clicked. */
+ selectedIcon: Resource
+}
+
+/**
+ * All the items in the bottom navigation bar.
+ * 应用底部导航栏中导航项列表。
+ */
+export const indexBottomNavItemList: IndexBottomNavItem[] = [{
+ title: '首页',
+ normalIcon: $r("app.media.ic_public_home"),
+ selectedIcon: $r("app.media.ic_public_home_filled")
+}, {
+ title: '视频',
+ normalIcon: $r("app.media.ic_public_video"),
+ selectedIcon: $r("app.media.ic_public_video_filled")
+}, {
+ title: '任务',
+ normalIcon: $r("app.media.ic_public_calendar"),
+ selectedIcon: $r("app.media.ic_public_calendar_filled")
+}, {
+ title: '个人',
+ normalIcon: $r("app.media.ic_public_contacts"),
+ selectedIcon: $r("app.media.ic_public_contacts_filled")
+}]
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedArticle.ets b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedArticle.ets
new file mode 100644
index 0000000000000000000000000000000000000000..4d9eab56fdfc42f0e1c04d3ebaf50fea3aa4dc43
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedArticle.ets
@@ -0,0 +1,29 @@
+/**
+ * Type definition for the articles in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中文章的类型定义。
+ */
+export interface RecommendedArticle {
+ isPinned: boolean
+ shortTitle: string
+ fullTitle: string
+ author: string
+ date: string
+ fwdCnt: number
+ cmtCnt: number
+ likeCnt: number
+ favCnt: number
+
+ // TODO
+}
+
+
+/**
+ * Type definition for the articles in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中所展示的文章概览组件的传入参数。
+ */
+export interface RecommendedArticleOverview {
+ isPinned: boolean
+ title: string
+ author: string
+ cmtCnt: number
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedVideo.ets b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedVideo.ets
new file mode 100644
index 0000000000000000000000000000000000000000..1aea2739e19aff146895d40bae59c30295cd7195
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/RecommendedVideo.ets
@@ -0,0 +1,31 @@
+/**
+ * Type definition for the videos in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中视频的类型定义。
+ */
+export interface RecommendedVideo {
+ shortTitle: string
+ fullTitle: string
+ author: string
+ date: string
+ cover: Resource
+ duration: number
+ fwdCnt: number
+ cmtCnt: number
+ likeCnt: number
+ favCnt: number
+
+ // TODO
+}
+
+
+/**
+ * Type definition for the articles in `HomePage - RecommendationTab`.
+ * `首页-推荐` 中所展示的视频概览组件的传入参数。
+ */
+export interface RecommendedVideoOverview {
+ title: string
+ cover: Resource
+ duration: number
+ author: string
+ cmtCnt: number
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/UserInfo.ets b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/UserInfo.ets
new file mode 100644
index 0000000000000000000000000000000000000000..6874acbe41a24f1d2d4270118cc29b1e9fdf4e71
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/ets/viewmodel/UserInfo.ets
@@ -0,0 +1,20 @@
+/**
+ * Type definition for user information.
+ * 用户信息的类型定义。
+ */
+@Observed
+export class UserInfo {
+ username: string
+ userId: string
+ brief: string
+ avatar: Resource
+ isFollowing: boolean // Is followed by current user?
+
+ constructor(username: string, userId: string, brief: string, avatar: Resource, isFollowing: boolean) {
+ this.username = username
+ this.userId = userId
+ this.brief = brief
+ this.avatar = avatar
+ this.isFollowing = isFollowing
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/main/mock/RecommendedArticleList.ets b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedArticleList.ets
new file mode 100644
index 0000000000000000000000000000000000000000..a5112904e3fd0b2fcfb451460857aeb5c51d0c6d
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedArticleList.ets
@@ -0,0 +1,33 @@
+import { RecommendedArticle } from '../ets/viewmodel/RecommendedArticle'
+
+export const recommendedArticleList: RecommendedArticle[] = [{
+ isPinned: true,
+ shortTitle: '聚首亚心之都,拥抱开源力量',
+ fullTitle: '“聚首亚心之都,拥抱开源力量”| 2023 OpenHarmony城市推介会•乌鲁木齐市成功举办',
+ author: 'OpenAtom OpenHarmony',
+ date: '2023-11-08 21:00',
+ fwdCnt: 152,
+ cmtCnt: 38,
+ likeCnt: 735,
+ favCnt: 62
+}, {
+ isPinned: true,
+ shortTitle: 'HarmonyOS语言基础类库开发指南上线啦!',
+ fullTitle: 'HarmonyOS语言基础类库开发指南上线啦!',
+ author: 'HarmonyOS开发者',
+ date: '2023-10-18 20:07',
+ fwdCnt: 93,
+ cmtCnt: 0,
+ likeCnt: 952,
+ favCnt: 14
+}, {
+ isPinned: false,
+ shortTitle: '第二届OpenHarmony技术大会精彩瞬间',
+ fullTitle: '高能回顾 | 第二届OpenHarmony技术大会精彩瞬间',
+ author: 'OpenAtom OpenHarmony',
+ date: '2023-11-07 21:00',
+ fwdCnt: 63,
+ cmtCnt: 94,
+ likeCnt: 1852,
+ favCnt: 54
+}]
diff --git a/scenario/arkui/MyNews/entry/src/main/mock/RecommendedUserList.ets b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedUserList.ets
new file mode 100644
index 0000000000000000000000000000000000000000..c1107b65a841c12dd3df5c9a072cca758708a741
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedUserList.ets
@@ -0,0 +1,10 @@
+import { UserInfo } from '../ets/viewmodel/UserInfo'
+
+export const recommendedUserList: UserInfo[] = [
+ new UserInfo('用户1', '001', '用户1的个人简介,用户1的个人简介', $r('app.media.avatar_1'), false),
+ new UserInfo('用户2', '002', '用户2的个人简介', $r('app.media.avatar_2'), false),
+ new UserInfo('用户3', '003', '用户3的个人简介', $r('app.media.avatar_3'), false),
+ new UserInfo('用户4', '004', '用户4的个人简介,用户4的个人简介', $r('app.media.avatar_4'), false),
+ new UserInfo('用户5', '005', '用户5的个人简介', $r('app.media.avatar_5'), false),
+ new UserInfo('用户6', '006', '用户6的个人简介', $r('app.media.avatar_6'), false)
+]
diff --git a/scenario/arkui/MyNews/entry/src/main/mock/RecommendedVideoList.ets b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedVideoList.ets
new file mode 100644
index 0000000000000000000000000000000000000000..0406090ef350e6ac757ed7644a5913f9f8e8d41b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/mock/RecommendedVideoList.ets
@@ -0,0 +1,25 @@
+import { RecommendedVideo } from '../ets/viewmodel/RecommendedVideo'
+
+export const recommendedVideoList: RecommendedVideo[] = [{
+ shortTitle: '打造高校生态链 OpenHarmony技术俱乐部分论坛召开',
+ fullTitle: '打造高校生态链 OpenHarmony技术俱乐部分论坛召开',
+ author: 'OpenAtom OpenHarmony',
+ date: '2023-11-07 21:00',
+ cover: $r("app.media.fig_1"),
+ duration: 362,
+ fwdCnt: 84,
+ cmtCnt: 275,
+ likeCnt: 1856,
+ favCnt: 52
+}, {
+ shortTitle: '首届OpenHarmony竞赛训练营结营颁奖',
+ fullTitle: '首届OpenHarmony竞赛训练营结营颁奖,75所高校学子助力建设开源生态',
+ author: 'OpenAtom OpenHarmony',
+ date: '2023-11-06 19:30',
+ cover: $r("app.media.fig_2"),
+ duration: 298,
+ fwdCnt: 22,
+ cmtCnt: 17,
+ likeCnt: 1362,
+ favCnt: 410
+}]
diff --git a/scenario/arkui/MyNews/entry/src/main/module.json5 b/scenario/arkui/MyNews/entry/src/main/module.json5
new file mode 100644
index 0000000000000000000000000000000000000000..f2ff2c7f12fb46a0656dd93857b72f7a1b057001
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/module.json5
@@ -0,0 +1,37 @@
+{
+ "module": {
+ "name": "entry",
+ "type": "entry",
+ "description": "$string:module_desc",
+ "mainElement": "EntryAbility",
+ "deviceTypes": [
+ "default",
+ "tablet"
+ ],
+ "deliveryWithInstall": true,
+ "installationFree": false,
+ "pages": "$profile:main_pages",
+ "abilities": [
+ {
+ "name": "EntryAbility",
+ "srcEntry": "./ets/entryability/EntryAbility.ets",
+ "description": "$string:EntryAbility_desc",
+ "icon": "$media:icon",
+ "label": "$string:EntryAbility_label",
+ "startWindowIcon": "$media:startIcon",
+ "startWindowBackground": "$color:start_window_background",
+ "exported": true,
+ "skills": [
+ {
+ "entities": [
+ "entity.system.home"
+ ],
+ "actions": [
+ "action.system.home"
+ ]
+ }
+ ]
+ }
+ ]
+ }
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/element/color.json b/scenario/arkui/MyNews/entry/src/main/resources/base/element/color.json
new file mode 100644
index 0000000000000000000000000000000000000000..fb6612af9407458f421a8e70fff8f2d69e33496d
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/element/color.json
@@ -0,0 +1,36 @@
+{
+ "color": [
+ {
+ "name": "start_window_background",
+ "value": "#FFFFFF"
+ },
+ {
+ "name": "primary",
+ "value": "#F54340"
+ },
+ {
+ "name": "primary_fg",
+ "value": "#FFFFFF"
+ },
+ {
+ "name": "light_bg",
+ "value": "#FFFFFF"
+ },
+ {
+ "name": "light_fg",
+ "value": "#222222"
+ },
+ {
+ "name": "light_fg_shallow",
+ "value": "#A4A4A4"
+ },
+ {
+ "name": "light_border",
+ "value": "#E8E8E8"
+ },
+ {
+ "name": "index_tab_bg",
+ "value": "#F8F8F8"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/element/string.json b/scenario/arkui/MyNews/entry/src/main/resources/base/element/string.json
new file mode 100644
index 0000000000000000000000000000000000000000..f94595515a99e0c828807e243494f57f09251930
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "module description"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "label"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_1.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_1.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..1702b684aa64318fcb886e892f42c2889b9a59d6
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_1.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_2.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_2.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..59f7c2d7be6e5ac54cab5d1ec4c237a893605d17
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_2.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_3.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_3.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..8e5b58e8c6d8e101fa1d9a6ce8fd3a63f8b9542b
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_3.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_4.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_4.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..32380875eadd8fb9a00bb7045ec7b5339d64bf93
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_4.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_5.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_5.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..bae35fc91d1fce373ae46da13999b00a9675ea41
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_5.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_6.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_6.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..9ee21df5d82cf2f678382caebb9617c7ddd203ae
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/avatar_6.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_1.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_1.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..173289b9aa7dee3a8b983db84a1b5d6d3e02fc84
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_1.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_2.jpg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_2.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..8b3b42bd916179b3157ae5a39769631b8de55eaa
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/fig_2.jpg differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_device_earphone.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_device_earphone.svg
new file mode 100644
index 0000000000000000000000000000000000000000..b138da29f372899f5069ce228a0eef7207de140b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_device_earphone.svg
@@ -0,0 +1,22 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add.svg
new file mode 100644
index 0000000000000000000000000000000000000000..7a9225d08425eeb35c70aa420497c56b0cbd4631
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add.svg
@@ -0,0 +1,30 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add_norm_filled.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add_norm_filled.svg
new file mode 100644
index 0000000000000000000000000000000000000000..9e78c6e9df522bbd87c32e3eff2ed7d2e09632f2
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_add_norm_filled.svg
@@ -0,0 +1,30 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar.svg
new file mode 100644
index 0000000000000000000000000000000000000000..162d749234c3c578468b57ee74f4196a657baa8a
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar_filled.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar_filled.svg
new file mode 100644
index 0000000000000000000000000000000000000000..30030b0e77357cbb2d406f4ac8720c2c82b11b7c
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_calendar_filled.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_cancel.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_cancel.svg
new file mode 100644
index 0000000000000000000000000000000000000000..dc70bebb51e6640f48ed009d40b91fc013645a8a
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_cancel.svg
@@ -0,0 +1,30 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts.svg
new file mode 100644
index 0000000000000000000000000000000000000000..ff3f3bd2338c8d48312ed03307ffffedab4b10e6
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts_filled.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts_filled.svg
new file mode 100644
index 0000000000000000000000000000000000000000..741f8cf8f9b016753308ea959e3f3c13f6a53752
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_contacts_filled.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home.svg
new file mode 100644
index 0000000000000000000000000000000000000000..444c565872c14fd55987539fa0a1eced4cbb5372
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home_filled.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home_filled.svg
new file mode 100644
index 0000000000000000000000000000000000000000..d25f5fb2d3c716d8e59a21e1db5fa7573d9c25f1
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_home_filled.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_play.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_play.svg
new file mode 100644
index 0000000000000000000000000000000000000000..96074424d17c6339c819b9ca17f42422cc11297a
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_play.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video.svg
new file mode 100644
index 0000000000000000000000000000000000000000..3649ddb027d178f61e5db3079253cc8cdc70a96b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video.svg
@@ -0,0 +1,22 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video_filled.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video_filled.svg
new file mode 100644
index 0000000000000000000000000000000000000000..b91995a902bc4a306686d34ca9b288f8708dd385
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_video_filled.svg
@@ -0,0 +1,21 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_view_list.svg b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_view_list.svg
new file mode 100644
index 0000000000000000000000000000000000000000..3dbb8f73f94728401917142b6508ebae62632df8
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/media/ic_public_view_list.svg
@@ -0,0 +1,31 @@
+
+
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/icon.png b/scenario/arkui/MyNews/entry/src/main/resources/base/media/icon.png
new file mode 100644
index 0000000000000000000000000000000000000000..cd45accb1dfd2fd0da16c732c72faa6e46b26521
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/icon.png differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/media/startIcon.png b/scenario/arkui/MyNews/entry/src/main/resources/base/media/startIcon.png
new file mode 100644
index 0000000000000000000000000000000000000000..366f76459ffd4494ec40d0ddd5c59385b9c5da11
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/main/resources/base/media/startIcon.png differ
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/base/profile/main_pages.json b/scenario/arkui/MyNews/entry/src/main/resources/base/profile/main_pages.json
new file mode 100644
index 0000000000000000000000000000000000000000..0b1c09c31fca7f3649cf2806829a02a8ce77a9e7
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/base/profile/main_pages.json
@@ -0,0 +1,6 @@
+{
+ "src": [
+ "pages/Index",
+ "view/PublishPage"
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/en_US/element/string.json b/scenario/arkui/MyNews/entry/src/main/resources/en_US/element/string.json
new file mode 100644
index 0000000000000000000000000000000000000000..f94595515a99e0c828807e243494f57f09251930
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/en_US/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "module description"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "label"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/main/resources/zh_CN/element/string.json b/scenario/arkui/MyNews/entry/src/main/resources/zh_CN/element/string.json
new file mode 100644
index 0000000000000000000000000000000000000000..597ecf95e61d7e30367c22fe2f8638008361b044
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/main/resources/zh_CN/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "模块描述"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "label"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/Ability.test.ets b/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/Ability.test.ets
new file mode 100644
index 0000000000000000000000000000000000000000..526d4a387590e0a4a069918b6896159f6008c58b
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/Ability.test.ets
@@ -0,0 +1,35 @@
+import hilog from '@ohos.hilog';
+import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it } from '@ohos/hypium';
+
+export default function abilityTest() {
+ describe('ActsAbilityTest', () => {
+ // Defines a test suite. Two parameters are supported: test suite name and test suite function.
+ beforeAll(() => {
+ // Presets an action, which is performed only once before all test cases of the test suite start.
+ // This API supports only one parameter: preset action function.
+ })
+ beforeEach(() => {
+ // Presets an action, which is performed before each unit test case starts.
+ // The number of execution times is the same as the number of test cases defined by **it**.
+ // This API supports only one parameter: preset action function.
+ })
+ afterEach(() => {
+ // Presets a clear action, which is performed after each unit test case ends.
+ // The number of execution times is the same as the number of test cases defined by **it**.
+ // This API supports only one parameter: clear action function.
+ })
+ afterAll(() => {
+ // Presets a clear action, which is performed after all test cases of the test suite end.
+ // This API supports only one parameter: clear action function.
+ })
+ it('assertContain', 0, () => {
+ // Defines a test case. This API supports three parameters: test case name, filter parameter, and test case function.
+ hilog.info(0x0000, 'testTag', '%{public}s', 'it begin');
+ let a = 'abc';
+ let b = 'b';
+ // Defines a variety of assertion methods, which are used to declare expected boolean conditions.
+ expect(a).assertContain(b);
+ expect(a).assertEqual(a);
+ })
+ })
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/List.test.ets b/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/List.test.ets
new file mode 100644
index 0000000000000000000000000000000000000000..794c7dc4ed66bd98fa3865e07922906e2fcef545
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/ets/test/List.test.ets
@@ -0,0 +1,5 @@
+import abilityTest from './Ability.test';
+
+export default function testsuite() {
+ abilityTest();
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/TestAbility.ets b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/TestAbility.ets
new file mode 100644
index 0000000000000000000000000000000000000000..9484761ccb050eef78e0468e36be45f045964d51
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/TestAbility.ets
@@ -0,0 +1,50 @@
+import UIAbility from '@ohos.app.ability.UIAbility';
+import AbilityDelegatorRegistry from '@ohos.app.ability.abilityDelegatorRegistry';
+import hilog from '@ohos.hilog';
+import { Hypium } from '@ohos/hypium';
+import testsuite from '../test/List.test';
+import window from '@ohos.window';
+import Want from '@ohos.app.ability.Want';
+import AbilityConstant from '@ohos.app.ability.AbilityConstant';
+
+export default class TestAbility extends UIAbility {
+ onCreate(want: Want, launchParam: AbilityConstant.LaunchParam) {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onCreate');
+ hilog.info(0x0000, 'testTag', '%{public}s', 'want param:' + JSON.stringify(want) ?? '');
+ hilog.info(0x0000, 'testTag', '%{public}s', 'launchParam:' + JSON.stringify(launchParam) ?? '');
+ let abilityDelegator: AbilityDelegatorRegistry.AbilityDelegator;
+ abilityDelegator = AbilityDelegatorRegistry.getAbilityDelegator();
+ let abilityDelegatorArguments: AbilityDelegatorRegistry.AbilityDelegatorArgs;
+ abilityDelegatorArguments = AbilityDelegatorRegistry.getArguments();
+ hilog.info(0x0000, 'testTag', '%{public}s', 'start run testcase!!!');
+ Hypium.hypiumTest(abilityDelegator, abilityDelegatorArguments, testsuite);
+ }
+
+ onDestroy() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onDestroy');
+ }
+
+ onWindowStageCreate(windowStage: window.WindowStage) {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onWindowStageCreate');
+ windowStage.loadContent('testability/pages/Index', (err, data) => {
+ if (err.code) {
+ hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
+ return;
+ }
+ hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s',
+ JSON.stringify(data) ?? '');
+ });
+ }
+
+ onWindowStageDestroy() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onWindowStageDestroy');
+ }
+
+ onForeground() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onForeground');
+ }
+
+ onBackground() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'TestAbility onBackground');
+ }
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/pages/Index.ets b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/pages/Index.ets
new file mode 100644
index 0000000000000000000000000000000000000000..423b4276eccd1d04d56471ae7a863c48015e8ae5
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testability/pages/Index.ets
@@ -0,0 +1,17 @@
+@Entry
+@Component
+struct Index {
+ @State message: string = 'Hello World';
+
+ build() {
+ Row() {
+ Column() {
+ Text(this.message)
+ .fontSize(50)
+ .fontWeight(FontWeight.Bold)
+ }
+ .width('100%')
+ }
+ .height('100%')
+ }
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/ets/testrunner/OpenHarmonyTestRunner.ets b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testrunner/OpenHarmonyTestRunner.ets
new file mode 100644
index 0000000000000000000000000000000000000000..4577210f25b97706c5f48bd6a40fdd5464fd1510
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/ets/testrunner/OpenHarmonyTestRunner.ets
@@ -0,0 +1,47 @@
+import hilog from '@ohos.hilog';
+import TestRunner from '@ohos.application.testRunner';
+import AbilityDelegatorRegistry from '@ohos.app.ability.abilityDelegatorRegistry';
+import Want from '@ohos.app.ability.Want';
+
+let abilityDelegator: AbilityDelegatorRegistry.AbilityDelegator | undefined = undefined
+let abilityDelegatorArguments: AbilityDelegatorRegistry.AbilityDelegatorArgs | undefined = undefined
+
+async function onAbilityCreateCallback() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'onAbilityCreateCallback');
+}
+
+async function addAbilityMonitorCallback(err: Error) {
+ hilog.info(0x0000, 'testTag', 'addAbilityMonitorCallback : %{public}s', JSON.stringify(err) ?? '');
+}
+
+export default class OpenHarmonyTestRunner implements TestRunner {
+ constructor() {
+ }
+
+ onPrepare() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner OnPrepare ');
+ }
+
+ async onRun() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner onRun run');
+ abilityDelegatorArguments = AbilityDelegatorRegistry.getArguments()
+ abilityDelegator = AbilityDelegatorRegistry.getAbilityDelegator()
+ const bundleName = abilityDelegatorArguments.bundleName;
+ const testAbilityName = 'TestAbility';
+ let lMonitor: AbilityDelegatorRegistry.AbilityMonitor = {
+ abilityName: testAbilityName,
+ onAbilityCreate: onAbilityCreateCallback,
+ };
+ abilityDelegator.addAbilityMonitor(lMonitor, addAbilityMonitorCallback)
+ const want: Want = {
+ bundleName: bundleName,
+ abilityName: testAbilityName
+ };
+ abilityDelegator = AbilityDelegatorRegistry.getAbilityDelegator();
+ abilityDelegator.startAbility(want, (err, data) => {
+ hilog.info(0x0000, 'testTag', 'startAbility : err : %{public}s', JSON.stringify(err) ?? '');
+ hilog.info(0x0000, 'testTag', 'startAbility : data : %{public}s', JSON.stringify(data) ?? '');
+ })
+ hilog.info(0x0000, 'testTag', '%{public}s', 'OpenHarmonyTestRunner onRun end');
+ }
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/module.json5 b/scenario/arkui/MyNews/entry/src/ohosTest/module.json5
new file mode 100644
index 0000000000000000000000000000000000000000..4fc9701701e879512edc8da21b356970a89e5166
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/module.json5
@@ -0,0 +1,37 @@
+{
+ "module": {
+ "name": "entry_test",
+ "type": "feature",
+ "description": "$string:module_test_desc",
+ "mainElement": "TestAbility",
+ "deviceTypes": [
+ "default",
+ "tablet"
+ ],
+ "deliveryWithInstall": true,
+ "installationFree": false,
+ "pages": "$profile:test_pages",
+ "abilities": [
+ {
+ "name": "TestAbility",
+ "srcEntry": "./ets/testability/TestAbility.ets",
+ "description": "$string:TestAbility_desc",
+ "icon": "$media:icon",
+ "label": "$string:TestAbility_label",
+ "exported": true,
+ "startWindowIcon": "$media:icon",
+ "startWindowBackground": "$color:start_window_background",
+ "skills": [
+ {
+ "actions": [
+ "action.system.home"
+ ],
+ "entities": [
+ "entity.system.home"
+ ]
+ }
+ ]
+ }
+ ]
+ }
+}
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/color.json b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/color.json
new file mode 100644
index 0000000000000000000000000000000000000000..3c712962da3c2751c2b9ddb53559afcbd2b54a02
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/color.json
@@ -0,0 +1,8 @@
+{
+ "color": [
+ {
+ "name": "start_window_background",
+ "value": "#FFFFFF"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/string.json b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/string.json
new file mode 100644
index 0000000000000000000000000000000000000000..65d8fa5a7cf54aa3943dcd0214f58d1771bc1f6c
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_test_desc",
+ "value": "test ability description"
+ },
+ {
+ "name": "TestAbility_desc",
+ "value": "the test ability"
+ },
+ {
+ "name": "TestAbility_label",
+ "value": "test label"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/media/icon.png b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/media/icon.png
new file mode 100644
index 0000000000000000000000000000000000000000..ce307a8827bd75456441ceb57d530e4c8d45d36c
Binary files /dev/null and b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/media/icon.png differ
diff --git a/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/profile/test_pages.json b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/profile/test_pages.json
new file mode 100644
index 0000000000000000000000000000000000000000..b7e7343cacb32ce982a45e76daad86e435e054fe
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/ohosTest/resources/base/profile/test_pages.json
@@ -0,0 +1,5 @@
+{
+ "src": [
+ "testability/pages/Index"
+ ]
+}
diff --git a/scenario/arkui/MyNews/entry/src/test/List.test.ets b/scenario/arkui/MyNews/entry/src/test/List.test.ets
new file mode 100644
index 0000000000000000000000000000000000000000..bb5b5c3731e283dd507c847560ee59bde477bbc7
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/test/List.test.ets
@@ -0,0 +1,5 @@
+import localUnitTest from './LocalUnit.test';
+
+export default function testsuite() {
+ localUnitTest();
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/entry/src/test/LocalUnit.test.ets b/scenario/arkui/MyNews/entry/src/test/LocalUnit.test.ets
new file mode 100644
index 0000000000000000000000000000000000000000..969668d0890d5c282797bc1b6a1f22786ed53a7c
--- /dev/null
+++ b/scenario/arkui/MyNews/entry/src/test/LocalUnit.test.ets
@@ -0,0 +1,33 @@
+import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it } from '@ohos/hypium';
+
+export default function localUnitTest() {
+ describe('localUnitTest', () => {
+ // Defines a test suite. Two parameters are supported: test suite name and test suite function.
+ beforeAll(() => {
+ // Presets an action, which is performed only once before all test cases of the test suite start.
+ // This API supports only one parameter: preset action function.
+ });
+ beforeEach(() => {
+ // Presets an action, which is performed before each unit test case starts.
+ // The number of execution times is the same as the number of test cases defined by **it**.
+ // This API supports only one parameter: preset action function.
+ });
+ afterEach(() => {
+ // Presets a clear action, which is performed after each unit test case ends.
+ // The number of execution times is the same as the number of test cases defined by **it**.
+ // This API supports only one parameter: clear action function.
+ });
+ afterAll(() => {
+ // Presets a clear action, which is performed after all test cases of the test suite end.
+ // This API supports only one parameter: clear action function.
+ });
+ it('assertContain', 0, () => {
+ // Defines a test case. This API supports three parameters: test case name, filter parameter, and test case function.
+ let a = 'abc';
+ let b = 'b';
+ // Defines a variety of assertion methods, which are used to declare expected boolean conditions.
+ expect(a).assertContain(b);
+ expect(a).assertEqual(a);
+ });
+ });
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/hvigor/hvigor-config.json5 b/scenario/arkui/MyNews/hvigor/hvigor-config.json5
new file mode 100644
index 0000000000000000000000000000000000000000..1a474900a688269946dbf37db2a9038a27824fc5
--- /dev/null
+++ b/scenario/arkui/MyNews/hvigor/hvigor-config.json5
@@ -0,0 +1,18 @@
+{
+ "hvigorVersion": "3.0.9",
+ "dependencies": {
+ "@ohos/hvigor-ohos-plugin": "3.0.9"
+ },
+ "execution": {
+ // "daemon": true, /* Enable daemon compilation. Default: true */
+ // "incremental": true, /* Enable incremental compilation. Default: true */
+ // "parallel": true, /* Enable parallel compilation. Default: true */
+ // "typeCheck": false, /* Enable typeCheck. Default: false */
+ },
+ "logging": {
+ // "level": "info" /* Define the log level. Value: [ "debug" | "info" | "warn" | "error" ]. Default: "info" */
+ },
+ "debugging": {
+ // "stacktrace": false /* Disable stacktrace compilation. Default: false */
+ }
+}
diff --git a/scenario/arkui/MyNews/hvigor/hvigor-wrapper.js b/scenario/arkui/MyNews/hvigor/hvigor-wrapper.js
new file mode 100644
index 0000000000000000000000000000000000000000..4274badb0001e0f3d054292925f62043c4a201b6
--- /dev/null
+++ b/scenario/arkui/MyNews/hvigor/hvigor-wrapper.js
@@ -0,0 +1,880 @@
+"use strict";
+var u = require("path"), D = require("os"), e = require("fs"), t = require("crypto"), r = require("child_process"),
+ n = "undefined" != typeof globalThis ? globalThis : "undefined" != typeof window ? window : "undefined" != typeof global ? global : "undefined" != typeof self ? self : {
+ }, i = {}, C = {}, F = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+ };
+Object.defineProperty(C, "__esModule", { value: !0 }), C.maxPathLength = C.isMac = C.isLinux = C.isWindows = void 0;
+const E = F(D), A = "Windows_NT", o = "Darwin";
+
+function a() {
+ return E.default.type() === A
+}
+
+function c() {
+ return E.default.type() === o
+}
+
+C.isWindows = a, C.isLinux = function () {
+ return "Linux" === E.default.type()
+}, C.isMac = c, C.maxPathLength = function () {
+ return c() ? 1016 : a() ? 259 : 4095
+}, function(e){
+ var t = n && n.__createBinding || (Object.create ? function (u, D, e, t) {
+ void 0 === t && (t = e);
+ var r = Object.getOwnPropertyDescriptor(D, e);
+ r && !("get" in r ? !D.__esModule : r.writable || r.configurable) || (r = { enumerable: !0, get: function () {
+ return D[e]
+ } }), Object.defineProperty(u, t, r)
+ } : function (u, D, e, t) {
+ void 0 === t && (t = e), u[t] = D[e]
+ }), r = n && n.__setModuleDefault || (Object.create ? function (u, D) {
+ Object.defineProperty(u, "default", { enumerable: !0, value: D })
+ } : function (u, D) {
+ u.default = D
+ }), i = n && n.__importStar || function (u) {
+ if (u && u.__esModule) return u;
+ var D = {};
+ if (null != u) for (var e in u) "default" !== e && Object.prototype.hasOwnProperty.call(u, e) && t(D, u, e);
+ return r(D, u), D
+ };
+ Object.defineProperty(e, "__esModule", {
+ value: !0
+ }), e.WORK_SPACE = e.HVIGOR_PROJECT_WRAPPER_HOME = e.HVIGOR_PROJECT_ROOT_DIR = e.HVIGOR_PROJECT_CACHES_HOME = e.HVIGOR_PNPM_STORE_PATH = e.HVIGOR_WRAPPER_PNPM_SCRIPT_PATH = e.PROJECT_CACHES = e.HVIGOR_WRAPPER_TOOLS_HOME = e.HVIGOR_USER_HOME = e.DEFAULT_PACKAGE_JSON = e.DEFAULT_HVIGOR_CONFIG_JSON_FILE_NAME = e.PNPM = e.HVIGOR = e.NPM_TOOL = e.PNPM_TOOL = e.HVIGOR_ENGINE_PACKAGE_NAME = void 0;
+ const F = i(D), E = i(u), A = C;
+ e.HVIGOR_ENGINE_PACKAGE_NAME = "@ohos/hvigor", e.PNPM_TOOL = (0, A.isWindows)() ? "pnpm.cmd" : "pnpm", e.NPM_TOOL = (0, A.isWindows)() ? "npm.cmd" : "npm", e.HVIGOR = "hvigor", e.PNPM = "pnpm", e.DEFAULT_HVIGOR_CONFIG_JSON_FILE_NAME = "hvigor-config.json5", e.DEFAULT_PACKAGE_JSON = "package.json", e.HVIGOR_USER_HOME = E.resolve(F.homedir(), ".hvigor"), e.HVIGOR_WRAPPER_TOOLS_HOME = E.resolve(e.HVIGOR_USER_HOME, "wrapper", "tools"), e.PROJECT_CACHES = "project_caches", e.HVIGOR_WRAPPER_PNPM_SCRIPT_PATH = E.resolve(e.HVIGOR_WRAPPER_TOOLS_HOME, "node_modules", ".bin", e.PNPM_TOOL), e.HVIGOR_PNPM_STORE_PATH = E.resolve(e.HVIGOR_USER_HOME, "caches"), e.HVIGOR_PROJECT_CACHES_HOME = E.resolve(e.HVIGOR_USER_HOME, e.PROJECT_CACHES), e.HVIGOR_PROJECT_ROOT_DIR = process.cwd(), e.HVIGOR_PROJECT_WRAPPER_HOME = E.resolve(e.HVIGOR_PROJECT_ROOT_DIR, e.HVIGOR), e.WORK_SPACE = "workspace"
+}(i);
+var s = {}, l = {};
+Object.defineProperty(l, "__esModule", {
+ value: !0
+}), l.logInfoPrintConsole = l.logErrorAndExit = void 0, l.logErrorAndExit = function (u) {
+ u instanceof Error ? console.error(u.message) : console.error(u), process.exit(-1)
+}, l.logInfoPrintConsole = function (u) {
+ console.log(u)
+};
+var B = n && n.__createBinding || (Object.create ? function (u, D, e, t) {
+ void 0 === t && (t = e);
+ var r = Object.getOwnPropertyDescriptor(D, e);
+ r && !("get" in r ? !D.__esModule : r.writable || r.configurable) || (r = { enumerable: !0, get: function () {
+ return D[e]
+ } }), Object.defineProperty(u, t, r)
+} : function (u, D, e, t) {
+ void 0 === t && (t = e), u[t] = D[e]
+}), d = n && n.__setModuleDefault || (Object.create ? function (u, D) {
+ Object.defineProperty(u, "default", { enumerable: !0, value: D })
+} : function (u, D) {
+ u.default = D
+}), f = n && n.__importStar || function (u) {
+ if (u && u.__esModule) return u;
+ var D = {};
+ if (null != u) for (var e in u) "default" !== e && Object.prototype.hasOwnProperty.call(u, e) && B(D, u, e);
+ return d(D, u), D
+};
+Object.defineProperty(s, "__esModule", { value: !0 });
+var _ = s.executeBuild = void 0;
+const p = f(e), O = f(u), h = l;
+_ = s.executeBuild = function (u) {
+ const D = O.resolve(u, "node_modules", "@ohos", "hvigor", "bin", "hvigor.js");
+ try {
+ const u = p.realpathSync(D);
+ require(u)
+ } catch (e) {
+ (0, h.logErrorAndExit)(`Error: ENOENT: no such file ${D},delete ${u} and retry.`)
+ }
+};
+var P = {}, v = {};
+!function(u){
+ var D = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+ };
+ Object.defineProperty(u, "__esModule", { value: !0 }), u.hashFile = u.hash = u.createHash = void 0;
+ const r = D(t), i = D(e);
+ u.createHash = (u = "MD5") => r.default.createHash(u);
+ u.hash = (D, e) => (0, u.createHash)(e).update(D).digest("hex");
+ u.hashFile = (D, e) => {
+ if (i.default.existsSync(D)) return (0, u.hash)(i.default.readFileSync(D, "utf-8"), e)
+ }
+}(v);
+var g = {}, m = {}, R = {};
+Object.defineProperty(R, "__esModule", { value: !0 }), R.Unicode = void 0;
+
+class y {
+}
+
+R.Unicode = y, y.SPACE_SEPARATOR = /[\u1680\u2000-\u200A\u202F\u205F\u3000]/, y.ID_START = /[\xAA\xB5\xBA\xC0-\xD6\xD8-\xF6\xF8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u037F\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u052F\u0531-\u0556\u0559\u0561-\u0587\u05D0-\u05EA\u05F0-\u05F2\u0620-\u064A\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE\u06EF\u06FA-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07CA-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u0860-\u086A\u08A0-\u08B4\u08B6-\u08BD\u0904-\u0939\u093D\u0950\u0958-\u0961\u0971-\u0980\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09F0\u09F1\u09FC\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0AF9\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B71\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C39\u0C3D\u0C58-\u0C5A\u0C60\u0C61\u0C80\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CF1\u0CF2\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D54-\u0D56\u0D5F-\u0D61\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0EDC-\u0EDF\u0F00\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1380-\u138F\u13A0-\u13F5\u13F8-\u13FD\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u16EE-\u16F8\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u1820-\u1877\u1880-\u1884\u1887-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191E\u1950-\u196D\u1970-\u1974\u1980-\u19AB\u19B0-\u19C9\u1A00-\u1A16\u1A20-\u1A54\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B83-\u1BA0\u1BAE\u1BAF\u1BBA-\u1BE5\u1C00-\u1C23\u1C4D-\u1C4F\u1C5A-\u1C7D\u1C80-\u1C88\u1CE9-\u1CEC\u1CEE-\u1CF1\u1CF5\u1CF6\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2071\u207F\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2160-\u2188\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005-\u3007\u3021-\u3029\u3031-\u3035\u3038-\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312E\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FEA\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA61F\uA62A\uA62B\uA640-\uA66E\uA67F-\uA69D\uA6A0-\uA6EF\uA717-\uA71F\uA722-\uA788\uA78B-\uA7AE\uA7B0-\uA7B7\uA7F7-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA840-\uA873\uA882-\uA8B3\uA8F2-\uA8F7\uA8FB\uA8FD\uA90A-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF\uA9E0-\uA9E4\uA9E6-\uA9EF\uA9FA-\uA9FE\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA60-\uAA76\uAA7A\uAA7E-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uAB30-\uAB5A\uAB5C-\uAB65\uAB70-\uABE2\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]|\uD800[\uDC00-\uDC0B\uDC0D-\uDC26\uDC28-\uDC3A\uDC3C\uDC3D\uDC3F-\uDC4D\uDC50-\uDC5D\uDC80-\uDCFA\uDD40-\uDD74\uDE80-\uDE9C\uDEA0-\uDED0\uDF00-\uDF1F\uDF2D-\uDF4A\uDF50-\uDF75\uDF80-\uDF9D\uDFA0-\uDFC3\uDFC8-\uDFCF\uDFD1-\uDFD5]|\uD801[\uDC00-\uDC9D\uDCB0-\uDCD3\uDCD8-\uDCFB\uDD00-\uDD27\uDD30-\uDD63\uDE00-\uDF36\uDF40-\uDF55\uDF60-\uDF67]|\uD802[\uDC00-\uDC05\uDC08\uDC0A-\uDC35\uDC37\uDC38\uDC3C\uDC3F-\uDC55\uDC60-\uDC76\uDC80-\uDC9E\uDCE0-\uDCF2\uDCF4\uDCF5\uDD00-\uDD15\uDD20-\uDD39\uDD80-\uDDB7\uDDBE\uDDBF\uDE00\uDE10-\uDE13\uDE15-\uDE17\uDE19-\uDE33\uDE60-\uDE7C\uDE80-\uDE9C\uDEC0-\uDEC7\uDEC9-\uDEE4\uDF00-\uDF35\uDF40-\uDF55\uDF60-\uDF72\uDF80-\uDF91]|\uD803[\uDC00-\uDC48\uDC80-\uDCB2\uDCC0-\uDCF2]|\uD804[\uDC03-\uDC37\uDC83-\uDCAF\uDCD0-\uDCE8\uDD03-\uDD26\uDD50-\uDD72\uDD76\uDD83-\uDDB2\uDDC1-\uDDC4\uDDDA\uDDDC\uDE00-\uDE11\uDE13-\uDE2B\uDE80-\uDE86\uDE88\uDE8A-\uDE8D\uDE8F-\uDE9D\uDE9F-\uDEA8\uDEB0-\uDEDE\uDF05-\uDF0C\uDF0F\uDF10\uDF13-\uDF28\uDF2A-\uDF30\uDF32\uDF33\uDF35-\uDF39\uDF3D\uDF50\uDF5D-\uDF61]|\uD805[\uDC00-\uDC34\uDC47-\uDC4A\uDC80-\uDCAF\uDCC4\uDCC5\uDCC7\uDD80-\uDDAE\uDDD8-\uDDDB\uDE00-\uDE2F\uDE44\uDE80-\uDEAA\uDF00-\uDF19]|\uD806[\uDCA0-\uDCDF\uDCFF\uDE00\uDE0B-\uDE32\uDE3A\uDE50\uDE5C-\uDE83\uDE86-\uDE89\uDEC0-\uDEF8]|\uD807[\uDC00-\uDC08\uDC0A-\uDC2E\uDC40\uDC72-\uDC8F\uDD00-\uDD06\uDD08\uDD09\uDD0B-\uDD30\uDD46]|\uD808[\uDC00-\uDF99]|\uD809[\uDC00-\uDC6E\uDC80-\uDD43]|[\uD80C\uD81C-\uD820\uD840-\uD868\uD86A-\uD86C\uD86F-\uD872\uD874-\uD879][\uDC00-\uDFFF]|\uD80D[\uDC00-\uDC2E]|\uD811[\uDC00-\uDE46]|\uD81A[\uDC00-\uDE38\uDE40-\uDE5E\uDED0-\uDEED\uDF00-\uDF2F\uDF40-\uDF43\uDF63-\uDF77\uDF7D-\uDF8F]|\uD81B[\uDF00-\uDF44\uDF50\uDF93-\uDF9F\uDFE0\uDFE1]|\uD821[\uDC00-\uDFEC]|\uD822[\uDC00-\uDEF2]|\uD82C[\uDC00-\uDD1E\uDD70-\uDEFB]|\uD82F[\uDC00-\uDC6A\uDC70-\uDC7C\uDC80-\uDC88\uDC90-\uDC99]|\uD835[\uDC00-\uDC54\uDC56-\uDC9C\uDC9E\uDC9F\uDCA2\uDCA5\uDCA6\uDCA9-\uDCAC\uDCAE-\uDCB9\uDCBB\uDCBD-\uDCC3\uDCC5-\uDD05\uDD07-\uDD0A\uDD0D-\uDD14\uDD16-\uDD1C\uDD1E-\uDD39\uDD3B-\uDD3E\uDD40-\uDD44\uDD46\uDD4A-\uDD50\uDD52-\uDEA5\uDEA8-\uDEC0\uDEC2-\uDEDA\uDEDC-\uDEFA\uDEFC-\uDF14\uDF16-\uDF34\uDF36-\uDF4E\uDF50-\uDF6E\uDF70-\uDF88\uDF8A-\uDFA8\uDFAA-\uDFC2\uDFC4-\uDFCB]|\uD83A[\uDC00-\uDCC4\uDD00-\uDD43]|\uD83B[\uDE00-\uDE03\uDE05-\uDE1F\uDE21\uDE22\uDE24\uDE27\uDE29-\uDE32\uDE34-\uDE37\uDE39\uDE3B\uDE42\uDE47\uDE49\uDE4B\uDE4D-\uDE4F\uDE51\uDE52\uDE54\uDE57\uDE59\uDE5B\uDE5D\uDE5F\uDE61\uDE62\uDE64\uDE67-\uDE6A\uDE6C-\uDE72\uDE74-\uDE77\uDE79-\uDE7C\uDE7E\uDE80-\uDE89\uDE8B-\uDE9B\uDEA1-\uDEA3\uDEA5-\uDEA9\uDEAB-\uDEBB]|\uD869[\uDC00-\uDED6\uDF00-\uDFFF]|\uD86D[\uDC00-\uDF34\uDF40-\uDFFF]|\uD86E[\uDC00-\uDC1D\uDC20-\uDFFF]|\uD873[\uDC00-\uDEA1\uDEB0-\uDFFF]|\uD87A[\uDC00-\uDFE0]|\uD87E[\uDC00-\uDE1D]/, y.ID_CONTINUE = /[\xAA\xB5\xBA\xC0-\xD6\xD8-\xF6\xF8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0300-\u0374\u0376\u0377\u037A-\u037D\u037F\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u0483-\u0487\u048A-\u052F\u0531-\u0556\u0559\u0561-\u0587\u0591-\u05BD\u05BF\u05C1\u05C2\u05C4\u05C5\u05C7\u05D0-\u05EA\u05F0-\u05F2\u0610-\u061A\u0620-\u0669\u066E-\u06D3\u06D5-\u06DC\u06DF-\u06E8\u06EA-\u06FC\u06FF\u0710-\u074A\u074D-\u07B1\u07C0-\u07F5\u07FA\u0800-\u082D\u0840-\u085B\u0860-\u086A\u08A0-\u08B4\u08B6-\u08BD\u08D4-\u08E1\u08E3-\u0963\u0966-\u096F\u0971-\u0983\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BC-\u09C4\u09C7\u09C8\u09CB-\u09CE\u09D7\u09DC\u09DD\u09DF-\u09E3\u09E6-\u09F1\u09FC\u0A01-\u0A03\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A3C\u0A3E-\u0A42\u0A47\u0A48\u0A4B-\u0A4D\u0A51\u0A59-\u0A5C\u0A5E\u0A66-\u0A75\u0A81-\u0A83\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABC-\u0AC5\u0AC7-\u0AC9\u0ACB-\u0ACD\u0AD0\u0AE0-\u0AE3\u0AE6-\u0AEF\u0AF9-\u0AFF\u0B01-\u0B03\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3C-\u0B44\u0B47\u0B48\u0B4B-\u0B4D\u0B56\u0B57\u0B5C\u0B5D\u0B5F-\u0B63\u0B66-\u0B6F\u0B71\u0B82\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BBE-\u0BC2\u0BC6-\u0BC8\u0BCA-\u0BCD\u0BD0\u0BD7\u0BE6-\u0BEF\u0C00-\u0C03\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C39\u0C3D-\u0C44\u0C46-\u0C48\u0C4A-\u0C4D\u0C55\u0C56\u0C58-\u0C5A\u0C60-\u0C63\u0C66-\u0C6F\u0C80-\u0C83\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBC-\u0CC4\u0CC6-\u0CC8\u0CCA-\u0CCD\u0CD5\u0CD6\u0CDE\u0CE0-\u0CE3\u0CE6-\u0CEF\u0CF1\u0CF2\u0D00-\u0D03\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D44\u0D46-\u0D48\u0D4A-\u0D4E\u0D54-\u0D57\u0D5F-\u0D63\u0D66-\u0D6F\u0D7A-\u0D7F\u0D82\u0D83\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0DCA\u0DCF-\u0DD4\u0DD6\u0DD8-\u0DDF\u0DE6-\u0DEF\u0DF2\u0DF3\u0E01-\u0E3A\u0E40-\u0E4E\u0E50-\u0E59\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB9\u0EBB-\u0EBD\u0EC0-\u0EC4\u0EC6\u0EC8-\u0ECD\u0ED0-\u0ED9\u0EDC-\u0EDF\u0F00\u0F18\u0F19\u0F20-\u0F29\u0F35\u0F37\u0F39\u0F3E-\u0F47\u0F49-\u0F6C\u0F71-\u0F84\u0F86-\u0F97\u0F99-\u0FBC\u0FC6\u1000-\u1049\u1050-\u109D\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u135D-\u135F\u1380-\u138F\u13A0-\u13F5\u13F8-\u13FD\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u16EE-\u16F8\u1700-\u170C\u170E-\u1714\u1720-\u1734\u1740-\u1753\u1760-\u176C\u176E-\u1770\u1772\u1773\u1780-\u17D3\u17D7\u17DC\u17DD\u17E0-\u17E9\u180B-\u180D\u1810-\u1819\u1820-\u1877\u1880-\u18AA\u18B0-\u18F5\u1900-\u191E\u1920-\u192B\u1930-\u193B\u1946-\u196D\u1970-\u1974\u1980-\u19AB\u19B0-\u19C9\u19D0-\u19D9\u1A00-\u1A1B\u1A20-\u1A5E\u1A60-\u1A7C\u1A7F-\u1A89\u1A90-\u1A99\u1AA7\u1AB0-\u1ABD\u1B00-\u1B4B\u1B50-\u1B59\u1B6B-\u1B73\u1B80-\u1BF3\u1C00-\u1C37\u1C40-\u1C49\u1C4D-\u1C7D\u1C80-\u1C88\u1CD0-\u1CD2\u1CD4-\u1CF9\u1D00-\u1DF9\u1DFB-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u203F\u2040\u2054\u2071\u207F\u2090-\u209C\u20D0-\u20DC\u20E1\u20E5-\u20F0\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2160-\u2188\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D7F-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2DE0-\u2DFF\u2E2F\u3005-\u3007\u3021-\u302F\u3031-\u3035\u3038-\u303C\u3041-\u3096\u3099\u309A\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312E\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FEA\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA62B\uA640-\uA66F\uA674-\uA67D\uA67F-\uA6F1\uA717-\uA71F\uA722-\uA788\uA78B-\uA7AE\uA7B0-\uA7B7\uA7F7-\uA827\uA840-\uA873\uA880-\uA8C5\uA8D0-\uA8D9\uA8E0-\uA8F7\uA8FB\uA8FD\uA900-\uA92D\uA930-\uA953\uA960-\uA97C\uA980-\uA9C0\uA9CF-\uA9D9\uA9E0-\uA9FE\uAA00-\uAA36\uAA40-\uAA4D\uAA50-\uAA59\uAA60-\uAA76\uAA7A-\uAAC2\uAADB-\uAADD\uAAE0-\uAAEF\uAAF2-\uAAF6\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uAB30-\uAB5A\uAB5C-\uAB65\uAB70-\uABEA\uABEC\uABED\uABF0-\uABF9\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE00-\uFE0F\uFE20-\uFE2F\uFE33\uFE34\uFE4D-\uFE4F\uFE70-\uFE74\uFE76-\uFEFC\uFF10-\uFF19\uFF21-\uFF3A\uFF3F\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]|\uD800[\uDC00-\uDC0B\uDC0D-\uDC26\uDC28-\uDC3A\uDC3C\uDC3D\uDC3F-\uDC4D\uDC50-\uDC5D\uDC80-\uDCFA\uDD40-\uDD74\uDDFD\uDE80-\uDE9C\uDEA0-\uDED0\uDEE0\uDF00-\uDF1F\uDF2D-\uDF4A\uDF50-\uDF7A\uDF80-\uDF9D\uDFA0-\uDFC3\uDFC8-\uDFCF\uDFD1-\uDFD5]|\uD801[\uDC00-\uDC9D\uDCA0-\uDCA9\uDCB0-\uDCD3\uDCD8-\uDCFB\uDD00-\uDD27\uDD30-\uDD63\uDE00-\uDF36\uDF40-\uDF55\uDF60-\uDF67]|\uD802[\uDC00-\uDC05\uDC08\uDC0A-\uDC35\uDC37\uDC38\uDC3C\uDC3F-\uDC55\uDC60-\uDC76\uDC80-\uDC9E\uDCE0-\uDCF2\uDCF4\uDCF5\uDD00-\uDD15\uDD20-\uDD39\uDD80-\uDDB7\uDDBE\uDDBF\uDE00-\uDE03\uDE05\uDE06\uDE0C-\uDE13\uDE15-\uDE17\uDE19-\uDE33\uDE38-\uDE3A\uDE3F\uDE60-\uDE7C\uDE80-\uDE9C\uDEC0-\uDEC7\uDEC9-\uDEE6\uDF00-\uDF35\uDF40-\uDF55\uDF60-\uDF72\uDF80-\uDF91]|\uD803[\uDC00-\uDC48\uDC80-\uDCB2\uDCC0-\uDCF2]|\uD804[\uDC00-\uDC46\uDC66-\uDC6F\uDC7F-\uDCBA\uDCD0-\uDCE8\uDCF0-\uDCF9\uDD00-\uDD34\uDD36-\uDD3F\uDD50-\uDD73\uDD76\uDD80-\uDDC4\uDDCA-\uDDCC\uDDD0-\uDDDA\uDDDC\uDE00-\uDE11\uDE13-\uDE37\uDE3E\uDE80-\uDE86\uDE88\uDE8A-\uDE8D\uDE8F-\uDE9D\uDE9F-\uDEA8\uDEB0-\uDEEA\uDEF0-\uDEF9\uDF00-\uDF03\uDF05-\uDF0C\uDF0F\uDF10\uDF13-\uDF28\uDF2A-\uDF30\uDF32\uDF33\uDF35-\uDF39\uDF3C-\uDF44\uDF47\uDF48\uDF4B-\uDF4D\uDF50\uDF57\uDF5D-\uDF63\uDF66-\uDF6C\uDF70-\uDF74]|\uD805[\uDC00-\uDC4A\uDC50-\uDC59\uDC80-\uDCC5\uDCC7\uDCD0-\uDCD9\uDD80-\uDDB5\uDDB8-\uDDC0\uDDD8-\uDDDD\uDE00-\uDE40\uDE44\uDE50-\uDE59\uDE80-\uDEB7\uDEC0-\uDEC9\uDF00-\uDF19\uDF1D-\uDF2B\uDF30-\uDF39]|\uD806[\uDCA0-\uDCE9\uDCFF\uDE00-\uDE3E\uDE47\uDE50-\uDE83\uDE86-\uDE99\uDEC0-\uDEF8]|\uD807[\uDC00-\uDC08\uDC0A-\uDC36\uDC38-\uDC40\uDC50-\uDC59\uDC72-\uDC8F\uDC92-\uDCA7\uDCA9-\uDCB6\uDD00-\uDD06\uDD08\uDD09\uDD0B-\uDD36\uDD3A\uDD3C\uDD3D\uDD3F-\uDD47\uDD50-\uDD59]|\uD808[\uDC00-\uDF99]|\uD809[\uDC00-\uDC6E\uDC80-\uDD43]|[\uD80C\uD81C-\uD820\uD840-\uD868\uD86A-\uD86C\uD86F-\uD872\uD874-\uD879][\uDC00-\uDFFF]|\uD80D[\uDC00-\uDC2E]|\uD811[\uDC00-\uDE46]|\uD81A[\uDC00-\uDE38\uDE40-\uDE5E\uDE60-\uDE69\uDED0-\uDEED\uDEF0-\uDEF4\uDF00-\uDF36\uDF40-\uDF43\uDF50-\uDF59\uDF63-\uDF77\uDF7D-\uDF8F]|\uD81B[\uDF00-\uDF44\uDF50-\uDF7E\uDF8F-\uDF9F\uDFE0\uDFE1]|\uD821[\uDC00-\uDFEC]|\uD822[\uDC00-\uDEF2]|\uD82C[\uDC00-\uDD1E\uDD70-\uDEFB]|\uD82F[\uDC00-\uDC6A\uDC70-\uDC7C\uDC80-\uDC88\uDC90-\uDC99\uDC9D\uDC9E]|\uD834[\uDD65-\uDD69\uDD6D-\uDD72\uDD7B-\uDD82\uDD85-\uDD8B\uDDAA-\uDDAD\uDE42-\uDE44]|\uD835[\uDC00-\uDC54\uDC56-\uDC9C\uDC9E\uDC9F\uDCA2\uDCA5\uDCA6\uDCA9-\uDCAC\uDCAE-\uDCB9\uDCBB\uDCBD-\uDCC3\uDCC5-\uDD05\uDD07-\uDD0A\uDD0D-\uDD14\uDD16-\uDD1C\uDD1E-\uDD39\uDD3B-\uDD3E\uDD40-\uDD44\uDD46\uDD4A-\uDD50\uDD52-\uDEA5\uDEA8-\uDEC0\uDEC2-\uDEDA\uDEDC-\uDEFA\uDEFC-\uDF14\uDF16-\uDF34\uDF36-\uDF4E\uDF50-\uDF6E\uDF70-\uDF88\uDF8A-\uDFA8\uDFAA-\uDFC2\uDFC4-\uDFCB\uDFCE-\uDFFF]|\uD836[\uDE00-\uDE36\uDE3B-\uDE6C\uDE75\uDE84\uDE9B-\uDE9F\uDEA1-\uDEAF]|\uD838[\uDC00-\uDC06\uDC08-\uDC18\uDC1B-\uDC21\uDC23\uDC24\uDC26-\uDC2A]|\uD83A[\uDC00-\uDCC4\uDCD0-\uDCD6\uDD00-\uDD4A\uDD50-\uDD59]|\uD83B[\uDE00-\uDE03\uDE05-\uDE1F\uDE21\uDE22\uDE24\uDE27\uDE29-\uDE32\uDE34-\uDE37\uDE39\uDE3B\uDE42\uDE47\uDE49\uDE4B\uDE4D-\uDE4F\uDE51\uDE52\uDE54\uDE57\uDE59\uDE5B\uDE5D\uDE5F\uDE61\uDE62\uDE64\uDE67-\uDE6A\uDE6C-\uDE72\uDE74-\uDE77\uDE79-\uDE7C\uDE7E\uDE80-\uDE89\uDE8B-\uDE9B\uDEA1-\uDEA3\uDEA5-\uDEA9\uDEAB-\uDEBB]|\uD869[\uDC00-\uDED6\uDF00-\uDFFF]|\uD86D[\uDC00-\uDF34\uDF40-\uDFFF]|\uD86E[\uDC00-\uDC1D\uDC20-\uDFFF]|\uD873[\uDC00-\uDEA1\uDEB0-\uDFFF]|\uD87A[\uDC00-\uDFE0]|\uD87E[\uDC00-\uDE1D]|\uDB40[\uDD00-\uDDEF]/, Object.defineProperty(m, "__esModule", {
+ value: !0
+}), m.JudgeUtil = void 0;
+const I = R;
+m.JudgeUtil = class {
+ static isIgnoreChar(u) {
+ return "string" == typeof u && ("\t" === u || "\v" === u || "\f" === u || " " === u || " " === u || "\ufeff" === u || "\n" === u || "\r" === u || "\u2028" === u || "\u2029" === u)
+ }
+
+ static isSpaceSeparator(u) {
+ return "string" == typeof u && I.Unicode.SPACE_SEPARATOR.test(u)
+ }
+
+ static isIdStartChar(u) {
+ return "string" == typeof u && (u >= "a" && u <= "z" || u >= "A" && u <= "Z" || "$" === u || "_" === u || I.Unicode.ID_START.test(u))
+ }
+
+ static isIdContinueChar(u) {
+ return "string" == typeof u && (u >= "a" && u <= "z" || u >= "A" && u <= "Z" || u >= "0" && u <= "9" || "$" === u || "_" === u || "" === u || "" === u || I.Unicode.ID_CONTINUE.test(u))
+ }
+
+ static isDigitWithoutZero(u) {
+ return /[1-9]/.test(u)
+ }
+
+ static isDigit(u) {
+ return "string" == typeof u && /[0-9]/.test(u)
+ }
+
+ static isHexDigit(u) {
+ return "string" == typeof u && /[0-9A-Fa-f]/.test(u)
+ }
+};
+var N = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+};
+Object.defineProperty(g, "__esModule", { value: !0 }), g.parseJsonText = g.parseJsonFile = void 0;
+const b = N(e), S = N(D), w = N(u), H = m;
+var x;
+!function(u){
+ u[u.Char=0] = "Char", u[u.EOF=1] = "EOF", u[u.Identifier=2] = "Identifier"
+}(x || (x = {}));
+let M, T, V, G, j, J, W = "start", U = [], L = 0, $ = 1, k = 0, K = !1, z = "default", q = "'", Z = 1;
+
+function X(u, D = !1) {
+ T = String(u), W = "start", U = [], L = 0, $ = 1, k = 0, G = void 0, K = D;
+ do {
+ M = Q(), nu[W]()
+ } while ("eof" !== M.type);
+ return G
+}
+
+function Q() {
+ for (z = "default", j = "", q = "'", Z = 1;; ) {
+ J = Y();
+ const u = Du[z]();
+ if (u) return u
+ }
+}
+
+function Y() {
+ if (T[L]) return String.fromCodePoint(T.codePointAt(L))
+}
+
+function uu() {
+ const u = Y();
+ return "\n" === u ? ($++, k = 0) : u ? k += u.length : k++, u && (L += u.length), u
+}
+
+g.parseJsonFile = function (u, D = !1, e = "utf-8") {
+ const t = b.default.readFileSync(w.default.resolve(u), { encoding: e });
+ try {
+ return X(t, D)
+ } catch (D) {
+ if (D instanceof SyntaxError) {
+ const e = D.message.split("at");
+ if (2 === e.length) throw new Error(`${e[0].trim()}${S.default.EOL}\t at ${u}:${e[1].trim()}`)
+ }
+ throw new Error(`${u} is not in valid JSON/JSON5 format.`)
+ }
+}, g.parseJsonText = X;
+const Du = {
+ default() {
+ switch (J) {
+ case "/":
+ return uu(), void (z = "comment");
+ case void 0:
+ return uu(), eu("eof")
+ }
+ if (!H.JudgeUtil.isIgnoreChar(J) && !H.JudgeUtil.isSpaceSeparator(J)) return Du[W]();
+ uu()
+ },
+ start() {
+ z = "value"
+ },
+ beforePropertyName() {
+ switch (J) {
+ case "$":
+ case "_":
+ return j = uu(), void (z = "identifierName");
+ case "\\":
+ return uu(), void (z = "identifierNameStartEscape");
+ case "}":
+ return eu("punctuator", uu());
+ case '"':
+ case "'":
+ return q = J, uu(), void (z = "string")
+ }
+ if (H.JudgeUtil.isIdStartChar(J)) return j += uu(), void (z = "identifierName");
+ throw Eu(x.Char, uu())
+ },
+ afterPropertyName() {
+ if (":" === J) return eu("punctuator", uu());
+ throw Eu(x.Char, uu())
+ },
+ beforePropertyValue() {
+ z = "value"
+ },
+ afterPropertyValue() {
+ switch (J) {
+ case ",":
+ case "}":
+ return eu("punctuator", uu())
+ }
+ throw Eu(x.Char, uu())
+ },
+ beforeArrayValue() {
+ if ("]" === J) return eu("punctuator", uu());
+ z = "value"
+ },
+ afterArrayValue() {
+ switch (J) {
+ case ",":
+ case "]":
+ return eu("punctuator", uu())
+ }
+ throw Eu(x.Char, uu())
+ },
+ end() {
+ throw Eu(x.Char, uu())
+ },
+ comment() {
+ switch (J) {
+ case "*":
+ return uu(), void (z = "multiLineComment");
+ case "/":
+ return uu(), void (z = "singleLineComment")
+ }
+ throw Eu(x.Char, uu())
+ },
+ multiLineComment() {
+ switch (J) {
+ case "*":
+ return uu(), void (z = "multiLineCommentAsterisk");
+ case void 0:
+ throw Eu(x.Char, uu())
+ }
+ uu()
+ },
+ multiLineCommentAsterisk() {
+ switch (J) {
+ case "*":
+ return void uu();
+ case "/":
+ return uu(), void (z = "default");
+ case void 0:
+ throw Eu(x.Char, uu())
+ }
+ uu(), z = "multiLineComment"
+ },
+ singleLineComment() {
+ switch (J) {
+ case "\n":
+ case "\r":
+ case "\u2028":
+ case "\u2029":
+ return uu(), void (z = "default");
+ case void 0:
+ return uu(), eu("eof")
+ }
+ uu()
+ },
+ value() {
+ switch (J) {
+ case "{":
+ case "[":
+ return eu("punctuator", uu());
+ case "n":
+ return uu(), tu("ull"), eu("null", null);
+ case "t":
+ return uu(), tu("rue"), eu("boolean",!0);
+ case "f":
+ return uu(), tu("alse"), eu("boolean",!1);
+ case "-":
+ case "+":
+ return "-" === uu() && (Z = -1), void (z = "numerical");
+ case ".":
+ case "0":
+ case "I":
+ case "N":
+ return void (z = "numerical");
+ case '"':
+ case "'":
+ return q = J, uu(), j = "", void (z = "string")
+ }
+ if (void 0 === J || !H.JudgeUtil.isDigitWithoutZero(J)) throw Eu(x.Char, uu());
+ z = "numerical"
+ },
+ numerical() {
+ switch (J) {
+ case ".":
+ return j = uu(), void (z = "decimalPointLeading");
+ case "0":
+ return j = uu(), void (z = "zero");
+ case "I":
+ return uu(), tu("nfinity"), eu("numeric", Z * (1 / 0));
+ case "N":
+ return uu(), tu("aN"), eu("numeric", NaN)
+ }
+ if (void 0 !== J && H.JudgeUtil.isDigitWithoutZero(J)) return j = uu(), void (z = "decimalInteger");
+ throw Eu(x.Char, uu())
+ },
+ zero() {
+ switch (J) {
+ case ".":
+ case "e":
+ case "E":
+ return void (z = "decimal");
+ case "x":
+ case "X":
+ return j += uu(), void (z = "hexadecimal")
+ }
+ return eu("numeric", 0)
+ },
+ decimalInteger() {
+ switch (J) {
+ case ".":
+ case "e":
+ case "E":
+ return void (z = "decimal")
+ }
+ if (!H.JudgeUtil.isDigit(J)) return eu("numeric", Z * Number(j));
+ j += uu()
+ },
+ decimal() {
+ switch (J) {
+ case ".":
+ j += uu(), z = "decimalFraction";
+ break;
+ case "e":
+ case "E":
+ j += uu(), z = "decimalExponent"
+ }
+ },
+ decimalPointLeading() {
+ if (H.JudgeUtil.isDigit(J)) return j += uu(), void (z = "decimalFraction");
+ throw Eu(x.Char, uu())
+ },
+ decimalFraction() {
+ switch (J) {
+ case "e":
+ case "E":
+ return j += uu(), void (z = "decimalExponent")
+ }
+ if (!H.JudgeUtil.isDigit(J)) return eu("numeric", Z * Number(j));
+ j += uu()
+ },
+ decimalExponent() {
+ switch (J) {
+ case "+":
+ case "-":
+ return j += uu(), void (z = "decimalExponentSign")
+ }
+ if (H.JudgeUtil.isDigit(J)) return j += uu(), void (z = "decimalExponentInteger");
+ throw Eu(x.Char, uu())
+ },
+ decimalExponentSign() {
+ if (H.JudgeUtil.isDigit(J)) return j += uu(), void (z = "decimalExponentInteger");
+ throw Eu(x.Char, uu())
+ },
+ decimalExponentInteger() {
+ if (!H.JudgeUtil.isDigit(J)) return eu("numeric", Z * Number(j));
+ j += uu()
+ },
+ hexadecimal() {
+ if (H.JudgeUtil.isHexDigit(J)) return j += uu(), void (z = "hexadecimalInteger");
+ throw Eu(x.Char, uu())
+ },
+ hexadecimalInteger() {
+ if (!H.JudgeUtil.isHexDigit(J)) return eu("numeric", Z * Number(j));
+ j += uu()
+ },
+ identifierNameStartEscape() {
+ if ("u" !== J) throw Eu(x.Char, uu());
+ uu();
+ const u = ru();
+ switch (u) {
+ case "$":
+ case "_":
+ break;
+ default:
+ if (!H.JudgeUtil.isIdStartChar(u)) throw Eu(x.Identifier)
+ }
+ j += u, z = "identifierName"
+ },
+ identifierName() {
+ switch (J) {
+ case "$":
+ case "_":
+ case "":
+ case "":
+ return void (j += uu());
+ case "\\":
+ return uu(), void (z = "identifierNameEscape")
+ }
+ if (!H.JudgeUtil.isIdContinueChar(J)) return eu("identifier", j);
+ j += uu()
+ },
+ identifierNameEscape() {
+ if ("u" !== J) throw Eu(x.Char, uu());
+ uu();
+ const u = ru();
+ switch (u) {
+ case "$":
+ case "_":
+ case "":
+ case "":
+ break;
+ default:
+ if (!H.JudgeUtil.isIdContinueChar(u)) throw Eu(x.Identifier)
+ }
+ j += u, z = "identifierName"
+ },
+ string() {
+ switch (J) {
+ case "\\":
+ return uu(), void (j += function(){
+ const u = Y(), D = function(){
+ switch (Y()) {
+ case "b":
+ return uu(), "\b";
+ case "f":
+ return uu(), "\f";
+ case "n":
+ return uu(), "\n";
+ case "r":
+ return uu(), "\r";
+ case "t":
+ return uu(), "\t";
+ case "v":
+ return uu(), "\v"
+ }
+ return
+ }();
+ if (D) return D;
+ switch (u) {
+ case "0":
+ if (uu(), H.JudgeUtil.isDigit(Y())) throw Eu(x.Char, uu());
+ return "\0";
+ case "x":
+ return uu(), function(){
+ let u = "", D = Y();
+ if (!H.JudgeUtil.isHexDigit(D)) throw Eu(x.Char, uu());
+ if (u += uu(), D = Y(),!H.JudgeUtil.isHexDigit(D)) throw Eu(x.Char, uu());
+ return u += uu(), String.fromCodePoint(parseInt(u, 16))
+ }();
+ case "u":
+ return uu(), ru();
+ case "\n":
+ case "\u2028":
+ case "\u2029":
+ return uu(), "";
+ case "\r":
+ return uu(), "\n" === Y() && uu(), ""
+ }
+ if (void 0 === u || H.JudgeUtil.isDigitWithoutZero(u)) throw Eu(x.Char, uu());
+ return uu()
+ }());
+ case '"':
+ case "'":
+ if (J === q) {
+ const u = eu("string", j);
+ return uu(), u
+ }
+ return void (j += uu());
+ case "\n":
+ case "\r":
+ case void 0:
+ throw Eu(x.Char, uu());
+ case "\u2028":
+ case "\u2029":
+ !function(u){
+ console.warn(`JSON5: '${Fu(u)}' in strings is not valid ECMAScript; consider escaping.`)
+ }(J)
+ }
+ j += uu()
+ }
+};
+
+function eu(u, D) {
+ return { type: u, value: D, line: $, column: k }
+}
+
+function tu(u) {
+ for (const D of u) {
+ if (Y() !== D) throw Eu(x.Char, uu());
+ uu()
+ }
+}
+
+function ru() {
+ let u = "", D = 4;
+ for (; D-- > 0; ) {
+ const D = Y();
+ if (!H.JudgeUtil.isHexDigit(D)) throw Eu(x.Char, uu());
+ u += uu()
+ }
+ return String.fromCodePoint(parseInt(u, 16))
+}
+
+const nu = {
+ start() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ iu()
+ },
+ beforePropertyName() {
+ switch (M.type) {
+ case "identifier":
+ case "string":
+ return V = M.value, void (W = "afterPropertyName");
+ case "punctuator":
+ return void Cu();
+ case "eof":
+ throw Eu(x.EOF)
+ }
+ },
+ afterPropertyName() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ W = "beforePropertyValue"
+ },
+ beforePropertyValue() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ iu()
+ },
+ afterPropertyValue() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ switch (M.value) {
+ case ",":
+ return void (W = "beforePropertyName");
+ case "}":
+ Cu()
+ }
+ },
+ beforeArrayValue() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ "punctuator" !== M.type || "]" !== M.value ? iu() : Cu()
+ },
+ afterArrayValue() {
+ if ("eof" === M.type) throw Eu(x.EOF);
+ switch (M.value) {
+ case ",":
+ return void (W = "beforeArrayValue");
+ case "]":
+ Cu()
+ }
+ },
+ end() {
+ }
+};
+
+function iu() {
+ const u = function(){
+ let u;
+ switch (M.type) {
+ case "punctuator":
+ switch (M.value) {
+ case "{":
+ u = {};
+ break;
+ case "[":
+ u = []
+ }
+ break;
+ case "null":
+ case "boolean":
+ case "numeric":
+ case "string":
+ u = M.value
+ }
+ return u
+ }();
+ if (K && "object" == typeof u && (u._line = $, u._column = k), void 0 === G) G = u; else {
+ const D = U[U.length-1];
+ Array.isArray(D) ? K && "object" != typeof u ? D.push({
+ value: u,
+ _line: $,
+ _column: k
+ }) : D.push(u) : D[V] = K && "object" != typeof u ? { value: u, _line: $, _column: k } : u
+ }
+ !function(u){
+ if (u && "object" == typeof u) U.push(u), W = Array.isArray(u) ? "beforeArrayValue" : "beforePropertyName"; else {
+ const u = U[U.length-1];
+ W = u ? Array.isArray(u) ? "afterArrayValue" : "afterPropertyValue" : "end"
+ }
+ }(u)
+}
+
+function Cu() {
+ U.pop();
+ const u = U[U.length-1];
+ W = u ? Array.isArray(u) ? "afterArrayValue" : "afterPropertyValue" : "end"
+}
+
+function Fu(u) {
+ const D = {
+ "'": "\\'",
+ '"': '\\"',
+ "\\": "\\\\",
+ "\b": "\\b",
+ "\f": "\\f",
+ "\n": "\\n",
+ "\r": "\\r",
+ "\t": "\\t",
+ "\v": "\\v",
+ "\0": "\\0",
+ "\u2028": "\\u2028",
+ "\u2029": "\\u2029"
+ };
+ if (D[u]) return D[u];
+ if (u < " ") {
+ const D = u.charCodeAt(0).toString(16);
+ return `\\x${`00${D}`.substring(D.length)}`
+ }
+ return u
+}
+
+function Eu(u, D) {
+ let e = "";
+ switch (u) {
+ case x.Char:
+ e = void 0 === D ? `JSON5: invalid end of input at ${$}:${k}` : `JSON5: invalid character '${Fu(D)}' at ${$}:${k}`;
+ break;
+ case x.EOF:
+ e = `JSON5: invalid end of input at ${$}:${k}`;
+ break;
+ case x.Identifier:
+ k -= 5, e = `JSON5: invalid identifier character at ${$}:${k}`
+ }
+ const t = new Au(e);
+ return t.lineNumber = $, t.columnNumber = k, t
+}
+
+class Au extends SyntaxError {
+}
+
+var ou = {}, au = n && n.__createBinding || (Object.create ? function (u, D, e, t) {
+ void 0 === t && (t = e);
+ var r = Object.getOwnPropertyDescriptor(D, e);
+ r && !("get" in r ? !D.__esModule : r.writable || r.configurable) || (r = { enumerable: !0, get: function () {
+ return D[e]
+ } }), Object.defineProperty(u, t, r)
+} : function (u, D, e, t) {
+ void 0 === t && (t = e), u[t] = D[e]
+}), cu = n && n.__setModuleDefault || (Object.create ? function (u, D) {
+ Object.defineProperty(u, "default", { enumerable: !0, value: D })
+} : function (u, D) {
+ u.default = D
+}), su = n && n.__importStar || function (u) {
+ if (u && u.__esModule) return u;
+ var D = {};
+ if (null != u) for (var e in u) "default" !== e && Object.prototype.hasOwnProperty.call(u, e) && au(D, u, e);
+ return cu(D, u), D
+}, lu = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+};
+Object.defineProperty(ou, "__esModule", {
+ value: !0
+}), ou.isFileExists = ou.offlinePluginConversion = ou.executeCommand = ou.getNpmPath = ou.hasNpmPackInPaths = void 0;
+const Bu = r, du = lu(e), fu = su(u), _u = i, pu = l;
+ou.hasNpmPackInPaths = function (u, D) {
+ try {
+ return require.resolve(u, { paths: [...D] }),!0
+ } catch (u) {
+ return !1
+ }
+}, ou.getNpmPath = function () {
+ const u = process.execPath;
+ return fu.join(fu.dirname(u), _u.NPM_TOOL)
+}, ou.executeCommand = function (u, D, e) {
+ 0 !== (0, Bu.spawnSync)(u, D, e).status && (0, pu.logErrorAndExit)(`Error: ${u} ${D} execute failed.See above for details.`)
+}, ou.offlinePluginConversion = function (u, D) {
+ return D.startsWith("file:") || D.endsWith(".tgz") ? fu.resolve(u, _u.HVIGOR, D.replace("file:", "")) : D
+}, ou.isFileExists = function (u) {
+ return du.default.existsSync(u) && du.default.statSync(u).isFile()
+};
+var Ou = n && n.__createBinding || (Object.create ? function (u, D, e, t) {
+ void 0 === t && (t = e);
+ var r = Object.getOwnPropertyDescriptor(D, e);
+ r && !("get" in r ? !D.__esModule : r.writable || r.configurable) || (r = { enumerable: !0, get: function () {
+ return D[e]
+ } }), Object.defineProperty(u, t, r)
+} : function (u, D, e, t) {
+ void 0 === t && (t = e), u[t] = D[e]
+}), hu = n && n.__setModuleDefault || (Object.create ? function (u, D) {
+ Object.defineProperty(u, "default", { enumerable: !0, value: D })
+} : function (u, D) {
+ u.default = D
+}), Pu = n && n.__importStar || function (u) {
+ if (u && u.__esModule) return u;
+ var D = {};
+ if (null != u) for (var e in u) "default" !== e && Object.prototype.hasOwnProperty.call(u, e) && Ou(D, u, e);
+ return hu(D, u), D
+}, vu = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+};
+Object.defineProperty(P, "__esModule", { value: !0 });
+var gu = P.initProjectWorkSpace = void 0;
+const mu = Pu(e), Ru = vu(D), yu = Pu(u), Iu = v, Nu = i, bu = g, Su = l, wu = ou;
+let Hu, xu, Mu;
+
+function Tu(u, D, e) {
+ return void 0 !== e.dependencies && (0, wu.offlinePluginConversion)(Nu.HVIGOR_PROJECT_ROOT_DIR, D.dependencies[u]) === yu.normalize(e.dependencies[u])
+}
+
+function Vu() {
+ const u = yu.join(Mu, Nu.WORK_SPACE);
+ if ((0, Su.logInfoPrintConsole)("Hvigor cleaning..."),!mu.existsSync(u)) return;
+ const D = mu.readdirSync(u);
+ if (!D || 0 === D.length) return;
+ const e = yu.resolve(Mu, "node_modules", "@ohos", "hvigor", "bin", "hvigor.js");
+ mu.existsSync(e) && (0, wu.executeCommand)(process.argv[0], [e, "--stop-daemon"], {});
+ try {
+ D.forEach((D => {
+ mu.rmSync(yu.resolve(u, D), { recursive: !0 })
+ }))
+ } catch (D) {
+ (0, Su.logErrorAndExit)(`The hvigor build tool cannot be installed. Please manually clear the workspace directory and synchronize the project again.\n\n Workspace Path: ${u}.`)
+ }
+}
+
+gu = P.initProjectWorkSpace = function () {
+ if (Hu = function(){
+ const u = yu.resolve(Nu.HVIGOR_PROJECT_WRAPPER_HOME, Nu.DEFAULT_HVIGOR_CONFIG_JSON_FILE_NAME);
+ mu.existsSync(u) || (0, Su.logErrorAndExit)(`Error: Hvigor config file ${u} does not exist.`);
+ return (0, bu.parseJsonFile)(u)
+ }(), Mu = function(u){
+ let D;
+ D = function(u){
+ let D = u.hvigorVersion;
+ if (D.startsWith("file:") || D.endsWith(".tgz")) return !1;
+ const e = u.dependencies, t = Object.getOwnPropertyNames(e);
+ for (const u of t) {
+ const D = e[u];
+ if (D.startsWith("file:") || D.endsWith(".tgz")) return !1
+ }
+ if (1 === t.length && "@ohos/hvigor-ohos-plugin" === t[0]) return D > "2.5.0";
+ return !1
+ }(u) ? function(u){
+ let D = `${Nu.HVIGOR_ENGINE_PACKAGE_NAME}@${u.hvigorVersion}`;
+ const e = u.dependencies;
+ if (e) {
+ Object.getOwnPropertyNames(e).sort().forEach((u => {
+ D += `,${u}@${e[u]}`
+ }))
+ }
+ return (0, Iu.hash)(D)
+ }(u) : (0, Iu.hash)(process.cwd());
+ return yu.resolve(Ru.default.homedir(), ".hvigor", "project_caches", D)
+ }(Hu), xu = function(){
+ const u = yu.resolve(Mu, Nu.WORK_SPACE, Nu.DEFAULT_PACKAGE_JSON);
+ return mu.existsSync(u) ? (0, bu.parseJsonFile)(u) : { dependencies: {} }
+ }(),!(0, wu.hasNpmPackInPaths)(Nu.HVIGOR_ENGINE_PACKAGE_NAME, [yu.join(Mu, Nu.WORK_SPACE)]) || (0, wu.offlinePluginConversion)(Nu.HVIGOR_PROJECT_ROOT_DIR, Hu.hvigorVersion) !== xu.dependencies[Nu.HVIGOR_ENGINE_PACKAGE_NAME] || !function(){
+
+ function u(u) {
+ const D = null == u ? void 0 : u.dependencies;
+ return void 0 === D ? 0 : Object.getOwnPropertyNames(D).length
+ }
+
+ const D = u(Hu), e = u(xu);
+ if (D + 1 !== e) return !1;
+ for (const u in null == Hu ? void 0 : Hu.dependencies) if (!(0, wu.hasNpmPackInPaths)(u, [yu.join(Mu, Nu.WORK_SPACE)]) || !Tu(u, Hu, xu)) return !1;
+ return !0
+ }()) {
+ Vu();
+ try {
+ !function(){
+ (0, Su.logInfoPrintConsole)("Hvigor installing...");
+ for (const u in Hu.dependencies) Hu.dependencies[u] && (Hu.dependencies[u] = (0, wu.offlinePluginConversion)(Nu.HVIGOR_PROJECT_ROOT_DIR, Hu.dependencies[u]));
+ const u = { dependencies: { ...Hu.dependencies } };
+ u.dependencies[Nu.HVIGOR_ENGINE_PACKAGE_NAME] = (0, wu.offlinePluginConversion)(Nu.HVIGOR_PROJECT_ROOT_DIR, Hu.hvigorVersion);
+ const D = yu.join(Mu, Nu.WORK_SPACE);
+ try {
+ mu.mkdirSync(D, { recursive: !0 });
+ const e = yu.resolve(D, Nu.DEFAULT_PACKAGE_JSON);
+ mu.writeFileSync(e, JSON.stringify(u))
+ } catch (u) {
+ (0, Su.logErrorAndExit)(u)
+ }
+ (function () {
+ const u = ["config", "set", "store-dir", Nu.HVIGOR_PNPM_STORE_PATH],
+ D = { cwd: yu.join(Mu, Nu.WORK_SPACE), stdio: ["inherit", "inherit", "inherit"] };
+ (0, wu.executeCommand)(Nu.HVIGOR_WRAPPER_PNPM_SCRIPT_PATH, u, D)
+ })(), function(){
+ const u = ["install"],
+ D = { cwd: yu.join(Mu, Nu.WORK_SPACE), stdio: ["inherit", "inherit", "inherit"] };
+ (0, wu.executeCommand)(Nu.HVIGOR_WRAPPER_PNPM_SCRIPT_PATH, u, D)
+ }(), (0, Su.logInfoPrintConsole)("Hvigor install success.")
+ }()
+ } catch (u) {
+ Vu()
+ }
+ }
+ return Mu
+};
+var Gu = {};
+!function(t){
+ var C = n && n.__createBinding || (Object.create ? function (u, D, e, t) {
+ void 0 === t && (t = e);
+ var r = Object.getOwnPropertyDescriptor(D, e);
+ r && !("get" in r ? !D.__esModule : r.writable || r.configurable) || (r = { enumerable: !0, get: function () {
+ return D[e]
+ } }), Object.defineProperty(u, t, r)
+ } : function (u, D, e, t) {
+ void 0 === t && (t = e), u[t] = D[e]
+ }), F = n && n.__setModuleDefault || (Object.create ? function (u, D) {
+ Object.defineProperty(u, "default", { enumerable: !0, value: D })
+ } : function (u, D) {
+ u.default = D
+ }), E = n && n.__importStar || function (u) {
+ if (u && u.__esModule) return u;
+ var D = {};
+ if (null != u) for (var e in u) "default" !== e && Object.prototype.hasOwnProperty.call(u, e) && C(D, u, e);
+ return F(D, u), D
+ }, A = n && n.__importDefault || function (u) {
+ return u && u.__esModule ? u : { default: u }
+ };
+ Object.defineProperty(t, "__esModule", {
+ value: !0
+ }), t.executeInstallPnpm = t.isPnpmInstalled = t.environmentHandler = t.checkNpmConifg = t.PNPM_VERSION = void 0;
+ const o = r, a = E(e), c = A(D), s = E(u), B = i, d = l, f = ou;
+ t.PNPM_VERSION = "7.30.0", t.checkNpmConifg = function () {
+ const u = s.resolve(B.HVIGOR_PROJECT_ROOT_DIR, ".npmrc"), D = s.resolve(c.default.homedir(), ".npmrc");
+ if ((0, f.isFileExists)(u) || (0, f.isFileExists)(D)) return;
+ const e = (0, f.getNpmPath)(),
+ t = (0, o.spawnSync)(e, ["config", "get", "prefix"], { cwd: B.HVIGOR_PROJECT_ROOT_DIR });
+ if (0 !== t.status || !t.stdout) return void (0, d.logErrorAndExit)("Error: The hvigor depends on the npmrc file. Configure the npmrc file first.");
+ const r = s.resolve(`${t.stdout}`.replace(/[\r\n]/gi, ""), ".npmrc");
+ (0, f.isFileExists)(r) || (0, d.logErrorAndExit)("Error: The hvigor depends on the npmrc file. Configure the npmrc file first.")
+ }, t.environmentHandler = function () {
+ process.env["npm_config_update-notifier"] = "false"
+ }, t.isPnpmInstalled = function () {
+ return!!a.existsSync(B.HVIGOR_WRAPPER_PNPM_SCRIPT_PATH) && (0, f.hasNpmPackInPaths)("pnpm", [B.HVIGOR_WRAPPER_TOOLS_HOME])
+ }, t.executeInstallPnpm = function () {
+ (0, d.logInfoPrintConsole)(`Installing pnpm@${t.PNPM_VERSION}...`);
+ const u = (0, f.getNpmPath)();
+ !function(){
+ const u = s.resolve(B.HVIGOR_WRAPPER_TOOLS_HOME, B.DEFAULT_PACKAGE_JSON);
+ try {
+ a.existsSync(B.HVIGOR_WRAPPER_TOOLS_HOME) || a.mkdirSync(B.HVIGOR_WRAPPER_TOOLS_HOME, {
+ recursive: !0
+ });
+ const D = { dependencies: {} };
+ D.dependencies[B.PNPM] = t.PNPM_VERSION, a.writeFileSync(u, JSON.stringify(D))
+ } catch (D) {
+ (0, d.logErrorAndExit)(`Error: EPERM: operation not permitted,create ${u} failed.`)
+ }
+ }(), (0, f.executeCommand)(u, ["install", "pnpm"], {
+ cwd: B.HVIGOR_WRAPPER_TOOLS_HOME,
+ stdio: ["inherit", "inherit", "inherit"],
+ env: process.env
+ }), (0, d.logInfoPrintConsole)("Pnpm install success.")
+ }
+}(Gu), function(){
+ Gu.checkNpmConifg(), Gu.environmentHandler(), Gu.isPnpmInstalled() || Gu.executeInstallPnpm();
+ const D = gu();
+ _(u.join(D, i.WORK_SPACE))
+}();
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/hvigorfile.ts b/scenario/arkui/MyNews/hvigorfile.ts
new file mode 100644
index 0000000000000000000000000000000000000000..20a8c119149bca149dc03919c5c1c022cd4ff077
--- /dev/null
+++ b/scenario/arkui/MyNews/hvigorfile.ts
@@ -0,0 +1,6 @@
+import { appTasks } from '@ohos/hvigor-ohos-plugin';
+
+export default {
+ system: appTasks, /* Built-in plugin of Hvigor. It cannot be modified. */
+ plugins: [] /* Custom plugin to extend the functionality of Hvigor. */
+}
diff --git a/scenario/arkui/MyNews/hvigorw b/scenario/arkui/MyNews/hvigorw
new file mode 100644
index 0000000000000000000000000000000000000000..ff6a29a2ac9e13860657c8e33464123f4da4b7ed
--- /dev/null
+++ b/scenario/arkui/MyNews/hvigorw
@@ -0,0 +1,48 @@
+#!/bin/bash
+
+# ----------------------------------------------------------------------------
+# Hvigor startup script, version 1.0.0
+#
+# Required ENV vars:
+# ------------------
+# NODE_HOME - location of a Node home dir
+# or
+# Add /usr/local/nodejs/bin to the PATH environment variable
+# ----------------------------------------------------------------------------
+
+HVIGOR_APP_HOME="`pwd -P`"
+HVIGOR_WRAPPER_SCRIPT=${HVIGOR_APP_HOME}/hvigor/hvigor-wrapper.js
+warn() {
+ echo ""
+ echo -e "\033[1;33m`date '+[%Y-%m-%d %H:%M:%S]'`$@\033[0m"
+}
+
+error() {
+ echo ""
+ echo -e "\033[1;31m`date '+[%Y-%m-%d %H:%M:%S]'`$@\033[0m"
+}
+
+fail() {
+ error "$@"
+ exit 1
+}
+
+# Determine node to start hvigor wrapper script
+if [ -n "${NODE_HOME}" ];then
+ EXECUTABLE_NODE="${NODE_HOME}/bin/node"
+ if [ ! -x "$EXECUTABLE_NODE" ];then
+ fail "ERROR: NODE_HOME is set to an invalid directory,check $NODE_HOME\n\nPlease set NODE_HOME in your environment to the location where your nodejs installed"
+ fi
+else
+ EXECUTABLE_NODE="node"
+ which ${EXECUTABLE_NODE} > /dev/null 2>&1 || fail "ERROR: NODE_HOME is not set and not 'node' command found in your path"
+fi
+
+# Check hvigor wrapper script
+if [ ! -r "$HVIGOR_WRAPPER_SCRIPT" ];then
+ fail "ERROR: Couldn't find hvigor/hvigor-wrapper.js in ${HVIGOR_APP_HOME}"
+fi
+
+# start hvigor-wrapper script
+exec "${EXECUTABLE_NODE}" \
+ "${HVIGOR_WRAPPER_SCRIPT}" "$@"
diff --git a/scenario/arkui/MyNews/hvigorw.bat b/scenario/arkui/MyNews/hvigorw.bat
new file mode 100644
index 0000000000000000000000000000000000000000..d570007e8a178639a230d95b94a09dcda89969d8
--- /dev/null
+++ b/scenario/arkui/MyNews/hvigorw.bat
@@ -0,0 +1,64 @@
+@if "%DEBUG%" == "" @echo off
+@rem ##########################################################################
+@rem
+@rem Hvigor startup script for Windows
+@rem
+@rem ##########################################################################
+
+@rem Set local scope for the variables with windows NT shell
+if "%OS%"=="Windows_NT" setlocal
+
+set DIRNAME=%~dp0
+if "%DIRNAME%" == "" set DIRNAME=.
+set APP_BASE_NAME=%~n0
+set APP_HOME=%DIRNAME%
+
+@rem Resolve any "." and ".." in APP_HOME to make it shorter.
+for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
+
+set WRAPPER_MODULE_PATH=%APP_HOME%\hvigor\hvigor-wrapper.js
+set NODE_EXE=node.exe
+
+goto start
+
+:start
+@rem Find node.exe
+if defined NODE_HOME goto findNodeFromNodeHome
+
+%NODE_EXE% --version >NUL 2>&1
+if "%ERRORLEVEL%" == "0" goto execute
+
+echo.
+echo ERROR: NODE_HOME is not set and no 'node' command could be found in your PATH.
+echo.
+echo Please set the NODE_HOME variable in your environment to match the
+echo location of your NodeJs installation.
+
+goto fail
+
+:findNodeFromNodeHome
+set NODE_HOME=%NODE_HOME:"=%
+set NODE_EXE_PATH=%NODE_HOME%/%NODE_EXE%
+
+if exist "%NODE_EXE_PATH%" goto execute
+echo.
+echo ERROR: NODE_HOME is not set and no 'node' command could be found in your PATH.
+echo.
+echo Please set the NODE_HOME variable in your environment to match the
+echo location of your NodeJs installation.
+
+goto fail
+
+:execute
+@rem Execute hvigor
+"%NODE_EXE%" "%WRAPPER_MODULE_PATH%" %*
+
+if "%ERRORLEVEL%" == "0" goto hvigorwEnd
+
+:fail
+exit /b 1
+
+:hvigorwEnd
+if "%OS%" == "Windows_NT" endlocal
+
+:end
diff --git a/scenario/arkui/MyNews/oh-package-lock.json5 b/scenario/arkui/MyNews/oh-package-lock.json5
new file mode 100644
index 0000000000000000000000000000000000000000..ddc8a54dc3a0148e1eab04cfb0c1d72e5077c01f
--- /dev/null
+++ b/scenario/arkui/MyNews/oh-package-lock.json5
@@ -0,0 +1,13 @@
+{
+ "lockfileVersion": 1,
+ "ATTENTION": "THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.",
+ "specifiers": {
+ "@ohos/hypium@1.0.11": "@ohos/hypium@1.0.11"
+ },
+ "packages": {
+ "@ohos/hypium@1.0.11": {
+ "resolved": "https://repo.harmonyos.com/ohpm/@ohos/hypium/-/hypium-1.0.11.tgz",
+ "integrity": "sha512-KawcLnv43C3QIYv1UbDnKCFX3MohtDxGuFvzlUxT/qf2DBilR56Ws6zrj90LdH6PjloJQwOPESuBQIHBACAK7w=="
+ }
+ }
+}
\ No newline at end of file
diff --git a/scenario/arkui/MyNews/oh-package.json5 b/scenario/arkui/MyNews/oh-package.json5
new file mode 100644
index 0000000000000000000000000000000000000000..f6e0108275a18c8e881f75e44b19c6aecb0c8193
--- /dev/null
+++ b/scenario/arkui/MyNews/oh-package.json5
@@ -0,0 +1,12 @@
+{
+ "license": "",
+ "devDependencies": {
+ "@ohos/hypium": "1.0.11"
+ },
+ "author": "",
+ "name": "mynews",
+ "description": "Please describe the basic information.",
+ "main": "",
+ "version": "1.0.0",
+ "dependencies": {}
+}
diff --git a/scenario/arkui/MyNews/screenshots/home-following.png b/scenario/arkui/MyNews/screenshots/home-following.png
new file mode 100644
index 0000000000000000000000000000000000000000..682043541d5e281fd30678d5d5dbfd988f41a8d6
Binary files /dev/null and b/scenario/arkui/MyNews/screenshots/home-following.png differ
diff --git a/scenario/arkui/MyNews/screenshots/home-recommendation.png b/scenario/arkui/MyNews/screenshots/home-recommendation.png
new file mode 100644
index 0000000000000000000000000000000000000000..969bfe7631da45abf29e0ba1c74809b7f2515ce6
Binary files /dev/null and b/scenario/arkui/MyNews/screenshots/home-recommendation.png differ