微信小程序商城开发:界面设计实战
上QQ阅读APP看书,第一时间看更新

第2章 小程序基础知识

第1章为小程序入门的各项事宜,接下来本章将介绍微信小程序开发的基础知识:项目有哪些配置文件,微信小程序的各种配置,WXSS样式语言,逻辑层.js脚本,WXML视图层开发等。

2.1 项目配置文件

可以在项目根目录使用project.config.json文件(参见1.3.1节)对项目进行配置,项目配置文件的内容参见表2-1。

表2-1 项目配置文件

其中,compileType的有效值如下。

❏ miniprogram:当前为普通小程序项目。

❏ plugin:当前为小程序插件项目。

setting中可以指定的内容如下:

scripts中指定自定义预处理的命令如下。

❏ beforeCompile:编译前预处理命令。

❏ beforePreview:预览前预处理命令。

❏ beforeUpload:上传前预处理命令。

packOptions用于配置项目在打包过程中的选项。打包是预览、上传时对项目进行的必须步骤。目前可以指定packOptions.ignore字段,忽略配置打包时符合指定规则的文件或文件夹,以跳过打包过程,这些文件或文件夹将不会出现在预览或上传的结果内。

packOptions.ignore为一对象数组,对象元素类型如下:

其中,type可以取值为folder、file、suffix、prefix、regexp、glob,分别对应文件夹、文件、后缀、前缀、正则表达式、Glob规则。所有规则值都会自动忽略大小写。

提示

value字段的值若表示文件或文件夹路径,以小程序目录(miniprogramRoot)为根目录。regexp、glob仅1.02.1809260及以上版本工具支持。

配置示例代码如下:

        {
          "packOptions": {
            "ignore": [{
              "type": "file",
              "value": "test/test.js"
            }, {
              "type": "folder",
              "value": "test"
            }, {
              "type": "suffix",
              "value": ".webp"
            }, {
              "type": "prefix",
              "value": "test-"
            }, {
              "type": "glob",
              "value": "test/**/*.js"
            }, {
              "type": "regexp",
              "value": "\\.jsx$"
            }]
          }
        }

注意

这部分设置的更改可能需要重新打开项目才能生效。

debugOptions用于配置在对项目代码进行调试时的选项。目前可以指定debugOptions. hidedInDevtools字段,用于配置是否显示调试器的源代码。

hidedInDevtools的配置规则和packOptions.ignore是一致的。当某个.js文件符合此规则时,调试器Sources面板中此文件源代码正文内容将被隐藏,显示代码示例如下:

        // xxx.js has been hided by project.config.json
        注:配置此规则后,可能需要关闭并重新打开项目才能看到效果。

项目配置代码示例如下:

        {
          "miniprogramRoot": "./src",
          "qcloudRoot": "./svr",
          "setting": {
            "postcss": true,
            "es6": true,
            "minified": true,
            "urlCheck": false
          },
          "packOptions": {
            "ignore": []
          },
          "debugOptions": {}
        }

2.2 全局配置和页面配置

每个微信小程序项目都有一个全局配置文件和多个页面配置文件。全局配置文件针对整个微信小程序项目的相关配置信息;页面配置文件只针对对应的页面,每个微信小程序都有一个对应的页面配置文件。全局配置文件和页面配置文件如果有相同的配置项目,页面配置文件的优先级高于全局配置文件,也就是以页面配置文件的效果为主。

2.2.1 全局配置

我们利用小程序根目录下的app.json文件对微信小程序进行全局配置,决定页面文件的路径、窗口表现、设置网络超时时间、设置多tab等。

每个微信小程序项目只有一个全局配置文件。下面是一个包含了部分常用配置选项的app.json:

        {
          "pages": [
            "pages/index/index",
            "pages/logs/index"
          ],
          "window": {
            "navigationBarTitleText": "Demo"
          },
          "tabBar": {
            "list": [{
              "pagePath": "pages/index/index",
              "text": "首页"
            }, {
              "pagePath": "pages/logs/logs",
              "text": "日志"
            }]
          },
          "networkTimeout": {
            "request": 10000,
            "downloadFile": 10000
          },
          "debug": true,
          "navigateToMiniProgramAppIdList": [
            "wxe5f52902cf4de896"
          ]
        }

app.json配置项列表参见表2-2。

表2-2 app.json配置项列表

(1)pages

pages用于指定小程序由哪些页面组成,是一个数组,数组中每一项都对应一个页面的“路径+文件名”信息。文件名不需要写文件后缀,开发框架会自动去寻找对应位置的.json、.js、.wxml、.wxss四个文件进行处理。数组的第一项代表小程序的初始页面(首页)。小程序中新增/减少页面,都需要对pages数组进行修改。例如,如下开发目录中:

        ├—— app.js
        ├—— app.json
        ├—— app.wxss
        ├—— pages
        |   |—— index
        |   |   ├—— index.wxml
        |   |   ├—— index.js
        |   |   ├—— index.json
        |   |   └—— index.wxss
        |   └—— logs
        |       ├—— logs.wxml
        |       └—— logs.js
        └—— utils

要在app.json中编写页面,则需要配置pages数组,代码如下所示。

        {
          "pages":[
            "pages/index/index",
            "pages/logs/logs"
          ]
        }

(2)window

window用于设置小程序的状态栏、导航条、标题、窗口背景色等,属性参见表2-3。

表2-3 window属性

其中,HexColor为十六进制颜色值,#ffffff表示白色,#000000表示黑色。

注意

navigationStyle只在app.json中生效。开启custom后,低版本客户端需要做好兼容。开发者工具基础库版本切到客户端6.7.2版本开始,navigationStyle: custom对<web-view>组件无效。

app.json示例代码如下,界面示例如图2-1所示:

图2-1 app.json示例界面

        {
          "window":{
            "navigationBarBackgroundColor": "#ffffff",
            "navigationBarTextStyle": "black",
            "navigationBarTitleText": "微信接口功能演示",
            "backgroundColor": "#eeeeee",
            "backgroundTextStyle": "light"
          }
        }

(3)tabBar

tabBar用于设置小程序多tab。例如,客户端窗口的底部或顶部有tab栏可以切换页面,可以通过tabBar配置项指定tab栏的表现,以及tab切换时显示的对应页面。tabBar属性参见表2-4。

表2-4 tabBar属性

其中,list接受一个数组,只能配置最少2个、最多5个tab。tab按数组的顺序排序,每个项都是一个对象,其属性值如下:

当属性iconPath和selectedIconPath的postion为top时,不显示icon。list属性如图2-2所示。

图2-2 list属性示例

(4)networkTimeout

networkTimeout用于指定各类网络请求的超时时间,单位均为毫秒,networkTimeout属性参见表2-5。

表2-5 networkTimeout属性

(5)debug

可以在开发者工具中开启debug模式,在开发者工具的控制台面板,调试信息以info的形式给出,其信息有Page的注册、页面路由、数据更新、事件触发等,可以帮助开发者快速定位一些常见的问题。

(6)functionalPages

基础库2.1.0开始支持,低版本需做兼容处理。启用插件功能页时,插件所有者小程序需要将functionalPages设置为true。

(7)subpackages

微信客户端6.6.0、基础库1.7.3及以上版本支持。启用分包加载时,声明项目分包结构。写成subPackages也支持。

(8)workers

使用Worker处理多线程任务时,设置Worker代码放置的目录。

(9)requiredBackgroundModes

微信客户端6.7.2及以上版本支持。声明需要后台运行的能力,类型为数组。目前支持以下项目:Audio后台音乐播放,代码示例如下:

        {
          "pages": ["pages/index/index"],
          "requiredBackgroundModes": ["audio"]
        }

注意

此处声明了后台运行的接口,开发版和体验版上可以直接生效,正式版还需通过审核。

(10)plugins

基础库1.9.6开始支持,低版本需做兼容处理。声明小程序需要使用的插件。

(11)preloadRule

基础库2.3.0开始支持,低版本需做兼容处理。声明分包预下载的规则。

(12)resizable

基础库2.3.0开始支持,低版本需做兼容处理。在iPad上运行的小程序可以设置支持屏幕旋转。

(13)navigateToMiniProgramAppIdList

基础库2.4.0开始支持,低版本需做兼容处理。当小程序需要使用wx.navigateToMini-Program接口跳转到其他小程序时,需要先在配置文件中声明需要跳转的小程序的AppID列表,最多允许填写10个。

2.2.2 页面配置

每一个小程序的页面可以使用.json文件对本页面的窗口表现进行配置。页面配置只能设置app.json中部分window配置项的内容,页面中配置项会覆盖app.json的window中相同的配置项,页面配置属性参见表2-6。

表2-6 页面配置属性

配置样例代码如下:

        {
          "navigationBarBackgroundColor": "#ffffff",
          "navigationBarTextStyle": "black",
          "navigationBarTitleText": "微信接口功能演示",
          "backgroundColor": "#eeeeee",
          "backgroundTextStyle": "light"
        }

提示

页面的.json只能设置window相关的配置项,以决定本页面的窗口表现,所以无须写window这个键。

2.3 WXSS样式语言

WXSS(WeiXin Style Sheets)是一套样式语言,用于描述WXML的组件样式。WXSS用来决定WXML的组件应该怎么显示。为了适应广大的前端开发者,WXSS具有CSS大部分特性。同时,为了更适合开发微信小程序,WXSS对CSS进行了修改和扩充。

与CSS相比,在微信小程序WXSS扩展的特性有:

❏ 尺寸单位。

❏ 样式导入。

❏ 全局样式和局部样式。

内联样式和选择器沿用了CSS的功能写法。

1.尺寸单位

尺寸单位为rpx(responsive pixel),可以根据屏幕宽度进行自适应。规定屏幕宽为750rpx。如在iPhone6上,屏幕宽度为375px,共有750个物理像素,则750rpx=375px=750物理像素,1rpx=0.5px=1物理像素,不同设备的换算方式如下:

在开发微信小程序时,建议设计师使用iPhone6作为视觉稿的标准。

注意

在较小的屏幕上不可避免会有一些毛刺,在开发时尽量避免这种情况。

2.样式导入

可以使用@import语句导入外联样式表,@import后跟需要导入的外联样式表的相对路径,用分号“; ”表示语句结束。代码示例如下:

        /** common.wxss **/
        .small-p {
          padding:5px;
        }

        /** app.wxss **/
        @import "common.wxss";
        .middle-p {
          padding:15px;
        }

3.全局样式与局部样式

定义在app.wxss中的样式为全局样式,作用于每一个页面。在page的.wxss文件中定义的样式为局部样式,只作用于对应的页面,并会覆盖app.wxss中相同的选择器。

4.内联样式

框架组件上支持使用style和class属性来控制组件的样式,说明如下。

❏ style:静态的样式统一写到class中,style接收动态的样式,在运行时会进行解析,尽量不要将静态的样式写进style,以免影响渲染速度,代码示例如下:

          <view style="color:{{color}}; " />

❏ class:用于指定样式规则,其属性值是样式规则中类选择器名(样式类名)的集合,样式类名不需要带上.,样式类名之间用空格分隔,代码示例如下:

          <view class="normal_view" />

5.选择器

目前支持的选择器有:

还有很多支持的选择器,不在这里一一列出,读者可以自行尝试,我们在后面第3章会讲解常用选择器在微信小程序中的使用。

2.4 逻辑层.js脚本

小程序开发框架的逻辑层使用JavaScript引擎,向小程序开发者提供JavaScript代码的运行环境以及微信小程序的特有功能。逻辑层对数据进行处理并发送给视图层,同时接受视图层的事件反馈。开发者写的所有代码最终将打包成一份JavaScript文件,并在小程序启动的时候运行,直到小程序销毁。这一行为类似于ServiceWorker,所以逻辑层也称为App Service。

在JavaScript的基础上,为了方便小程序开发增加了以下功能:

❏ 增加App和Page方法,进行程序和页面的注册。

❏ 增加getApp和getCurrentPages方法,用来获取App实例和当前页面栈。

❏ 提供丰富的API,如微信用户数据、扫一扫、支付等微信特有能力。

❏ 每个页面有独立的作用域,并提供模块化能力。

注意

小程序框架的逻辑层并非运行在浏览器中,因此JavaScript在Web中的一些能力无法使用,如window、document等。

2.4.1 App方法

小程序的App方法包含一系列函数,例如:App(Object)、onLaunch(Object)、on-Show(Object)、onHide()、onError(String error)、onPageNotFound(Object)、getApp(Object)等。

1. App(Object)

