当前位置:   article > 正文

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

微信小程序网易云代码

目录

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

二.编写视频项(组件)

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

四.编写视频详情页


成品图:

准备工作:

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

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

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

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

  1. //数据处理
  2. function formatCount(count){
  3. var counter = parseInt(count)
  4. if(counter > 100000000){
  5. return (counter / 100000000).toFixed(1) + "亿"
  6. }else if(counter > 10000){
  7. return (counter / 10000).toFixed(1) + "万"
  8. }else {
  9. return counter + ""
  10. }
  11. }
  12. //往左边加零
  13. function padLeftZero(time){
  14. time = time + ""
  15. //先在前面加两个0,然后从字符串长度位开始截取
  16. return ("00" + time).slice(time.length)
  17. }
  18. //时间转换
  19. function formatDuration(duration)
  20. {
  21. //转成秒数
  22. duration = duration / 1000
  23. //取整数部分,除以60得到分钟数
  24. var minute = Math.floor(duration / 60)
  25. //取余数部分的秒数
  26. var second = Math.floor(duration % 60)
  27. return padLeftZero(minute) + ":" + padLeftZero(second)
  28. }
  29. module.exports = {
  30. formatCount:formatCount,
  31. formatDuration:formatDuration
  32. }

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

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

①service/index.js

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

②service/api_video.js

  1. import musicRequest from './index';
  2. //获取热门视频
  3. export function getTopMV(offset,limit=10){
  4. return musicRequest.get("/top/mv",{offset,limit})
  5. }
  6. //获取视频地址
  7. export function getMVURL(id){
  8. return musicRequest.get("/mv/url",{id})
  9. }
  10. //获取视频详情信息
  11. export function getMVDetail(id){
  12. return musicRequest.get("/mv/detail",{mvid:id})
  13. }
  14. //获取其他相关的视频
  15. export function getRelatedVideo(id){
  16. return musicRequest.get("/related/allvideo",{id})
  17. }

二.编写视频项(组件)

