ui-ts-creating-simple-page.md 16.9 KB
Newer Older
Z
zengyawen 已提交
1 2
# 创建简单视图

T
tianyu 已提交
3
在这一小节中,我们将开始食物详情页的开发,学习如何通过容器组件Stack、Flex和基础组件Image、Text,构建用户自定义组件,完成图文并茂的食物介绍。
Z
zengyawen 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26


## 构建Stack布局

1. 创建食物名称。
   删掉工程模板的build方法的代码,创建Stack组件,将Text组件放进Stack组件的花括号中,使其成为Stack组件的子组件。Stack组件为堆叠组件,可以包含一个或多个子组件,其特点是后一个子组件覆盖前一个子组件。
   ```
   @Entry
   @Component
   struct MyComponent {
     build() {
       Stack() {
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
     }
   }
   ```

   ![zh-cn_image_0000001214128687](figures/zh-cn_image_0000001214128687.png)

2. 食物图片展示。
T
tianyu 已提交
27 28
   创建Image组件,指定Image组件的url,Image组件是必选构造参数组件。为了让Text组件在Image组件上方显示,所以要先声明Image组件。图片资源放在resources下的rawfile文件夹内,引用rawfile下资源时使用`$rawfile('filename')`的形式,filename为rawfile目录下的文件相对路径。当前`$rawfile`仅支持Image控件引用图片资源。
   
Z
zengyawen 已提交
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
   ```
   @Entry
   @Component
   struct MyComponent {
     build() {
       Stack() {
           Image($rawfile('Tomato.png'))
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
     }
   }
   ```

T
tianyu 已提交
44 45

![zh-cn_image_0000001168410342](figures/zh-cn_image_0000001168410342.png)
Z
zengyawen 已提交
46 47

3. 通过资源访问图片。
H
HelloCrease 已提交
48
   除指定图片路径外,也可以使用引用媒体资源符$r引用资源,需要遵循resources文件夹的资源限定词的规则。右键resources文件夹,点击New>Resource Directory,选择Resource Type为Media(图片资源)。
Z
zengyawen 已提交
49

H
HelloCrease 已提交
50
   将Tomato.png放入media文件夹内。就可以通过`$r('app.type.name')`的形式引用应用资源,即`$r('app.media.Tomato')`
Z
zengyawen 已提交
51

T
tianyu 已提交
52 53
```
@Entry
Z
zengyawen 已提交
54 55 56 57 58 59 60 61 62 63 64 65 66
   @Component
   struct MyComponent {
     build() {
       Stack() {
           Image($r('app.media.Tomato'))
               .objectFit(ImageFit.Contain)
               .height(357)
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
     }
   }
T
tianyu 已提交
67 68
```

Z
zengyawen 已提交
69 70 71 72

4. 设置Image宽高,并且将image的objectFit属性设置为ImageFit.Contain,即保持图片长宽比的情况下,使得图片完整地显示在边界内。
   如果Image填满了整个屏幕,原因如下:
   1. Image没有设置宽高。
H
HelloCrease 已提交
73

Z
zengyawen 已提交
74 75
   2. Image的objectFit默认属性是ImageFit.Cover,即在保持长宽比的情况下放大或缩小,使其填满整个显示边界。

T
tianyu 已提交
76 77
```
@Entry
Z
zengyawen 已提交
78 79 80 81 82 83 84 85 86 87 88 89 90
   @Component
   struct MyComponent {
     build() {
       Stack() {
           Image($r('app.media.Tomato'))
               .objectFit(ImageFit.Contain)
               .height(357)
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
     }
   }
T
tianyu 已提交
91 92
```

Z
zengyawen 已提交
93 94 95 96

   ![zh-cn_image_0000001214210217](figures/zh-cn_image_0000001214210217.png)

5. 设置食物图片和名称布局。设置Stack的对齐方式为底部起始端对齐,Stack默认为居中对齐。设置Stack构造参数alignContent为Alignment.BottomStart。其中Alignment和FontWeight一样,都是框架提供的内置枚举类型。
T
tianyu 已提交
97 98 99

```
@Entry
Z
zengyawen 已提交
100 101 102 103 104 105 106 107 108 109 110 111 112
   @Component
   struct MyComponent {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
           Image($r('app.media.Tomato'))
               .objectFit(ImageFit.Contain)
               .height(357)
            Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
     }
   }
T
tianyu 已提交
113 114
```

