uni-app App端內置了一個基于 weex 改進的原生渲染引擎,提供了原生渲染能力。
在App端,如果使用vue頁面,則使用webview渲染;如果使用nvue頁面(native vue的縮寫),則使用原生渲染。一個App中可以同時使用兩種頁面,比如首頁使用nvue,二級頁使用vue頁面,hello uni-app示例就是如此。
雖然nvue也可以多端編譯,輸出H5和小程序,但nvue的css寫法受限,所以如果你不開發(fā)App,那么不需要使用nvue。
以往的 weex ,有個很大的問題是它只是一個高性能的渲染器,沒有足夠的API能力(比如各種push sdk集成、藍牙等能力調用),使得開發(fā)時非常依賴原生工程師協(xié)作,開發(fā)者本來想節(jié)約成本,結果需要前端、iOS、Android 3撥人開發(fā),適得其反。 nvue 解決了這個問題,讓前端工程師可以直接開發(fā)完整 App,并提供豐富的插件生態(tài)和云打包。這些組合方案,幫助開發(fā)者切實的提高效率、降低成本。
同時uni-app擴展了weex原生渲染引擎的很多排版能力,修復了很多bug。比如
nvue的組件和API寫法與vue頁面一致,其內置組件還比vue頁面內置組件增加了更多,詳見。但nvue在css的寫法限制較多,具體如下:
如果你熟悉 weex或react native 開發(fā),那么 nvue 是你的更優(yōu)選擇,能切實提升你的開發(fā)效率,降低成本。
如果你是web前端,不熟悉原生排版,那么建議你仍然以使用vue頁面為主,在App端某些vue頁面表現(xiàn)不佳的場景下使用 nvue 作為強化補充。這些場景如下:
但注意,在某些場景下,nvue不如vue頁面,如下:
uni-app在App端,支持vue頁面和nvue頁面混搭、互相跳轉。也支持純nvue原生渲染。
啟用純原生渲染模式,可以減少App端的包體積、減少使用時的內存占用。因為webview渲染模式的相關模塊將被移除。
在manifest.json源碼視圖的"app-plus"下配置"renderer":"native",即代表App端啟用純原生渲染模式。此時pages.json注冊的vue頁面將被忽略,vue組件也將被原生渲染引擎來渲染。
如果不指定該值,默認是不啟動純原生渲染的。
// manifest.json
{
// ...
/* App平臺特有配置 */
"app-plus": {
"renderer": "native", //App端純原生渲染模式
}
}
如你之前是weex開發(fā)者,可以繼續(xù)查閱本章節(jié),否則可以跳過看下一節(jié)快速上手。
weex的組件和jsapi,與uni-app不同。uni-app與微信小程序相同。
考慮到weex用戶的遷移,uni-app 也支持weex的代碼寫法。在manifest.json中可以配置使用weex編譯模式或uni-app編譯模式。選擇weex編譯模式時將不支持uni-app的組件和jsapi,需要開發(fā)者參考weex官方文檔的寫法來寫代碼。 比如 weex 編譯模式用<div>。uni-app 編譯模式則使用<view>。
一般情況建議使用uni-app模式,除非歷史weex代碼較多,需要逐步過渡。同時注意weex編譯模式的切換是項目級的,不支持同項目下某個nvue頁面使用weex模式,另一個nvue頁面使用uni-app模式。
weex編譯模式 | uni-app編譯模式 | |
---|---|---|
平臺 | 僅App | 所有端,包含小程序和H5 |
組件 | weex組件如div | uni-app組件如view |
生命周期 | 只支持weex生命周期 | 支持所有uni-app生命周期 |
JS API | weex API、uni API、Plus API | weex API、uni API、Plus API |
單位 | 750px是屏幕寬度,wx是固定像素單位 | 750rpx是屏幕寬度,px是固定像素單位 |
全局樣式 | 手動引入 | app.vue的樣式即為全局樣式 |
頁面滾動 | 必須給頁面套或組件 | 默認支持頁面滾動 |
在 manifest.json 中修改2種編譯模式,manifest.json -> app-plus -> nvueCompiler 切換編譯模式。
nvueCompiler 有兩個值:
// manifest.json
{
// ...
/* App平臺特有配置 */
"app-plus": {
"nvueCompiler":"uni-app" //是否啟用 uni-app 模式
}
}
如果沒有在manifest里明確配置,默認是weex模式。這是為了向下兼容。
weex 編譯模式不支持 onNavigationBarButtonTap 生命周期函數(shù)的寫法。在 nvue 中監(jiān)聽原生標題欄按鈕點擊事件,詳見:uni.onNavigationBarButtonTap。
weex編譯模式不支持onShow生命周期,但熟悉5+的話,可利用監(jiān)聽webview的addEventListener show事件實現(xiàn)onShow效果。
weex 編譯模式不支持vuex。
nvue 的頁面跳轉,與 weex 不同,仍然遵循 uni-app 的路由模型。vue 頁面和 nvue 頁面之間不管怎么跳轉,都遵循這個模型。包括 nvue 頁面跳向 nvue 頁面。每個頁面都需要在 pages.json 中注冊,調用 uni-app 的 路由 API 進行跳轉。
原生開發(fā)沒有頁面滾動的概念,頁面內容高過屏幕高度并不會自動滾動,只有部分組件可滾動(list、waterfall、scroll-view/scroller),要滾得內容需要套在可滾動組件下。這不符合前端開發(fā)的習慣,所以在 nvue 編譯為 uni-app模式時,給頁面外層自動套了一個 scroller,頁面內容過高會自動滾動。(組件不會套,頁面有recycle-list時也不會套)。 可以設置不自動套。
{
"path": "",
"style": {
"disableScroll": true // 不嵌套 scroller
}
}
weex 編譯模式下支持使用 weex ui ,例子見:https://ext.dcloud.net.cn/plugin?id=442。但相比uni-app插件市場及官方uni ui而言,weex語法的組件生態(tài)還是比較欠缺的。
在HBuilderX的 uni-app 項目中,新建頁面,彈出界面右上角可以選擇是建立vue頁面還是nvue頁面,或者2個同時建。
不管是vue頁面還是nvue頁面,都需要在pages.json中注冊。如果在HBuilderX中新建頁面是會自動注冊的,如果使用其他編輯器,則需要自行在pages.json里注冊。
如果一個頁面路由下同時有vue頁面和nvue頁面,即出現(xiàn)同名的vue和nvue文件。那么在App端,會僅使用nvue頁面,同名的vue文件將不會被編譯到App端。而在非App端,會優(yōu)先使用vue頁面。
如果不同名,只有nvue頁面,則在非app端,只有uni-app編譯模式的nvue文件才會編譯。
nvue 頁面結構同 vue, 由 template、style、script 構成。
HBuilderX內置了weex調試工具的強化版,包括審查界面元素、看log、debug打斷點,詳見
在 uni-app 中,nvue 和 vue 頁面可以混搭使用。
推薦使用uni.$on,uni.$emit的方式進行頁面通訊,舊的通訊方式(uni.postMessage及plus.webview.postMessageToUniNView)不再推薦使用。
通訊實現(xiàn)方式
// 接收信息的頁面
// $on(eventName, callback)
uni.$on('page-popup', (data) => {
console.log('標題:' + data.title)
console.log('內容:' + data.content)
})
// 發(fā)送信息的頁面
// $emit(eventName, data)
uni.$emit('page-popup', {
title: '我是title',
content: '我是content'
});
使用此頁面通訊時注意事項:要在頁面卸載前,使用 uni.$off 移除事件監(jiān)聽器。參考
步驟:
代碼示例:
//test.nvue
<template>
<div @click="test">
<text>點擊頁面發(fā)送數(shù)據(jù)</text>
</div>
</template>
<script>
export default {
methods: {
test(e) {
uni.postMessage({test: "數(shù)據(jù)",value:"數(shù)據(jù)"});
}
}
}
</script>
//App.vue
<script>
export default {
onUniNViewMessage:function(e){
console.log("App.vue收到數(shù)據(jù)")
console.log(JSON.stringify(e.data))
},
onLaunch: function() {
console.log('App Launch');
}
}
</script>
步驟:
const globalEvent = weex.requireModule('globalEvent');
globalEvent.addEventListener("plusMessage", e => {
console.log(e.data);//得到數(shù)據(jù)
});
代碼示例:
//index.nvue
<template>
<div @click="test">
<text>點擊頁面發(fā)送數(shù)據(jù){{num}}</text>
</div>
</template>
<script>
const globalEvent = weex.requireModule('globalEvent');
export default {
data() {
return {
num: "0"
}
},
created() {
globalEvent.addEventListener("plusMessage", e => {
console.log(e.data);
if (e.data.num) { //存在num時才賦值,在nvue里調用uni的API也會觸發(fā)plusMessage事件,所以需要判斷需要的數(shù)據(jù)是否存在
this.num = e.data.num
}
});
},
methods: {
test(e) {
uni.navigateTo({
url: '../test/test'
})
}
}
}
//test.vue
<template>
<view>
<button type="primary" @click="test">點擊改變nvue的數(shù)據(jù)</button>
</view>
</template>
<script>
export default {
methods: {
test() {
var pages = getCurrentPages();
var page = pages[pages.length - 2];
var currentWebview = page.$getAppWebview();
plus.webview.postMessageToUniNView({
num: "123"
}, currentWebview.id);
uni.navigateBack()
}
}
}
</script>
除了通信事件,vue 和 nvue 頁面之間還可以共享變量和存儲。 uni-app提供的共享變量和數(shù)據(jù)的方案如下:
1. vuex:
自HBuilderX 2.2.5起,nvue支持vuex。這是vue官方的狀態(tài)管理工具。
注意:
2. uni.storage:
vue和nvue頁面可以使用相同的uni.storage存儲。這個存儲是持久化的。 比如登陸狀態(tài)可以保存在這里。
App端還支持plus.sqlite,也是共享通用的。
3. globalData:
小程序有globalData機制,這套機制在uni-app里也可以使用,全端通用。 在App.vue文件里定義globalData,如下:
<script>
export default {
globalData: {
text: 'text'
},
onLaunch: function() {
console.log('App Launch')
},
onShow: function() {
console.log('App Show')
},
onHide: function() {
console.log('App Hide')
}
}
</script>
js中操作globalData的方式如下: getApp().globalData.text = 'test'
如果需要把globalData的數(shù)據(jù)綁定到頁面上,可在頁面的onShow聲明周期里進行變量重賦值。
uni-app是邏輯層和視圖層分離的。此時會產生兩層通信成本。比如拖動視圖層的元素,如果在邏輯層不停接收事件,因為通信損耗會產生不順滑的體驗。
BindingX是weex提供的一種預描述交互語法。由原生解析BindingX規(guī)則,按此規(guī)則處理視圖層的交互和動效。不再實時去js邏輯層運行和通信。
BindingX類似一種強化版的css,運行性能高,但沒有js那樣足夠強的編程靈活性。
uni-app 內置了 BindingX,可在 nvue 中使用 BindingX 完成復雜的動畫效果。
代碼示例
<template>
<div class="container">
<div ref="b1" class="btn" style="background-color:#6A1B9A" @click="clickBtn">
<text class="text">A</text>
</div>
<div ref="b2" class="btn" style="background-color:#0277BD" @click="clickBtn">
<text class="text">B</text>
</div>
<div ref="b3" class="btn" style="background-color:#FF9800" @click="clickBtn">
<text class="text">C</text>
</div>
<div ref="main_btn" class="btn" @click="clickBtn">
<image class="image" ref="main_image" src="https://gw.alicdn.com/tfs/TB1PZ25antYBeNjy1XdXXXXyVXa-128-128.png" rel="external nofollow" />
</div>
</div>
</template>
<script>
const Binding = uni.requireNativePlugin('bindingx');
module.exports = {
data: {
isExpanded: false
},
methods: {
getEl: function(el) {
if (typeof el === 'string' || typeof el === 'number') return el;
if (WXEnvironment) {
return el.ref;
} else {
return el instanceof HTMLElement ? el : el.$el;
}
},
collapse: function() {
let main_btn = this.getEl(this.$refs.main_btn);
let main_image = this.getEl(this.$refs.main_image);
let b1 = this.getEl(this.$refs.b1);
let b2 = this.getEl(this.$refs.b2);
let b3 = this.getEl(this.$refs.b3);
let main_binding = Binding.bind({
eventType: 'timing',
exitExpression: 't>800',
props: [{
element: main_image,
property: 'transform.rotateZ',
expression: 'easeOutQuint(t,45,0-45,800)'
}, {
element: main_btn,
property: 'background-color',
expression: "evaluateColor('#607D8B','#ff0000',min(t,800)/800)"
}]
}, function(res) {
if (res.state === 'exit') {
Binding.unbind({
token: main_binding
})
}
});
let btn_binding = Binding.bind({
eventType: 'timing',
exitExpression: 't>800',
props: [{
element: b1,
property: 'transform.translateY',
expression: "easeOutQuint(t,-150,150,800)"
}, {
element: b2,
property: 'transform.translateY',
expression: "t<=100?0:easeOutQuint(t-100,-300,300,700)"
}, {
element: b3,
property: 'transform.translateY',
expression: "t<=200?0:easeOutQuint(t-200,-450,450,600)"
}]
}, function(res) {
if (res.state === 'exit') {
Binding.unbind({
token: btn_binding
})
}
})
},
expand: function() {
let main_btn = this.getEl(this.$refs.main_btn);
let main_image = this.getEl(this.$refs.main_image);
let b1 = this.getEl(this.$refs.b1);
let b2 = this.getEl(this.$refs.b2);
let b3 = this.getEl(this.$refs.b3);
let main_binding = Binding.bind({
eventType: 'timing',
exitExpression: 't>100',
props: [{
element: main_image,
property: 'transform.rotateZ',
expression: 'linear(t,0,45,100)'
}, {
element: main_btn,
property: 'background-color',
expression: "evaluateColor('#ff0000','#607D8B',min(t,100)/100)"
}]
}, function(res) {
if (res.state === 'exit') {
Binding.unbind({
token: main_binding
})
}
});
let btn_binding = Binding.bind({
eventType: 'timing',
exitExpression: 't>800',
props: [{
element: b1,
property: 'transform.translateY',
expression: "easeOutBounce(t,0,0-150,800)"
}, {
element: b2,
property: 'transform.translateY',
expression: "t<=100?0:easeOutBounce(t-100,0,0-300,700)"
}, {
element: b3,
property: 'transform.translateY',
expression: "t<=200?0:easeOutBounce(t-200,0,0-450,600)"
}]
}, function(res) {
if (res.state === 'exit') {
Binding.unbind({
token: btn_binding
})
}
})
},
clickBtn: function(e) {
if (this.isExpanded) {
this.collapse();
} else {
this.expand();
}
this.isExpanded = !this.isExpanded;
}
}
}
</script>
<style>
.container {
flex: 1;
}
.image {
width: 60px;
height: 60px;
}
.text {
color: #ffffff;
font-size: 30px;
}
.btn {
width: 100px;
height: 100px;
background-color: #ff0000;
align-items: center;
justify-content: center;
position: absolute;
border-radius: 50px;
bottom: 25px;
right: 25px;
}
</style>
注意
nvue 支持大部分 uni-app API ,下面只列舉目前還不支持的 API 。
動畫
API | 說明 |
---|---|
uni.createAnimation() | 創(chuàng)建一個動畫實例 |
滾動
API | 說明 |
---|---|
uni.pageScrollTo() | 將頁面滾動到目標位置 |
繪畫
canvas API使用,詳見canvas文檔。
節(jié)點布局交互
API | 說明 |
---|---|
uni.createIntersectionObserver() | 創(chuàng)建并返回一個 IntersectionObserver 對象實例 |
基于原生引擎的渲染,雖然還是前端技術棧,但和web開發(fā)肯定是有區(qū)別的。
下面有些正確和錯誤的寫法示例對比:
/ 錯誤 /
#id {} .a .b .c {} .a > .b {}
/ 正確 / .class {}
- border 不支持簡寫
```css
/* 錯誤 */
.class {
border: 1px red solid;
}
/* 正確 */
.class {
border-width: 1px;
border-style: solid;
border-color: red;
}
/* 錯誤 */
.class {
background: red;
}
/* 正確 */
.class {
background-color: red;
}
/ 正確 / .class { border-width: 1px; border-style: solid; border-color: red; / #ifndef APP-PLUS-NVUE / -webkit-transform: scaleY(.5); / #endif/ }
## Android平臺陰影(box-shadow)問題
Android平臺weex對陰影樣式(`box-shadow`)支持不完善,如設置圓角邊框時陰影樣式顯示不正常、設置動畫時在`Android7`上顯示不正常等。為解決這些問題,從HBuilderX 2.4.7起,新增`elevation`屬性(組件的屬性,不是css樣式)設置組件的層級,`Number`類型,層級值越大陰影越明顯,陰影效果也與組件位置有關,越靠近頁面底部陰影效果越明顯
**用法**
```html
<view elevation="5px"></view>
注意
更多建議: