README.md 7.0 KB
Newer Older
雪洛's avatar
雪洛 已提交
1 2 3 4
# 编译到web端

> 新增于4.0版本

W
web  
wanganxp 已提交
5 6 7 8 9 10 11 12
uni-app x 编译到web平台时,并非是与uni-app js引擎版一致。而是基于uts的统一规范,和编译到安卓端的一致性较高。

与App版相比,web版有几个较大的差别:
1. web版是一个spa的单页应用,而app是多页的。
2. pages.json配置的导航栏和tabbar,在web端并非原生的,而是网页的一部分。虽然uvue页面仍然是在导航栏和tabbar之间的,但在web平台,开发者可以直接操作导航栏和tabbar的dom。
3. web版默认有页面滚动;app没有。

本文档会介绍与web和Android的差异及注意事项。
13

W
wanganxp 已提交
14 15 16 17
## web运行失败注意事项@faq

- 在比较早的HBuilderX版本中创建的uni-app x项目,特征是项目根目录是app.vue而不是app.uvue。这样的项目运行web会失败。需替换index.html和app.uvue这2个文件,可以新建项目从中复制,也可以从hello uni-app x示例项目复制。

雪洛's avatar
雪洛 已提交
18 19 20 21 22 23 24 25
## vue

### 特性支持情况

uni-app x编译到web端时遵循vue规范,目前有部分vue特性暂不支持。

不支持的特性如下

雪洛's avatar
雪洛 已提交
26
- 指令:`v-once``v-memo`
雪洛's avatar
雪洛 已提交
27
- render函数
DCloud-WZF's avatar
DCloud-WZF 已提交
28
- 不支持组件中监听页面 `onPageScroll``onReachBottom` 生命周期
雪洛's avatar
雪洛 已提交
29 30 31 32 33

部分支持的特性

- mixin:需要使用`defineMixin`函数定义mixin,不可直接使用对象字面量定义mixin

雪洛's avatar
雪洛 已提交
34 35 36 37
自4.11版本起支持如下特性

- 组合式API:`defineOptions``defineModel``toValue``toRef``toRefs``hasInjectionContext`

雪洛's avatar
雪洛 已提交
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
### refs@refs

使用refs获取内置组件实例时会获取到对应的Element,而不是vue组件实例。

```vue
<template>
  <view ref="view"></view>
</template>
<script>
export default {
  onReady() {
    console.log(this.$refs['view']) // 此时获取到的是UniViewElement
  }
}
</script>
```

雪洛's avatar
雪洛 已提交
55
### vue实例相关属性类型问题
雪洛's avatar
雪洛 已提交
56

雪洛's avatar
雪洛 已提交
57
为保证运行性能,app安卓端部分属性(如:$data、$refs)被转为了Map类型(安卓端map支持使用下标访问),而web端仍是普通对象或proxy。为保证多端代码一致,在使用这些属性时可以统一为下标访问。
雪洛's avatar
雪洛 已提交
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76

```vue
<template>
  <view></view>
</template>
<script>
export default {
  data() {
    return {
      a: 1
    }
  },
  onReady() {
    console.log(this.$data['a']) // 1
  }
}
</script>
```

雪洛's avatar
雪洛 已提交
77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
### ComponentPublicInstance类型

目前已知组件使用emits会导致this不能直接传递给ComponentPublicInstance类型,需要as一下。

```ts
function log(ins: ComponentPublicInstance) {
  console.log(ins);
}

export default {
  emits: ['change', 'input'],
  methods: {
    buttonClick() {
      log(this as ComponentPublicInstance)
    }
  }
}
```

如下为进阶说明,可以不看

我们在代码中用到的ComponentPublicInstance为省略泛型的写法,最终推导出的ComponentPublicInstance类型为:

```ts
// 此处为方便说明,省略了很多属性、简化了写法
type ComponentPublicInstance = {
  $emit: (event: string, ...args: any[]) => void
}
```

对于使用了emits的组件,最终推导出来的this对应的类型为:

```ts
// 此处为方便说明,省略了很多属性、简化了写法
type XxxComponentPublicInstance = {
  $emit: (event: 'change' | 'input', ...args: any[]) => void
}
```

雪洛's avatar
雪洛 已提交
116
在将XxxComponentPublicInstance类型的this值赋值给ComponentPublicInstance类型的参数时,由于$emit类型无法兼容导致编译报错。
雪洛's avatar
雪洛 已提交
117 118 119 120 121 122 123 124 125 126 127 128

下面我们只看$emit的类型

```ts
type ComponentPublicInstanceEmit = (event: string, ...args: any[]) => void
type XxxComponentPublicInstanceEmit = (event: 'change' | 'input', ...args: any[]) => void
```

对于ComponentPublicInstanceEmit类型的函数A来说,event参数可以接收任意字符串,如果将其赋值为只能接收'change' | 'input'作为event参数的函数B,这时候用户如果将event参数设为'click'则函数B无法处理,而函数A的类型定义又可以接收'click',所以ts并不允许这种行为。