App()函数用来注册一个小程序。接受一个Object参数,指定小程序的生命周期回调等。App()必须在app.js中调用且只能调用一次,否则会出现无法预期的后果。Object参数说明参见表2-7。

表2-7 App()函数的Object参数表

前台、后台的含义是,当用户点击左上角关闭,或者按了设备Home键离开微信,小程序并没有直接销毁,而是进入了后台;当再次进入微信或再次打开小程序,小程序又会从后台进入前台。需要注意的是,只有当小程序进入后台一定时间,或者系统资源占用过高,小程序才会被真正销毁。

关闭小程序(基础库版本1.1.0开始支持)是指,当用户从扫一扫、转发等入口(场景值为1007, 1008, 1011, 1025)进入小程序,且没有置顶小程序的情况下退出,小程序会被销毁。

小程序运行机制在基础库版本1.4.0有所改变:上面“关闭小程序”逻辑在新版本已不适用。

代码示例如下:

        App({
          onLaunch: function(options) {
            // Do something initial when launch.
          },
          onShow: function(options) {
            // Do something when show.
          },
          onHide: function() {
            // Do something when hide.
          },
          onError: function(msg) {
            console.log(msg)
          },
          globalData: 'I am global data'
        })

2. onLaunch(Object)

onLaunch()函数在小程序初始化完成时触发,全局只触发一次,其中,Object参数说明参见表2-8。

表2-8 onLaunch()函数的Object参数表

其中,referrerInfo.appId场景值参见表2-9。

表2-9 referrerInfo.appId场景值

3. onShow(Object)

小程序启动时或从后台进入前台显示时触发onShow()函数。Object参数说明与on-Launch()函数一致。

4. onHide()

小程序从前台进入后台时触发onHide()函数。

5. onError(String error)

小程序发生脚本错误或者API调用失败时触发onError()函数。参数说明如下:

6. onPageNotFound(Object)

基础库1.9.90开始支持,低版本需做兼容处理。小程序要打开的页面不存在时触发onPageNotFound()函数。Object参数说明如下:

开发者可以在onPageNotFound回调中进行重定向,但必须在回调中同步处理,异步处理(例如setTimeout异步执行)无效。代码示例如下:

        App({
          onPageNotFound(res) {
            wx.redirectTo({
              url: 'pages/...'
            })     //如果是tabBar页面,请使用wx.switchTab
          }
        })

注意

● 如果开发者没有添加onPageNotFound监听,当跳转页面不存在时,将推入“微信客户端原生的页面不存在”提示页面。

● 如果onPageNotFound回调中又重定向到另一个不存在的页面,将推入“微信客户端原生的页面不存在”提示页面,并且不再回调onPageNotFound。

7. getApp(Object)

getApp()函数是全局函数,可以用来获取小程序App实例。Object参数说明如下:

代码示例如下:

        // other.js
        var appInstance = getApp()
        console.log(appInstance.globalData)      // I am global data

注意

❏ 不要在定义于App()内的函数中调用getApp(),使用this就可以获取App实例。

❏ 通过getApp()获取实例之后,不要私自调用生命周期函数。

2.4.2 运行机制

小程序启动会有两种情况,一种是“冷启动”,一种是“热启动”。假如用户已经打开过某小程序,然后在一定时间内再次打开该小程序,此时无须重新启动,只需将后台状态的小程序切换到前台,这个过程就是“热启动”。“冷启动”指的是用户首次打开小程序,或小程序被微信主动销毁后再次打开,此时小程序需要重新加载启动。

小程序冷启动时如果发现有新版本,将会异步下载新版本的代码包,并同时用客户端本地的包进行启动,即新版本的小程序需要等下一次冷启动才会应用。如果需要马上应用最新版本,可以使用wx.getUpdateManager API进行处理。

小程序没有重启的概念。当小程序进入后台,客户端会维持一段时间的运行状态,超过一定时间后(目前是5分钟)会被微信主动销毁。当短时间内(5分钟)连续收到两次以上系统内存告警,会对小程序进行销毁。

再次打开逻辑:基础库1.4.0开始支持,低版本需做兼容处理。

打开小程序有A和B两类场景。

A场景,打开首页。场景值有以下几项:

B场景,打开小程序指定的某个页面。场景值为除上面场景以外的其他内容,再次打开一个小程序的逻辑如下:

2.4.3 场景值

当前支持的场景值参数见表2-10。基础库1.1.0开始支持,低版本需做兼容处理。

表2-10 场景值参数

(续)

(续)

可以在App的onLaunch和onShow中获取上述场景值,部分场景值下还可以获取来源应用、公众号或小程序的AppID。

提示

由于Android系统限制,目前还无法获取到按Home键退出到桌面,然后从桌面再次进小程序的场景值,对于这种情况,会保留上一次的场景值。

2.4.4 Page方法

小程序的Page方法用于页面的注册和配置。

1.页面Page()函数

Page()函数用来注册一个页面。接受一个Object类型参数,指定页面的初始数据、生命周期回调、事件处理函数等。Object参数说明参见表2-11。

表2-11 Page()函数的Object参数

代码示例如下:

    //index.js
    Page({
      data: {
        text: "This is page data."
      },
      onLoad: function(options) {
        // Do some initialize when page load.
      },
      onReady: function() {
        // Do something when page ready.
      },
      onShow: function() {
        // Do something when page show.
      },
      onHide: function() {
        // Do something when page hide.
      },
      onUnload: function() {
        // Do something when page close.
      },
      onPullDownRefresh: function() {
        // Do something when pull down.
      },
      onReachBottom: function() {
        // Do something when page reach bottom.
      },
      onShareAppMessage: function () {
        // return custom share data when user share.
      },
      onPageScroll: function() {
      // Do something when page scroll
      },
      onResize: function() {
      // Do something when page resize
      },
      onTabItemTap(item) {
            console.log(item.index)
            console.log(item.pagePath)
            console.log(item.text)
          },
          // Event handler.
          viewTap: function() {
            this.setData({
              text: 'Set some data for updating view.'
            }, function() {
              // this is setData callback
            })
          },
          customData: {
            hi: 'MINA'
          }
        })

页面可以像自定义组件一样使用Component来创建,这样就可以使用自定义组件的特性。

2.初始数据data

data是页面第一次渲染使用的初始数据。页面加载时,data将会以JSON字符串的形式由逻辑层传至渲染层,因此data中的数据必须可以转成JSON的类型,如字符串、数字、布尔值、对象、数组等。

