• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# HML语法参考
2
3HML(OpenHarmony Markup Language)是一套类HTML的标记语言,通过组件,事件构建出页面的内容。页面具备数据绑定、事件绑定、列表渲染、条件渲染和逻辑控制等高级能力。
4
5
6## 页面结构
7
8```html
9<!-- xxx.hml -->
10<div class="item-container">
11  <text class="item-title">Image Show</text>
12  <div class="item-content">
13    <image src="/common/xxx.png" class="image"></image>
14  </div>
15</div>
16```
17
18
19## 数据绑定
20
21```html
22<!-- xxx.hml -->
23<div onclick="changeText">
24  <text> {{content[1]}} </text>
25</div>
26```
27```css
28/*xxx.css*/
29.container{
30    margin: 200px;
31}
32```
33```js
34// xxx.js
35export default {
36  data: {
37    content: ['Hello World!', 'Welcome to my world!']
38  },
39  changeText: function() {
40    this.content.splice(1, 1, this.content[0]);
41  }
42}
43```
44
45> **说明:**
46> - 针对数组内的数据修改,请使用splice方法生效数据绑定变更。
47>
48> - hml文件中的js表达式不支持ES6语法。
49
50![zh-cn_image_0000001118642003](figures/zh-cn_image_0000001118642003.png)
51
52## 普通事件绑定
53
54事件通过'on'或者'\@'绑定在组件上,当组件触发事件时会执行JS文件中对应的事件处理函数。
55
56事件支持的写法有:
57
58- "funcName":funcName为事件回调函数名(在JS文件中定义相应的函数实现)。
59
60- "funcName(a,b)":函数参数例如a,b可以是常量,或者是在JS文件中的data中定义的变量(前面不用写this.)。
61
62- 示例
63  ```html
64  <!-- xxx.hml -->
65  <div class="container">
66      <text class="title">{{count}}</text>
67      <div class="box">
68          <input type="button" class="btn" value="increase" onclick="increase" />
69          <input type="button" class="btn" value="decrease" @click="decrease" />
70          <!-- 传递额外参数 -->
71          <input type="button" class="btn" value="double" @click="multiply(2)" />
72          <input type="button" class="btn" value="decuple" @click="multiply(10)" />
73          <input type="button" class="btn" value="square" @click="multiply(count)" />
74      </div>
75  </div>
76  ```
77
78  ```js
79  // xxx.js
80  export default {
81    data: {
82      count: 0
83    },
84    increase() {
85      this.count++;
86    },
87    decrease() {
88      this.count--;
89    },
90    multiply(multiplier) {
91      this.count = multiplier * this.count;
92    }
93  };
94  ```
95
96  ```css
97  /* xxx.css */
98  .container {
99      display: flex;
100      flex-direction: column;
101      justify-content: center;
102      align-items: center;
103      left: 0px;
104      top: 0px;
105      width: 454px;
106      height: 454px;
107  }
108  .title {
109      font-size: 30px;
110      text-align: center;
111      width: 200px;
112      height: 100px;
113  }
114  .box {
115      width: 454px;
116      height: 200px;
117      justify-content: center;
118      align-items: center;
119      flex-wrap: wrap;
120  }
121  .btn {
122      width: 200px;
123      border-radius: 0;
124      margin-top: 10px;
125      margin-left: 10px;
126  }
127  ```
128
129![zh-cn_image_0000001118642002](figures/zh-cn_image_0000001118642002.gif)
130
131## 冒泡事件绑定<sup>5+</sup>
132
133冒泡事件绑定包括:
134
135- 绑定冒泡事件:on:{event}.bubble。on:{event}等价于on:{event}.bubble。
136
137- 绑定并阻止冒泡事件向上冒泡:grab:{event}.bubble。grab:{event}等价于grab:{event}.bubble。
138  > **说明:**
139  >
140  > 冒泡事件是指多个组件嵌套时,组件之间会有层次关系,当这些组件注册了相同的事件时,这个事件会首先运行在该元素上的处理程序,然后运行其父元素上的处理程序,一直向上到其他祖先上的处理程序。如果当一个组件触发了这个事件,它会首先触发该组件的回调函数,然后触发其父元素上的回调函数,然后触发其他祖先上的处理程序。
141  >
142  > 详细冒泡事件说明参见[通用事件](../reference/arkui-js/js-components-common-events.md)章节。
143- 示例
144  ```html
145  <!-- xxx.hml -->
146  <div>
147     <!-- 使用事件冒泡模式绑定事件回调函数。5+ -->;
148      <div on:touchstart.bubble="touchstartfunc"></div>
149      <div on:touchstart="touchstartfunc"></div>
150      <!-- 绑定事件回调函数,但阻止事件向上传递。5+ -->
151      <div grab:touchstart.bubble="touchstartfunc"></div>
152      <div grab:touchstart="touchstartfunc"></div>
153      <!-- 使用事件冒泡模式绑定事件回调函数。6+ -->
154      <div on:click.bubble="clickfunc"></div>
155      <div on:click="clickfunc"></div>
156      <!-- 绑定事件回调函数,但阻止事件向上传递。6+ -->
157      <div grab:click.bubble="clickfunc"></div>
158      <div grab:click="clickfunc"></div>
159  </div>
160  ```
161
162  ```js
163  // xxx.js
164  export default {
165      clickfunc: function(e) {
166          console.log(e);
167      },
168      touchstartfuc: function(e) {
169          console.log(e);
170      },
171  }
172  ```
173
174> **说明:**
175> 采用旧写法(onclick)的事件绑定在最小API版本6以下时采用不冒泡处理,在最小API版本为6及6以上时采用冒泡处理。
176
177## 捕获事件绑定<sup>5+</sup>
178
179Touch触摸类事件支持捕获,捕获阶段位于冒泡阶段之前,捕获事件先到达父组件然后达到子组件。
180
181捕获事件绑定包括:
182
183- 绑定捕获事件:on:{event}.capture。
184
185- 绑定并阻止事件向下传递:grab:{event}.capture。
186
187- 示例
188  ```html
189  <!-- xxx.hml -->
190  <div>
191      <!-- 使用事件捕获模式绑定事件回调函数。5+ -->
192      <div on:touchstart.capture="touchstartfunc"></div>
193      <!-- 绑定事件回调函数,但阻止事件向下传递。5+ -->
194      <div grab:touchstart.capture="touchstartfunc"></div>
195  </div>
196  ```
197
198  ```js
199  // xxx.js
200  export default {
201      touchstartfuc: function(e) {
202          console.log(e);
203      },
204  }
205  ```
206
207
208## 列表渲染
209
210```html
211<!-- xxx.hml -->
212<div class="array-container" style="flex-direction: column;margin: 200px;">
213  <!-- div列表渲染 -->
214  <!-- 默认$item代表数组中的元素, $idx代表数组中的元素索引 -->
215  <div for="{{array}}" tid="id" onclick="changeText">
216    <text>{{$idx}}.{{$item.name}}</text>
217  </div>
218  <!-- 自定义元素变量名称 -->
219  <div for="{{value in array}}" tid="id" onclick="changeText">
220    <text>{{$idx}}.{{value.name}}</text>
221  </div>
222  <!-- 自定义元素变量、索引名称 -->
223  <div for="{{(index, value) in array}}" tid="id" onclick="changeText">
224    <text>{{index}}.{{value.name}}</text>
225  </div>
226</div>
227
228```
229
230```js
231// xxx.js
232export default {
233  data: {
234    array: [
235      {id: 1, name: 'jack', age: 18},
236      {id: 2, name: 'tony', age: 18},
237    ],
238  },
239  changeText: function() {
240    if (this.array[1].name === "tony"){
241      this.array.splice(1, 1, {id:2, name: 'Isabella', age: 18});
242    } else {
243      this.array.splice(2, 1, {id:3, name: 'Bary', age: 18});
244    }
245  },
246}
247```
248
249tid属性主要用来加速for循环的重渲染,旨在列表中的数据有变更时,提高重新渲染的效率。tid属性是用来指定数组中每个元素的唯一标识,如果未指定,数组中每个元素的索引为该元素的唯一id。例如上述tid="id"表示数组中的每个元素的id属性为该元素的唯一标识。for循环支持的写法如下:
250
251- for="array":其中array为数组对象,array的元素变量默认为$item。
252
253- for="v in array":其中v为自定义的元素变量,元素索引默认为$idx。
254
255- for="(i, v) in array":其中元素索引为i,元素变量为v,遍历数组对象array。
256
257> **说明:**
258> - 数组中的每个元素必须存在tid指定的数据属性,否则运行时可能会导致异常。
259>
260> - 数组中被tid指定的属性要保证唯一性,如果不是则会造成性能损耗。比如,示例中只有id和name可以作为tid字段,因为它们属于唯一字段。
261>
262> - tid不支持表达式。
263
264![zh-cn_image_0000001118642001](figures/zh-cn_image_0000001118642001.gif)
265
266## 条件渲染
267
268条件渲染分为2种:if/elif/else和show。两种写法的区别在于:第一种写法里if为false时,组件不会在vdom中构建,也不会渲染,而第二种写法里show为false时虽然也不渲染,但会在vdom中构建;另外,当使用if/elif/else写法时,节点必须是兄弟节点,否则编译无法通过。实例如下:
269
270```html
271<!-- xxx.hml -->
272<div class="container">
273  <button class="btn" type="capsule" value="toggleShow" onclick="toggleShow"></button>
274  <button class="btn" type="capsule" value="toggleDisplay" onclick="toggleDisplay"></button>
275  <text if="{{visible}}"> Hello-world1 </text>
276  <text elif="{{display}}"> Hello-world2 </text>
277  <text else> Hello-World </text>
278</div>
279```
280
281```css
282/* xxx.css */
283.container{
284  flex-direction: column;
285  align-items: center;
286}
287.btn{
288  width: 280px;
289  font-size: 26px;
290  margin: 10px 0;
291}
292```
293
294```js
295// xxx.js
296export default {
297  data: {
298    visible: false,
299    display: true,
300  },
301  toggleShow: function() {
302    this.visible = !this.visible;
303  },
304  toggleDisplay: function() {
305    this.display = !this.display;
306  }
307}
308```
309
310![zh-cn_image_0000001118642004](figures/zh-cn_image_0000001118642004.gif)
311
312优化渲染优化:show方法。当show为true时,节点正常渲染;当为false时,仅仅设置display样式为none。
313
314```html
315<!-- xxx.hml -->
316<div class="container">
317  <button class="btn" type="capsule" value="toggle" onclick="toggle"></button>
318  <text show="{{visible}}" > Hello World </text>
319</div>
320```
321
322```css
323/* xxx.css */
324.container{
325  flex-direction: column;
326  align-items: center;
327}
328.btn{
329  width: 280px;
330  font-size: 26px;
331  margin: 10px 0;
332}
333```
334
335```js
336// xxx.js
337export default {
338  data: {
339    visible: false,
340  },
341  toggle: function() {
342    this.visible = !this.visible;
343  },
344}
345```
346
347> **说明:**
348> 禁止在同一个元素上同时设置for和if属性。
349
350![zh-cn_image_0000001118642005](figures/zh-cn_image_0000001118642005.gif)
351
352## 逻辑控制块
353
354&lt;block&gt;控制块使得循环渲染和条件渲染变得更加灵活;block在构建时不会被当作真实的节点编译。注意block标签只支持for和if属性。
355
356```html
357<!-- xxx.hml -->
358<list>
359  <block for="glasses">
360    <list-item type="glasses">
361      <text>{{$item.name}}</text>
362    </list-item>
363    <block for="$item.kinds">
364      <list-item type="kind">
365        <text>{{$item.color}}</text>
366      </list-item>
367    </block>
368  </block>
369</list>
370```
371
372```js
373// xxx.js
374export default {
375  data: {
376    glasses: [
377      {name:'sunglasses', kinds:[{name:'XXX',color:'XXX'},{name:'XXX',color:'XXX'}]},
378      {name:'nearsightedness mirror', kinds:[{name:'XXX',color:'XXX'}]},
379    ],
380  },
381}
382```
383
384![zh-cn_image_0000001118642006](figures/zh-cn_image_0000001118642006.png)
385
386## 模板引用
387
388HML可以通过element引用模板文件,详细介绍可参考[自定义组件](../reference/arkui-js/js-components-custom-basic-usage.md)章节。
389
390```html
391<!-- template.hml -->
392<div class="item">
393  <text>Name: {{name}}</text>
394  <text>Age: {{age}}</text>
395</div>
396```
397
398```html
399<!-- index.hml -->
400<element name='comp' src='../../common/template.hml'></element>
401<div>
402  <comp name="Tony" age="18"></comp>
403</div>
404```
405