但是反过来则是可以的XxxComponentPublicInstanceEmit类型的函数A可以被重新赋值为ComponentPublicInstanceEmit类型的函数。

雪洛's avatar
雪洛 已提交
129
总结一下就是ts在比较函数参数是否兼容时使用逆变(contravariance)而非协变(covariance),这两个名词定义比较复杂,结合上述示例理解即可。
雪洛's avatar
雪洛 已提交
130

W
web  
wanganxp 已提交
131
### 注意事项
雪洛's avatar
雪洛 已提交
132

雪洛's avatar
雪洛 已提交
133
- data内$开头的属性不可直接使用`this.$xxx`访问,需要使用`this.$data['$xxx']`,这是vue的规范。目前安卓端可以使用this.$xxx访问是Bug而非特性,请勿使用此特性。
雪洛's avatar
雪洛 已提交
134
- 安卓端由于kotlin特性组件内部使用组件data内定义的属性时this可以省略,请勿在web端使用此特性。
W
web  
wanganxp 已提交
135
- web端由于是一个单页应用,使用`$root`会获取应用根组件,而不是页面根组件。而安卓端是多页应用,`$root`获取的是页面根组件。
雪洛's avatar
雪洛 已提交
136 137 138 139
- web端使用`$parent`会获取父组件(含内置组件),安卓端只会获取父级非内置组件,web端后续会调整,请勿利用此特性。

## uts

雪洛's avatar
雪洛 已提交
140
参考:[类型兼容性](../uts/type-compatibility.md)
雪洛's avatar
雪洛 已提交
141

雪洛's avatar
雪洛 已提交
142 143
## css

雪洛's avatar
雪洛 已提交
144 145
### 默认样式

雪洛's avatar
雪洛 已提交
146 147
为保证多端统一,uni-app-x编译到web端时,内置组件根元素带有一些默认样式,详情参考:[uvue css使用](../css/README.md)。如果是使用`document.createElement`等方式自行创建的html元素不会有这些默认样式。

雪洛's avatar
雪洛 已提交
148 149
### 样式格式自动转化

雪洛's avatar
雪洛 已提交
150 151 152 153 154 155 156 157 158
通过element.style.xxx设置样式时,web端会自动将样式进行转化,具体取决于浏览器。

例如:

```ts
element.style.color = '##FF0000'
element.style.color === 'rgb(255, 0, 0)' // true
```

雪洛's avatar
雪洛 已提交
159 160
### fixed定位

W
web  
wanganxp 已提交
161
position: fixed定位时,web端为相对于整个浏览器页面进行定位,app端为相对于页面(除导航栏、tabbar)定位。可以使用[css变量](../css/README.md#variable)使两端表现一致
雪洛's avatar
雪洛 已提交
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178

```css
.fixed {
  position: fixed;
  width: 100px;
  height: 100px;
  background-color: #FF0000;
  left: 10px;
  /* #ifdef WEB */
  top: calc(--window-top + 10px);
  /* #endif */
  /* #ifdef APP */
  top: 10px;  /* App端暂不支持calc */
  /* #endif */
}
```

雪洛's avatar
雪洛 已提交
179 180
## api

雪洛's avatar
雪洛 已提交
181
uni相关的异步api在web端不传回调时会返回promise(详情参考:[API Promise 化](https://uniapp.dcloud.net.cn/api/#api-promise-%E5%8C%96))。但是由于目前缺少此用法的类型定义,使用返回的Promise无法通过编译。另外安卓端暂未实现此功能,建议不要使用此特性。
雪洛's avatar
雪洛 已提交
182

雪洛's avatar
雪洛 已提交
183 184 185 186 187 188
## 特性支持

- [x] 宽屏适配
- [x] 国际化
- [x] 地图
- [x] uni-push2.0
雪洛's avatar
雪洛 已提交
189 190
- [ ] 服务端渲染
- [ ] 接口Promise化
雪洛's avatar
雪洛 已提交
191

雪洛's avatar
雪洛 已提交
192 193 194 195 196 197 198 199
## 运行与发行

运行到web端时,uni-app-x编译器不会对语法进行转化来兼容低版本浏览器。发行时会对代码进行转化,以保证低版本浏览器能正常运行。

发行时支持的最低浏览器版本为:`chrome 64``safari 11.1``firefox 62``edge 79``safari on iOS 12`

## 其他注意事项

雪洛's avatar
雪洛 已提交
200
- 内置组件的tagName、nodeName与安卓端不同,目前web端和安卓端相比多了`UNI-`前缀,例如web端为`UNI-VIEW``UNI-IMAGE`,安卓端为`VIEW``IMAGE`。此问题已在HBuilderX 4.02版本修复,web端移除了UNI-前缀。