亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

Vue2.0 實現歌手列表滾動及右側快速入口功能

2019-11-19 13:18:59
字體:
來源:轉載
供稿:網友

1 歌手列表

歌手列表頁類似于手機通訊錄,我們也將其作為一個基礎組件獨立出來,這部分的邏輯比較簡單,這里不做過多的講解

// base/listview/listview.vue<template>  <scroll class="listview" :data="data">    <ul>      <li v-for="(group, index) in data" :key="index" class="list-group">        <h2 class="list-group-title">{{group.title}}</h2>        <uL>          <li v-for="(item, index) in group.items" :key="index" class="list-group-item">            <img class="avatar" v-lazy="item.avatar">            <span class="name">{{item.name}}</span>          </li>        </uL>      </li>    </ul>  </scroll></template><script type="text/ecmascript-6">  import Scroll from 'base/scroll/scroll'  export default {    props: {      data: {        type: Array,        default: () => []      }    },    components: {      Scroll    }  }</script><style scoped lang="stylus" rel="stylesheet/stylus">  @import "~common/stylus/variable"  .listview    position: relative    width: 100%    height: 100%    overflow: hidden    background: $color-background    .list-group      padding-bottom: 30px      .list-group-title        height: 30px        line-height: 30px        padding-left: 20px        font-size: $font-size-small        color: $color-text-l        background: $color-highlight-background      .list-group-item        display: flex        align-items: center        padding: 20px 0 0 30px        .avatar          width: 50px          height: 50px          border-radius: 50%        .name          margin-left: 20px          color: $color-text-l          font-size: $font-size-medium    .list-shortcut      position: absolute      z-index: 30      right: 0      top: 50%      transform: translateY(-50%)      width: 20px      padding: 20px 0      border-radius: 10px      text-align: center      background: $color-background-d      font-family: Helvetica      .item        padding: 3px        line-height: 1        color: $color-text-l        font-size: $font-size-small        &.current          color: $color-theme          font-weight: bolder    .list-fixed      position: absolute      top: -1px      left: 0      width: 100%      .fixed-title        height: 30px        line-height: 30px        padding-left: 20px        font-size: $font-size-small        color: $color-text-l        background: $color-highlight-background    .loading-container      position: absolute      width: 100%      top: 50%      transform: translateY(-50%)</style>// singer.vue<template> <div class="singer">  <list-view :data="singerList"></list-view> </div></template><script type="text/ecmascript-6"> import ListView from 'base/listview/listview' export default {  ...  components: {   ListView  } }</script>

 

運行結果

2 右側快速入口_點擊滾動

同樣是類比于手機通訊錄,懸浮于屏幕右側的 A-Z 可以幫助我們快速找到對應的歌手,為此,我們需要獲取 title 的集合數組

// listview.vue<div class="list-shortcut">  <ul>    <li v-for="(item, index) in shortcutList" :key="index" class="item">{{item}}</li>  </ul></div><script type="text/ecmascript-6">  export default {    ...    computed: {      shortcutList() {        return this.data.map((group) => {          return group.title.substr(0, 1)        })      }    }  }</script>

 

運行結果

快速入口出現了之后,我們接下來就為其添加點擊事件,當我們點擊對應字母時,需要獲取其索引,這里我們直接獲取 v-for 提供的 index 即可

// listview.vue<ul>  <li v-for="(item, index) in shortcutList" :key="index" @touchstart="onShortcutTouchStart($even, index)" class="item">{{item}}</li></ul>export default {  ...  methods: {    onShortcutTouchStart(e, index) {      console.log(index)    }  }}

點擊之后,我們需要頁面滾動到相應位置,這里需要擴展 scroll 組件的方法,這里擴展的方法都是來自 better-scroll 組件所封裝的方法,這里提一下 scrollToElement 方法的第二個參數是動畫時間,可根據自身需求進行設置

// scroll.vuemethods: { ... scrollTo() {  this.scroll && this.scroll.scrollTo.apply(this.scroll, arguments) }, scrollToElement() {  this.scroll && this.scroll.scrollToElement.apply(this.scroll, arguments) }}

隨后給 scroll 組件添加 ref="listview" 以及歌手列表添加 ref="listGroup" 方便我們調用

