Skip to content

2019-web/lab2-MiniProgram

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

28 Commits
 
 
 
 
 
 

Repository files navigation

高级Web技术 Lab 2:微信小程序

1. 实验环境的搭建

1.1 微信开发者工具

在微信小程序官方网站上下载自己电脑对应操作系统版本的微信小程序开发者工具,在本地的电脑上安装。

微信小程序的新建项目是需要 Appid 的,需要自己注册一个微信小程序的账号来获得 Appid。

如果自己没有注册过微信小程序的账号,可以用助教账号的"appid": "wx21776ebfd5bca378"来完成实验。

关于微信开发者工具的使用,请参考微信小程序官方的文档: 概览

1.2 轻量级后端

最终的 PJ 是前后端分离的项目,小程序作为前端。lab3 是用 SpringBoot 作为Restful后端,这部分还没学习。所以,lab2 用轻量级的服务器,推荐用 json-server ,具体使用参考 Github 主页 : https://github.com/typicode/json-server

在本地电脑上安装 NodeJS 环境,具体安装过程百度,如果遇到问题无法解决可以联系助教。

# 在终端通过命令安装json-server包,如果因为权限问题失败了,用sudo npm install -g json-server
npm install -g json-server
# git clone 并进入文件夹
git clone https://github.com/2019-web/mini-edu-lab-server-code.git
cd mini-edu-lab-server-code
# 启动一个以db.josn文件的服务器,端口是5300
json-server --watch --port 5300 db.json

在浏览器上可以访问 http://localhost:5300 看到的界面如下,说明后端搭建成功了。

提醒 : coursecontentschatbox类似于数据库表名,可以点击相应的链接。在 db.json内部存放的是数据库中的数据,最终的PJ数据库结构请自己设计,这里只为了演示方便。 这里引入 json-server 的目的,是帮助大家快速搭建起来微信小程序的前后端分离的项目demo,方便调试。微信小程序向后端发起请求,获得请求的数据,然后在微信小程序界面上显示。

2. 实验步骤

2.1 总体介绍

在微信开发者工具中,新建项目,进入的界面如下:

./assests/img/6.png

项目名称、目录、AppID(如果自己没有的话,可以用wx21776ebfd5bca378)可以按照自己的实际情况配置。

新建项目完成之后,进入的界面如下:

./assests/img/newProject.png

小程序的目录结构请参考微信小程序的官方文档:目录结构,后面的引用部分也是如此,不再赘述,要学会多去看微信小程序的官方文档。

2.1.1 注意

在网页上的微信小程序的管理后台,request url 是只允许设置域名的,也就是说如果你的小程序要对外发布,后端服务器是必须需要进行域名解析的。

考虑到在本地电脑上的实验阶段,不需要注册域名,在微信开发者工具中的“详细”勾选:

  • 不校验合法域名、web-view(业务域名)、TLS 版本以及 HTTPS 证书

这样才可以请求本地的 http://localhost:5300 域名,否则是请求错误的。

2.1.2 总体介绍

在此基础上,新建course课程目录、contents目录目录、chatbox聊天目录。

在这个三个文件夹内部新建对应名称的Page。

Page页面说明:

  • Course课程Page:课程列表

  • Contents目录Page:对应课程目录、章节信息、知识点等信息

  • Chatbox聊天Page:一个知识点,按照对话式教学的,对话框界面

流程说明: 点击Course课程Page上的若干课程列表,点击其中的一门课程,进入Contents目录Page,显示的是对应的课程的知识点或者章节信息,点击其中的知识点或者章节信息,进入其中的知识点的Chatbox聊天Page。

2.2 具体代码说明

2.2.1 index 首页

pages/index/index.wxml中,代码修改如下 :

<!--index.wxml-->
<view class="container">
  <view class="userinfo">
    <button wx:if="{{!hasUserInfo && canIUse}}" open-type="getUserInfo" bindgetuserinfo="getUserInfo"> 获取头像昵称 </button>
    <block wx:else>
      <image bindtap="bindViewTap" class="userinfo-avatar" src="{{userInfo.avatarUrl}}" mode="cover"></image>
      <text class="userinfo-nickname">{{userInfo.nickName}}</text>
    </block>
  </view>
  <view style="margin-top:20px">
    <navigator
    url="../course/course"
    hover-class="navigator-hover"
    >
    <button>进入课程</button>
    </navigator>
  </view>