①components/video-item/index.wxml

  1. <!--components/video-item/index.wxml-->
  2. <wxs src="../../utils/format.wxs" module="format"/>
  3. <view class="item">
  4. <view class="album">
  5. <image class="image" src="{{item.cover}}" mode="widthFix" />
  6. <view class="info">
  7. <view class="count">
  8. {{format.formatCount(item.playCount)}}
  9. </view>
  10. <view class="duration">
  11. {{format.formatDuration(item.mv.videos[0].duration)}}
  12. </view>
  13. </view>
  14. </view>
  15. <view class="content">
  16. {{item.name}} - {{item.artistName}}
  17. </view>
  18. </view>

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

  1. // components/video-item/index.js
  2. Component({
  3. /**
  4. * 组件的属性列表
  5. */
  6. properties: {
  7. item:Object
  8. },
  9. /**
  10. * 组件的初始数据
  11. */
  12. data: {
  13. },
  14. /**
  15. * 组件的方法列表
  16. */
  17. methods: {
  18. }
  19. })

 ③components/video-item/index.wxss

  1. .item{
  2. width: 100%;
  3. }
  4. .album{
  5. position: relative;
  6. border-radius: 12rpx;
  7. overflow: hidden;
  8. display: flex;
  9. }
  10. .image{
  11. width: 100%;
  12. }
  13. .album .info{
  14. width: 100%;
  15. position: absolute;
  16. bottom: 10rpx;
  17. color: white;
  18. display: flex;
  19. justify-content: space-between;
  20. font-size: 24rpx;
  21. padding: 0 10rpx;
  22. }
  23. .info .count{
  24. padding-left: 36rpx;
  25. position: relative;
  26. }
  27. .info .duration{
  28. padding-right: 18rpx;
  29. }
  30. .info .count::before{
  31. content: "";
  32. position: absolute;
  33. left: -2rpx;
  34. top: 4rpx;
  35. width: 30rpx;
  36. height: 24rpx;
  37. background-size: cover;
  38. background-image: url("");
  39. }
  40. .content{
  41. margin-top: 10rpx;
  42. font-size: 28rpx;
  43. /*显示两行,超出部分使用省略号显示*/
  44. text-overflow: ellipsis;
  45. display: -webkit-box;
  46. -webkit-line-clamp: 2;
  47. -webkit-box-orient: vertical;
  48. display: -moz-box;
  49. -moz-line-clamp: 2;
  50. -moz-box-orient: vertical;
  51. word-wrap: break-word;
  52. word-break: break-all;
  53. white-space: normal;
  54. overflow: hidden;
  55. }

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

 ①pages/index/index.js

  1. import {getTopMV} from '../../service/api_video'
  2. Page({
  3. /**
  4. * 页面的初始数据
  5. */
  6. data: {
  7. MVData:[],//视频列表信息
  8. hasMore:true,//是否还有更多
  9. swiperHeight:"",//轮播图的高度
  10. orignIndex:Math.floor(Math.random()*30),//从[0,30)中取任意一个小数(向下取整)作为初始点
  11. orignCount:10,//每次请求获取10条数据
  12. },
  13. /**
  14. * 生命周期函数--监听页面加载
  15. */
  16. onLoad: function (options) {
  17. this.getTopMVData(this.data.orignIndex)
  18. console.log("初始点:"+this.data.orignIndex);
  19. },
  20. /**
  21. * 由于载入图片之后,无法与swiper同高,所以加入下面的逻辑
  22. * bindload载入时触发,获取载入图片之后dom的矩形信息==>
  23. * wx.createSelectorQuery().select(".swiper-image").boundingClientRect();
  24. * 如果是e.detail.height就是原图片的高度,性质不同。
  25. */
  26. queryRect:function(e) {
  27. let $this = this;
  28. const query = wx.createSelectorQuery();
  29. query.select(".swiper-image").boundingClientRect();
  30. query.exec(res=>{
  31. $this.setData({
  32. swiperHeight:res[0].height+"px"
  33. })
  34. })
  35. },
  36. //点击视频跳入详情页
  37. videoItemClick:function(e) {
  38. //获取当前视频的id
  39. const id = e.currentTarget.dataset.id
  40. //navigateTo会保留当前页面,并跳转到下个页面
  41. wx.navigateTo({
  42. url: '/pages/video_detail/index?id='+id,
  43. })
  44. },
  45. getTopMVData:async function(offset,str) {
  46. //如果是上拉,并且当前数据组的下次请求没有更多的数据了,则终止请求
  47. if(str === "reachbottom" && !this.data.hasMore) return
  48. // 获取从某个点开始的视频列表
  49. const res = await getTopMV(offset)
  50. // 获取当前的视频列表
  51. let newData = this.data.MVData
  52. // 如果是下拉刷新,直接覆盖视频列表
  53. if(str==="pulldown") {
  54. console.log("下拉了");
  55. newData = res.data
  56. }
  57. else{newData = newData.concat(res.data)}// 如果是上拉获取更多视频,则将新旧列表进行拼接
  58. // 更新视频列表,以及请求获取视频列表之后是否还有更多数据的判断变量
  59. this.setData({
  60. MVData:newData,
  61. hasMore: res.hasMore //是否还有更多
  62. })
  63. },
  64. /**
  65. * 页面相关事件处理函数--监听用户下拉动作
  66. */
  67. onPullDownRefresh: function () {
  68. //下拉刷新->随机刷新视频列表的初始点
  69. let orignIndex = Math.floor(Math.random()*30)
  70. this.setData({orignIndex})
  71. console.log("刷新后:"+this.data.orignIndex);
  72. this.getTopMVData(orignIndex,"pulldown")
  73. },
  74. /**
  75. * 页面上拉触底事件的处理函数
  76. */
  77. onReachBottom: function () {
  78. //上拉获取更多数据->在原本的初始点上+当前视频列表的长度来作为此次请求的初始点
  79. this.getTopMVData(this.data.orignIndex+this.data.MVData.length,"reachbottom")
  80. },
  81. })

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

  1. {
  2. "enablePullDownRefresh": true,
  3. "backgroundTextStyle": "dark",
  4. "usingComponents": {
  5. "video-item":"/components/video-item/index"
  6. }
  7. }

③pages/index/index.wxml 

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

 ④pages/index/index.wxss

  1. page{
  2. padding: 0 20rpx;
  3. }
  4. .swiper{
  5. width: 95%;
  6. border-radius: 10rpx;
  7. overflow: hidden;
  8. transform: translateY(0);
  9. }
  10. .swiper-image{
  11. width: 100%;
  12. }
  13. .video{
  14. margin-top: 20rpx;
  15. width: 95%;
  16. display: flex;
  17. flex-wrap: wrap;
  18. justify-content: space-between;
  19. }
  20. .item{
  21. width: 48%;
  22. margin-top: 20rpx;
  23. }

四.编写视频详情页

