用Vue来实现音乐播放器(10):Scroll组件的抽象和应用
了解better-scroll什么时候是需要refresh计算的??通常我们遇到的better-scroll不能滚动的问题的根源是什么??better-scroll的渲染原理是:根据初始化的时机 或者调用refresh()的时机的那个时候的scroll的父元素的高度和子元素的高度之差去做一个计算 计算出他可以滚动的位置 所以我们去实例化或者refresh这个scroll的时候 一定要保证 这个dom是已经渲染好的 我们能正确计算到高度的
如果我们有数据变化以及dom变化的场景的时候 我们一定要去让better-scroll重新refresh() 然后我们封装了scroll组件之后 我们就不用在外部掉获取到数据之后 再this.$refs.scroll.refresh() 我们只需要把数据传递给scroll scroll就能监听到数据变化 scroll组件就能自己计算高度
<template>
<div class="recommend">
<scroll class="recommend-content">
<!-- //scroll插件作用于他的第一个子元素 -->
<!-- 局部轮播的父级 -->
<div> <div class="slider-wrapper" v-if="recommend.length"> //这里也是为了在初始化实例和refresh()之前确保dom已经渲染好了
<slider>
<!-- 填写插槽 -->
<div v-for="(item,index) in recommend" :key="index">
<a :href="item.linkUrl">
<img :src="item.picUrl">
</a>
</div>
</slider>
</div>
<div class="recommend-list">
<h1 class="list-title">热门歌单推荐</h1>
<div class="mod-playlist">
<ul class="playlist-list">
<li v-for="(item,index) in discList" :key="index" class="playlist-item">
<div class="playlist-item-box">
<div class="playlist-cover mod-cover">
<a href="#">
<img :src="item.imgurl" class="pic">
</a>
</div>
<h4 class="playlist-title">
<span class="playlist-title-txt">{{item.dissname}}</span>
</h4>
<div class="playlist-author">
<a href="#">{{item.creator.name}}</a>
</div> </div> </li>
</ul>
</div>
</div> </div>
</scroll>
</div>
</template>
<script> import Scroll from '../../base/scroll/scroll.vue'
import Slider from '../../base/slider/slider.vue'
import {getRecommend,getDiscList} from '../../api/recommend.js'
import { ERR_OK } from '../../api/config.js' export default {
data(){//为了将实例里面的轮播图数据与dom相关 要有一个data()方法
return {
recommend:[],
discList:[]
} },
created(){//一般在这个生命钩子里面加载数据
this._getRecommend(),
this._getDiscList()
},
methods:{
_getRecommend(){
getRecommend().then((res)=>{//这个res就是我们抓取到的数据对象
if(res.code===ERR_OK){
// console.log(res.data.slider)
this.recommend=res.data.slider
}
}) },
_getDiscList(){
getDiscList().then((res)=>{
if(res.code===ERR_OK){
// console.log(res.data.list)
this.discList=res.data.list
}
})
},
}, components:{
Slider,Scroll
}
}
<template>
<div ref="wrapper">
<slot></slot>
</div>
</template> <script> //将better-scroll相关都放在这里 可以实现代码的复用 import BScroll from 'better-scroll' export default {
props: {
probeType: {
type: Number,
default: 1
},
click: {
type: Boolean,
default: true
},
data: {//因为我们是滚动的 所以数据会改变 需要一个数组来监听发生的数据变化 从而可以refresh()scroll之后 重新计算滚动的高度
type:Array,
default:null
}
},
//vue在mounted()这个生命周期钩子里面加载dom到页面上 一般在这个钩子里面还可以初始化插件 以及操作dom //要写个定时器,,因为一般数据的请求是异步的 那么异步的数据请求和同样也是异步的mounted()的顺序是什么呢?
//可能是mounted()先执行 那么就会在数据更新之前将dom挂载了 这样会产生虚拟dom
//所以我们要保证dom的挂载是在数据更新之后 所以使用setTimeout() 或者是this.$nextTick
//可以用nextTick 将回调延迟到下次dom更新循环之后执行 在修改数据之后立即使用它,然后等待dom更新 //浏览器渲染一般是17ms 这里延迟到20ms mounted(){
setTimeout(()=>{
this._initScroll(); //better-scroll的渲染原理是 //根据初始化的时机 或者调用refresh()的时机的那个时候的scroll的父元素的高度和子元素的高度之差去做一个计算
//计算出他可以滚动的位置 所以我们去实例化或者refresh这个scroll的时候 一定要保证 这个dom是已经渲染好的 我们能正确计算到高度的
//所以我们在这里一定要保证我们能计算到正确的高度 因此我们要判断一下 此时dom有没有渲染好 在recommend.vue组件中确认
},20)
},
methods:{
_initScroll(){
//在这里先判断一下this.$refs.wrapper是不是undefined
//如果是undefined 那么BScroll类由于第一个参数为undefined会报错
if(!this.$refs.wrapper){
return
}
this.scroll=new BScroll(this.$refs.wrapper,{
probeType:this.probeType,
click:this.click
})
},
enable(){
this.scroll&&this.scroll.enable()
},
disable(){
this.scroll&&this.scroll.disable()
},
refresh(){
this.scroll&&this.scroll.refresh()
}
},
//监控data的变化 如果data变化了 刷新scroll
watch:{
data(){
setTimeout(()=>{
//这里也是操作dom 只要涉及到操作dom就需要延迟
this.refresh()
},20)
}
}
}
</script> <style lang="stylus" scoped> </style>
完成以上这些操作 scroll组件已经初始化了 但是并不能滚动 这是因为 我们初始化这个scroll插件的时机是在scroll.vue组件中的mounted()里面初始化到的 然而我的scroll插件里面的数据是动态渲染拿到的 也就是一个异步操作 即mounted()挂载的前提是 也可以说是实例化或者是refresh()这个插件的前提是 动态获取到数据之后 将dom撑开了之后 能够计算到滚动的高度差之后 在挂载到页面 然后此时我们重新计算scroll 然后调用this.refresh()
才能让scroll正确的滚动
那么我们在recommend.vue中就要确保一下有数据 或者是直接父传子一个data属性 如果recommend.vue中的数据变化了 就会触发scroll.vue中的高度改变
<template>
<div class="recommend">
<!-- 这里的data可以看成是父传子的属性 即<scroll>组件中的props属性下的data
当这里的data改变了的话 scroll组件的data也会改变 因为监听(watch)了data 所以data改变之后就会刷新scroll
即重新计算滚动的高度等等
如果这里先没有传一个data过去 那么我们可能在初始化scroll组件里面的时候可能拿不到数据 如果没有数据
那么scroll组件可能认为不需要滚动
--> <scroll class="recommend-content" :data="discList" ref="scroll">
<!-- //scroll插件作用于他的第一个子元素 -->
<!-- 局部轮播的父级 -->
<div> <!-- 注意这里要写当轮播图的数据都拿到了之后 即slider.vue组件里面的mounted()方法能执行成功之后
执行下面操作
--> <!-- 因为我们的recommend是通过异步拿到的 可能会有延时 但是如果不加上v-if的话 我们可能在没有拿到数据的情况下就
把插槽插入进去了 那么slider.vue里面的mounted()也就执行了
-->
<div class="slider-wrapper" v-if="recommend.length">
<slider>
<!-- 填写插槽 -->
<div v-for="(item,index) in recommend" :key="index">
<a :href="item.linkUrl">
<img :src="item.picUrl" @load="loadImage" class="needsclick">
</a>
</div>
</slider>
</div>
<div class="recommend-list">
<h1 class="list-title">热门歌单推荐</h1>
<div class="mod-playlist">
<ul class="playlist-list">
<li v-for="(item,index) in discList" :key="index" class="playlist-item">
<div class="playlist-item-box">
<div class="playlist-cover mod-cover">
<a href="#">
<img v-lazy="item.imgurl" class="pic">
</a>
</div>
<h4 class="playlist-title">
<span class="playlist-title-txt">{{item.dissname}}</span>
</h4>
<div class="playlist-author">
<a href="#">{{item.creator.name}}</a>
</div> </div> </li>
</ul>
</div>
</div> </div>
</scroll>
</div>
</template>
用Vue来实现音乐播放器(10):Scroll组件的抽象和应用的更多相关文章
- vue小练习--音乐播放器
1 首先建一个文件夹 放几首歌曲 2 看代码 1)基本版本 <!DOCTYPE html> <html lang="zh-CN"> <head> ...
- 用Vue来实现音乐播放器(三十八):歌词滚动列表的问题
1.频繁切换歌曲时,歌词会跳来跳去 原因: // 歌词跳跃是因为内部有一个currentLyric对像内部有一些功能来完成歌词的跳跃 //每个currentLyric能实现歌曲的播放跳到相应的位置 是 ...
- 用Vue来实现音乐播放器(十六):滚动列表的实现
滚动列表是一个基础组件 他是基于scroll组件实现的 在base文件夹下面创建一个list-view文件夹 里面有list-view.vue组件 <template> < ...
- Vue实战:音乐播放器(一) 页面效果
先看一下效果图 首页 歌单详情页 歌手列表 歌手详情页 排行页面 榜单的详情页(排序样式) 搜索页面 搜索结果 播放器内核 歌词自动滚动 播放列表 用户中心
- 用Vue来实现音乐播放器(十四):歌手数据接口抓取
第一步:在api文件夹下创建一个singer.js文件 返回一个getSingerList()方法 使他能够在singer.vue中调用 import jsonp from '../common/j ...
- 用Vue来实现音乐播放器(九):歌单数据接口分析
z这里如果我们和之前获取轮播图的数据一样来获取表单的数据 发现根本获取不到 原因是qq音乐在请求头里面加了authority和refer等 但是如果我们通过jsonp实现跨域来请求数据的话 是根本 ...
- 用Vue来实现音乐播放器(八):自动轮播图啊
slider.vue组件的模板部分 <template> <div class="slider" ref="slider"> <d ...
- 用Vue来实现音乐播放器(二十三):音乐列表
当我们将音乐列表往上滑的时候 我们上面的歌手图片部分也会变小 当我们将音乐列表向下拉的时候 我们的图片会放大 当我们将音乐列表向上滑的时候 我们的图片有一个高斯模糊的效果 并且随着我们的列 ...
- 用Vue来实现音乐播放器(二十一):歌手详情数据抓取
第一步:在api文件夹下的singer.js中抛出getSingerDetail方法 第二步:在singer-detail.vue组件中引入api文件夹下的singer.js和config.js 第三 ...
随机推荐
- Scrapy 教程(五)-分页策略
scrapy 爬取分页网站的策略 1. 检测当前页是否存在“下一页” 2. 如果存在,把“下一页”的链接交给本方法或者其他方法 3. 如果不存在,结束 图示 示例代码 def parse(self, ...
- 使用axios发送ajax请求
1.安装 npm install axios 2.在Home.vue中引入 import axios from 'axios' export default { name: 'Home', c ...
- 2019-11-22-Roslyn-在多开发框架让-msbuild-的-Target-仅运行一次
title author date CreateTime categories Roslyn 在多开发框架让 msbuild 的 Target 仅运行一次 lindexi 2019-11-22 09: ...
- Java设计模式之——代理设计模式
1.什么是代理设计模式 所谓代理模式是指客户端并不直接调用实际的对象,而是通过调用代理,来间接的调用实际的对象. 这里举一个栗子:就拿我们平时租房子来举例子,好比租客和房主之间的关系,我们租房子往往不 ...
- 关于Mysql5.6 Failed to open file error2的记录
今天在执行mysql命令行下的\.命令时,它总说Failed to open file “...........”error2 找了半天原因,最后发现是百度云在往那个文件夹里下载东西,所以访问文件的时 ...
- 005-(已测试成功的方案)kickstart模式实现批量安装centos7.x系统
1.1 安装系统的方法 l 光盘(ISO文件,光盘的镜像文件)===>>每一台物理机都得给一个光驱,如果用外置光驱的话,是不是每台机器都需要插一下 l U盘:ISO镜像刻录到U盘==& ...
- hdu 5890 01背包 bitset
注意不能每个T都mem 不然会T #include <bits/stdc++.h> #define PI acos(-1.0) #define mem(a,b) memset((a),b, ...
- 最简单的注册美区Apple ID方法
最简单方法注册苹果美区Apple ID 1.打开苹果官网链接 苹果官网 2. 点击右下角的 United States 3. 点击图片中的选项 4.点击右上角的选项创建新的Apple ID 注意是新的 ...
- pyqt5-橡皮筋控件QRubberBand
提供一个矩形或线来指示选择或边界 一般结合鼠标事件一同协作 继承于 QWidget import sys from PyQt5.QtWidgets import QApplication, QWidg ...
- LeetCode--128--最长连续序列(python)
给定一个未排序的整数数组,找出最长连续序列的长度. 要求算法的时间复杂度为 O(n). 示例: 输入: [100, 4, 200, 1, 3, 2]输出: 4解释: 最长连续序列是 [1, 2, 3, ...