</view>

其它的文件index.json 、index.js、index.wxss保持默认即可。

主要增加的代码如下 :

  <view style="margin-top:20px">
    <navigator
    url="../course/course"
    hover-class="navigator-hover">
    <button>进入课程</button>
    </navigator>
  </view>

<navigator></navigator>可以实现当前小程序内的跳转链接,点击<button></button>按钮,可以实现跳转到 Course 文件夹内的Course页面。微信小程序中,可以用 hover-class 属性来指定元素的点击态效果。

实际的界面如下 :

具体内容请参考微信小程序的官方文档: 小程序代码构成

2.2.2 course课程部分

/pages/course/course.wxml中,代码修改如下 :

<text>pages/course/course.wxml</text>

<view class="container">
  <view class="course">
    <navigator
    url="../contents/contents?content_id=network"
    hover-class="navigator-hover">
    <button>计算机网络</button>
    </navigator>
  </view>
  <view class="course">
    <navigator
    url="../contents/contents?content_id=datastructure"
    hover-class="navigator-hover">
    <button>数据结构</button>
    </navigator>
  </view>
</view>  

/pages/course/course.wxss中,代码修改如下:

/* pages/course/course.wxss */
.course {
  /* 保持上下两个按钮的间距 */
  margin-top: 20px; 
}

/pages/course/course.json中,代码修改如下:

{
  "usingComponents": {},
  "navigationBarTitleText": "课程页面"
}

navigationBarTitleText是进行导航栏标题文字内容的配置,具体参考微信小程序的官方文档: 页面配置

/pages/course/course.js中,暂时保持默认。

后来又引入了 flex 布局,来做两列的课程显示,具体还是参考相关的文档吧,不再赘述了。

请参考相关博客文章:Flex 布局教程:语法篇

lab是为了尽量简单,PJ是要多多注意细节的。

2.2.3 contents目录部分

/pages/contents/content.js中,代码修改如下 :

// pages/course/course.js
Page({

  /**
   * 页面的初始数据
   */
  data: {
    contents_name:[]
  },

  /**
   * 生命周期函数--监听页面加载
   */
  onLoad: function (options) {
    // 从options取得Course页面传过来的URL参数course_id,打印方便调试
    console.log(options.course_id);
    //把this对象复制到临时变量that
    var that = this;
    wx.request({
      url: 'http://localhost:5300/contents?content_id=' + options.content_id,
      headers: {
        'Content-Type': 'application/json'
      },
      success: function (res) {
        console.log(res);
        console.log(res.data[0].contents_name);
        that.setData({
          contents_name: res.data[0].contents_name
        })
      }
    })
  },

  /**
   * 生命周期函数--监听页面初次渲染完成
   */
  onReady: function () {

  },

  /**
   * 生命周期函数--监听页面显示
   */
  onShow: function () {

  },

  /**
   * 生命周期函数--监听页面隐藏
   */
  onHide: function () {

  },

  /**
   * 生命周期函数--监听页面卸载
   */
  onUnload: function () {

  },

  /**
   * 页面相关事件处理函数--监听用户下拉动作
   */
  onPullDownRefresh: function () {

  },

  /**
   * 页面上拉触底事件的处理函数
   */
  onReachBottom: function () {

  },

  /**
   * 用户点击右上角分享
   */
  onShareAppMessage: function () {

  }
})

/pages/contents/content.js中,注意这里的:

    wx.request({
      url: 'http://localhost:5300/contents?content_id=' + options.content_id,
      headers: {
        'Content-Type': 'application/json'
      },
      success: function (res) {
        // 对于url的HTTP GET请求返回的res JSON类型的数据打印,在微信开发者工具的Console终端可以查看输出信息
        console.log(res);
        // 从res数据中获得需要的课程名称数据
        that.setData({
          contents_name: res.data[0].contents_name
        })
      }
    })

按照url发起HTTP请求后端服务器,返回得到的数据res,在浏览器中的控制台的显示如下,关于调试部分,参见微信小程序的官方文档:console

