微信小程序之网易云音乐小案例

news/2024/11/28 20:36:51/

目录

一.编写对网易云音乐api发起请求的代码

二.编写视频项(组件)

三.编写mv列表:包含(轮播图+视频列表[每个视频项引用组件来呈现])

四.编写视频详情页


成品图:

准备工作:

——在pages下新建两个page(index,video_detail)mv首页和视频详情

——在根目录下创建components,并在其中创建一个component(video-item)视频项

——在根目录下创建service,并在其中创建api_video.js和index.js,用于对api发起请求

——在utils下创建format.wxs,用于数据格式的转换【下附format的代码】

//数据处理
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 + ""//先在前面加两个0,然后从字符串长度位开始截取return ("00" + time).slice(time.length)
}
//时间转换
function formatDuration(duration)
{//转成秒数duration = duration / 1000//取整数部分,除以60得到分钟数var minute = Math.floor(duration / 60)//取余数部分的秒数var second = Math.floor(duration % 60)return padLeftZero(minute) + ":" + padLeftZero(second) 
}
module.exports = {formatCount:formatCount,formatDuration:formatDuration
}

 ——在根目录下创建images,并在其中创建bannar,用于轮播图的展示【下附图片】

一.编写对网易云音乐api发起请求的代码

①service/index.js

const BASE_URL = "https://autumnfish.cn" //网易云音乐api接口,如果接口失效了,可以百度重新搜索一个进行替换,或者在本地开一个api服务器也行.
//创建一个类,赋予它请求接口的方式
class MusicRequest{request(url,method,params){return new Promise((resolve,reject)=>{wx.request({url: BASE_URL+url,data:params,method:method,success:res=>resolve(res.data),fail:err=>reject(err)})})}get(url,params){return this.request(url,"GET",params)}post(url,params){return this.request(url,"POST",params)}
}
//向外抛出一个对象
export default new MusicRequest();

②service/api_video.js

import musicRequest from './index';//获取热门视频
export function getTopMV(offset,limit=10){return musicRequest.get("/top/mv",{offset,limit})
}//获取视频地址
export function getMVURL(id){return musicRequest.get("/mv/url",{id})
}//获取视频详情信息
export function getMVDetail(id){return musicRequest.get("/mv/detail",{mvid:id})
}//获取其他相关的视频
export function getRelatedVideo(id){return musicRequest.get("/related/allvideo",{id})
}

二.编写视频项(组件)

①components/video-item/index.wxml

<!--components/video-item/index.wxml-->
<wxs src="../../utils/format.wxs" module="format"/>
<view class="item"><view class="album"><image class="image" 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.artistName}}</view>
</view>

②components/video-item/index.js (这里主要接收外部传进组件的数据item)

// components/video-item/index.js
Component({/*** 组件的属性列表*/properties: {item:Object},/*** 组件的初始数据*/data: {},/*** 组件的方法列表*/methods: {}
})

 ③components/video-item/index.wxss

.item{width: 100%;
}
.album{position: relative;border-radius: 12rpx;overflow: hidden;display: flex;
}
.image{width: 100%;
}
.album .info{width: 100%;position: absolute;bottom: 10rpx;color: white;display: flex;justify-content: space-between;font-size: 24rpx;padding: 0 10rpx;
}
.info .count{padding-left: 36rpx;position: relative;
}
.info .duration{padding-right: 18rpx;
}
.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;
}

三.编写mv列表:包含(轮播图+视频列表[每个视频项引用组件来呈现])

 ①pages/index/index.js