渲染层可以通过WXML对数据进行绑定。代码示例如下:

        <view>{{text}}</view>
        <view>{{array[0].msg}}</view>
        Page({
          data: {
            text: 'init data',
            array: [{msg: '1'}, {msg: '2'}]
          }
        })

3.生命周期回调函数

注册页面时的生命周期回调函数包括onLoad()、onShow()、onReady()、onHide()、on-Unload()。

onLoad(Object query)页面加载时触发。一个页面只会调用一次,可以在onLoad的参数中获取打开当前页面路径中的参数。

参数说明如下:

onShow()页面显示/切入前台时触发。

onReady()页面初次渲染完成时触发。一个页面只会调用一次,该函数执行完毕代表页面已经准备妥当,可以和视图层进行交互。

注意

对界面内容进行设置的API如wx.setNavigationBarTitle,请在onReady之后进行。

onHide()页面隐藏/切入后台时触发,如navigateTo或底部tab切换到其他页面,小程序切入后台等。

onUnload()页面卸载时触发,如redirectTo或navigateBack到其他页面时。

4.页面事件处理函数

(1)onPullDownRefresh()

监听用户下拉刷新事件。需要在app.json的window选项中或页面配置中开启enable-PullDownRefresh。可以通过wx.startPullDownRefresh触发下拉刷新,调用后触发下拉刷新动画,效果与用户手动下拉刷新一致。当处理完数据刷新后,wx.stopPullDownRefresh可以停止当前页面的下拉刷新。

(2)onReachBottom()

监听用户上拉触底事件。可以在app.json的window选项中或页面配置中设置触发距离onReachBottomDistance。在触发距离内滑动,本事件只会被触发一次。

(3)onPageScroll(Object)

监听用户滑动页面事件。Object参数如下:

(4)onShareAppMessage(Object)

监听用户点击页面内转发按钮(<button>组件open-type="share")或右上角菜单“转发”按钮的行为,并自定义转发内容。

注意

只有定义了此事件处理函数,右上角菜单才会显示“转发”按钮。

Object参数如下:

此事件需要返回一个Object,用于自定义转发内容,返回内容如下:

图片路径可以是本地文件路径、代码包文件路径或者网络图片路径。支持PNG及JPG。显示图片长宽比是5∶4。

代码示例如下:

        Page({
          onShareAppMessage: function (res) {
            if (res.from === 'button') {
              //来自页面内转发按钮
              console.log(res.target)
            }
            return {
              title: ’自定义转发标题’,
              path: '/page/user? id=123'
            }
          }
        })

(5)onTabItemTap(Object)

基础库1.9.0开始支持,低版本需做兼容处理。点击tab时触发,Object参数如下:

代码示例如下:

        Page({
          onTabItemTap(item) {
            console.log(item.index)
            console.log(item.pagePath)
            console.log(item.text)
          }
        })

5.组件事件处理函数

Page中还可以定义组件事件处理函数。在渲染层的组件中加入事件绑定,当事件被触发时,执行Page中定义的事件处理函数。

代码示例如下:

        <view bindtap="viewTap"> click me </view>

        Page({
          viewTap: function() {
            console.log('view tap')
          }
        })

6. route

Page.route表示到当前页面的路径,类型为String,基础库1.2.0开始支持,低版本需做兼容处理。代码示例如下:

        Page({
          onShow: function() {
            console.log(this.route)
          }
        })

7. setData

Page.prototype.setData(Object data, Function callback)函数用于将数据从逻辑层发送到视图层(异步),同时改变对应的this.data的值(同步)。参数说明如下:

Object以key: value的形式表示,将this.data中key对应的值改变成value。

其中key可以用数据路径的形式给出,支持改变数组中的某一项或对象的某个属性,如array[2].message, a.b.c.d,并且不需要在this.data中预先定义。

注意

● 直接修改this.data而不调用this.setData是无法改变页面的状态的,并会造成数据不一致。

● 仅支持设置可JSON化的数据。单次设置的数据不能超过1024kb,请尽量避免一次设置过多的数据。请不要把data中任何一项的value设为undefined,否则这一项将不被设置并可能遗留一些潜在问题。

代码示例如下:

    <! --index.wxml-->
    <view>{{text}}</view>
    <button bindtap="changeText"> Change normal data </button>
    <view>{{num}}</view>
    <button bindtap="changeNum"> Change normal num </button>
    <view>{{array[0].text}}</view>
    <button bindtap="changeItemInArray"> Change Array data </button>
    <view>{{object.text}}</view>
    <button bindtap="changeItemInObject"> Change Object data </button>
    <view>{{newField.text}}</view>
    <button bindtap="addNewField"> Add new data </button>

    // index.js
    Page({
      data: {
        text: 'init data',
        num: 0,
        array: [{text: 'init data'}],
        object: {
          text: 'init data'
        }
      },
      changeText: function() {
        // this.data.text = 'changed data' //不要直接修改this.data
        //应该使用setData
        this.setData({
          text: 'changed data'
        })
      },
      changeNum: function() {
        //或者,可以修改this.data之后马上用setData设置一下修改了的字段
        this.data.num = 1
        this.setData({
          num: this.data.num
        })
      },
      changeItemInArray: function() {
        //对于对象或数组字段,可以直接修改一个其下的子字段,这样做通常比修改整个对象或数组更好
        this.setData({
          'array[0].text':'changed data'
        })
      },
      changeItemInObject: function(){
        this.setData({
          'object.text': 'changed data'
        });
      },
      addNewField: function() {
        this.setData({
          'newField.text': 'new data'
        })
      }
    })

8.生命周期

理解生命周期的含义,将会帮助你理解开发。图2-3为Page实例的生命周期。

图2-3 Page实例的生命周期

2.4.5 路由

在小程序中,所有页面的路由全部由框架进行管理。框架以栈的形式维护了当前的所有页面。当发生路由切换的时候,页面栈的表现如表2-12所示。

表2-12 当路由切换时页面栈的表现

getCurrentPages()函数用于获取当前页面栈的实例,以数组形式按栈的顺序给出,第一个元素为首页,最后一个元素为当前页面。

注意

● 不要尝试修改页面栈,会导致路由以及页面状态错误。

● 不要在App.onLaunch的时候调用getCurrentPages(),此时page还没有生成。