/pages/contents/content.js中,注意这里的:

that.setData({
  contents_name: res.data[0].contents_name
})

setData 函数用于将数据从逻辑层发送到视图层,同时改变对应的 this.data 的值。

想要的这部分数据res.data[0].contents_name然后通过 setData 方法复制给 contents_name 变量,这个变量在 Page 的 Data 内部定义的。

/pages/content/content.wxml中,代码修改如下:

<!--pages/contents/contents.wxml-->
<text>pages/contents/contents.wxml</text>
<view class="container">
  <view class="contents">
    <view wx:for="{{contents_name}}">
    {{item.id}}-{{item.text}}-{{item.image_url}}-{{item.chatbox_id}}
    
    <view class="chatbox">
        <navigator
        url="../chatbox/chatbox?chatbox_id={{item.chatbox_id}}"
        hover-class="navigator-hover">
        <button>{{item.text}}</button>
        </navigator>
    </view>
    </view>
  </view>
</view>  

说明 : WXML 中的动态数据均来自对应 Page 的 data。在组件上使用 wx:for 控制属性绑定一个数组,即可使用数组中各项的数据重复渲染该组件。将contents_name数组中按照循环打印出来,item即为contents_name的默认变量名,默认数组的当前项的下标变量名默认为 index,数组当前项的变量名默认为 item,具体参考微信小程序官方的文档: 列表渲染

同时,在微信开发者工具中,在调试工具中有“AppData”,去查看具体的变量数据:

/pages/content/content.js中,与data内定义的变量是一一对应关系。微信小程序开发的类似于 MVVM 框架,只要修改 Page 的 data 内部定义的变量contents_name/pages/content/content.wxml{{contents_name}}的变量就能保持同步的更新。

