ui-ts-layout-mediaquery.md 8.5 KB
Newer Older
K
kukixi 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
# 媒体查询


媒体查询(Media Query)在移动设备上应用十分广泛,开发者经常需要根据设备的大致类型或者特定的特征和设备参数(例如屏幕分辨率)来修改应用的样式。为此媒体查询提供了如下功能:


1. 针对设备和应用的属性信息,可以设计出相匹配的布局样式。

2. 当屏幕发生动态改变时(比如分屏、横竖屏切换),应用页面布局同步更新。


## 如何使用

通过调用媒体查询接口,设置媒体查询条件和查询结果的回调函数,在对应的条件的回调函数里更改页面布局或者实现业务逻辑。

首先导入媒体查询模块,例如:
```
  import mediaquery from '@ohos.mediaquery'
```
然后通过matchMediaSync接口设置媒体查询条件,并保存返回的条件监听句柄,例如:
```
  listener = mediaquery.matchMediaSync('(orientation: landscape)')
```
最后通过上面保存的条件监听句柄listener去注册回调函数,在回调函数里更改页面布局或者实现业务逻辑,当匹配到媒体查询条件时会触发此回调函数,例如:
```
onPortrait(mediaQueryResult) {
    if (mediaQueryResult.matches) {
        // do something here
    } else {
        // do something here
    }
}
listener.on('change', onPortrait)
```

## 媒体查询条件语法规则
```
[media-type] [and|not|only] [(media-feature)]
```
例如:
H
HelloCrease 已提交
41

K
kukixi 已提交
42
screen and (round-screen: true) // 当设备屏幕是圆形时条件成立
H
HelloCrease 已提交
43

K
kukixi 已提交
44
(max-height: 800) // 当高度小于800时条件成立
H
HelloCrease 已提交
45

K
kukixi 已提交
46
(height <= 800)  // 当高度小于800时条件成立
H
HelloCrease 已提交
47

K
kukixi 已提交
48 49 50 51
screen and (device-type: tv) or (resolution < 2) // 包含多个媒体特征的多条件复杂语句查询

###  媒体类型(media-type)

H
HelloCrease 已提交
52 53 54
| 类型     | 说明             |
| ------ | -------------- |
| screen | 按屏幕相关参数进行媒体查询。 |
K
kukixi 已提交
55 56 57 58 59 60 61 62


###  媒体逻辑操作(and|not|only)

媒体逻辑操作符:and、or、not、only用于构成复杂媒体查询,也可以通过comma(,)将其组合起来,详细解释说明如下表。

  **表1** 媒体逻辑操作符

H
HelloCrease 已提交
63 64 65 66 67 68 69
| 类型       | 说明                                       |
| -------- | ---------------------------------------- |
| and      | 将多个媒体特征(Media&nbsp;Feature)以“与”的方式连接成一个媒体查询,只有当所有媒体特征都为true,查询条件成立。另外,它还可以将媒体类型和媒体功能结合起来。<br/>例如:screen&nbsp;and&nbsp;(device-type:&nbsp;wearable)&nbsp;and&nbsp;(max-height:&nbsp;600)&nbsp;表示当设备类型是智能穿戴同时应用的最大高度小于等于600个像素单位时成立。 |
| not      | 取反媒体查询结果,媒体查询结果不成立时返回true,否则返回false。在媒体查询列表中应用not,则not仅取反应用它的媒体查询。<br/>例如:not&nbsp;screen&nbsp;and&nbsp;(min-height:&nbsp;50)&nbsp;and&nbsp;(max-height:&nbsp;600)&nbsp;表示当应用高度小于50个像素单位或者大于600个像素单位时成立。<br/>>&nbsp;![icon-note.gif](public_sys-resources/icon-note.gif)&nbsp;**说明:**<br/>>&nbsp;使用not运算符时必须指定媒体类型。 |
| only     | 当整个表达式都匹配时,才会应用选择的样式,可以应用在防止某些较早的版本的浏览器上产生歧义的场景。一些较早版本的浏览器对于同时包含了媒体类型和媒体特征的语句会产生歧义,比如:<br/>screen&nbsp;and&nbsp;(min-height:&nbsp;50)<br/>老版本浏览器会将这句话理解成screen,从而导致仅仅匹配到媒体类型(screen),就应用了指定样式,使用only可以很好地规避这种情况。<br/>>&nbsp;![icon-note.gif](public_sys-resources/icon-note.gif)&nbsp;**说明:**<br/>>&nbsp;使用only时必须指定媒体类型。 |
| ,(comma) | 将多个媒体特征以“或”的方式连接成一个媒体查询,如果存在结果为true的媒体特征,则查询条件成立。其效果等同于or运算符。<br/>例如:screen&nbsp;and&nbsp;(min-height:&nbsp;1000),&nbsp;&nbsp;(round-screen:true)&nbsp;表示当应用高度大于等于1000个像素单位或者设备屏幕是圆形时,条件成立。 |
| or       | 将多个媒体特征以“或”的方式连接成一个媒体查询,如果存在结果为true的媒体特征,则查询条件成立。<br/>例如:screen&nbsp;and&nbsp;(max-height:&nbsp;1000)&nbsp;or&nbsp;&nbsp;(round-screen:true)表示当应用高度小于等于1000个像素单位或者设备屏幕是圆形时,条件成立。 |
K
kukixi 已提交
70 71 72 73 74