路由的触发方式以及页面生命周期函数参见表2-13。

表2-13 路由的触发方式以及页面生命周期函数

例如,A、B页面为TabBar页面,C是从A页面打开的页面,D页面是从C页面打开的页面,Tab切换对应的生命周期如下:

提示

● navigateTo或redirectTo只能打开非tabBar页面。

● switchTab只能打开tabBar页面。

● reLaunch可以打开任意页面。

● 页面底部的tabBar由页面决定,即只要是定义为tabBar的页面,底部都有tabBar。

● 调用页面路由所带的参数可以在目标页面的onLoad中获取。

2.4.6 模块化

在JavaScript文件中声明的变量和函数只在该文件中有效;不同的文件中可以声明相同名字的变量和函数,不会互相影响。通过全局函数getApp()可以获取全局的应用实例,如果需要全局的数据可以在App()中设置,代码示例如下:

        // app.js
        App({
          globalData: 1
        })

        // a.js
        // The localValue can only be used in file a.js.
        var localValue = 'a'
        // Get the app instance.
        var app = getApp()
        // Get the global data and change it.
        app.globalData++

        // b.js
        // You can redefine localValue in file b.js, without interference with the local-
            Value in a.js.
        var localValue = 'b'
        // If a.js it run before b.js, now the globalData shoule be 2.
        console.log(getApp().globalData)

可以将一些公共的代码抽离成一个单独的.js文件,作为一个模块。模块只有通过module.exports或exports才能对外暴露接口。

注意

exports是module.exports的一个引用,在模块里面随意更改exports的指向会造成未知的错误。所以推荐开发者采用module.exports来暴露模块接口,除非你已经清晰知道这两者的关系。小程序目前不支持直接引入node_modules,开发者需要使用到node_modules时建议拷贝相关的代码到小程序的目录中,或者使用小程序支持的npm功能。

代码示例如下:

        // common.js
        function sayHello(name) {
          console.log(`Hello ${name} ! `)
        }
        function sayGoodbye(name) {
          console.log(`Goodbye ${name} ! `)
        }
        module.exports.sayHello = sayHello
        exports.sayGoodbye = sayGoodbye

在需要使用这些模块的文件中,使用require(path)将公共代码引入,代码示例如下:

        var common = require('common.js')
        Page({
          helloMINA: function() {
            common.sayHello('MINA')
          },
          goodbyeMINA: function() {
            common.sayGoodbye('MINA')
          }
        })

提示

require暂时不支持绝对路径。

2.4.7 API

小程序开发框架提供丰富的微信原生API,可以方便地调用微信提供的能力,如获取用户信息、本地存储、支付功能等。通常,小程序API有以下几种类型:事件监听API,同步API,异步API。

1.事件监听API

以on开头的API为事件监听API,用来监听某个事件是否触发,如wx.onSocketOpen、wx.onCompassChange等。这类API接受一个回调函数作为参数,当事件触发时会调用这个回调函数,并将相关数据以参数形式传入。代码示例如下:

        wx.onCompassChange(function (res) {
          console.log(res.direction)
        })

2.同步API

以Sync结尾的API都是同步API,如wx.setStorageSync、wx.getSystemInfoSync等。此外,也有一些其他的同步API,如wx.createWorker、wx.getBackgroundAudioManager等。同步API的执行结果可以通过函数返回值直接获取,如果执行出错会抛出异常。代码示例如下:

        try {
          wx.setStorageSync('key', 'value')
        } catch (e) {
          console.error(e)
        }

3.异步API

大多数API都是异步的,如wx.request、wx.login等。这类API接口通常都接受一个Object类型的参数,这个参数用于指定如何接收接口调用结果。

异步API的Object参数说明如下:

API通常的回调函数有三个:success、fail和complete。回调函数调用时会传入一个Object类型参数,包含以下字段:

异步API的执行结果需要通过Object类型的参数中传入的对应回调函数获取。部分异步API也会有返回值,可以用来实现更丰富的功能,如wx.request、wx.connectSockets等。代码示例如下:

        wx.login({
          success(res) {
            console.log(res.code)
          }
        })

2.5 WXML视图层开发

WXML(WeiXin Markup Language)是框架设计的类似于HTML的标签语言,结合基础组件、事件系统就可以构建出页面的结构,组成.wxml文件。WXML中的动态数据均来自对应Page的data。本节主要讲解视图层开发中.wxml文件常用的语法,包含数据绑定、列表渲染、条件渲染、模板、事件、引用等处理。

2.5.1 数据绑定

数据绑定是指在小程序的.js文件里,将data定义的各类数据显示在.wxml页面中。当然data里面定义的各类数据可以通过其他方式进行变更。

1.简单绑定

数据简单绑定是指使用Mustache语法(双大括号)将变量包起来。

.wxml文件代码示例如下:

    <view> {{ message }} </view>

.js文件代码示例如下:

    Page({
      data: {
        message: 'Hello MINA! '
      }
    })

组件属性需要在双引号之内,.wxml文件代码示例如下:

    <view id="item-{{id}}"> </view>

.js文件代码示例如下:

    Page({
      data: {
        id: 0
      }
    })

控制属性需要在双引号之内,.wxml文件代码示例如下:

    <view wx:if="{{condition}}"> </view>

.js文件代码示例如下:

    Page({
      data: {
        condition: true
      }
    })

关键字(需要在双引号之内)包括。

❏ true:boolean类型的true,代表真值。

❏ false:boolean类型的false,代表假值。

.wxml文件代码示例如下:

    <checkbox checked="{{false}}"> </checkbox>

注意

不要直接写checked="false",其计算结果是一个字符串,转成boolean类型后代表真值。

2.运算

可以在{{}}内进行简单的运算,支持如下几种运算:

三元运算,.wxml文件代码示例如下:

    <view hidden="{{flag ? true : false}}"> Hidden </view>

算数运算,.wxml文件代码示例如下:

    <view> {{a + b}} + {{c}} + d </view>

.js文件代码示例如下:

    Page({
      data: {
        a: 1,
        b: 2,
        c: 3
      }
    })

view中的内容为3+3+d。

逻辑判断,.wxml文件代码示例如下:

    <view wx:if="{{length > 5}}"> </view>

字符串运算,.wxml文件代码示例如下:

    <view>{{"hello" + name}}</view>

.js文件代码示例如下:

    Page({
      data:{
        name: 'MINA'
      }
    })