// listview.vueexport default {  ...  methods: {    onShortcutTouchStart(e, index) {      this.$refs.listview.scrollToElement(this.$refs.listGroup[index], 0)    }  }}

 

運行結果

3 右側快速入口_滑動滾動

當我們的手指在右側快速入口上滑動時,歌手列表也會同步進行滾動,當我們滾動右側快速入口時,我們需要阻止歌手列表滾動,以及瀏覽器原生滾動,所以要使用 @touchmove.stop.prevent 阻止冒泡,并且在 onShortcutTouchStart 事件中記錄觸碰點的初始位置,以及 onShortcutTouchMove 事件中觸碰點的位置,通過兩個位置的像素差,來滾動歌手列表

// listview.vue<div class="list-shortcut" @touchmove.stop.prevent="onShortcutTouchMove">  <ul>    <li v-for="(item, index) in shortcutList" :key="index" @touchstart="onShortcutTouchStart($event, index)" class="item">{{item}}</li>  </ul></div><script type="text/ecmascript-6">  const ANCHOR_HEIGHT = 18  export default {    created() {      this.touch = {}    },    ...    methods: {      onShortcutTouchStart(e, index) {        let firstTouch = e.touches[0]        this.touch.y1 = firstTouch.pageY        this.touch.anchorIndex = index        this._scrollTo(index)      },      onShortcutTouchMove(e) {        let firstTouch = e.touches[0]        this.touch.y2 = firstTouch.pageY        let delta = (this.touch.y2 - this.touch.y1) / ANCHOR_HEIGHT | 0        let anchorIndex = this.touch.anchorIndex + delta        this._scrollTo(anchorIndex)      },      _scrollTo(index) {        this.$refs.listview.scrollToElement(this.$refs.listGroup[index], 0)      }    },    components: {      Scroll    }  }</script>

 

運行結果

4 右側快速入口_高亮設置

當歌手列表滾動時,我們想要在右側快速入口中,高亮當前顯示的 title ,這就需要我們監聽 scroll 組件的滾動事件,來獲取當前滾動的位置

// scroll.vue<script type="text/ecmascript-6"> export default {  props: {   ...   listenScroll: {    type: Boolean,    default: false   }  },  methods: {   _initScroll() {    ...    if (this.listenScroll) {     let me = this     this.scroll.on('scroll', (pos) => {      me.$emit('scroll', pos)     })    }   }  } }</script>

我們當初給參數 probeType 設的默認值為 1,即會非實時(屏幕滑動超過一定時間后)派發 scroll 事件,我們在屏幕滑動的過程中,需要實時派發 scroll 事件,所以在 listview 中將 probeType 的值設為 3

// listview.vue
<template>
    <scroll class="listview"
            :data="data"
            ref="listview"
            :probe-type="probeType"
            :listenScroll="listenScroll"
            @scroll="scroll">
        <ul>
            ...
        </ul>
        <div class="list-shortcut" @touchmove.stop.prevent="onShortcutTouchMove">
            <ul>
                <li v-for="(item, index) in shortcutList"
                    :key="index"
                    :class="{'current':currentIndex===index}"
                    @touchstart="onShortcutTouchStart($event, index)"
                    class="item">{{item}}</li>
            </ul>
        </div>
    </scroll>
</template>
<script type="text/ecmascript-6">
    export default {
        created() {
            ...
            this.listHeight = []
            this.probeType = 3
        },
        data() {
            return {
                scrollY: -1,
                currentIndex: 0
            }
        },
        methods: {
            ...
            scroll(pos) {
                this.scrollY = pos.y
            },
            _scrollTo(index) {
                this.scrollY = -this.listHeight[index]
                this.$refs.listview.scrollToElement(this.$refs.listGroup[index], 0)
            },
            _calculateHeight() {
                this.listHeight = []
                const list = this.$refs.listGroup
                let height = 0
                this.listHeight.push(height)
                for (let i = 0; i < list.length; i++) {
                    let item = list[i]
                    height += item.clientHeight
                    this.listHeight.push(height)
                }
            }
        },
        watch: {
            data() {
                this.$nextTick(() => {
                    this._calculateHeight()
                })
            },
             scrollY(newY) {
                const listHeight = this.listHeight
                // 當滾動到頂部,newY>0
                if (newY > 0) {
                    this.currentIndex = 0
                    return
                }
                // 在中間部分滾動
                for (let i = 0; i < listHeight.length - 1; i++) {
                    let height1 = listHeight[i]
                    let height2 = listHeight[i + 1]
                    if (-newY >= height1 && -newY < height2) {
                        this.currentIndex = i
                        return
                    }
                }
                // 當滾動到底部,且-newY大于最后一個元素的上限
                this.currentIndex = listHeight.length - 2
            }
        },
        components: {
            Scroll
        }
    }