Z
zengyawen 已提交
115 116 117

   ![zh-cn_image_0000001168728872](figures/zh-cn_image_0000001168728872.png)

T
tianyu 已提交
118
6. 通过设置Stack的背景颜色来改变食物图片的背景颜色,设置颜色有四种方式:
Z
zengyawen 已提交
119 120
   1. 通过框架提供的Color内置枚举值来设置,比如backgroundColor(Color.Red),即设置背景颜色为红色。
   2. string类型参数,支持的颜色格式有:rgb、rgba和HEX颜色码。比如backgroundColor('\#0000FF'),即设置背景颜色为蓝色,backgroundColor('rgb(255, 255, 255)'),即设置背景颜色为白色。
T
tianyu 已提交
121 122 123 124
   3. number类型参数,支持十六进制颜色值。比如backgroundColor(0xFF0000),即设置背景颜色为红色。 
   
   4. Resource类型参数请参考[资源访问](ts-resource-access.md)
   
Z
zengyawen 已提交
125

T
tianyu 已提交
126 127
```
@Entry
Z
zengyawen 已提交
128 129 130 131 132 133 134 135 136 137 138 139 140
   @Component
   struct MyComponent {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
           Image($r('app.media.Tomato'))
               .objectFit(ImageFit.Contain)
               .height(357)
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
       }
       .backgroundColor('#FFedf2f5')
     }
T
tianyu 已提交
141 142 143
}
```

Z
zengyawen 已提交
144 145 146 147 148 149 150

   ![zh-cn_image_0000001168888822](figures/zh-cn_image_0000001168888822.png)

7. 调整Text组件的外边距margin,使其距离左侧和底部有一定的距离。margin是简写属性,可以统一指定四个边的外边距,也可以分别指定。具体设置方式如下:
   1. 参数为Length时,即统一指定四个边的外边距,比如margin(20),即上、右、下、左四个边的外边距都是20。
   2. 参数为{top?: Length, right?: Length, bottom?: Length, left?:Length},即分别指定四个边的边距,比如margin({ left: 26, bottom: 17.4 }),即左边距为26,下边距为17.4。

T
tianyu 已提交
151 152
```
@Entry
Z
zengyawen 已提交
153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
   @Component
   struct MyComponent {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
           Image($r('app.media.Tomato'))
               .objectFit(ImageFit.Contain)
               .height(357)
           Text('Tomato')
               .fontSize(26)
               .fontWeight(500)
               .margin({left: 26, bottom: 17.4})
       }
        .backgroundColor('#FFedf2f5')
     }
   }
T
tianyu 已提交
168 169
```

Z
zengyawen 已提交
170 171 172 173 174 175

   ![zh-cn_image_0000001213968747](figures/zh-cn_image_0000001213968747.png)

8. 调整组件间的结构,语义化组件名称。创建页面入口组件为FoodDetail,在FoodDetail中创建Column,设置水平方向上居中对齐 alignItems(HorizontalAlign.Center)。MyComponent组件名改为FoodImageDisplay,为FoodDetail的子组件。
   Column是子组件竖直排列的容器组件,本质为线性布局,所以只能设置交叉轴方向的对齐。

T
tianyu 已提交
176 177
```
@Component
Z
zengyawen 已提交
178 179 180 181 182 183 184 185 186 187 188 189 190 191
   struct FoodImageDisplay {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
         Image($r('app.media.Tomato'))
           .objectFit(ImageFit.Contain)
         Text('Tomato')
           .fontSize(26)
           .fontWeight(500)
           .margin({ left: 26, bottom: 17.4 })
       }
       .height(357)
       .backgroundColor('#FFedf2f5')
     }
   }
H
HelloCrease 已提交
192

Z
zengyawen 已提交
193 194 195 196 197 198 199 200 201 202
   @Entry
   @Component
   struct FoodDetail {
     build() {
       Column() {
         FoodImageDisplay()
       }
       .alignItems(HorizontalAlign.Center)
     }
   }
T
tianyu 已提交
203 204
```

Z
zengyawen 已提交
205 206 207 208 209 210 211


## 构建Flex布局

开发者可以使用Flex弹性布局来构建食物的食物成分表,弹性布局在本场景的优势在于可以免去多余的宽高计算,通过比例来设置不同单元格的大小,更加灵活。