Page({

  /**
   * 页面的初始数据
   */
  data: {
    contents_name:[]
  },

这部分是小程序开发的重点部分,具体参考微信小程序的官方文档: 数据绑定Page(Object object)

2.2.4 chatbox聊天部分

(1) chatbox聊天

比较关键的就是点击底部的“添加信息”按钮,然乎对话信息一条一条的显示出来,这里的做法比较简陋,直接将整个信息通过 setData 赋值给变量,可能以后要考虑到“单次设置的数据不能超过1024KB,尽量避免一次设置过多的数据”的问题。

pages/chatbox/chatbox.js中, 修改代码如下 :

// pages/chatbox/chatbox.js
Page({

  /**
   * 页面的初始数据
   */
  data: {
    message:[],      // message数组默认为空
    index: 0,        // 下标默认值为 0
    totalMessage:[]  //将一个知识点所有的message数据都存放到这个数组里
  },

  /**
   * 生命周期函数--监听页面加载
   */
  onLoad: function (options) {
    console.log(options.chatbox_id);
    var that = this;

    wx.request({
      url: 'http://localhost:5300/chatbox?chatbox_id=' + options.chatbox_id,
      headers: {
        'Content-Type': 'application/json'
      },
      success: function (res) {
        console.log(res);
        that.setData({
          totalMessage : res.data[0].message
        })
      }
    })    
  },

  /**
   * 生命周期函数--监听页面初次渲染完成
   */
  onReady: function () {

  },

  /**
   * 生命周期函数--监听页面显示
   */
  onShow: function () {
  },

  /**
   * 生命周期函数--监听页面隐藏
   */
  onHide: function () {

  },

  /**
   * 生命周期函数--监听页面卸载
   */
  onUnload: function () {

  },

  /**
   * 页面相关事件处理函数--监听用户下拉动作
   */
  onPullDownRefresh: function () {

  },

  /**
   * 页面上拉触底事件的处理函数
   */
  onReachBottom: function () {

  },

  /**
   * 用户点击右上角分享
   */
  onShareAppMessage: function () {

  },

  addMessage:function () {
    var that = this;
    // 将 Page 内部定义的 Data 中的 index 赋值给 i
    var i = this.data.index ;
    console.log(this.data.totalMessage.length);

    if(i>=this.data.totalMessage.length){
      // 跳出程序,这种实现比较简陋
      return;
    }

    console.log(this.data.totalMessage[i]);
    // 通过下标的移动,给message数组不断连接(concat)值的方式增加值
    var message = this.data.message.concat(this.data.totalMessage[i]);
    
    // 将函数内部的message变量通过setData赋值给Page中的Data内的变量message,页面的数据随之更新
    that.setData({  
      message: message
    })

    i++;
    // 将函数内部的i变量通过setData赋值给Page中的Data内的变量index,目的是保存下标的值
    that.setData({
      index: i
    })    
  }
})
(2) 界面坐标问题

pages/chatbox/chatbox.wxml中,修改代码如下 :

<!--pages/chatbox/chatbox.wxml-->
<text>pages/chatbox/chatbox.wxml</text>
<!-- padding-bottom:80px;限制文本内容区域距离底端大小,微信小程序中使用padding-bottom解决position:fixed遮挡正文内容的问题 -->
<view id="container" style="padding-bottom:80px;">
  <view>
    <view wx:for="{{message}}" class='messageHeight'>
    <!--{{item.uuid}}-{{item.text}}-{{item.image_url}}-{{item.type}}-->
	  <block wx:if="{{item.type === 'teacher'}}">
      <view class="chat teacher" >
      <image src="{{item.image_url}}" class='user-photo'></image>
    <text class='chat-message'>
    {{item.text}}
    </text>
     </view>
     </block>
    <block wx:if="{{item.type === 'student'}}">
      <view class="chat student" >
      <image src="{{item.image_url}}" class='user-photo'></image>
      <text class='chat-message'>
    {{item.text}}
    </text>
     </view>
     </block>
    </view>
  </view>
</view>  

<view class="bottom" style="width:100%;height:80px;border-top:1px solid red;background-color:#f0f0f0;position:fixed; bottom:0;">
  <button bindtap="addMessage" disable="true">增加信息</button>
</view>

上面的代码中 id="container"<view></view>区域显示聊天信息区域。

下面的代码要获取id="container"<view></view>区域的相关属性值。

    var query = wx.createSelectorQuery()
    query.select('#container').boundingClientRect()
    query.exec(function (res) {
    //  rect.id       节点的ID
    //  rect.dataset  节点的dataset
    //  rect.left     节点的左边界坐标
    //  rect.right    节点的右边界坐标
    //  rect.top      节点的上边界坐标
    //  rect.bottom   节点的下边界坐标
    //  rect.width    节点的宽度
    //  rect.height   节点的高度 
    //动态获取view元素的宽高
    console.log(res);
    console.log("节点的上边界坐标");
    console.dir(res[0].top);
    console.log("节点的下边界坐标"); 
    console.dir(res[0].bottom);
    console.log("节点的高度");
    console.log(res[0].height);
    console.log("节点的左边界坐标");
    console.log(res[0].left);
    console.log("节点的右边界坐标");
    console.log(res[0].right);
  
    wx.pageScrollTo({
      // scrollTop	Number	是	滚动到页面的目标位置(单位px)
      scrollTop: res[0].height,
      // duration: 300
    })
  })  

这里相关的函数还是参考微信小程序官方的文档吧,不再赘述。

在上面的代码中打印输出了res的值,对于res打印输出的值进行说明:

在上面的三张图中,第一张图可以看到console.log(res)打印出res的变量值。第二张图可以看到<text></text>的区域大小是220.95x22,第三张图可以看到<view id="container"></view>的区域大小是320x150,因此可以得到<view id="container"></view> 节点的上边界坐标是22,节点的下边界坐标是150+22=172。

考虑一个问题,每次点击按钮"增加信息"后需要往view层中添加数据内容,在添加后需要将内容滚动到最下面一条,以便有更好的用户体验。 选择了使用wx.pageScrollTo接口,通过获取当前view层的高度来动态的调整pageScrollTo中的scrollTop值。

scrollTop:设置或获取位于对象最顶端和窗口中可见内容的最顶端之间的距离

2.3 对于 wx.request 的封装

考虑到之前我提的问题,对于wx.request封装的问题,对于HTTP请求进行封装,这个是通常的做法。

在项目根目录下,新建文件夹apis,在其中新建request.js文件,封装具体的类request,具体的代码如下:

class request {
  constructor() {
    this._header = {}
  }

  /**
   * 设置统一的异常处理
   */
  setErrorHandler(handler) {
    this._errorHandler = handler;
  }

  /**
   * GET类型的网络请求
   */
  getRequest(url, data, header = this._header) {
    return this.requestAll(url, data, header, 'GET')
  }

  /**
   * DELETE类型的网络请求
   */
  deleteRequest(url, data, header = this._header) {
    return this.requestAll(url, data, header, 'DELETE')
  }

  /**
   * PUT类型的网络请求
   */
  putRequest(url, data, header = this._header) {
    return this.requestAll(url, data, header, 'PUT')
  }

  /**
   * POST类型的网络请求
   */
  postRequest(url, data, header = this._header) {
    return this.requestAll(url, data, header, 'POST')
  }

  /**
   * 网络请求
   */
  requestAll(url, data, header, method) {
    return new Promise((resolve, reject) => {
      wx.request({
        url: url,
        data: data,
        header: header,
        method: method,
        success: (res => {
          if (res.statusCode === 200) {
          	//200: 服务端业务处理正常结束
            resolve(res)
          } else {
          	//其它错误,提示用户错误信息
            if (this._errorHandler != null) {
            //如果有统一的异常处理,就先调用统一异常处理函数对异常进行处理
              this._errorHandler(res)
            }
            reject(res)
          }
        }),
        fail: (res => {
          if (this._errorHandler != null) {
            this._errorHandler(res)
          }
          reject(res)
        })
      })
    })
  }
}

export default request

在上面的代码中,难理解的部分是Promise代码,这是ECMAScript 6新的语法,提供异步操作,具体的内容请参考阮一峰的文档:Promise 对象

同样地,在apis文件夹下新建文件chatboxApi.js,具体代码如下:

import request from './request.js'

class chatboxApi {
  constructor() {
    const app = getApp();
    console.log(app);
    this._baseUrl = 'http://localhost:5300';
    this._defaultHeader = { 'data-tupe': 'application/json' }
    this._request = new request
    this._request.setErrorHandler(this.errorHander)
  }

  /**
   * 统一的异常处理方法
   */
  errorHander(res) {
    console.error(res)
  }

  /**
   * 获取所有信息
   */
  getMessage(chatbox_id, key = null) {
    let data = key != null ? { chatbox_id:chatbox_id, queryValue: key } : { chatbox_id:chatbox_id }
    return this._request.getRequest(this._baseUrl + '/chatbox', data).then(res => res.data)
  }

  /**
   * HTTP 请求中的 PUT 、DELETE、POST等网络类型等等函数
  */
}
export default chatboxApi

3. 说明

前面的叙述,主要是搭建起一个新的项目,介绍了原生开发微信小程序。这个lab2-MiniProgram是帮助大家向最终的PJ靠近的,由于lab篇幅有限,只是起到一点点帮助作用,有更多的内容是需要大家去探索学习。

目前开发微信小程序时,可选的技术方案大概有四种,分别是 :

  1. 微信小程序原生开发
  2. 使用wepy框架
  3. 使用mpvue框架
  4. 使用taro框架

四种开发方案,各有优劣。使用第三方框架开发,可以享受框架带来的开发便利,但对于底层的小程序的功能未必能得到很好的应用。而原生小程序的开发模式,又过于简陋,但对于理解功能很有帮助。

最终的PJ项目,我还是建议用第三方框架去开发微信小程序,考虑到整个课程在学习 MVVM 框架,在这里可以应用上,方便考核。

4. 课后作业

截止时间:2018-5-12 23:59:59

提交方式:将压缩包提交到超星指定的lab作业栏里

题目:

  • 任务 : 在这个项目中,多处地方都用到了wx.request,后面我将wx.request这里封装成Request.js,然后提供给chatboxApi.js来调用,请模仿chatboxApi.js写出contentsApi.js与courseApi.js,然后相关页面调用和修改。

最后的代码压缩成“姓名+学号+miniprogram.zip”的文件压缩包。

题目2题目3取消,在5月12日23:59:59之前,请联系助教,进行第一阶段PJ的基本检查,作为本次Lab检查,请各位同学相互通知一下!

About

实验二 : 微信小程序

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published