数据路径运算,.wxml文件代码示例如下:

    <view>{{object.key}} {{array[0]}}</view>

.js文件代码示例如下:

    Page({
      data: {
        object: {
          key: 'Hello '
        },
        array: ['MINA']
      }
    })

3.组合

也可以在Mustache内直接进行组合,构成新的数组或者对象。

(1)数组

.wxml文件代码示例如下:

    <view wx:for="{{[zero, 1, 2, 3, 4]}}"> {{item}} </view>

.js文件代码示例如下:

    Page({
      data: {
        zero: 0
      }
    })

最终组合成数组[0, 1, 2, 3, 4]。

(2)对象

.wxml文件代码示例如下:

    <template is="objectCombine" data="{{for: a, bar: b}}"></template>

.js文件代码示例如下:

    Page({
      data: {
        a: 1,
        b: 2
      }
    })

最终组合成对象{for: 1, bar: 2}。

也可以用扩展运算符...将一个对象展开。.wxml文件代码示例如下:

    <template is="objectCombine" data="{{...obj1, ...obj2, e: 5}}"></template>

.js文件代码示例如下:

    Page({
      data: {
        obj1: {
          a: 1,
          b: 2
        },
        obj2: {
          c: 3,
          d: 4
        }
      }
    })

最终组合成对象{a: 1, b: 2, c: 3, d: 4, e: 5}。

如果对象的key和value相同,也可以间接地表达。.wxml文件代码示例如下:

    <template is="objectCombine" data="{{foo, bar}}"></template>

.js文件代码示例如下:

        Page({
          data: {
            foo: 'my-foo',
            bar: 'my-bar'
          }
        })

最终组合成对象{foo: 'my-foo', bar:'my-bar'}。

注意

上述方式可以随意组合,但如存在变量名相同的情况,后边的会覆盖前面的,如:

          <template is="objectCombine" data="{{...obj1, ...obj2, a, c: 6}}"></template>
          Page({
            data: {
              obj1: {
                a: 1,
                b: 2
              },
              obj2: {
                b: 3,
                c: 4
              },
              a: 5
            }
          })

最终组合成的对象是{a: 5, b: 3, c: 6}。花括号和引号之间如果有空格,将最终被解析成字符串。

.wxml文件代码示例如下:

        <view wx:for="{{[1,2,3]}} ">
          {{item}}
        </view>

等同于代码:

        <view wx:for="{{[1,2,3] + ' '}}">
          {{item}}
        </view>

2.5.2 列表渲染

列表渲染是指,在小程序.js文件里,把data定义的数组数据通过for循环语句显示在.wxml页面中。当然data里面定义的各类数据可以通过其他方式进行变更。在实际的应用中,通过列表渲染可以输出产品列表、新闻列表等。

1. wx:for

在组件上使用wx:for控制属性绑定一个数组,即可使用数组中各项的数据重复渲染该组件。

数组的当前项的下标变量名默认为index,数组当前项的变量名默认为item, .wxml文件代码示例如下:

        <view wx:for="{{array}}">
          {{index}}: {{item.message}}
        </view>

.js文件代码示例如下:

        Page({
          data: {
            array: [{
              message: 'foo',
            }, {
              message: 'bar'
            }]
          }
        })

使用wx:for-item可以指定数组当前元素的变量名,使用wx:for-index可以指定数组当前下标的变量名,.wxml文件代码示例如下:

        <view wx:for="{{array}}" wx:for-index="idx" wx:for-item="itemName">
          {{idx}}: {{itemName.message}}
        </view>

wx:for也可以嵌套,下面是一个九九乘法表的.wxml文件代码示例:

        <view wx:for="{{[1, 2, 3, 4, 5, 6, 7, 8, 9]}}" wx:for-item="i">
          <view wx:for="{{[1, 2, 3, 4, 5, 6, 7, 8, 9]}}" wx:for-item="j">
            <view wx:if="{{i <= j}}">
              {{i}} * {{j}} = {{i * j}}
            </view>
          </view>
        </view>

2. block wx:for

可以将wx:for用在<block/>标签上,以渲染一个包含多节点的结构块。.wxml文件代码示例如下:

        <block wx:for="{{[1, 2, 3]}}">
          <view> {{index}}: </view>
          <view> {{item}} </view>
        </block>

注意

<block/>并不是一个组件,它仅仅是一个包装元素,不会在页面中做任何渲染,只接受控制属性。

3. wx:key

如果列表中项目的位置会动态改变或者有新的项目添加到列表中,并且希望列表中的项目保持自己的特征和状态(如<input/>中的输入内容,<switch/>的选中状态),需要使用wx:key来指定列表中项的唯一标识符。

wx:key的值有以下两种形式。

❏ 字符串:代表在for循环的array中item的某个property,该property的值需要是列表中唯一的字符串或数字,且不能动态改变。

❏ 保留关键字(*this):代表在for循环中的item本身,这种表示需要item本身是一个唯一的字符串或者数字,例如,当数据改变触发渲染层重新渲染的时候,会校正带有key的组件,框架会将它们重新排序,而不是重新创建,以确保组件保持自身的状态,并且提高列表渲染时的效率。如不提供wx:key,会报错;如果明确知道该列表是静态的,或者不必关注其顺序,可以选择忽略。

.wxml文件代码示例如下:

        <switch wx:for="{{objectArray}}" wx:key="unique" style="display: block; "> {{item.
            id}} </switch>
        <button bindtap="switch"> Switch </button>
        <button bindtap="addToFront"> Add to the front </button>
        <switch wx:for="{{numberArray}}" wx:key="*this" style="display: block; "> {{item}}
            </switch>
        <button bindtap="addNumberToFront"> Add to the front </button>

.js文件代码示例如下:

        Page({
          data: {
            objectArray: [
              {id: 5, unique: 'unique_5'},
              {id: 4, unique: 'unique_4'},
              {id: 3, unique: 'unique_3'},
              {id: 2, unique: 'unique_2'},
              {id: 1, unique: 'unique_1'},
              {id: 0, unique: 'unique_0'},
            ],
            numberArray: [1, 2, 3, 4]
          },
          switch: function(e) {
            const length = this.data.objectArray.length
            for (let i = 0; i < length; ++i) {
              const x = Math.floor(Math.random() * length)
              const y = Math.floor(Math.random() * length)
              const temp = this.data.objectArray[x]
              this.data.objectArray[x] = this.data.objectArray[y]
              this.data.objectArray[y] = temp
            }
            this.setData({
              objectArray: this.data.objectArray
            })
          },
          addToFront: function(e) {
            const length = this.data.objectArray.length
            this.data.objectArray  =  [{id:  length,  unique:  'unique_'  +  length}].concat
                (this.data.objectArray)
            this.setData({
              objectArray: this.data.objectArray
            })
          },
          addNumberToFront: function(e){
            this.data.numberArray  =  [  this.data.numberArray.length  +  1  ].concat(this.
                data.numberArray)
            this.setData({
              numberArray: this.data.numberArray
            })
          }
        })