1. 创建ContentTable组件,使其成为页面入口组件FoodDetail的子组件。
T
tianyu 已提交
212 213 214

```
@Component
Z
zengyawen 已提交
215 216 217 218 219 220 221 222 223 224 225 226 227 228
   struct FoodImageDisplay {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
         Image($r('app.media.Tomato'))
           .objectFit(ImageFit.Contain)
           .height(357)
         Text('Tomato')
           .fontSize(26)
           .fontWeight(500)
           .margin({ left: 26, bottom: 17.4 })
       }
       .backgroundColor('#FFedf2f5')
     }
   }
H
HelloCrease 已提交
229

Z
zengyawen 已提交
230 231 232 233
   @Component
   struct ContentTable {
     build() {}
   }
H
HelloCrease 已提交
234

Z
zengyawen 已提交
235 236 237 238 239 240 241 242 243 244 245
   @Entry
   @Component
   struct FoodDetail {
     build() {
       Column() {
         FoodImageDisplay()
         ContentTable()
       }
       .alignItems(HorizontalAlign.Center)
     }
   }
T
tianyu 已提交
246 247
```

Z
zengyawen 已提交
248 249 250 251 252 253 254 255

2. 创建Flex组件展示Tomato两类成分。
   一类是热量Calories,包含卡路里(Calories);一类是营养成分Nutrition,包含蛋白质(Protein)、脂肪(Fat)、碳水化合物(Carbohydrates)和维生素C(VitaminC)。

   先创建热量这一类。创建Flex组件,高度为280,上、右、左内边距为30,包含三个Text子组件分别代表类别名(Calories),含量名称(Calories)和含量数值(17kcal)。Flex组件默认为水平排列方式。

   已省略FoodImageDisplay代码,只针对ContentTable进行扩展。

T
tianyu 已提交
256 257
```
@Component
Z
zengyawen 已提交
258 259 260 261 262 263 264 265 266 267 268 269 270 271 272
   struct ContentTable {
     build() {
       Flex() {
         Text('Calories')
           .fontSize(17.4)
           .fontWeight(FontWeight.Bold)
         Text('Calories')
           .fontSize(17.4)
         Text('17kcal')
           .fontSize(17.4)
       }
       .height(280)
       .padding({ top: 30, right: 30, left: 30 })
     }
   }
H
HelloCrease 已提交
273

Z
zengyawen 已提交
274 275 276 277 278 279 280 281 282 283 284
   @Entry
   @Component
   struct FoodDetail {
     build() {
       Column() {
         FoodImageDisplay()
         ContentTable()
       }
       .alignItems(HorizontalAlign.Center)
     }
   }
T
tianyu 已提交
285 286
```

Z
zengyawen 已提交
287 288 289 290

   ![zh-cn_image_0000001169759552](figures/zh-cn_image_0000001169759552.png)

3. 调整布局,设置各部分占比。分类名占比(layoutWeight)为1,成分名和成分含量一共占比(layoutWeight)2。成分名和成分含量位于同一个Flex中,成分名占据所有剩余空间flexGrow(1)。
T
tianyu 已提交
291 292 293

```
@Component
Z
zengyawen 已提交
294 295 296
   struct FoodImageDisplay {
     build() {
       Stack({ alignContent: Alignment.BottomStart }) {
297
         Image($r('app.media.Tomato'))
Z
zengyawen 已提交
298 299 300 301 302 303 304 305 306 307
           .objectFit(ImageFit.Contain)
           .height(357)
         Text('Tomato')
           .fontSize(26)
           .fontWeight(500)
           .margin({ left: 26, bottom: 17.4 })
       }
       .backgroundColor('#FFedf2f5')
     }
   }
H
HelloCrease 已提交
308

Z
zengyawen 已提交
309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
   @Component
   struct ContentTable {
     build() {
       Flex() {
         Text('Calories')
           .fontSize(17.4)
           .fontWeight(FontWeight.Bold)
           .layoutWeight(1)
         Flex() {
           Text('Calories')
             .fontSize(17.4)
             .flexGrow(1)
           Text('17kcal')
             .fontSize(17.4)
         }
         .layoutWeight(2)
       }
       .height(280)
       .padding({ top: 30, right: 30, left: 30 })
     }
   }
H
HelloCrease 已提交
330

Z
zengyawen 已提交
331 332 333 334 335 336 337 338 339 340 341
   @Entry
   @Component
   struct FoodDetail {
     build() {
       Column() {
         FoodImageDisplay()
         ContentTable()
       }
       .alignItems(HorizontalAlign.Center)
     }
   }
T
tianyu 已提交
342 343
```

