- ✅初始化项目
- ✅配置tabbar
- ✅封装请求函数
- ✅分层架构二次封装请求函数
- ✅封装格式化方法 (utils/format.wxs)
- ✅封装视频列表item组件
- ✅完成视频列表页展示
- 今日完成效果如下
\
初始化项目
\
删除无关项目文件
- 删除pages目录下的index和logs,删除utils目录下的util.js
- 初始化app.js为
// app.js
App({
})
- 初始化app.json为
{
"pages": [
],
"window": {
"backgroundTextStyle": "light",
"navigationBarBackgroundColor": "#fff",
"navigationBarTitleText": "Weixin",
"navigationBarTextStyle": "black"
},
"style": "v2",
"sitemapLocation": "sitemap.json",
"lazyCodeLoading": "requiredComponents"
}
- 初始化app.wxss为
/**app.wxss**/
- 目前为止编译器/模拟器会报错
\
配置tabbar
新建两个页面
- 在pages右键创建文件夹
home-music和home-video - 在
home-video和home-music上右键新建page为index
- 小程序开发工具会自动在目录下创建四个文件,并自动在
app.json中添加url
导入静态资源assets
- 在文件管理器中打开项目,复制已经准备好的assets资源到根目录
\
在app.json中配置tabbar
"tabBar": {
"list": [
{
"pagePath": "pages/home-music/index",
"text": "音乐",
"iconPath": "assets/images/tabbar/music_normal.png",
"selectedIconPath": "assets/images/tabbar/music_active.png"
},
{
"pagePath": "pages/home-video/index",
"text": "视频",
"iconPath": "assets/images/tabbar/video_normal.png",
"selectedIconPath": "assets/images/tabbar/video_active.png"
}
]
},
- 效果
\
封装请求函数
由于微信小程序给我们提供的wx.request请求许多参数都可以复用,所以可以封装一个请求函数
- 在根目录下新建
service文件夹 - 新建
service/index.js
const BASE_URL = "http://123.207.32.32:9001";
class HYRequest {
request(url, method, params) {
return new Promise((resolve, reject) => {
wx.request({
url: BASE_URL + url,
method: method,
data: params,
success: function (res) {
resolve(res.data);
},
fail: function (err) {
reject(err);
},
});
});
}
get(url, params) {
return this.request(url, "GET", params);
}
post(url, data) {
return this.request(url, "POST", data);
}
}
const hyRequest = new HYRequest();
export default hyRequest;
二次封装请求函数
某些动作需要反复调用请求函数,但是有不变的参数,没必要每次传递,可以再封装一次
- 新建
service/api_video.js
import hyRequest from './index'
export function getTopMV(offset,limit=10){
return hyRequest.get('/top/mv',{
offset,
limit
})
}
- 由于我们的api后台需要接收两个参数,一个是偏移量,一个是数量,我们默认每次截取十个即可
调用请求获得数据
- 在
home-video/index.js中引入getTopMV方法
import { getTopMV } from "../../service/api_video";
- 在
home-video/index.js的data中定义两个变量
data: {
topMVs: [], //用以保存请求到的数据列表
hasMore: true, // 判断还有没有更多数据能够请求
},
- 封装专属于
home-video的网络请求方法,用以在不同操作下复用
/**
* 封装网络请求的方法
*/
async getTopMVData(offset) {
// 判断是否可以请求
if (!this.data.hasMore) return;
// 展示加载动画
wx.showNavigationBarLoading();
// 真正请求数据
const res = await getTopMV(offset);
let newData = this.data.topMVs;
if (offset === 0) {
newData = res.data;
} else {
newData = newData.concat(res.data);
}
this.setData({
topMVs: newData,
});
this.setData({
hasMore: res.hasMore,
});
// 隐藏navigation刷新动画
wx.hideNavigationBarLoading();
// 当请求数据完成 关闭下拉刷新动画
if (offset === 0) {
wx.stopPullDownRefresh();
}
},
- 当生命周期onLoad执行时,调用请求方法
/**
* 生命周期函数--监听页面加载
* async await
*/
onLoad: function (options) {
this.getTopMVData(0);
},
- 当下拉刷新时调用请求方法
/**
* 下拉刷新
*/
onPullDownRefresh: function () {
this.getTopMVData(0);
},
- 当页面触底时调用请求方法
/**
* 页面上拉触底事件的处理函数
*/
onReachBottom: function () {
this.getTopMVData(this.data.topMVs.length);
},
home-video/index.js完整代码
// pages/home-video/index.js
import { getTopMV } from "../../service/api_video";
Page({
/**
* 页面的初始数据
*/
data: {
topMVs: [],
hasMore: true,
},
/**
* 生命周期函数--监听页面加载
* async await
*/
onLoad: function (options) {
this.getTopMVData(0);
},
/**
* 封装网络请求的方法
*/
async getTopMVData(offset) {
// 判断是否可以请求
if (!this.data.hasMore) return;
// 展示加载动画
wx.showNavigationBarLoading();
// 真正请求数据
const res = await getTopMV(offset);
let newData = this.data.topMVs;
if (offset === 0) {
newData = res.data;
} else {
newData = newData.concat(res.data);
}
this.setData({
topMVs: newData,
});
this.setData({
hasMore: res.hasMore,
});
// 隐藏navigation刷新动画
wx.hideNavigationBarLoading();
// 当请求数据完成 关闭下拉刷新动画
if (offset === 0) {
wx.stopPullDownRefresh();
}
},
/**
* 封装事件处理的方法
*/
handleVideoItemClick: function (event) {
const id = event.currentTarget.dataset.item.id;
// 页面跳转
wx.navigateTo({
url: `/pages/detail-video/index?id=${id}`,
});
},
/**
* 下拉刷新
*/
onPullDownRefresh: function () {
this.getTopMVData(0);
},
/**
* 生命周期函数--监听页面初次渲染完成
*/
onReady: function () {},
/**
* 生命周期函数--监听页面显示
*/
onShow: function () {},
/**
* 生命周期函数--监听页面隐藏
*/
onHide: function () {},
/**
* 生命周期函数--监听页面卸载
*/
onUnload: function () {},
/**
* 页面相关事件处理函数--监听用户下拉动作
*/
onPullDownRefresh: function () {},
/**
* 页面上拉触底事件的处理函数
*/
onReachBottom: function () {
this.getTopMVData(this.data.topMVs.length);
},
/**
* 用户点击右上角分享
*/
onShareAppMessage: function () {},
});
遍历数据到页面
前面我们已经请求到数据了,可以直接在home-video/index.wxml中调用,由于列表的item也可以复用,我们将它封装成一个component
<!--pages/home-video/index.wxml-->
<view class="video">
<view class="item" wx:for="{{topMVs}}" wx:key="id">
<!-- <video-item-v1 item="{{item}}"
bindtap="handleVideoItemClick"
data-item="{{item}}"></video-item-v1> -->
</view>
</view>
\
封装item组件
- 在根目录下新建文件夹
components/video-item-v1,右键video-item-v1新建component为index
- 可以看到
component的index.js和普通组件不同,其中有一个专门接收父组件传递过来数据的函数
/**
* 组件的属性列表
*/
properties: {
},
- 可以在这儿定义接收的数据,就可以直接在wxml页面中使用了
/**
* 组件的属性列表
*/
properties: {
item:{
type:Object,
default:{}
}
},
- 由于服务器返回的数据里面
播放量和时间都需要转换格式,我们可以在utils目录下定义一个format.wxs用于格式化数据,wxml中只能调用wxs定义的方法
function formatCount(count) {
var counter = parseInt(count);
if (counter > 100000000) {
return (counter / 100000000).toFixed(1) + "亿";
} else if (counter > 10000) {
return (counter / 10000).toFixed(1) + "万";
} else {
return counter + "";
}
}
function padLeftZero(time) {
time = time + "";
return ("00" + time).slice(time.length);
}
function formatDuration(duration) {
duration = duration / 1000;
var minute = Math.floor(duration / 60);
// 计算秒钟
var second = duration % 60;
return padLeftZero(minute) + ":" + padLeftZero(second);
}
// commonjs
module.exports = {
formatCount: formatCount,
formatDuration: formatDuration,
};
- 现在我们可以在wxml中使用父组件传递过来的数据了
<!--components/video-item-v1/index.wxml-->
<wxs src="../../utils/format.wxs" module="format"></wxs>
<view class="item">
<view class="album">
<image class="images" src="{{item.cover}}" mode="widthFix" />
<view class="info">
<view class="count">{{format.formatCount(item.playCount)}}</view>
<view class="duration">{{format.formatDuration(item.mv.videos[0].duration)}}</view>
</view>
</view>
<view class="content">
{{item.name}} - {{item.artisName}}
</view>
</view>
- 为item组件配置样式
/* components/video-item-v1/index.wxss */
.item {
width: 100%;
margin-bottom: 30rpx;
}
.album {
position: relative;
border-radius: 12rpx;
overflow: hidden;
display: flex;
}
.album .image {
width: 100%;
}
.info {
position: absolute;
padding: 0 10rpx;
box-sizing: border-box;
width: 100%;
bottom: 8rpx;
display: flex;
justify-content: space-between;
color: #fff;
font-size: 24rpx;
}
.info .count {
padding-left: 36rpx;
position: relative;
}
.info .count::before {
content: "";
position: absolute;
left: -2rpx;
top: 4rpx;
width: 30rpx;
height: 24rpx;
background-size: cover;
background-image: url("");
}
.content {
margin-top: 10rpx;
font-size: 28rpx;
/* 显示两行 */
text-overflow: ellipsis;
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
display: -moz-box;
-moz-line-clamp: 2;
-moz-box-orient: vertical;
word-wrap: break-word;
word-break: break-all;
white-space: normal;
overflow: hidden;
}
父组件引入子组件
- 在
home-video/index.json中引入子组件
{
"enablePullDownRefresh": true,
"backgroundTextStyle": "dark",
"usingComponents": {
"video-item-v1": "/components/video-item-v1/index"
}
}
- 改写
home-video/index.wxml使用组件
<!--pages/home-video/index.wxml-->
<view class="video">
<view class="item" wx:for="{{topMVs}}" wx:key="id">
<video-item-v1 item="{{item}}"
bindtap="handleVideoItemClick"
data-item="{{item}}"></video-item-v1>
</view>
</view>
父组件传值就是通过data-item传递的
\
- 最后我们还需要修改
home-video/index.wxss
/* pages/home-video/index.wxss */
.video {
display: flex;
flex-wrap: wrap;
justify-content: space-around;
}
.item {
width: 48%;
}
\
其他:
- 项目地址: gitee.com/LUNIONT/xhu…