注意,当wx:for的值为字符串时,会将字符串解析成字符串数组,.wxml文件代码示例如下:

        <view wx:for="array">
          {{item}}
        </view>

等同于:

        <view wx:for="{{['a', 'r', 'r', 'a', 'y']}}">
          {{item}}
        </view>

花括号和引号之间如果有空格,将最终被解析成字符串,.wxml文件代码示例如下:

        <view wx:for="{{[1,2,3]}} ">
          {{item}}
        </view>

等同于:

        <view wx:for="{{[1,2,3] + ' '}}" >
          {{item}}
        </view>

2.5.3 条件渲染

条件渲染是指,根据if语句中的条件来决定if语句所在区块是显示还是隐藏,如果if条件语句为True,则渲染显示;如果if条件语句为False,则隐藏不做渲染显示。在实际开发中,可以通过按钮来改变if条件语句的变量状态(True和False之间转换)来实现某个区块的显示或隐藏。

1. wx:if

在框架中,使用wx:if="{{condition}}"来判断是否需要渲染该代码块,.wxml文件代码示例如下:

        <view wx:if="{{condition}}"> True </view>

也可以用wx:elif和wx:else来添加一个else块,.wxml文件代码示例如下:

        <view wx:if="{{length > 5}}"> 1 </view>
        <view wx:elif="{{length > 2}}"> 2 </view>
        <view wx:else> 3 </view>

2. block wx:if

因为wx:if是一个控制属性,需要将它添加到一个标签上。如果要一次性判断多个组件标签,可以使用一个<block/>标签将多个组件包装起来,并在上边使用wx:if控制属性,.wxml文件代码示例如下:

        <block wx:if="{{true}}">
          <view> view1 </view>
          <view> view2 </view>
        </block>

3. wx:if vs hidden

因为wx:if之中的模板也可能包含数据绑定,所以当wx:if的条件值切换时,框架有一个局部渲染的过程,以确保条件块在切换时销毁或重新渲染。

同时,wx:if也是惰性的,如果初始渲染条件为False,框架什么也不做,在条件第一次变成真的时候才开始局部渲染。相比之下,hidden就简单多了,组件始终会被渲染,只是简单地控制显示与隐藏。一般来说,wx:if有更高的切换消耗,而hidden有更高的初始渲染消耗。因此,在需要频繁切换的情景下,用hidden更好;而在运行时条件不大可能改变时,则用wx:if较好。

2.5.4 模板

模板用于将一些公用的WXML代码单独整理成一个.wxml文件,然后在有需要的地方直接引入即可。可以在模板中定义代码片段,然后在不同的地方调用。

1.定义模板

使用name属性作为模板的名字,然后在<template/>内定义代码片段。.wxml文件代码示例如下:

        <! --
          index: int
          msg: string
          time: string
        -->
        <template name="msgItem">
          <view>
            <text> {{index}}: {{msg}} </text>
            <text> Time: {{time}} </text>
          </view>
        </template>

2.使用模板

使用is属性声明需要使用的模板,然后将模板所需要的data传入。.wxml文件代码示例如下:

        <template is="msgItem" data="{{...item}}"/>

.js文件代码示例如下:

        Page({
          data: {
          item: {
            index: 0,
            msg: 'this is a template',
            time: '2016-09-15'
            }
          }
        })

is属性可以使用Mustache语法,动态决定具体需要渲染哪个模板。.wxml文件代码示例如下:

        <template name="odd">
          <view> odd </view>
        </template>
          <template name="even">
          <view> even </view>
        </template>
        <block wx:for="{{[1, 2, 3, 4, 5]}}">
          <template is="{{item % 2 == 0 ? 'even' : 'odd'}}"/>
        </block>

模板拥有自己的作用域,只能使用data传入的数据以及模板定义文件中定义的<wxs />模块。

2.5.5 事件

事件是控件可以识别的操作,如按下某个按钮,选择某个复选框。每一种控件有自己可以识别的事件,如小程序页面的加载、按钮的单击、表单的提交等事件,也包括编辑框(文本框)的文本改变事件等。

事件有如下特征:

❏ 事件是视图层到逻辑层的通信方式。

❏ 事件可以将用户的行为反馈到逻辑层进行处理。

❏ 事件可以绑定在组件上,当达到触发事件,就会执行逻辑层中对应的事件处理函数。

❏ 事件对象可以携带额外信息,如id、dataset、touches。

1.事件的使用方式

在组件中绑定一个事件处理函数,如bindtap,当用户点击该组件的时候,会在该页面对应的Page中找到相应的事件处理函数,.wxml文件代码示例如下:

        <view id="tapTest" data-hi="WeChat" bindtap="tapName"> Click me! </view>

在相应的Page定义中写入相应的事件处理函数,参数是event, .js文件代码示例如下:

        Page({
          tapName: function(event) {
            console.log(event)
          }
        })

可以看到,log出来的信息大致如下:

        {
          "type":"tap",
          "timeStamp":895,
          "target": {
            "id": "tapTest",
            "dataset":  {
              "hi":"WeChat"
            }
          },
          "currentTarget":  {
            "id": "tapTest",
            "dataset": {
              "hi":"WeChat"
            }
          },
          "detail": {
            "x":53,
            "y":14
          },
        "touches":[{
          "identifier":0,
          "pageX":53,
          "pageY":14,
          "clientX":53,
          "clientY":14
        }],
        "changedTouches":[{
          "identifier":0,
          "pageX":53,
          "pageY":14,
          "clientX":53,
          "clientY":14
        }]
      }

2.事件分类

事件分为冒泡事件和非冒泡事件。

❏ 冒泡事件:当一个组件上的事件被触发后,该事件会向父节点传递。