Z
zengyawen 已提交
344 345 346 347 348 349

   ![zh-cn_image_0000001215079443](figures/zh-cn_image_0000001215079443.png)

4. 仿照热量分类创建营养成分分类。营养成分部分(Nutrition)包含:蛋白质(Protein)、脂肪(Fat)、碳水化合物(Carbohydrates)和维生素C(VitaminC)四个成分,后三个成分在表格中省略分类名,用空格代替。
   设置外层Flex为竖直排列FlexDirection.Column, 在主轴方向(竖直方向)上等距排列FlexAlign.SpaceBetween,在交叉轴方向(水平轴方向)上首部对齐排列ItemAlign.Start。

T
tianyu 已提交
350 351
```
@Component
Z
zengyawen 已提交
352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429
   struct ContentTable {
     build() {
       Flex({ direction: FlexDirection.Column, justifyContent: FlexAlign.SpaceBetween, alignItems: ItemAlign.Start }) {
         Flex() {
           Text('Calories')
             .fontSize(17.4)
             .fontWeight(FontWeight.Bold)
             .layoutWeight(1)
           Flex() {
             Text('Calories')
               .fontSize(17.4)
               .flexGrow(1)
             Text('17kcal')
               .fontSize(17.4)
           }
           .layoutWeight(2)
         }
         Flex() {
           Text('Nutrition')
             .fontSize(17.4)
             .fontWeight(FontWeight.Bold)
             .layoutWeight(1)
           Flex() {
             Text('Protein')
               .fontSize(17.4)
               .flexGrow(1)
             Text('0.9g')
               .fontSize(17.4)
           }
           .layoutWeight(2)
         }
         Flex() {
           Text(' ')
             .fontSize(17.4)
             .fontWeight(FontWeight.Bold)
             .layoutWeight(1)
           Flex() {
             Text('Fat')
               .fontSize(17.4)
               .flexGrow(1)
             Text('0.2g')
               .fontSize(17.4)
           }
           .layoutWeight(2)
         }
         Flex() {
           Text(' ')
             .fontSize(17.4)
             .fontWeight(FontWeight.Bold)
             .layoutWeight(1)
           Flex() {
             Text('Carbohydrates')
               .fontSize(17.4)
               .flexGrow(1)
             Text('3.9g')
               .fontSize(17.4)
           }
           .layoutWeight(2)
         }
         Flex() {
           Text(' ')
             .fontSize(17.4)
             .fontWeight(FontWeight.Bold)
             .layoutWeight(1)
           Flex() {
             Text('vitaminC')
               .fontSize(17.4)
               .flexGrow(1)
             Text('17.8mg')
               .fontSize(17.4)
           }
           .layoutWeight(2)
         }
       }
       .height(280)
       .padding({ top: 30, right: 30, left: 30 })
     }
   }
H
HelloCrease 已提交
430

Z
zengyawen 已提交
431 432 433 434 435 436 437 438 439 440 441
   @Entry
   @Component
   struct FoodDetail {
       build() {
           Column() {
               FoodImageDisplay()
               ContentTable()
           }
           .alignItems(HorizontalAlign.Center)
       }
   }
T
tianyu 已提交
442 443
```

Z
zengyawen 已提交
444 445 446 447

5. 使用自定义构造函数\@Builder简化代码。可以发现,每个成分表中的成分单元其实都是一样的UI结构。
   ![zh-cn_image_0000001169599582](figures/zh-cn_image_0000001169599582.png)

L
luoying_ace_admin 已提交
448
   当前对每个成分单元都进行了声明,造成了代码的重复和冗余。可以使用\@Builder来构建自定义方法,抽象出相同的UI结构声明。\@Builder修饰的方法和Component的build方法都是为了声明一些UI渲染结构,遵循一样的ArkTS语法。开发者可以定义一个或者多个\@Builder修饰的方法,但Component的build方法必须只有一个。
Z
zengyawen 已提交
449 450 451

   在ContentTable内声明\@Builder修饰的IngredientItem方法,用于声明分类名、成分名称和成分含量UI描述。