</script>

 

運行結果

5 滾動固定標題

當我們滾動歌手列表頁時,希望該歌手的 title 一直顯示在頂部,并且滾動到下一個 title 時,新的 title 將舊的 title 頂替掉,這里就需要我們計算一個 title 的高度

// listview.vue
<template>
    <scroll class="listview"
            :data="data"
            ref="listview"
            :probe-type="probeType"
            :listenScroll="listenScroll"
            @scroll="scroll">
        ...
        <div class="list-fixed" ref="fixed" v-show="fixedTitle">
            <div class="fixed-title">{{fixedTitle}}</div>
        </div>
    </scroll>
</template>
<script type="text/ecmascript-6">
    import Scroll from 'base/scroll/scroll'
    const TITLE_HEIGHT = 28
    const ANCHOR_HEIGHT = 18
    export default {
        ...
        data() {
            return {
                scrollY: -1,
                currentIndex: 0,
                diff: -1
            }
        },
        computed: {
            ...
            fixedTitle() {
                if (this.scrollY > 0) {
                    return ''
                }
                return this.data[this.currentIndex] ? this.data[this.currentIndex].title : ''
            }
        },
        watch: {
            ...
            scrollY(newY) {
                ...
                for (let i = 0; i < listHeight.length - 1; i++) {
                    ...
                    if (-newY >= height1 && -newY < height2) {
                        ...
                        this.diff = height2 + newY
                        return
                    }
                }
                ...
            },
            diff(newVal) {
                let fixedTop = (newVal > 0 && newVal < TITLE_HEIGHT) ? newVal - TITLE_HEIGHT : 0
                if (this.fixedTop === fixedTop) {
                    return
                }
                this.fixedTop = fixedTop
                this.$refs.fixed.style.transform = `translate3d(0,${fixedTop}px,0)`
            }
        }
    }
</script>

 

運行結果

該章節的內容到這里就全部結束了,源碼我已經發到了 GitHub Vue_Music_06 上了,有需要的同學可自行下載

總結