在MediaQuery Level 4中引入了范围查询,使其能够使用max-,min-的同时,也支持了&lt;=,&gt;=,&lt;&gt;操作符。

  **表2** 媒体逻辑范围操作符

H
HelloCrease 已提交
75 76 77 78 79 80
| 类型    | 说明                                       |
| ----- | ---------------------------------------- |
| &lt;= | 小于等于,例如:screen&nbsp;and&nbsp;(50&nbsp;&lt;=&nbsp;height)。 |
| &gt;= | 大于等于,例如:screen&nbsp;and&nbsp;(600&nbsp;&gt;=&nbsp;height)。 |
| &lt;  | 小于,例如:screen&nbsp;and&nbsp;(50&nbsp;&lt;&nbsp;height)。 |
| &gt;  | 大于,例如:screen&nbsp;and&nbsp;(600&nbsp;&gt;&nbsp;height)。 |
K
kukixi 已提交
81 82 83 84


### 媒体特征(media-feature)

H
HelloCrease 已提交
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
| 类型                | 说明                                       |
| ----------------- | ---------------------------------------- |
| height            | 应用页面显示区域的高度。                             |
| min-height        | 应用页面显示区域的最小高度。                           |
| max-height        | 应用页面显示区域的最大高度。                           |
| width             | 应用页面显示区域的宽度。                             |
| min-width         | 应用页面显示区域的最小宽度。                           |
| max-width         | 应用页面显示区域的最大宽度。                           |
| resolution        | 设备的分辨率,支持dpi,dppx和dpcm单位。其中:<br/>-&nbsp;dpi表示每英寸中物理像素个数,1dpi≈0.39dpcm;<br/>-&nbsp;dpcm表示每厘米上的物理像素个数,1dpcm&nbsp;&nbsp;2.54dpi;<br/>-&nbsp;dppx表示每个px中的物理像素数(此单位按96px=1英寸为基准,与页面中的px单位计算方式不同),1dppx&nbsp;=&nbsp;96dpi。 |
| min-resolution    | 设备的最小分辨率。                                |
| max-resolution    | 设备的最大分辨率。                                |
| orientation       | 屏幕的方向。<br/>可选值:<br/>-&nbsp;orientation:&nbsp;portrait(设备竖屏)<br/>-&nbsp;orientation:&nbsp;landscape(设备横屏) |
| device-height     | 设备的高度。                                   |
| min-device-height | 设备的最小高度。                                 |
| max-device-height | 设备的最大高度。                                 |
| device-width      | 设备的宽度。                                   |
| min-device-width  | 设备的最小宽度。                                 |
| max-device-width  | 设备的最大宽度。                                 |
| round-screen      | 屏幕类型,圆形屏幕为true,&nbsp;非圆形屏幕为&nbsp;false。  |
| dark-mode         | 系统为深色模式时为true,否则为false。                  |
K
kukixi 已提交
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144

## 场景示例

通过使用媒体查询实现当屏幕发生横竖屏切换,给页面文本应用不同的内容和样式。

  ```
  import mediaquery from '@ohos.mediaquery'
  let portraitFunc = null
  
  @Entry
  @Component
  struct MediaQueryExample {
    @State color: string = '#DB7093'
    @State text: string = 'Portrait'
    @State fontSize: number = 24
    listener = mediaquery.matchMediaSync('(orientation: landscape)')
  
    onPortrait(mediaQueryResult) {
      if (mediaQueryResult.matches) {
        this.color = '#FFD700'
        this.text = 'Landscape'
      } else {
        this.color = '#DB7093'
        this.text = 'Portrait'
      }
    }
  
    aboutToAppear() {
      portraitFunc = this.onPortrait.bind(this) //bind current js instance
      this.listener.on('change', portraitFunc)
    }
  
    build() {
      Flex({ direction: FlexDirection.Column, alignItems: ItemAlign.Center, justifyContent: FlexAlign.Center }) {
        Text(this.text).fontSize(50).fontColor(this.color)
      }
      .width('100%').height('100%')
    }
  }
  ```
K
kukixi 已提交
145
横屏下文本内容为Landscape,颜色为#FFD700。<br/>
K
kukixi 已提交
146 147
![zh-cn_image_0000001262954829](figures/zh-cn_image_0000001262954829.png)

K
kukixi 已提交
148
非横屏下文本内容为Portrait,颜色为#DB7093。<br/>
K
kukixi 已提交
149
![zh-cn_image_0000001263074739](figures/zh-cn_image_0000001263074739.png)