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<block>控制块使得循环渲染和条件渲染变得更加灵活;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