以上所述是小編給大家介紹的Vue2.0 實現歌手列表滾動及右側快速入口功能,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對武林網網站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩免费av在线| 成人做爽爽免费视频| 国产精品一区=区| 亚洲一区二区三区在线免费观看| 久久久国产精品x99av| 欧美一区视频在线| 欧日韩不卡在线视频| 日韩在线免费视频观看| 亚洲jizzjizz日本少妇| 精品久久久999| 在线日韩精品视频| 日韩av电影国产| 久久亚洲影音av资源网| 亚洲精品www久久久久久广东| 在线精品视频视频中文字幕| 韩国欧美亚洲国产| 91九色视频在线| 91在线观看免费高清完整版在线观看| 国产一区二区在线免费| 亚洲人成啪啪网站| 欧美激情视频播放| 在线免费看av不卡| 富二代精品短视频| 日韩欧美在线一区| 中文字幕亚洲欧美在线| 国产一区二区精品丝袜| 午夜精品福利在线观看| 国产精品一区二区三区在线播放| 国产精品久久久久91| 一本一道久久a久久精品逆3p| 日韩第一页在线| 久久久久国产视频| 亚洲欧美自拍一区| 国产伦精品一区二区三区精品视频| 亚洲精品视频播放| 日韩中文字幕不卡视频| 欧美猛交免费看| 91免费视频网站| 亚洲欧美日韩国产精品| 久久久噜久噜久久综合| 国产欧美亚洲视频| 国产性猛交xxxx免费看久久| 欧美巨大黑人极品精男| 国产91精品久久久久久| 欧美成人午夜激情视频| 精品高清美女精品国产区| 亚洲加勒比久久88色综合| 亚洲桃花岛网站| 亚洲国产97在线精品一区| 色爱精品视频一区| 成人免费淫片aa视频免费| 亚洲已满18点击进入在线看片| 日韩美女毛茸茸| 国产成人亚洲精品| 日本精品中文字幕| 精品夜色国产国偷在线| 国内精品久久久久影院 日本资源| 亚洲欧美日韩另类| 欧美精品在线免费观看| 欧美激情在线观看| 日韩视频免费大全中文字幕| 国产欧美日韩中文| 欧美性生交xxxxx久久久| 日韩激情av在线播放| 精品久久久久久中文字幕| 欧美精品在线看| 福利一区视频在线观看| 51精品国产黑色丝袜高跟鞋| 中文字幕不卡在线视频极品| 国产精品一区二区三区毛片淫片| 国色天香2019中文字幕在线观看| 久久久亚洲福利精品午夜| 国产欧美欧洲在线观看| 日韩综合视频在线观看| 欧美精品少妇videofree| 91深夜福利视频| 狠狠色香婷婷久久亚洲精品| 国产精品久久久久久av福利| 久久成人一区二区| 91精品国产成人www| 日韩av在线直播| 欧美劲爆第一页| 亚洲xxx自由成熟| 亚洲qvod图片区电影| 7m精品福利视频导航| 91超碰中文字幕久久精品| 欧美视频国产精品| 38少妇精品导航| 亚洲欧美精品伊人久久| 538国产精品视频一区二区| 日韩在线视频网站| 在线观看亚洲区| 精品国产拍在线观看| 国产精品亚洲综合天堂夜夜| 欧美激情一二三| 欧美电影在线免费观看网站| 国产成人综合一区二区三区| 国产精品欧美久久久| 亚洲精品一区在线观看香蕉| 亚洲欧美国产精品久久久久久久| 国产伊人精品在线| 欧美激情一二三| 色悠久久久久综合先锋影音下载| 国产日韩精品一区二区| 久久国产精品电影| 国产精品第3页| 中日韩美女免费视频网站在线观看| 国产日韩av高清| 亚洲跨种族黑人xxx| 性欧美xxxx视频在线观看| 欧美日韩亚洲一区二区三区| 午夜美女久久久久爽久久| 欧美国产精品日韩| 亚洲精品小视频在线观看| 中文字幕精品—区二区| 欧美日韩爱爱视频| 日韩中文av在线| 亚洲欧美日韩另类| 日韩国产高清视频在线| 国产精品久久精品| 秋霞成人午夜鲁丝一区二区三区| 国产精品18久久久久久麻辣| 日本亚洲精品在线观看| 欧美男插女视频| 欧美日韩一区二区三区| 国产97在线观看| 亚洲成人精品视频| 亚洲字幕在线观看| 欧美日韩美女在线| 日韩午夜在线视频| 一区二区三区www| 在线免费观看羞羞视频一区二区| 亚洲精品久久久久中文字幕二区| 97香蕉久久超级碰碰高清版| 另类少妇人与禽zozz0性伦| 欧美在线免费看| www高清在线视频日韩欧美| 欧美乱大交做爰xxxⅹ性3| 亚洲成人在线视频播放| 91经典在线视频| 91免费人成网站在线观看18| 亚洲综合第一页| 日韩精品一区二区视频| 国产成人精品在线观看| 国产精品久久激情| www.日韩系列| 亚洲成成品网站| 亚洲午夜未删减在线观看| 91夜夜未满十八勿入爽爽影院| 日韩亚洲欧美成人| 精品无码久久久久久国产| 日韩中文第一页| 性亚洲最疯狂xxxx高清| 另类图片亚洲另类| 精品国产一区av| 国产成人涩涩涩视频在线观看| 精品亚洲一区二区三区在线观看| 日韩欧美高清在线视频| 久久久久久久久久久久久久久久久久av| 亚洲男人天堂2019| 亚洲一区二区免费在线| 成人天堂噜噜噜| 国产精品扒开腿爽爽爽视频| 动漫精品一区二区|