T
tianyu 已提交
452 453
```
 @Component
Z
zengyawen 已提交
454
   struct ContentTable {
H
HelloCrease 已提交
455
     @Builder IngredientItem(title:string, name: string, value: string) {
Z
zengyawen 已提交
456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471
       Flex() {
         Text(title)
           .fontSize(17.4)
           .fontWeight(FontWeight.Bold)
           .layoutWeight(1)
         Flex({ alignItems: ItemAlign.Center }) {
           Text(name)
             .fontSize(17.4)
             .flexGrow(1)
           Text(value)
             .fontSize(17.4)
         }
         .layoutWeight(2)
       }
     }
   }
T
tianyu 已提交
472
```
Z
zengyawen 已提交
473 474


T
tianyu 已提交
475 476 477 478
 在ContentTable的build方法内调用IngredientItem接口,需要用this去调用该Component作用域内的方法,以此来区分全局的方法调用。

```
@Component
Z
zengyawen 已提交
479 480 481 482 483 484 485 486 487 488 489 490 491 492
   struct ContentTable {
     ......
     build() {
       Flex({ direction: FlexDirection.Column, justifyContent: FlexAlign.SpaceBetween, alignItems: ItemAlign.Start }) {
         this.IngredientItem('Calories', 'Calories', '17kcal')
         this.IngredientItem('Nutrition', 'Protein', '0.9g')
         this.IngredientItem('', 'Fat', '0.2g')
         this.IngredientItem('', 'Carbohydrates', '3.9g')
         this.IngredientItem('', 'VitaminC', '17.8mg')
       }
       .height(280)
       .padding({ top: 30, right: 30, left: 30 })
     }
   }
T
tianyu 已提交
493
```
Z
zengyawen 已提交
494 495


T
tianyu 已提交
496 497 498 499
 ContentTable组件整体代码如下。

```
@Component
Z
zengyawen 已提交
500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516
   struct ContentTable {
     @Builder IngredientItem(title:string, name: string, value: string) {
       Flex() {
         Text(title)
           .fontSize(17.4)
           .fontWeight(FontWeight.Bold)
           .layoutWeight(1)
         Flex() {
           Text(name)
             .fontSize(17.4)
             .flexGrow(1)
           Text(value)
             .fontSize(17.4)
         }
         .layoutWeight(2)
       }
     }
H
HelloCrease 已提交
517

T
tianyu 已提交
518 519 520 521 522 523 524 525 526 527 528 529
 build() {
   Flex({ direction: FlexDirection.Column, justifyContent: FlexAlign.SpaceBetween, alignItems: ItemAlign.Start }) {
     this.IngredientItem('Calories', 'Calories', '17kcal')
     this.IngredientItem('Nutrition', 'Protein', '0.9g')
     this.IngredientItem('', 'Fat', '0.2g')
     this.IngredientItem('', 'Carbohydrates', '3.9g')
     this.IngredientItem('', 'VitaminC', '17.8mg')
   }
   .height(280)
   .padding({ top: 30, right: 30, left: 30 })
 }

Z
zengyawen 已提交
530
   }
H
HelloCrease 已提交
531

Z
zengyawen 已提交
532 533 534 535 536 537 538 539 540 541 542
   @Entry
   @Component
   struct FoodDetail {
       build() {
           Column() {
               FoodImageDisplay()
               ContentTable()
           }
           .alignItems(HorizontalAlign.Center)
       }
   }
T
tianyu 已提交
543 544
```

Z
zengyawen 已提交
545 546 547 548 549 550 551 552 553

   ![zh-cn_image_0000001215199399](figures/zh-cn_image_0000001215199399.png)

通过学习Stack布局和Flex布局已完成食物的图文展示和营养成分表,构建出第一个普通视图的食物详情页。在下一个章节中,将开发食物分类列表页,并完成食物分类列表页面和食物详情页面的跳转和数据传递,现在我们就进入下一个章节的学习吧。

## 相关实例

针对创建简单视图,有以下示例工程可供参考:

L
luoying_ace_admin 已提交
554
- [`BuildCommonView`:创建简单视图(ArkTS)(API8)](https://gitee.com/openharmony/applications_app_samples/tree/master/ETSUI/BuildCommonView)
555

Z
zengyawen 已提交
556
  本示例为构建了简单页面展示食物番茄的图片和营养信息,主要为了展示简单页面的Stack布局和Flex布局。