介绍

uni-app App端内置 weex 渲染引擎,提供了原生渲染能力。

uni-app 里根据编译配置不同,可以使用 weex 的组件,也可以使用小程序组件(即uni-app组件)。编写页面时页面后缀名为 .nvue(native vue的缩写)

nvue 相当于给 weex 补充了大量 uni-app 的组件和api,以及丰富的 Plus API、Native.js、原生插件。

以往的 weex ,有个很大的问题是它只是一个高性能的渲染器,没有足够的API能力,使得开发时非常依赖原生工程师协作,开发者本来想节约成本,结果需要前端、iOS、Android 3拨人开发,适得其反。而 nvue 解决了这个大问题,让前端工程师可以直接开发完整 App,并提供原生插件的市场交易和云打包。这些组合方案,开发者切实的提高效率、降低成本。

如果你已经是 weex 的开发者,具有 weex 的填坑能力,那么 nvue 是你的更优选择,能切实提升你的开发效率,降低成本。

如果你不开发App,那么你不太需要nvue。

如果你是web前端,不熟悉 weex,那么建议你仍然以使用 vue 为主,在App端某些 vue 表现不佳的场景下使用 nvue 作为强化补充:

  • 左右拖动的长列表。在webview里,通过swiper+scroll-view实现左右拖动的长列表,前端模拟下拉刷新,这套方案的性能不好。此时推荐使用nvue,比如新建uni-app项目时的新闻示例模板,就是首页采用了nvue
  • 如需要将软键盘右下角按钮文字改为“发送”,则需要使用nvue
  • 前端控件无法覆盖原生控件的问题。在nvue下,都是原生控件,覆盖map、video等不需要cover-view(如需要发布到小程序,仍然推荐写cover-view)
  • 同样因为层级问题得到解决,nvue可以实现video内嵌到swiper中,以实现抖音式视频滑动切换,例子见插件市场;nvue的视频全屏后,仍然可以通过cover-view实现内容覆盖,比如增加文字标题、分享按钮。
  • nvue下有live-pusher组件,和小程序对齐。而vue页面下使用直播,需在条件编译里单独调用plus.video的API。
  • nvue下的map组件,小程序对齐。而vue页面的map组件有一些差异。
  • App端实现粘性布局,比如滚动吸顶,则nvue才能保证高性能,例子见插件市场此外,App端,vue页面上也可以覆盖subnvue(一种非全屏的nvue页面覆盖在webview上),以解决App上的原生控件层级问题。详见

项目渲染模式

uni-app在App端,支持vue页面和nvue页面混搭、互相跳转。也支持纯nvue项目。

在manifest.json源码视图的"app-plus"下配置"renderer":"native",即代表App端启用纯原生渲染模式。此时pages.json注册的vue页面将被忽略,vue组件中的代码也需覆盖nvue规范,并会被原生渲染。

启动纯原生渲染,可以减少App端的包体积、加快App启动速度。因为webview渲染模式的相关模块将被移除。

如果不指定该值,默认是不启动纯原生渲染的。

  1. // manifest.json
  2. {
  3. // ...
  4. /* App平台特有配置 */
  5. "app-plus": {
  6. "renderer": "native", //App端纯原生渲染模式
  7. }
  8. }

nvue页面编译模式差异

uni-app 深度改进了 weex,提供了2种编译模式,一种是常规的 weex 组件模式,即编写<div>。另一种是 uni-app 组件模式,即编写<view>。后者更提供了编译为小程序和H5的能力,实现了全端输出。

也可以理解为uni-app做了一个原生渲染的小程序引擎。

weex编译模式uni-app编译模式
平台仅App所有端,包含小程序和H5
组件weex组件如divuni-app组件如view
生命周期只支持weex生命周期支持所有uni-app生命周期
JS APIweex API、uni API、Plus APIweex API、uni API、Plus API
单位750px是屏幕宽度,wx是固定像素单位750rpx是屏幕宽度,px是固定像素单位
全局样式手动引入app.vue的样式即为全局样式
页面滚动必须给页面套组件默认支持页面滚动