①pages/video_detail/index.js

  1. // pages/video_detail/index.js
  2. import {getMVDetail,getMVURL,getRelatedVideo} from '../../service/api_video'
  3. Page({
  4. /**
  5. * 页面的初始数据
  6. */
  7. data: {
  8. mvInfo:{},
  9. url:"",
  10. relatedData:[]
  11. },
  12. /**
  13. * 生命周期函数--监听页面加载
  14. */
  15. onLoad(options) {
  16. //获取跳转时传入的id
  17. var id = options.id
  18. this.getMVData(id)
  19. },
  20. getMVData:function(id){
  21. getMVDetail(id).then(res=>{
  22. this.setData({mvInfo:res.data})
  23. })
  24. getMVURL(id).then(res=>{
  25. this.setData({url:res.data.url})
  26. })
  27. getRelatedVideo(id).then(res=>{
  28. this.setData({relatedData:res.data})
  29. })
  30. console.log(this);
  31. },
  32. /**
  33. * 生命周期函数--监听页面初次渲染完成
  34. */
  35. onReady() {
  36. },
  37. /**
  38. * 生命周期函数--监听页面显示
  39. */
  40. onShow() {
  41. },
  42. /**
  43. * 生命周期函数--监听页面隐藏
  44. */
  45. onHide() {
  46. },
  47. /**
  48. * 生命周期函数--监听页面卸载
  49. */
  50. onUnload() {
  51. },
  52. /**
  53. * 页面相关事件处理函数--监听用户下拉动作
  54. */
  55. onPullDownRefresh() {
  56. },
  57. /**
  58. * 页面上拉触底事件的处理函数
  59. */
  60. onReachBottom() {
  61. },
  62. /**
  63. * 用户点击右上角分享
  64. */
  65. onShareAppMessage() {
  66. }
  67. })

②pages/video_detail/index.wxml

  1. <!--pages/video_detail/index.wxml-->
  2. <wxs src="../../utils/format.wxs" module="format"/>
  3. <video src="{{url}}" class="video"></video>
  4. <view class="detail">
  5. <view class="name">
  6. {{mvInfo.name}}
  7. </view>
  8. <view class="singer">
  9. {{mvInfo.artistName}}
  10. </view>
  11. <view class="info">
  12. {{format.formatCount(mvInfo.playCount)}}播放-{{mvInfo.publishTime}}
  13. </view>
  14. </view>
  15. <view class="related">
  16. <view wx:for="{{relatedData}}" class="item" wx:key="index">
  17. <view class="image">
  18. <image src="{{item.coverUrl}}" mode=""/>
  19. <view class="imageInfo">
  20. <view class="count">
  21. {{format.formatCount(item.playTime)}}
  22. </view>
  23. </view>
  24. </view>
  25. <view class="relatedInfo">
  26. <view class="title">
  27. {{item.title}}
  28. </view>
  29. <view class="user">
  30. <block wx:for="{{item.creator}}" wx:key="index">
  31. {{item.userName}}
  32. <block wx:if="{{!index==item.creator.length}}">/</block>
  33. </block>
  34. </view>
  35. </view>
  36. </view>
  37. </view>

③pages/video_detail/index.wxss

  1. /* pages/video_detail/index.wxss */
  2. .video{
  3. width: 100%;
  4. }
  5. .detail{
  6. margin: 20rpx 0 90rpx 50rpx;
  7. }
  8. .detail .name{
  9. font-weight: bold;
  10. }
  11. .detail .singer{
  12. font-size: 13px;
  13. margin: 10rpx 0;
  14. color: gray;
  15. }
  16. .detail .info{
  17. font-size: 13px;
  18. color: gray;
  19. }
  20. .related{
  21. margin: 0 50rpx;
  22. }
  23. .item{
  24. display: flex;
  25. margin-bottom: 20rpx;
  26. width: 100%;
  27. }
  28. .item .image{
  29. width: 40%;
  30. height: 150rpx;
  31. margin-right: 20rpx;
  32. position: relative;
  33. }
  34. .item image{
  35. width: 100%;
  36. height: 100%;
  37. border-radius: 5px;
  38. }
  39. .item .relatedInfo{
  40. width: 58%;
  41. }
  42. .item .relatedInfo .title{
  43. font-size: 15px;
  44. }
  45. .item .relatedInfo .user{
  46. font-size: 12px;
  47. margin-top: 20rpx;
  48. color: gray;
  49. }
  50. .imageInfo {
  51. width: 100%;
  52. position: absolute;
  53. bottom: 10rpx;
  54. color: white;
  55. display: flex;
  56. justify-content: flex-end;
  57. font-size: 24rpx;
  58. }
  59. .imageInfo .count{
  60. padding-left: 36rpx;
  61. padding-right: 10rpx;
  62. position: relative;
  63. }
  64. .imageInfo .count::before{
  65. content: "";
  66. position: absolute;
  67. left: -2rpx;
  68. top: 4rpx;
  69. width: 30rpx;
  70. height: 24rpx;
  71. background-size: cover;
  72. background-image: url("");
  73. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/77845
推荐阅读
相关标签
  

闽ICP备14008679号