import {getTopMV} from '../../service/api_video'
Page({/*** 页面的初始数据*/data: {MVData:[],//视频列表信息hasMore:true,//是否还有更多swiperHeight:"",//轮播图的高度orignIndex:Math.floor(Math.random()*30),//从[0,30)中取任意一个小数(向下取整)作为初始点orignCount:10,//每次请求获取10条数据},/*** 生命周期函数--监听页面加载*/onLoad: function (options) {this.getTopMVData(this.data.orignIndex)console.log("初始点:"+this.data.orignIndex);},
/*** 由于载入图片之后,无法与swiper同高,所以加入下面的逻辑 * bindload载入时触发,获取载入图片之后dom的矩形信息==>* wx.createSelectorQuery().select(".swiper-image").boundingClientRect();* 如果是e.detail.height就是原图片的高度,性质不同。*/queryRect:function(e) {let $this = this;const query = wx.createSelectorQuery();query.select(".swiper-image").boundingClientRect();query.exec(res=>{$this.setData({swiperHeight:res[0].height+"px"})})},//点击视频跳入详情页videoItemClick:function(e) {//获取当前视频的idconst id = e.currentTarget.dataset.id//navigateTo会保留当前页面,并跳转到下个页面wx.navigateTo({url: '/pages/video_detail/index?id='+id,})},getTopMVData:async function(offset,str) {//如果是上拉,并且当前数据组的下次请求没有更多的数据了,则终止请求if(str === "reachbottom" && !this.data.hasMore) return// 获取从某个点开始的视频列表const res = await getTopMV(offset)// 获取当前的视频列表let newData = this.data.MVData// 如果是下拉刷新,直接覆盖视频列表if(str==="pulldown") {console.log("下拉了");newData = res.data}else{newData = newData.concat(res.data)}// 如果是上拉获取更多视频,则将新旧列表进行拼接// 更新视频列表,以及请求获取视频列表之后是否还有更多数据的判断变量this.setData({MVData:newData,hasMore: res.hasMore //是否还有更多})},/*** 页面相关事件处理函数--监听用户下拉动作*/onPullDownRefresh: function () {//下拉刷新->随机刷新视频列表的初始点let orignIndex = Math.floor(Math.random()*30)this.setData({orignIndex})console.log("刷新后:"+this.data.orignIndex);this.getTopMVData(orignIndex,"pulldown")},/*** 页面上拉触底事件的处理函数*/onReachBottom: function () {//上拉获取更多数据->在原本的初始点上+当前视频列表的长度来作为此次请求的初始点this.getTopMVData(this.data.orignIndex+this.data.MVData.length,"reachbottom")},})

 ②pages/index/index.json (这里要引入组件以及启动page的下拉触发函数)

{"enablePullDownRefresh": true,"backgroundTextStyle": "dark","usingComponents": {"video-item":"/components/video-item/index"}
}

③pages/index/index.wxml 

<swiper class="swiper" autoplay indicator-dots circular interval="{{2000}}" style="height: {{swiperHeight}};"><block wx:for="{{[1,2,3,4,5,6,7]}}" wx:key="index"><swiper-item class="swiper-item"><image class="swiper-image" bindload="queryRect" src="/images/bannar/{{item}}.jpg" mode="widthFix" /></swiper-item></block>
</swiper>
<view class="video"><view class="item" wx:for="{{MVData}}" wx:key="index"><video-item item="{{item}}" data-id="{{item.id}}" bindtap="videoItemClick"></video-item></view>
</view>

 ④pages/index/index.wxss

page{padding: 0 20rpx;
}
.swiper{width: 95%;border-radius: 10rpx;overflow: hidden;transform: translateY(0);
}
.swiper-image{width: 100%;
}
.video{margin-top: 20rpx;width: 95%;display: flex;flex-wrap: wrap;justify-content: space-between;
}
.item{width: 48%;margin-top: 20rpx;
}

四.编写视频详情页

①pages/video_detail/index.js

// pages/video_detail/index.js
import {getMVDetail,getMVURL,getRelatedVideo} from '../../service/api_video'
Page({/*** 页面的初始数据*/data: {mvInfo:{},url:"",relatedData:[]},/*** 生命周期函数--监听页面加载*/onLoad(options) {//获取跳转时传入的idvar id = options.idthis.getMVData(id)},getMVData:function(id){getMVDetail(id).then(res=>{this.setData({mvInfo:res.data})})getMVURL(id).then(res=>{this.setData({url:res.data.url})})getRelatedVideo(id).then(res=>{this.setData({relatedData:res.data})})console.log(this);},/*** 生命周期函数--监听页面初次渲染完成*/onReady() {},/*** 生命周期函数--监听页面显示*/onShow() {},/*** 生命周期函数--监听页面隐藏*/onHide() {},/*** 生命周期函数--监听页面卸载*/onUnload() {},/*** 页面相关事件处理函数--监听用户下拉动作*/onPullDownRefresh() {},/*** 页面上拉触底事件的处理函数*/onReachBottom() {},/*** 用户点击右上角分享*/onShareAppMessage() {}
})

②pages/video_detail/index.wxml

<!--pages/video_detail/index.wxml-->
<wxs src="../../utils/format.wxs" module="format"/>
<video src="{{url}}" class="video"></video>
<view class="detail"><view class="name">{{mvInfo.name}}</view><view class="singer">{{mvInfo.artistName}}</view><view class="info">{{format.formatCount(mvInfo.playCount)}}播放-{{mvInfo.publishTime}}</view>
</view>
<view class="related"><view wx:for="{{relatedData}}" class="item"  wx:key="index"><view class="image"><image src="{{item.coverUrl}}" mode=""/><view class="imageInfo"><view class="count">{{format.formatCount(item.playTime)}}</view></view></view><view class="relatedInfo"><view class="title">{{item.title}}</view><view class="user"><block wx:for="{{item.creator}}" wx:key="index">{{item.userName}}<block wx:if="{{!index==item.creator.length}}">/</block></block></view></view></view>
</view>

③pages/video_detail/index.wxss

/* pages/video_detail/index.wxss */
.video{width: 100%;
}
.detail{margin: 20rpx 0 90rpx 50rpx;
}
.detail .name{font-weight: bold;
}
.detail .singer{font-size: 13px;margin: 10rpx 0; color: gray;
}
.detail .info{font-size: 13px;color: gray;
}
.related{margin: 0 50rpx;
}
.item{display: flex;margin-bottom: 20rpx;width: 100%;
}
.item .image{width: 40%;height: 150rpx;margin-right: 20rpx;position: relative;
}
.item image{width: 100%;height: 100%;border-radius: 5px;
}
.item .relatedInfo{width: 58%;
}
.item .relatedInfo .title{font-size: 15px;
}
.item .relatedInfo .user{font-size: 12px;margin-top: 20rpx;color: gray;
}
.imageInfo {width: 100%;position: absolute;bottom: 10rpx;color: white;display: flex;justify-content: flex-end;font-size: 24rpx;
}
.imageInfo .count{padding-left: 36rpx;padding-right: 10rpx;position: relative;
}
.imageInfo .count::before{content: "";position: absolute;left: -2rpx;top: 4rpx;width: 30rpx;height: 24rpx;background-size: cover;background-image: url("");
}


http://www.ppmy.cn/news/535595.html

相关文章

笔记 Docker 安装 Mysql (四) (2)Docker版Mysq主从复制

1.运行master容器 docker run -d -p 3307:3306 --privilegedtrue -v /data/mysql-master/log:/var/log/mysql -v /data/mysql-master/data:/var/lib/mysql -v /data/mysql-master/conf:/etc/mysql/ --name mysql-master -e MYSQL_ROOT_PASSWORDroot mysql:5.7.40-debian 1.1 …

Spring Boot + MDC 实现全链路调用日志跟踪,这才叫优雅!

点击下方“Java编程鸭”关注并标星 更多精彩 第一时间直达 1、简介&#xff1a; MDC&#xff08;Mapped Diagnostic Context&#xff0c;映射调试上下文&#xff09;是 log4j 、logback及log4j2 提供的一种方便在多线程条件下记录日志的功能。MDC 可以看成是一个与当前线程绑定…

03 处理文件

文章目录 1 创建文件&#xff08;touch命令&#xff09;2 复制文件&#xff08;cp命令&#xff09;2.1 递归复制&#xff08;-R参数&#xff09;2.2 也可以在cp命令中使用通配符 3 移动文件/重命名文件&#xff08;mv命令&#xff09;4 删除文件&#xff08;rm命令&#xff09;…

jenkins启动报:No such file or directory

新的jenkins服务器&#xff0c;自己配的maven&#xff0c;地址都是对的&#xff0c;就是在下载依赖的时候&#xff0c;报没有文件夹&#xff0c;如下 Non-resolvable parent POM for share.project:share-project:1.0.0-SNAPSHOT: Could not transfer artifact org.springfram…

termux怎么玩?

https://www.jianshu.com/p/f41bc63d4246 https://raw.githubusercontent.com/Neo-Oli/termux-ubuntu/master/ubuntu.sh chroot是个啥呀&#xff1f;这和虚拟Ubuntu啥子关系哦&#xff1f; https://www.ibm.com/developerworks/cn/linux/l-cn-chroot/ 那为啥我即使执行chroot命…

linux安装nodejs及服务部署(二)

一、安装nvm 1&#xff0c;下载 [rootizbp1b498epn4trb75oykez ~]# wget -qO- https://raw.githubusercontent.com/creationix/nvm/v0.34.0/install.sh | bash2&#xff0c;编辑.bashrc文件 [rootizbp1b498epn4trb75oykez ~]# vi .bash_profile3&#xff0c;在配置文件最后加入…

8.25css笔记

CSS的基本概念 全称&#xff1a;Cascading Style Sheet 层叠样式表 选择器 标签&#xff1a; …之类的标签 使用方式&#xff1a;直接输入标签名 类 (class)&#xff1a;class名字 使用方式&#xff1a;输入点class名 ID&#xff1a;ID名 使用方式&#xff1a;输入#id名 一…