❏ 非冒泡事件:当一个组件上的事件被触发后,该事件不会向父节点传递。

WXML的冒泡事件参见表2-14。

表2-14 WXML的冒泡事件列表

注意

除上表之外的其他组件自定义事件如无特殊声明都是非冒泡事件,如<form/>的submit事件,<input/>的input事件,<scroll-view/>的scroll事件。

3.事件绑定和冒泡

事件绑定的写法与组件的属性相关,分为key和value两种形式:

❏ key以bind或catch开头,后跟事件的类型,如bindtap、catchtouchstart。自基础库版本1.5.0起,在非原生组件中,bind和catch后可以紧跟一个冒号,其含义不变,如bind:tap、catch:touchstart。

❏ value是一个字符串,需要在对应的Page中定义同名的函数,否则当触发事件的时候会报错。

提示

bind事件绑定不会阻止冒泡事件向上冒泡,catch事件绑定可以阻止冒泡事件向上冒泡。

例如,在下面这个例子中,点击inner view会依次调用handleTap3和handleTap2(因为tap事件会冒泡到middle view,而middle view阻止了tap事件冒泡,不再向父节点传递),点击middle view会触发handleTap2,点击outer view会触发handleTap1, .wxml文件代码示例如下:

        <view id="outer" bindtap="handleTap1">
          outer view
          <view id="middle" catchtap="handleTap2">
            middle view
            <view id="inner" bindtap="handleTap3">
              inner view
            </view>
          </view>
        </view>

4.事件的捕获阶段

自基础库版本1.5.0起,触摸类事件支持捕获阶段。捕获阶段位于冒泡阶段之前,在捕获阶段中,事件到达节点的顺序与冒泡阶段恰好相反。需要在捕获阶段监听事件时,可以采用capture-bind、capture-catch关键字,后者将中断捕获阶段和取消冒泡阶段。

在下面的.wxml文件代码中,点击inner view会依次调用handleTap2、handleTap4、handleTap3、handleTap1:

        <view id="outer" bind:touchstart="handleTap1" capture-bind:touchstart="handleTap2">
          outer view
          <view id="inner" bind:touchstart="handleTap3" capture-bind:touchstart="handleTap4">
            inner view
          </view>
        </view>

如果将上面代码中的第一个capture-bind改为capture-catch,将只触发handleTap2。.wxml文件代码示例如下:

        <view id="outer" bind:touchstart="handleTap1" capture-catch:touchstart="handleTap2">
          outer view
          <view id="inner" bind:touchstart="handleTap3" capture-bind:touchstart="handleTap4">
            inner view
          </view>
        </view>

5.事件对象

如无特殊说明,当组件触发事件时,逻辑层绑定该事件的处理函数会收到一个事件对象。

BaseEvent基础事件对象属性如下:

其中,type代表事件的类型。

timeStamp页面打开到触发事件所经过的毫秒数。

target触发事件的源组件,属性如下:

currentTarget事件绑定的当前组件,属性如下:

说明:target和currentTarget可以参考上例,点击inner view时,handleTap3收到的事件对象target和currentTarget都是inner,而handleTap2收到的事件对象target就是inner, currentTarget就是middle。

CustomEvent自定义事件对象属性(继承BaseEvent)如下:

TouchEvent触摸事件对象属性(继承BaseEvent)如下:

特殊事件:<canvas/>中的触摸事件不可冒泡,所以没有currentTarget。

6. dataset

在组件中可以定义数据,这些数据将会通过事件传递给SERVICE。书写方式:以data-开头,多个单词由连字符-链接,不能有大写(大写会自动转成小写),如data-element-type,最终在event.currentTarget.dataset中会将连字符转成驼峰形式,如elementType。

.wxml文件代码示例如下:

        <view data-alpha-beta="1" data-alphaBeta="2" bindtap="bindViewTap"> DataSet Test
            </view>

.js文件代码示例如下:

        Page({
          bindViewTap:function(event){
            event.currentTarget.dataset.alphaBeta === 1 // -会转为驼峰写法
            event.currentTarget.dataset.alphabeta === 2 //大写会转为小写
          }
        })

7. touches

touches是一个数组,每个元素为一个Touch对象(canvas触摸事件中携带的touches是CanvasTouch数组)。表示当前停留在屏幕上的触摸点。

Touch对象属性如下:

8. changedTouches

changedTouches数据格式同touches,表示有变化的触摸点,如从无变有(touchstart),位置变化(touchmove),从有变无(touchend、touchcancel)。

9. detail

自定义事件所携带的数据,如表单组件的提交事件会携带用户的输入,媒体的错误事件会携带错误信息,详见组件定义中各个事件的定义。

点击事件的detail带有的x和y,同pageX和pageY,代表到文档左上角的距离。

2.5.6 引用

WXML提供两种文件引用方式:import和include。

1. import

import可以在该文件中使用目标文件定义的template,例如,在item.wxml中定义了一个名为item的template, .wxml文件代码示例如下:

        <! -- item.wxml -->
        <template name="item">
          <text>{{text}}</text>
        </template>

在index.wxml中引用了item.wxml,就可以使用item模板,.wxml文件代码示例如下:

        <import src="item.wxml"/>
        <template is="item" data="{{text: 'forbar'}}"/>

import有作用域的概念,即只输入目标文件中定义的模板,而不输入目标文件自己输入的模板。例如,C import B, B import A,在C中可以使用B定义的template,在B中可以使用A定义的template,但是C不能使用A定义的template。

.wxml文件代码示例如下:

        <! -- A.wxml -->
        <template name="A">
          <text> A template </text>
        </template>

.wxml文件代码示例如下:

        <! -- B.wxml -->
        <import src="a.wxml"/>
        <template name="B">
          <text> B template </text>
        </template>

.wxml文件代码示例如下:

        <! -- C.wxml -->
        <import src="b.wxml"/>
        <template is="A"/>  <! -- Error! Can not use tempalte when not import A. -->
        <template is="B"/>

2. include

include可以将目标文件除<template/> <wxs/>之外的整个代码引入,相当于拷贝到include位置,.wxml文件代码示例如下:

    <! -- index.wxml -->
    <include src="header.wxml"/>
    <view> body </view>
    <include src="footer.wxml"/>

.wxml文件代码示例如下:

    <! -- header.wxml -->
    <view> header </view>

    <! -- footer.wxml -->
    <view> footer </view>