• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 网页中安全区域计算和避让适配
2<!--Kit: ArkWeb-->
3<!--Subsystem: Web-->
4<!--Owner: @KeeGitee-->
5<!--Designer: @LongLie-->
6<!--Tester: @ghiker-->
7<!--Adviser: @HelloCrease-->
8
9早期设备屏幕多为矩形,应用界面能够完整展示。随着全面屏、刘海屏等异形屏幕的普及,屏幕边缘可能出现圆角、摄像头或系统导航条,导致界面部分被遮挡。
10
11安全区域是指屏幕中未被设备硬件或系统UI遮挡的区域,不与系统非安全区域(如状态栏、挖孔区和导航栏)重叠。应用的布局默认限定在安全区域内,但系统提供沉浸式布局能力,允许应用通过配置使界面扩展至非安全区域。
12
13在沉浸式效果下,Web组件中的网页元素可能会被状态栏、挖孔区及导航条遮挡。此情况下,需要网页开发者进行避让适配,确保网页中的文字、表单和交互组件等关键内容避让非安全区域,从而保证用户可以完整地阅读和操作。
14
15Web组件提供的利用W3C CSS进行安全区域计算和避让适配的能力,支持异形屏幕设备在沉浸式效果下网页的正常显示,网页开发者可以利用该能力对被遮挡的元素进行避让。
16
17## 开启Web组件沉浸式效果
18
19Web组件默认布局在安全区域内。开启<!--RP1-->沉浸式效果<!--RP1End-->后,Web网页将扩展至状态栏和导航栏,从而最大化利用屏幕可视区域,增强视觉连贯性,改善用户的UI体验。开发者可通过以下方式启用Web组件的沉浸式效果。
20
21- 通过[setWindowLayoutFullScreen](../reference/apis-arkui/arkts-apis-window-Window.md#setwindowlayoutfullscreen9)设置应用窗口全屏。窗口全屏时,Web组件可布局至非安全区域。
22
23  ```ts
24  // EntryAbility.ets
25  import { UIAbility } from '@kit.AbilityKit';
26  import { window } from '@kit.ArkUI';
27
28  export default class EntryAbility extends UIAbility {
29    // ...
30    onWindowStageCreate(windowStage: window.WindowStage): void {
31      windowStage.getMainWindow().then(window => {
32        // 设置窗口全屏
33        window.setWindowLayoutFullScreen(true);
34      });
35    }
36  }
37  ```
38
39  ```ts
40  // xxx.ets
41  import { webview } from '@kit.ArkWeb';
42
43  @Entry
44  @Component
45  struct WebComponent {
46    controller: webview.WebviewController = new webview.WebviewController();
47
48    build() {
49      Column() {
50        Web({ src: 'www.example.com', controller: this.controller })
51          .width('100%').height('100%')
52      }
53    }
54  }
55  ```
56
57- 通过[expandSafeArea](../reference/apis-arkui/arkui-ts/ts-universal-attributes-expand-safe-area.md)设置Web组件扩展安全区域,可以自定义扩展类型和方向。下面的示例中,Web组件可扩展至状态栏和导航栏,实现沉浸式效果。
58
59  ```ts
60  // xxx.ets
61  import { webview } from '@kit.ArkWeb';
62
63  @Entry
64  @Component
65  struct WebComponent {
66    controller: webview.WebviewController = new webview.WebviewController();
67
68    build() {
69      Column() {
70        Web({ src: 'www.example.com', controller: this.controller })
71          .width('100%').height('100%')
72          // 扩展至系统默认非安全区域(状态栏、导航栏),并设置只扩展上方区域和下方区域
73          .expandSafeArea([SafeAreaType.SYSTEM], [SafeAreaEdge.TOP, SafeAreaEdge.BOTTOM])
74      }
75    }
76  }
77  ```
78
79## 设置网页在可视窗口中的布局方式
80
81viewport-fit用于设置网页在可视窗口中的布局方式,是`<meta name="viewport">`标签的一个属性。设置方式如下:
82
83```html
84<meta name='viewport' content='viewport-fit=cover'>
85```
86
87如表1所示,viewport-fit默认为`auto`,与`contain`表现一致,表示网页内容全部包含在安全区域内。`cover`表示网页内容完全覆盖可视窗口,可能与非安全区域发生重叠。
88
89**表1** viewport-fit属性取值说明
90
91| viewport-fit取值 | 说明 | 适用场景 |
92| - | - | - |
93| auto | 默认值,与contain表现一致。 | 无需特殊适配的普通网页。 |
94| contain | 网页内容被严格限制在安全区域内,不与非安全区域重叠。 | 需要确保完整显示的网页。 |
95| cover | 网页内容完全覆盖可视窗口,可能与非安全区域重叠。 | 需要最大化渲染可视窗口的网页,由网页开发者进行避让适配。 |
96
97> **说明:**
98>
99> Web组件当前还不支持开启沉浸式效果时将网页内容限制在安全区域内。因此,当设置`viewport-fit=contain`时,表现与`cover`一致,网页内容完全填充Web组件区域。
100
101## 网页元素避让适配
102
103`safe-area-inset-*`是一组CSS环境变量,定义了安全区域与Web可视窗口边缘的距离,即网页内容要完整显示时,在top、right、bottom和left四个方向上需要避让的距离,如下图所示。不同于其他CSS属性,环境变量的属性名称对大小写敏感。
104
105**图1** safe-area-inset-*示意图
106
107![web-safe-area-inset](figures/arkweb_safe_area_inset.png)
108
109当设置`viewport-fit=cover`时,ArkWeb内核将持续监测Web组件及系统非安全区域的位置与尺寸,根据两者的重叠部分计算网页在四个方向上需避让的具体距离,并设置给环境变量`safe-area-inset-*`。在矩形显示器(如普通PC/2in1设备的屏幕)上,这些值为零。在非矩形显示器(如圆形表盘或移动设备屏幕)上,`safe-area-inset-*`所界定的内矩形区域即为安全区域,网页内容在该区域内可完整显示,避免被非矩形显示区域裁剪。
110
111网页元素的避让适配依赖CSS函数`env()`,该函数用于获取浏览器或系统提供的环境变量。使用`env()`函数可以获取`safe-area-inset-*`的值。网页开发者无需关注设备非安全区域的具体位置和尺寸,在CSS样式中应用`env(safe-area-inset-*)`即可定义网页需要避让的距离,实现跨设备的避让。语法如下:
112
113```
114/* 分别表示上、右、下、左,四个方向上的避让值 */
115env(safe-area-inset-top);
116env(safe-area-inset-right);
117env(safe-area-inset-bottom);
118env(safe-area-inset-left);
119
120/* 基于fallback设置避让值,第二个参数表示环境变量不存在时的回退值 */
121/* 下述长度单位参见:https://developer.mozilla.org/zh-CN/docs/Web/CSS/length */
122env(safe-area-inset-top, 20px);
123env(safe-area-inset-right, 1em);
124env(safe-area-inset-bottom, 0.5vh);
125env(safe-area-inset-left, 1.4rem);
126
127/* env()可基于部分数学计算函数`calc()`,`min()`,`max()`进行组合计算 */
128calc(env(safe-area-inset-top) + 10px)
129min(env(safe-area-inset-left), 50px)
130max(env(safe-area-inset-bottom), 30px)
131```
132
133> **说明:**
134>
135> 使用`env(safe-area-inset-*)`进行避让时需要设置`viewport-fit=cover`。`viewport-fit=contain`时,`env(safe-area-inset-*)`值为0。
136
137## 网页元素避让非安全区域最佳实践
138
139Web组件启用沉浸式效果时,渲染内容可能与非安全区域重叠,影响用户的阅读和交互,如图2所示。非安全区域包括顶部状态栏、屏幕挖孔区和底部导航条。在沉浸式效果下,index.html网页的标题栏被屏幕挖孔区遮挡,底部的Tab区域与导航条发生重叠。
140
141**图2** Web组件开启沉浸式效果时网页元素被非安全区域遮挡
142
143![web-safe-area-immersion](figures/arkweb_safe_area_immersion.png)
144
145```html
146<!-- index.html -->
147<!DOCTYPE html>
148<html lang="en">
149<head>
150    <meta charset="UTF-8">
151    <meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover">
152    <style>
153        body {
154            margin: 0;
155            background: #f6f6f6;
156        }
157        .edge {
158            position: fixed;
159            display: flex;
160            width: 100%;
161            background: #fefefe;
162        }
163        .title-bar {
164            align-items: center;
165            justify-content: center;
166            top: 0;
167            height: 40px;
168        }
169        .content {
170            margin: 8px;
171            padding-top: 40px;
172        }
173        .tabs {
174            justify-content: space-around;
175            bottom: 0;
176            height: 40px;
177        }
178        .tab {
179            padding: 10px;
180        }
181        .tab.active {
182            color: Blue;
183        }
184    </style>
185</head>
186<body>
187    <div>
188        <div class="edge title-bar">Example page</div>
189        <div class="content">
190            <p>Contents of page</p>
191        </div>
192    </div>
193    <div class="edge tabs">
194        <div class="tab active">Tab1</div>
195        <div class="tab">Tab2</div>
196        <div class="tab">Tab3</div>
197    </div>
198</body>
199</html>
200```
201
202网页开发者可利用`env(safe-area-inset-*)`定义CSS样式,确保文字、图片和交互组件避让非安全区域。在以下示例中,通过`env(safe-area-inset-*)`更新了index.html的CSS样式,使网页主要内容避让非安全区域,效果见图3。
203
204```
205.title-bar {
206    align-items: center;
207    justify-content: center;
208    top: 0;
209    height: 40px;
210    padding-top: env(safe-area-inset-top); /* 设置padding-top避让上方非安全区域 */
211}
212.content {
213    margin: 8px;
214    padding-top: calc(env(safe-area-inset-top) + 40px); /* 同步title-bar增加padding-top高度 */
215}
216.tabs {
217    justify-content: space-around;
218    bottom: 0;
219    height: calc(env(safe-area-inset-bottom) + 40px); /* 增加tab区域高度以避让下方非安全区域 */
220}
221```
222
223**图3** Web组件开启沉浸式效果时网页元素避让非安全区域
224
225![web-safe-area](figures/arkweb_safe_area_avoid.png)
226