在 manifest.json 中修改2种编译模式,manifest.json -> app-plus -> nvueCompiler 切换编译模式。

nvueCompiler 有两个值:

  • weex
  • uni-app
  1. // manifest.json
  2. {
  3. // ...
  4. /* App平台特有配置 */
  5. "app-plus": {
  6. "nvueCompiler":"uni-app" //是否启用 uni-app 模式
  7. }
  8. }
  • 如果没有在manifest里明确配置,默认是weex模式。这是为了向下兼容。

快速上手

1. 新建 nvue 页面

在HBuilderX的 uni-app 项目中,新建页面,弹出界面右上角可以选择是建立vue页面还是nvue页面,或者2个同时建。

不管是vue页面还是nvue页面,都需要在pages.json中注册。如果在HBuilderX中新建页面是会自动注册的,如果使用其他编辑器,则需要自行在pages.json里注册。

如果一个页面路由下同时有vue页面和nvue页面,即出现同名的vue和nvue文件。那么在App端,会优先使用nvue页面,同名的vue文件将不会被编译到App端。而在非App端,会优先使用vue页面。

如果不同名,只有nvue页面,则在非app端,只有uni-app编译模式的nvue文件才会编译。

2. 开发 nvue 页面

nvue 页面结构同 vue, 由 template、style、script 构成。

  • template: 模板写法、数据绑定同 vue。组件支持2种模式,1、 weex 组件,参考:weex 内置组件;2、uni-app组件,参考:nvue中支持的uni-app组件
  • style:由于采用原生渲染,并非所有浏览器的 css 均支持,布局模型只支持 flex 布局,虽然不会造成某些界面布局无法实现,但写法要注意。详见:weex 样式
  • script:写法同 vue,并支持3种API
    • weex API :使用前需先引入对应模块,参考:weex 内置模块
    • uni API:nvue可以使用部分 uni API,详细支持列表请参照:nvue 里可使用的 uni-app API
    • plus API:在自定义组件编译模式下,nvue里可直接使用plus APIweex组件代码示例(uni-app组件代码和普通uni-app的vue源码相同)
  1. <template>
  2. <div class="example">
  3. <div class="item" @click="changeNum"><text class="text">点击文字,改变数字大小: {{num}}</text></div>
  4. <div class="item" @click="showModal('native')"><text class="text">使用 weex 的 API 弹出模态框</text></div>
  5. <div class="item" @click="showModal('uni')"><text class="text">使用 uni-app 的 API 弹出模态框</text></div>
  6. <div class="item" @click="showModal('plus')"><text class="text">使用 plus 的 API 弹出模态框</text></div>
  7. </div>
  8. </template>
  9. <script>
  10. export default {
  11. data() {return {num: 1}},
  12. created() {console.log('页面created')},
  13. methods: {
  14. changeNum() { this.num += 1; },
  15. showModal(type) {
  16. if (type === 'uni') {
  17. uni.showModal({content: '使用 uni-app 的 API 弹出模态框'});
  18. }
  19. else if (type === 'plus') {
  20. plus.nativeUI.alert('使用 uni-app 的 API 弹出模态框');
  21. } else {
  22. var modal = weex.requireModule('modal')
  23. modal.alert({message: '使用 weex 的 API 弹出模态框'});
  24. }
  25. }
  26. }
  27. }
  28. </script>
  29. <style>
  30. .example {flex-direction: column;}
  31. .text {line-height: 80px;font-size: 32px;color: #333;}
  32. .item {height: 80px;width: 690px;margin: 30px;background-color: #f8f8f8;border-width: 1px;border-color: #eee;border-radius: 10px;align-items: center;}
  33. </style>

3. 调试 nvue 页面

HBuilderX内置了更好用的weex/uni-app调试工具,包括审查界面元素、看log、debug打断点,详见

生命周期

nvue 的 uni-app 编译模式的生命周期同普通vue页面。而 weex 编译模式,即普通 weex 生命周期函数如下:

Vue 生命周期钩子说明
beforeCreate在实例初始化之后,数据观测 (data observer) 和 event/watcher 事件配置之前被调用
created在实例创建完成后被立即调用
beforeMount在挂载开始之前被调用
mountedel 被新创建的 vm.$el 替换时调用
beforeUpdate数据更新时调用
updated页面元素更新后调用
beforeDestroy实例销毁之前调用
destroyed实例销毁后调用
errorCaptured当捕获一个来自子孙组件的错误时被调用

注意: weex 编译模式不支持 onNavigationBarButtonTap 生命周期函数的写法。在 nvue 中监听原生标题栏按钮点击事件,详见:uni.onNavigationBarButtonTap

页面跳转

nvue 的页面跳转,与 weex 不同,仍然遵循 uni-app 的路由模型。vue 页面和 nvue 页面之间不管怎么跳转,都遵循这个模型。包括 nvue 页面跳向 nvue 页面。

每个页面都需要在 pages.json 中注册,调用 uni-app路由 API 进行跳转。

nvue 和 vue 相互通讯

在 uni-app 中,nvue 和 vue 页面可以混搭使用。

nvue 向 vue 通讯

步骤:

  • nvue 使用 uni.postMessage(data) 发送数据通讯,dataJSON 格式(键值对的值仅支持String)。
  • App.vue 里使用 onUniNViewMessage 进行监听。代码示例:
  1. //test.nvue
  2. <template>
  3. <div @click="test">
  4. <text>点击页面发送数据</text>
  5. </div>
  6. </template>
  7. <script>
  8. export default {
  9. methods: {
  10. test(e) {
  11. uni.postMessage({test: "数据",value:"数据"});
  12. }
  13. }
  14. }
  15. </script>
  1. //App.vue
  2. <script>
  3. export default {
  4. onUniNViewMessage:function(e){
  5. console.log("App.vue收到数据")
  6. console.log(JSON.stringify(e.data))
  7. },
  8. onLaunch: function() {
  9. console.log('App Launch');
  10. }
  11. }
  12. </script>

vue 向 nvue 通讯

步骤:

  • vue 里使用 plus.webview.postMessageToUniNView(data,nvueId) 发送消息,dataJSON 格式(键值对的值仅支持String),nvueIdnvue 所在 webview 的 id,webview的 id 获取方式参考:$getAppWebview()
  • nvue 里引用 globalEvent 模块监听 plusMessage 事件,如下:
  1. const globalEvent = weex.requireModule('globalEvent');
  2. globalEvent.addEventListener("plusMessage", e => {
  3. console.log(e.data);//得到数据
  4. });

代码示例:

  1. //index.nvue
  2. <template>
  3. <div @click="test">
  4. <text>点击页面发送数据{{num}}</text>
  5. </div>
  6. </template>
  7. <script>
  8. const globalEvent = weex.requireModule('globalEvent');
  9. export default {
  10. data() {
  11. return {
  12. num: "0"
  13. }
  14. },
  15. created() {
  16. globalEvent.addEventListener("plusMessage", e => {
  17. console.log(e.data);
  18. if (e.data.num) { //存在num时才赋值,在nvue里调用uni的API也会触发plusMessage事件,所以需要判断需要的数据是否存在
  19. this.num = e.data.num
  20. }
  21. });
  22. },
  23. methods: {
  24. test(e) {
  25. uni.navigateTo({
  26. url: '../test/test'
  27. })
  28. }
  29. }
  30. }
  1. //test.vue
  2. <template>
  3. <view>
  4. <button type="primary" @click="test">点击改变nvue的数据</button>
  5. </view>
  6. </template>
  7. <script>
  8. export default {
  9. methods: {
  10. test() {
  11. var pages = getCurrentPages();
  12. var page = pages[pages.length - 2];
  13. var currentWebview = page.$getAppWebview();
  14. plus.webview.postMessageToUniNView({
  15. num: "123"
  16. }, currentWebview.id);
  17. uni.navigateBack()
  18. }
  19. }
  20. }
  21. </script>

vue 和 nvue 共享的变量和数据

除了通信事件,vue 和 nvue 页面之间还可以共享变量和存储。但注意nvue不支持vuex,uni-app提供的共享变量和数据的方案如下:

  • uni.storagevue和nvue页面可以使用相同的uni.storage存储。这个存储是持久化的。比如登陆状态可以保存在这里。

  • globalData小程序有globalData机制,这套机制在uni-app里也可以使用,全端通用。在App.vue文件里定义globalData,如下:

  1. <script>
  2. export default {
  3. globalData: {
  4. text: 'text'
  5. },
  6. onLaunch: function() {
  7. console.log('App Launch')
  8. },
  9. onShow: function() {
  10. console.log('App Show')
  11. },
  12. onHide: function() {
  13. console.log('App Hide')
  14. }
  15. }
  16. </script>

js中操作globalData的方式如下:getApp().globalData.text = 'test'

如果需要把globalData的数据绑定到页面上,可在页面的onShow声明周期里进行变量重赋值。HBuilderX 2.0.3起,nvue页面在uni-app编译模式下,也支持onShow。

weex编译模式不支持onShow,但熟悉5+的话,可利用监听webview的addEventListener show事件实现onShow效果。

weex 编译模式中使用 weex 第三方库

nvue 的 weex 编译模式中,可以使用weex的第三方库,这里以 Weex Ui 为例,介绍如何使用。

1.初始化工程

  1. npm init -y

2.安装依赖 (暂时只支持名称里包含weex的库)

  1. npm i weex-ui -S

3.在nvue里面使用

  1. <template>
  2. <div>
  3. <wxc-cell label="标题" title="Weex Ui" :has-arrow="true" @wxcCellClicked="wxcCellClicked" :has-margin="true"></wxc-cell>
  4. </div>
  5. </template>
  6. <script>
  7. import { WxcCell } from 'weex-ui';
  8. export default {
  9. components: { WxcCell },
  10. methods: {
  11. wxcCellClicked (e) {
  12. console.log(e)
  13. }
  14. }
  15. };
  16. </script>

Tis:

nvue 里使用 BindingX

uni-app 内置了 BindingX,可在 nvue 中使用 BindingX 完成复杂的动画效果。

  • 使用方式可参考 BindingX 快速开始,demo示例可参考 BindingX 示例vue 的相关示例,将实验田里的 vue 代码拷贝到 nvue 文件里即可。
  • 若引入 weex-bindingx 时发现不生效,检查项目路径,路径不能含有中文。
  • 使用npm时如果命令行报错,需要注意看命令行的提示代码示例
  1. <template>
  2. <div class="container">
  3. <div ref="b1" class="btn" style="background-color:#6A1B9A" @click="clickBtn">
  4. <text class="text">A</text>
  5. </div>
  6. <div ref="b2" class="btn" style="background-color:#0277BD" @click="clickBtn">
  7. <text class="text">B</text>
  8. </div>
  9. <div ref="b3" class="btn" style="background-color:#FF9800" @click="clickBtn">
  10. <text class="text">C</text>
  11. </div>
  12. <div ref="main_btn" class="btn" @click="clickBtn">
  13. <image class="image" ref="main_image" src="https://gw.alicdn.com/tfs/TB1PZ25antYBeNjy1XdXXXXyVXa-128-128.png" />
  14. </div>
  15. </div>
  16. </template>
  17. <script>
  18. import Binding from 'weex-bindingx';
  19. module.exports = {
  20. data: {
  21. isExpanded: false
  22. },
  23. methods: {
  24. getEl: function(el) {
  25. if (typeof el === 'string' || typeof el === 'number') return el;
  26. if (WXEnvironment) {
  27. return el.ref;
  28. } else {
  29. return el instanceof HTMLElement ? el : el.$el;
  30. }
  31. },
  32. collapse: function() {
  33. let main_btn = this.getEl(this.$refs.main_btn);
  34. let main_image = this.getEl(this.$refs.main_image);
  35. let b1 = this.getEl(this.$refs.b1);
  36. let b2 = this.getEl(this.$refs.b2);
  37. let b3 = this.getEl(this.$refs.b3);
  38. Binding.bind({
  39. eventType: 'timing',
  40. exitExpression: {
  41. origin: 't>800'
  42. },
  43. props: [{
  44. element: main_image,
  45. property: 'transform.rotateZ',
  46. expression: {
  47. origin: 'easeOutQuint(t,45,0-45,800)'
  48. }
  49. },{
  50. element: main_btn,
  51. property: 'background-color',
  52. expression: {
  53. origin: "evaluateColor('#607D8B','#ff0000',min(t,800)/800)"
  54. }
  55. }]
  56. });
  57. Binding.bind({
  58. eventType: 'timing',
  59. exitExpression: {
  60. origin: 't>800'
  61. },
  62. props: [{
  63. element: b1,
  64. property: 'transform.translateY',
  65. expression: {
  66. origin: "easeOutQuint(t,-150,150,800)"
  67. }
  68. },{
  69. element: b2,
  70. property: 'transform.translateY',
  71. expression: {
  72. origin: "t<=100?0:easeOutQuint(t-100,-300,300,700)"
  73. }
  74. },{
  75. element: b3,
  76. property: 'transform.translateY',
  77. expression: {
  78. origin: "t<=200?0:easeOutQuint(t-200,-450,450,600)"
  79. }
  80. }]
  81. })
  82. },
  83. expand: function() {
  84. let main_btn = this.getEl(this.$refs.main_btn);
  85. let main_image = this.getEl(this.$refs.main_image);
  86. let b1 = this.getEl(this.$refs.b1);
  87. let b2 = this.getEl(this.$refs.b2);
  88. let b3 = this.getEl(this.$refs.b3);
  89. Binding.bind({
  90. eventType: 'timing',
  91. exitExpression: {
  92. origin: 't>100'
  93. },
  94. props: [{
  95. element: main_image,
  96. property: 'transform.rotateZ',
  97. expression: {
  98. origin: 'linear(t,0,45,100)'
  99. }
  100. },{
  101. element: main_btn,
  102. property: 'background-color',
  103. expression: {
  104. origin: "evaluateColor('#ff0000','#607D8B',min(t,100)/100)"
  105. }
  106. }]
  107. });
  108. Binding.bind({
  109. eventType: 'timing',
  110. exitExpression: {
  111. origin: 't>800'
  112. },
  113. props: [{
  114. element: b1,
  115. property: 'transform.translateY',
  116. expression: {
  117. origin: "easeOutBounce(t,0,0-150,800)"
  118. }
  119. }, {
  120. element: b2,
  121. property: 'transform.translateY',
  122. expression: {
  123. origin: "t<=100?0:easeOutBounce(t-100,0,0-300,700)"
  124. }
  125. }, {
  126. element: b3,
  127. property: 'transform.translateY',
  128. expression: {
  129. origin: "t<=200?0:easeOutBounce(t-200,0,0-450,600)"
  130. }
  131. }]
  132. })
  133. },
  134. clickBtn: function(e) {
  135. if (this.isExpanded) {
  136. this.collapse();
  137. } else {
  138. this.expand();
  139. }
  140. this.isExpanded = !this.isExpanded;
  141. }
  142. }
  143. }
  144. </script>
  145. <style>
  146. .container {flex: 1;}
  147. .image {width: 60px;height: 60px;}
  148. .text {color: #ffffff;font-size: 30px;}
  149. .btn {width: 100px;height: 100px;background-color: #ff0000;align-items: center;justify-content: center;position: absolute;border-radius: 50px;bottom: 25px;right: 25px;}
  150. </style>

nvue 里使用 HTML5Plus API

  • 自定义组件编译模式,HBuilderX 1.9.8起,nvue页面可直接使用plus的API,并且同样不需要等待plus ready。
  • 非自定义组件编译模式,nvue无法直接调用 HTML5Plus 相关的 API,可以通过与vue通讯的方式,在 vue 页面调用 HTML5Plus 的API,传值给nvue页面,以达到类似的结果。以下为非自定义组件模式下,通过页面通讯在nvue页面获得设备uuid的示例(推荐老项目尽快升级为自定义组件,享受更高的性能和功能):nvue 页面
  1. <template>
  2. <div @click="test">
  3. <text>点击页面得到设备的uuid为:{{uuid}}</text>
  4. </div>
  5. </template>
  6. <script>
  7. const globalEvent = weex.requireModule('globalEvent');
  8. export default {
  9. data() {
  10. return {
  11. uuid: null
  12. }
  13. },
  14. created() {
  15. globalEvent.addEventListener("plusMessage", e => {
  16. console.log(e.data);
  17. if (e.data.detail) {
  18. this.uuid = e.data.detail
  19. }
  20. });
  21. },
  22. methods: {
  23. test(e) {
  24. uni.postMessage({module:"device",api:"uuid"});
  25. }
  26. }
  27. }
  28. </script>

App.vue

  1. <script>
  2. export default {
  3. onUniNViewMessage: function(e) {
  4. console.log("App.vue收到数据:" + JSON.stringify(e.data));
  5. console.log("uuid:" + plus[e.data.module][e.data.api])
  6. var pages = getCurrentPages();
  7. var page = pages[pages.length - 1];
  8. var currentWebview = page.$getAppWebview();
  9. plus.webview.postMessageToUniNView({
  10. detail: plus[e.data.module][e.data.api]
  11. }, currentWebview.id);
  12. }
  13. }
  14. </script>

nvue 里可使用的 uni-app API

nvue 支持大部分 uni-app API ,下面只列举目前还不支持的 API 。

地图

API说明
uni.createMapContext()创建并返回 map 上下文

视频

API说明
uni.createVideoContext()创建并返回 video 上下文

直播推流

API说明
uni.createLivePusherContext()创建并返回 livePusher 上下文

动画

API说明
uni.createAnimation()创建一个动画实例

滚动

API说明
uni.pageScrollTo()将页面滚动到目标位置

绘画

API说明
uni.createCanvasContext()创建 canvas 绘图上下文
uni.canvasToTempFilePath()把当前画布指定区域的内容导出生成指定大小的图片,并返回文件路径
uni.canvasGetImageData()返回一个数组,用来描述 canvas 区域隐含的像素数据
uni.canvasPutImageData()将像素数据绘制到画布的方法

下拉刷新

API说明
uni.onPullDownRefresh()监听该页面用户下拉刷新事件
uni.startPullDownRefresh()开始下拉刷新
uni.stopPullDownRefresh()停止当前页面下拉刷新

节点信息

API说明
uni.createSelectorQuery()返回一个 SelectorQuery 对象实例

节点布局交互

API说明
uni.createIntersectionObserver()创建并返回一个 IntersectionObserver 对象实例

nvue的新增API

为了解决nvue的weex编译模式不支持uni-app生命周期的问题,在nvue 里新增了几个特殊的 API。如果是uni-app编译模式,无需使用这些API:

uni.onNavigationBarButtonTap(CALLBACK)

监听原生标题栏按钮点击事件。

CALLBACK 参数说明:

属性类型说明
indexNumber原生标题栏按钮数组的下标

代码示例

  1. export default {
  2. created() {
  3. uni.onNavigationBarButtonTap((e) => {
  4. console.log("监听到原生标题栏按钮点击事件");
  5. console.log(e);
  6. })
  7. }
  8. }
uni.onNavigationBarSearchInputChanged(CALLBACK)

监听原生标题栏搜索输入框输入内容变化事件。

CALLBACK 参数说明:

属性类型说明
textString搜索输入框输入内容

代码示例

  1. export default {
  2. created() {
  3. uni.onNavigationBarSearchInputChanged((e) => {
  4. console.log("输入内容:"+ e.text);
  5. })
  6. }
  7. }
uni.onNavigationBarSearchInputConfirmed()

监听原生标题栏搜索输入框搜索事件,用户点击软键盘上的“搜索”按钮时触发。

代码示例

  1. export default {
  2. created() {
  3. uni.onNavigationBarSearchInputConfirmed(() => {
  4. console.log("用户点击软键盘搜索");
  5. })
  6. }
  7. }
uni.onNavigationBarSearchInputClicked()

监听原生标题栏搜索输入框点击事件。

代码示例

  1. export default {
  2. created() {
  3. uni.onNavigationBarSearchInputClicked(() => {
  4. console.log("点击输入框");
  5. })
  6. }
  7. }

nvue开发与vue开发的常见区别

基于原生引擎的渲染,虽然还是前端技术栈,但和web开发肯定是有区别的。

比如没有dom、window等对象,比如只支持flex布局,好在uni-app一直以来也是推动开发者不使用dom,默认使用flex布局。所以适应起来要好一些。

但仍然还有一些区别需要注意:

  • nvue 页面只能使用 flex 布局,不支持其他布局方式。
  • weex 下,页面内容高过屏幕高度并不会自动滚动,它没有页面滚动的概念,只有区域滚动,要滚得内容需要套在组件下。在 nvue 编译为 uni-app模式时,纠正了这个问题,页面内容过高会自动滚动。
  • weex 下,px是与屏幕宽度相关的动态单位,750px代表成屏幕宽度100%,它的静态单位是wx。在 nvue 编译为 uni-app模式时,纠正了这个问题,rpx是与屏幕宽度相关的动态单位,px是静态单位。
  • 页面开发前,首先想清楚这个页面的纵向内容有什么,哪些是要滚动的,然后每个纵向内容的横轴排布有什么,按 flex 布局设计好界面。
  • 文字内容,必须、只能在组件下。不能在、的text区域里直接写文字。

  • 支持的css有限,不过并不影响布局出你需要的界面,flex还是非常强大的。详见

  • class 进行绑定时只支持数组语法。

单位说明

  • weex的css单位支持如下:
    • px:以750宽的屏幕为基准动态计算的长度单位,与vue页面中的rpx理念相同。(一定要注意nvue里的px,和vue里的px逻辑不一样)
    • wx:与设备屏幕宽度无关的长度单位,与vue页面中的px理念相同
  • uni-app编译模式下单位和普通vue相同,rpx是以750宽屏幕为基准的动态单位。px则是固定像素单位。详见

注意事项

  • 现阶段 nvue 的定位是 vue 的补充。在 App 平台实现一些 vue 上无法实现或性能有问题的场景。
  • nvue 的各组件在安卓端默认是透明的,如果不设置background-color,可能会导致出现重影的问题。
  • 在 App.vue 中定义的全局js变量不会在 nvue 页面生效。globalData是生效的。
  • nvue 不支持 vue 里的 vuex
  • nvue 切换横竖屏时可能导致样式出现问题,建议有 nvue 的页面锁定手机方向。
  • 不能在 style 中引入字体文件,nvue 中字体图标的使用参考:weex 加载自定义字体
  • 目前不支持在 nvue 页面使用 typescript/ts。
  • nvue 页面 titleNview 设为 false时,想要模拟状态栏,可以参考:https://ask.dcloud.net.cn/article/35111

发现错误?想参与编辑?在 GitHub 上编辑此页面!