云开发 页面渲染
在数据绑定章节,我们已经掌握如何把data里面的数据渲染到页面,这一部分我们会介绍如何通过点击组件绑定的事件处理函数来修改data里面的数据,如何把事件处理函数获取到的数据打印到页面。
将变量值渲染到页面
还记得我们之前在控制台打印的Date对象、Math对象、字符串String对象以及常量么?在第一节里我们把这些对象赋值给了一个变量,然后通过控制台可以把这些值给console.log()打印出来,那这些值可不可以渲染到小程序的页面上呢?答案是肯定的。
将变量值渲染到页面
使用开发者工具新建一个页面比如data,然后在data.js的Page({})函数的前面,也就是不写在Page函数里面,写在data.js的第1行输入以下代码:
let lesson = "云开发技术训练营";
let enname = "CloudBase Camp";
let x = 3, y = 4, z = 5.001, a = -3, b = -4, c = -5;
let now = new Date();
注意上面这些是JavaScript函数的语句,所以用的是分号;分隔,这个不要和之前的逗号分隔给弄混了哦。如果语句是换行的,后面的分号;也可以不必写。
然后在data里面添加如下数据(注意没有双引号,单双引号里的是字符串)
data: {
charat: lesson.charAt(4),
concat: enname.concat(lesson),
uppercase:enname.toUpperCase(),
abs:Math.abs(b),
pow: Math.pow(x, y),
sign:Math.sign(a),
now:now,
fullyear:now.getFullYear(),
date:now.getDate(),
day: now.getDay(),
hours: now.getHours(),
minutes: now.getMinutes(),
seconds: now.getSeconds(),
time: now.getTime()
},
在data.wxml里输入以下代码:
<view>"云开发技术训练营"第5个字符 {{charat}}</view>
<view>两个字符串连接后的结果:{{concat}}</view>
<view>CloudBase Camp字母大写:{{uppercase}}</view>
<view>b的绝对值:{{abs}}</view>
<view>x的y次幂:{{pow}}</view>
<view>返回a是正还是负:{{sign}}</view>
<view>now对象:{{now}}</view>
<view>{{fullyear}}年</view>
<view>{{date}}日</view>
<view>星期{{day}}</view>
<view>{{hours}}时</view>
<view>{{minutes}}分</view>
<view>{{seconds}}秒</view>
<view>1970年1月1日至今的毫秒数:{{time}}</view>
因为data是一个对象Object,我们可以通过冒号:的方式将变量值赋值给data里的各个属性,而在数据绑定章节,这些数据是可以直接渲染到小程序的页面的。
toString()方法
我们发现{{now}}渲染的结果是一个对象[object Object],而并没有显示出字符串文本,这个时候就需要用到对象的toString()方法,得到对象的字符串。将data里now的赋值改为如下:
now:now.toString(),
技术文档:toString()方法
响应的数据绑定
逻辑层js文件里的data数据,无论是基础的字符串、数组、对象等,还是通过变量给赋的值,都可以渲染到页面。不仅如此,只要对逻辑层data里的数据进行修改,视图层也会做相应的更新,我们称之为响应的数据绑定,而这是通过Page的setData()方法来实现的。
使用开发者工具在data.wxml里输入:
<view style="background-color:{{bgcolor}};width:400rpx;height:300rpx;"></view>
<button bindtap="redTap">让背景变红</button>
<button bindtap="yellowTap">让背景变黄</button>
然后在data.js里添加一个数据
bgcolor:"#000000",
然后在js里添加两个button绑定的事件处理函数redTap和yellowTap:
redTap:function(){
this.setData({
bgcolor: "#cd584a"
})
},
yellowTap:function(){
this.setData({
bgcolor: "#f8ce5f"
})
},
点击button,原来view组件的背景颜色由黑色变成了其他颜色,这是因为点击组件触发事件处理函数,调用了Page的setData()方法修改了data里与之相应的属性的值(重复赋值就是修改),bgcolor由原来的”#000000″,变成了其他数据。
小任务:通过以往的学习我们了解到无论是组件的样式,图片、链接的路径,数组、对象里的数据,他们都是可以进行数据分离写到data里面的,这也就意味着,我们通过点击事件改变data里面的数据可以达到很多意想不到的效果,请发挥你的想象力做一些有意思的案例出来。
响应的布尔操作
在前面我们已经了解到,有些组件的私有属性的数据类型为Boolean布尔值,比如视频、Swiper轮播组件是否自动播放、是否轮播,视频组件是否显示播放按钮等等,这些我们都可以使用setData将true改为false,false改为true来达到控制的目的。
在交互方面,响应的布尔操作可以用于单一属性true与false的切换,比如显示与隐藏、展开与折叠、聚焦与失焦、选中与不选中。
我们来看一个案例,使用开发者工具在data.wxml里输入以下代码:
<video id="daxueVideo" src="http://wxsnsdy.tc.qq.com/105/20210/snsdyvideodownload?filekey=30280201010421301f0201690402534804102ca905ce620b1241b726bc41dcff44e00204012882540400&bizid=1023&hy=SH&fileparam=302c020101042530230204136ffd93020457e3c4ff02024ef202031e8d7f02030f42400204045a320a0201000400" autoplay loop muted="{{muted}}" initial-time="100" controls event-model="bubble">
</video>
<button bindtap="changeMuted">静音和取消静音</button>
然后给在data.js的data里新增
muted: true,
然后添加changeMuted事件处理函数
changeMuted: function (e) {
this.setData({
muted: !this.data.muted
})
},
在开发者工具的模拟器里点击按钮,我们发现静音和取消静音都是这个按钮。这里的感叹号 !是逻辑非的意思,可以理解为not。
this.setData和 this.data都用到了一个关键字 this。 this和中文里的“这个的”有类似的指代作用,在方法中, this 指代该方法所属的对象,比如这里的是Page对象, this.data就是指Page函数对象里的data对象。
响应的数组操作
结合点击事件以及数组操作的知识,我们再来看下面这个案例,了解如何通过点击按钮新增数组里的数据和删除数组里的数据。
使用开发者工具在data.wxml里输入以下代码,注意这里视图层只有一个{{text}},也就是说我们之后会把所有的数据都赋值给data里的text。
<view>{{text}}</view>
<button bindtap="addLine">新增一行</button>
<button bindtap="removeLine">删掉最后一行</button>
然后在data.js的Page()之前声明变量,这里声明extraLine为一个空数组,我们之后会往这个数组里添加和删除数据。
let initData = '只有一行原始数据'
let extraLine = [];
然后再在Page的data里添加一条数据,
text: initData,
我们先来看没有事件处理函数时,数据渲染的逻辑,首先我们把initData变量值赋值给text,这时渲染的结果只有initData里的数据,所以页面显示的是“只有一行原始数据”,而extraLine和text没有什么关系。
我们再来在Page里添加addLine和removeLine的事件处理函数:
addLine: function (e) {
extraLine.push('新增的内容')
this.setData({
text: initData + '\n' + extraLine.join('\n')
})
},
removeLine: function (e) {
if (extraLine.length > 0) {
extraLine.pop()
this.setData({
text: initData + '\n' + extraLine.join('\n')
})
}
},
首先回顾一下之前的数组操作知识,push为往数组的末尾新增数据,而pop则删除数组末尾一行的数据,join为数组数据之前的连接符。
点击按钮新增一行,触发绑定的事件处理函数addLine,首先会执行extraLine数组新增一条数据“新增的内容”,但是这时extraLine和text还没有关系,这时在setData()函数里将initData和extraLine进行拼接(注意extraLine本来是一个数组,但是调用join方法后返回的是数组的值拼接好的字符串)。点击按钮删除最后一行,会先删除extraLine数组里最后一行的数据。
小任务:新增内容过于单一,我们可以给它后面添加一个随机数,将
extraLine.push('新增的内容')
改成extraLine.push('新增的内容'+Math.random())
,再来看看新增数据的效果,关于Math.random()大家可以自行去MDN查阅。大家也可以把拼接的连接符由\n
换成其他字符。
函数与调用函数
函数的作用,可以写一次代码,然后反复地重用这个代码。JavaScript的函数本身也是对象,因此可以把函数赋值给变量,或者作为参数传递给其他函数。
函数的定义和结构
我们可以使用function关键词来定义一个函数,括号()里为函数的参数,参数可以有很多个,使用逗号,隔开;函数要执行的代码(语句)使用大括号{}包住:
function 函数名(参数 1, 参数 2, 参数 3) {
代码块内要执行的语句
}
不带参数的函数
比如,我们使用开发者工具在data.js的Page()函数前,添加如下代码:
function greet() {
console.log("你好,欢迎来到云开发训练营");
};
greet(); //调用greet()函数
保存之后,我们可以在控制台看到函数打印的字符串。定义一个函数并不会自动的执行它。定义了函数仅仅是赋予函数以名称并明确函数被调用时该做些什么。调用函数才会以给定的参数真正执行这些动作。greet()函数没有参数,调用函数时,直接写函数名+括号即可。
只有一个参数的函数
下面定义了一个简单的平方函数square(),square为函数名,number为函数的参数(名称可以自定义),使用return语句确定函数的返回值。我们继续在data.js的Page()函数前,输入以下代码:
function square(number) {
return number * number;
};
square(5);
square(5),就是把5赋值给变量number,然后执行numbernumber,也就是55,然后返回return这个值。
这里的number被称之为形参,而5被称之为实参。大家可以结合案例就能大致了解形参和实参的意思。
- 形参是在定义函数时使用的参数,目的是用来接收调用该函数时传进来的实际参数。
- 实参是在调用时传递给函数的参数
JavaScript允许传入任意个参数而不影响调用,因此传入的参数可以比定义的参数多,但是不能少。也就是说实参的数量可以多于形参但是不能少于形参。
对象的方法
在小程序里我们会经常将一个匿名函数赋值给对象的一个属性,而这个属性我们可以称之为对象的方法。
匿名函数
函数声明function在语法上是一个语句,但函数也可以由函数表达式创建,这样的函数没有函数名称(匿名)。
使用开发者工具在data.js的Page()函数前,输入以下代码:
let square = function(number) {
return number * number
};
console.log(square(4))//使用console.log()输出变量square
执行后,可以在控制台看到输出的结果为16。上面这个function函数没有函数名,相当于是把函数的返回值赋值给了变量square。
箭头函数
为什么叫箭头函数(Arrow Function),因为它定义一个函数用的就是一个箭头=>,我们来看两个例子,在data.js的Page()函数前输入以下代码:
const multiply = (x, y) => {
return x * y;
}
const sum= (x, y) => x + y;//连{}和return语句都可以省掉
console.log(multiply(20, 4));
console.log(sum(20, 4));
在控制台我们可以看到箭头函数打印的结果。箭头函数相当于匿名函数,它没有函数名,而且也简化了函数定义。箭头函数可以只包含一个表达式,甚至连{ … }和return都可以省略掉。大家可以先只需要了解这个写法就可以了,以后碰到不至于比较迷惑,见多了也试着尝试多写一下。
调用对象的方法
可以使用点表示法来调用对象的方法,这个和访问对象的属性没有区别。而调用对象的方法和调用一个函数也是大同小异。
调用对象的方法我们在前面就已经接触过大量的案例了,在前面我们已经说过,wx是小程序的全局对象,而在第一节我们打印的很多API,就是调用了wx对象里的方法。
JavaScript函数的写法
在点击事件章节里,我们创建的事件点击处理函数的写法如下:
scrollToPosition() {
},
而在这一节我们创建的事件点击函数的写法为:
yellowTap:function(){
},
这两种写法都是可以执行的,大家可以把这两种写法互相修改一下试试看~
currentTarget事件对象
在前面的列表渲染里,我们知道点击电影列表里的某一部电影,要进行页面跳转显示该电影的详情,我们需要给该电影创建一个页面,那如果要显示数千部的电影的详情,一一创建电影详情页显然不合适,毕竟所有电影的详情页都是同一一个结构,有没有办法所有电影详情都共用一个页面,但是根据点击的链接的不同,渲染相应的数据?答案是肯定的,要解决这个问题,首先我们要了解链接组件的点击信息。
当点击组件触发事件时,逻辑层绑定该事件的处理函数会收到一个事件对象,通过 event 对象可以获取事件触发时候的一些信息,比如时间戳、 detail 以及当前组件的一些属性值集合,尤其是事件源组件的id。
currentTarget是事件对象的一个属性,表示的是事件绑定的当前组件。使用开发者工具在data.wxml里输入以下代码
<view class="weui-navbar">
<block wx:for="{{tabs}}" wx:key="index">
<view id="{{index}}" class="weui-navbar__item {{activeIndex == index ? 'weui-bar__item_on' : ''}}" bindtap="tabClick">
<view class="weui-navbar__title">{{item}}</view>
</view>
</block>
</view>
<view class="weui-tab__panel">
<view hidden="{{activeIndex != 0}}">帝都</view>
<view hidden="{{activeIndex != 1}}">魔都</view>
<view hidden="{{activeIndex != 2}}">妖都</view>
<view hidden="{{activeIndex != 3}}">渔村</view>
</view>
然后再往data.js的data里添加以下数据:
tabs: ["北京", "上海", "广州", "深圳"],
activeIndex:0,
然后再添加事件处理函数tabClick。
tabClick: function (e) {
console.log(e)
this.setData({
activeIndex: e.currentTarget.id
});
},
编译之后在模拟器里预览。当我们点击上面的tab时,触发tabClick事件处理函数,这时候事件处理函数会收到一个事件对象e,我们可以看一下控制台打印的e对象的内容,关于e对象具体属性的解释可以看技术文档。
技术文档:事件对象
currentTarget就是事件对象的一个属性,我们可以使用点表示法获取到点击的组件的Id,并将其赋值给activeIndex,所谓active就是激活的意思,也就是我们点击哪个tab,哪个tab就激活。
- 当点击的id为0,也就是第一个tab时,activeIndex的值被事件处理函数修改为0;
- activeIndex == index相同的tab,也就是激活的tab就会有weui-bar__item_on的class,也就显示为绿色
- !=是不等于操作符,activeIndex != 0显然不成立条件为false,也就是组件hidden为false,即为显示;而activeIndex != 1,2,3则都会true,hidden生效,组件不显示,于是tab的效果就有了。
当我们对字符串、Math对象、Date对象、数组对象、函数对象、事件对象所包含的信息不了解时,把他们打印出来即可。打印出来的结果基本都是字符串、列表、对象,而在前面我们已经掌握如何操作它们。通过实战,通过打印日志,既有利于我们调试代码,也加强我们对逻辑的理解。