介紹
HarmonyOS ArkUI提供了豐富多樣的UI組件,您可以使用這些組件輕松地編寫出更加豐富、漂亮的界面。在本篇Codelab中,您將通過一個簡單的購物社交應用示例,學習如何使用常用的基礎組件和容器組件。
本示例主要包含:“登錄”、“首頁”、“我的”三個頁面,效果圖如下:
相關概念
- [Text]:顯示一段文本的組件。
- [Image]:圖片組件,支持本地圖片和網絡圖片的渲染展示。
- [TextInput]:可以輸入單行文本并支持響應輸入事件的組件。
- [Button]:按鈕組件,可快速創建不同樣式的按鈕。
- [LoadingProgress]:用于顯示加載動效的組件。
- [Flex]:應用彈性方式布局子組件的容器組件。
- [Column]:沿垂直方向布局的容器。
- [Row]:沿水平方向布局容器。
- [List]:列表包含一系列相同寬度的列表項。適合連續、多行呈現同類數據,例如圖片和文本。
- [Swiper]:滑動容器,提供切換子組件顯示的能力。
- [Grid]:網格容器,由“行”和“列”分割的單元格所組成,通過指定“項目”所在的單元格做出各種各樣的布局。
環境搭建
軟件要求
- [DevEco Studio]版本:DevEco Studio 3.1 Release。
- OpenHarmony SDK版本:API version 9。
硬件要求
- 開發板類型:[潤和RK3568開發板]。
- OpenHarmony系統:3.2 Release。
環境搭建
我們首先要完成開發環境的搭建,本示例以RK3568開發板為例,參照以下步驟進行:
- [獲取OpenHarmony系統版本]:標準系統解決方案(二進制)。以3.2 Release版本為例:
- 搭建燒錄環境。
- [完成DevEco Device Tool的安裝]
- [完成RK3568開發板的燒錄]
- 搭建開發環境。
- 開始前請參考[工具準備],完成DevEco Studio的安裝和開發環境配置。
- 開發環境配置完成后,請參考[使用工程向導]創建工程(模板選擇“Empty Ability”)。
- 工程創建完成后,選擇使用[真機進行調測]。
- 鴻蒙開發指導文檔:[
qr23.cn/FBD4cY
]
代碼結構解讀
本篇Codelab只對核心代碼進行講解,對于完整代碼,我們會在附件下載和gitee源碼中提供下載方式。
├──entry/src/main/ets // 代碼區
│ ├──common
│ │ └──constants
│ │ └──CommonConstants.ets // 公共常量類
│ ├──entryability
│ │ └──EntryAbility.ts // 程序入口類
│ ├──pages
│ │ ├──LoginPage.ets // 登錄界面
│ │ └──MainPage.ets // 主界面
│ ├──view
│ │ ├──Home.ets // 首頁
│ │ └──Setting.ets // 設置頁
│ └──viewmodel
│ ├──ItemData.ets // 列表數據實體類
│ └──MainViewModel.ets // 主界面視圖Model
└──entry/src/main/resources // 應用資源目錄
`HarmonyOS與OpenHarmony鴻蒙文檔籽料:mau123789是v直接拿`
實現“登錄”頁面
本節主要介紹“登錄”頁面的實現,效果圖如下:
界面使用Column容器組件布局,由Image、Text、TextInput、Button、LoadingProgress等基礎組件構成,主要代碼如下:
// LoginPage.ets
@Entry
@Component
struct LoginPage {
...
build() {
Column() {
Image($r('app.media.logo'))
...
Text($r('app.string.login_page'))
...
Text($r('app.string.login_more'))
...
TextInput({ placeholder: $r('app.string.account') })
...
TextInput({ placeholder: $r('app.string.password') })
...
Row() {
Text($r('app.string.message_login')).blueTextStyle()
Text($r('app.string.forgot_password')).blueTextStyle()
}
....
Button($r('app.string.login'), { type: ButtonType.Capsule })
....
Text($r('app.string.register_account'))
....
if (this.isShowProgress) {
LoadingProgress()
....
}
Blank()
Text($r('app.string.other_login_method'))
....
Row({ space: CommonConstants.LOGIN_METHODS_SPACE }) {
this.imageButton($r('app.media.login_method1'))
this.imageButton($r('app.media.login_method2'))
this.imageButton($r('app.media.login_method3'))
}
}
....
}
}
獲取用戶輸入
當用戶登錄前,需要獲取用戶輸入的帳號和密碼才能執行登錄邏輯。給TextInput設置onChange事件,在onChange事件里面實時獲取用戶輸入的文本信息。
// LoginPage.ets
TextInput({ placeholder: $r('app.string.account') })
.maxLength(CommonConstants.INPUT_ACCOUNT_LENGTH)
.type(InputType.Number)
.inputStyle()
.onChange((value: string) = > {
this.account = value;
})
控制LoadingProgress顯示和隱藏
給登錄按鈕綁定onClick事件,調用login方法模擬登錄。定義變量isShowProgress結合條件渲染if用來控制LoadingProgress的顯示和隱藏。當用戶點擊按鈕時設置isShowProgress為true,即顯示LoadingProgress;使用定時器setTimeout設置isShowProgress 2秒后為false,即隱藏LoadingProgress,然后執行跳轉到首頁的邏輯。
// LoginPage.ets
@Entry
@Component
struct LoginPage {
@State account: string = '';
@State password: string = '';
@State isShowProgress: boolean = false;
private timeOutId = null;
...
login() {
if (this.account === '' || this.password === '') {
prompt.showToast({
message: $r('app.string.input_empty_tips')
})
} else {
this.isShowProgress = true;
if (this.timeOutId === null) {
this.timeOutId = setTimeout(() = > {
this.isShowProgress = false;
this.timeOutId = null;
router.replaceUrl({ url: 'pages/MainPage' });
}, CommonConstants.LOGIN_DELAY_TIME);
}
}
}
...
build() {
Column() {
...
Button($r('app.string.login'), { type: ButtonType.Capsule })
....
.onClick(() = > {
this.login();
})
...
if (this.isShowProgress) {
LoadingProgress()
.color($r('app.color.loading_color'))
.width($r('app.float.login_progress_size'))
.height($r('app.float.login_progress_size'))
.margin({ top: $r('app.float.login_progress_margin_top') })
}
...
}
...
}
}
實現頁面跳轉
頁面間的跳轉可以使用router模塊相關API來實現,使用前需要先導入該模塊,然后使用router.replace()方法實現頁面跳轉。
// LoginPage.ets
import router from '@ohos.router';
login() {
if (this.account === '' || this.password === '') {
...
} else {
this.isShowProgress = true;
if (this.timeOutId === -1) {
this.timeOutId = setTimeout(() = > {
this.isShowProgress = false;
this.timeOutId = -1;
router.replaceUrl({ url: 'pages/MainPage' });
}, CommonConstants.LOGIN_DELAY_TIME);
}
}
}
實現“首頁”和“我的”頁面
定義資源數據
由于“首頁”和“我的”頁面中有多處圖片和文字的組合,因此提取出ItemData類。在MainViewModel.ets文件中對頁面使用的資源進行定義,在MainViewModel.ets文件中定義數據。
// ItemData.ets
export default class PageResource {
title: Resource;
img?: Resource;
others?: Resource;
constructor(title: Resource, img?: Resource, others?: Resource) {
this.title = title;
this.img = img;
this.others = others;
}
}
// MainViewModel.ets
import ItemData from './temData';
export class MainViewModel {
...
getFirstGridData(): Array< ItemData > {
let firstGridData: ItemData[] = [
new ItemData($r('app.string.my_love'), $r('app.media.love')),
new ItemData($r('app.string.history_record'), $r('app.media.record')),
...
];
return firstGridData;
}
...
}
export default new MainViewModel();
實現頁面框架
從前面介紹章節的示意圖可以看出,本示例由兩個tab頁組成,使用Tabs組件來實現,提取tabBar的公共樣式,同時設置TabContent和Tabs的backgroundColor來實現底部tabBar欄背景色突出的效果。
// MainPage.ets
Tabs({
barPosition: BarPosition.End,
controller: this.tabsController
}) {
TabContent() {
...
}
...
.backgroundColor($r('app.color.mainPage_backgroundColor')) // “首頁”的頁面背景色
.tabBar(this.TabBuilder(CommonConstants.HOME_TITLE, CommonConstants.HOME_TAB_INDEX,
$r('app.media.home_selected'), $r('app.media.home_normal')))
...
}
...
.backgroundColor(Color.White) // 底部tabBar欄背景色
...
.onChange((index: number) = > {
this.currentIndex = index;
})
...
實現“首頁”內容
“首頁”效果如下所示:
從上面效果如可以看出“首頁”由三部分內容組成分別是輪播圖、24柵格圖、44柵格圖。首先使用Swiper組件實現輪播圖,無需設置圖片大小。
// Home.ets
Swiper(this.swiperController) {
ForEach(mainViewModel.getSwiperImages(), (img: Resource) = > {
Image(img).borderRadius($r('app.float.home_swiper_borderRadius'))
}, (img: Resource) = > JSON.stringify(img.id))
}
...
.autoPlay(true)
...
然后使用Grid組件實現2*4柵格圖,代碼如下
// Home.ets
Grid() {
ForEach(mainViewModel.getFirstGridData(), (item: ItemData) = > {
GridItem() {
Column() {
Image(item.img)
.width($r('app.float.home_homeCell_size'))
.height($r('app.float.home_homeCell_size'))
Text(item.title)
.fontSize($r('app.float.little_text_size'))
.margin({ top: $r('app.float.home_homeCell_margin') })
}
}
}, (item: ItemData) = > JSON.stringify(item))
}
.columnsTemplate('1fr 1fr 1fr 1fr')
.rowsTemplate('1fr 1fr')
...
使用Grid組件實現4*4柵格列表欄,其中單個柵格中有一張背景圖片和兩行字體不同的文本,因此在Column組件中放置兩個Text組件,并設置背景圖,注意Grid組件必須設置高度,否則可能出現頁面空白。
// Home.ets
Grid() {
ForEach(mainViewModel.getSecondGridData(), (secondItem: ItemData) = > {
GridItem() {
Column() {
Text(secondItem.title)
...
Text(secondItem.others)
...
}
.alignItems(HorizontalAlign.Start)
}
...
.backgroundImage(secondItem.img)
.backgroundImageSize(ImageSize.Cover)
...
}, (secondItem: ItemData) = > JSON.stringify(secondItem))
}
...
.height($r('app.float.home_secondGrid_height'))
.columnsTemplate('1fr 1fr')
.rowsTemplate('1fr 1fr')
...
實現“我的”頁內容
“我的”頁面效果圖如下:
使用List組件結合ForEach語句來實現頁面列表內容,其中引用了settingCell子組件,列表間的灰色分割線可以使用Divider屬性實現,代碼實現如下:
// Setting.ets
List() {
ForEach(mainViewModel.getSettingListData(), (item: ItemData) = > {
ListItem() {
this.settingCell(item)
}
.height($r('app.float.setting_list_height'))
}, (item: ItemData) = > JSON.stringify(item))
}
...
.divider({ // 設置分隔線
...
})
...
@Builder settingCell(item: ItemData) {
Row() {
Row({ space: CommonConstants.COMMON_SPACE }) {
Image(item.img)
...
Text(item.title)
...
}
if (item.others === null) {
Image($r("app.media.right_grey"))
...
} else {
Toggle({ type: ToggleType.Switch, isOn: false })
}
}
.justifyContent(FlexAlign.SpaceBetween) // 相鄰元素之間距離相同
...
}
審核編輯 黃宇
-
開發板
+關注
關注
25文章
5120瀏覽量
97963 -
組件
+關注
關注
1文章
516瀏覽量
17887 -
鴻蒙
+關注
關注
57文章
2390瀏覽量
42968 -
HarmonyOS
+關注
關注
79文章
1982瀏覽量
30410 -
OpenHarmony
+關注
關注
25文章
3744瀏覽量
16476
發布評論請先 登錄
相關推薦
評論