真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

vue實(shí)現(xiàn)吸頂、錨點(diǎn)和滾動(dòng)高亮按鈕效果

因公司后臺(tái)管理系統(tǒng)很多功能技術(shù)老舊,最近在用vue重構(gòu)公司的后臺(tái)管理系統(tǒng),在做商品管理添加商品這一塊,借鑒淘寶的添加商品的交互,需要實(shí)現(xiàn)一個(gè)簡(jiǎn)單的吸頂、錨點(diǎn)和滾動(dòng)高亮按鈕的效果。

創(chuàng)新互聯(lián)公司主營(yíng)復(fù)興網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營(yíng)網(wǎng)站建設(shè)方案,App定制開(kāi)發(fā),復(fù)興h5微信小程序定制開(kāi)發(fā)搭建,復(fù)興網(wǎng)站營(yíng)銷(xiāo)推廣歡迎復(fù)興等地區(qū)企業(yè)咨詢

需求

  1. 滾動(dòng)頁(yè)面到頂部,實(shí)現(xiàn)某元素固定到頂部效果
  2. 點(diǎn)擊某個(gè)按鈕,頁(yè)面滾動(dòng)到相應(yīng)的位置
  3. 滾動(dòng)頁(yè)面,當(dāng)?shù)竭_(dá)某個(gè)位置時(shí),高亮對(duì)應(yīng)的相關(guān)按鈕

元素吸頂實(shí)現(xiàn)方式

關(guān)于元素吸頂效果,通過(guò)查閱相關(guān)資料和相關(guān)測(cè)試,有三種方式(還有一種是jquery的方法,這里就不介紹了)

一、使用position:sticky

1. 什么是position:sticky?

粘性定位元素相當(dāng)于position:relative和position:sticky的結(jié)合體,受限于父級(jí)元素,在不同的條件下呈現(xiàn)出不同的頁(yè)面效果

2. 如何使用sticky?

sticky元素效果完全受限于父級(jí)元素,使用條件:

1.sticky元素的父元素的overflow只能設(shè)置為visible,否則會(huì)導(dǎo)致沒(méi)有粘滯效果

2.sticky元素的父元素不能設(shè)置固定的高度,否則會(huì)導(dǎo)致沒(méi)有粘滯效果

3.sticky滿足條件變成fixed定位時(shí),與標(biāo)準(zhǔn)fixed元素不一樣,不會(huì)脫離文檔流

4.sticky 定位的元素不能添加一個(gè)只包含自身的父元素,會(huì)導(dǎo)致沒(méi)有粘滯效果

5.同一個(gè)父級(jí)元素中的sticky元素,如果定位值相等,則會(huì)重疊,如果屬于不同父級(jí)元素中,則會(huì)擠掉之前的元素,形成依次占位的效果 具體實(shí)現(xiàn)效果如下:

.sticky-box{ 
 position: sticky; 
 position: -webkit-sticky;
 top: 60px; //可通過(guò)js動(dòng)態(tài)設(shè)置
}

3.兼容性

通過(guò)查看can i use 可以看到相關(guān)的兼容性:

可以看出這個(gè)屬性的兼容性不是那么好,如果項(xiàng)目需要兼容到ie11等的話,就不是那么適用了

二、使用offsetTop**

HTMLElement.offsetTop 為只讀屬性,它返回當(dāng)前元素相對(duì)于其 offsetParent 元素的頂部?jī)?nèi)邊距的距離。因此我們需要注意的是,在監(jiān)聽(tīng)頁(yè)面滾動(dòng)的過(guò)程中,需要將定位父級(jí)元素的偏移量也計(jì)算在內(nèi),可以如下寫(xiě)法:

//獲取當(dāng)前元素的offsetTop
  getOffsetTop(obj) {
   let offsetTop = 0;
   while (obj != window.document.body && obj != null) {
    offsetTop += obj.offsetTop;
    obj = obj.offsetParent;
   }
   return offsetTop;
  }

通過(guò)在vue的mounted生命周期函數(shù)中添加監(jiān)聽(tīng)事件滾動(dòng)的事件:

mounted() {
  /**通過(guò)給變成固定定位的元素添加一個(gè)同等高度的父元素,防止該元素變成固定定位時(shí),脫離文檔流導(dǎo)致的頁(yè)面抖動(dòng) */
  this.tabsHeight = this.$refs.elTabs.offsetHeight;
  window.addEventListener("scroll", this.handleScroll);
 },
 destroyed() {
  //離開(kāi)該頁(yè)面需要移除這個(gè)監(jiān)聽(tīng)的事件
  window.removeEventListener("scroll", this.handleScroll);
 },
  methods: {
  /**滾動(dòng)事件 */
  handleScroll() {
    //獲取頁(yè)面滾動(dòng)條的高度
    let scrollTop = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop;
    let offsetTop = this.getOffsetTop(this.$refs.elTabs);
    this.isFixed = scrollTop > offsetTop;
  }
 }

同時(shí)如果這種吸頂方式在項(xiàng)目中會(huì)多次用到,就可以封裝成組件的形式

三、使用getBoundingClientRect().top**

還有一種更為直接的方式,可以實(shí)現(xiàn)吸頂效果,就是使用getBoundingClientRect().top來(lái)獲取元素相對(duì)于視口(瀏覽器窗口)的位置,相對(duì)于offsetTop,該方法不用考慮到吸頂元素的父級(jí)元素和頁(yè)面滾動(dòng)條的高度,直接對(duì)該元素進(jìn)行處理即可,實(shí)現(xiàn)如下: /* 滾動(dòng)事件 / handleScroll() { / * getBoundingClientRect().top 獲取某元素距離瀏覽器頂部的高度,不包含滾動(dòng)的距離 */ let tabOffsetTop = this.$refs.stickyBox.getBoundingClientRect().top; this.isFixed = tabOffsetTop < this.offsetTop }

/**滾動(dòng)事件 */
  handleScroll() {
   /**
    * getBoundingClientRect().top 獲取某元素距離瀏覽器頂部的高度,不包含滾動(dòng)的距離
    this.offsetTop 表示的是吸頂元素距離頂部的條件值(一般項(xiàng)目需求是0)
    */
   let tabOffsetTop = this.$refs.stickyBox.getBoundingClientRect().top;
   this.isFixed = tabOffsetTop < this.offsetTop
  }

錨點(diǎn)定位

點(diǎn)擊相應(yīng)的按鈕,頁(yè)面滾動(dòng)到相應(yīng)的位置,目前我知道實(shí)現(xiàn)該功能的方式有兩種:

1. 使用a標(biāo)簽定位 2. 使用js模擬錨點(diǎn)定位

使用a標(biāo)簽定位

這是一種常見(jiàn)的定位方式,它有兩種實(shí)現(xiàn)方式:

1. 通過(guò)href屬性鏈接到指定元素的id

2.另一種是添加一個(gè) a 標(biāo)簽,再將 href 屬性鏈接到這個(gè) a 標(biāo)簽的 name 屬性

按鈕1 
 按鈕1 
 
視圖1

這種定位方式很簡(jiǎn)單,支持任意標(biāo)簽的定位,但是a標(biāo)簽的定位會(huì)改變路由的hash,如果有相關(guān)路由會(huì)進(jìn)行路由跳轉(zhuǎn)

使用js模擬錨點(diǎn)定位

通過(guò)js獲取元素的scrollTop值,使其滾動(dòng)到指定的位置,就能實(shí)現(xiàn)錨點(diǎn)定位效果,這里的tab切換選項(xiàng),用到是的element-ui的el-tabs組件,具體實(shí)現(xiàn)如下:


 
  


methods:{
 //獲取當(dāng)前元素的offsetTop
 getOffsetTop(obj) {
  let offsetTop = 0;
  while (obj != window.document.body && obj != null) {
  offsetTop += obj.offsetTop;
  obj = obj.offsetParent;
 }
  return offsetTop;
 },
 
 
 tabClick(e) {
  let _this = this;
  //獲取當(dāng)前選中的index以便后面滾動(dòng)高亮
  this.index = parseInt(e.index);
  //給定一個(gè)標(biāo)識(shí),錨點(diǎn)事件不觸發(fā)滾動(dòng)
  this.isScroll = false;
  this.isChange = false;
  //獲取當(dāng)前選中元素的top值(給元素綁定對(duì)應(yīng)的ref值)
  let offsetTop = this.getOffsetTop(this.$refs[this.activeName]);
  let scrollTop = offsetTop - this.fixedHeight;
  window.scrollTo({
    top: scrollTop
  });
}

不得不提的一個(gè)方法就是scrollIntoView,Element.scrollIntoView() 方法讓當(dāng)前的元素滾動(dòng)到瀏覽器窗口的可視區(qū)域內(nèi),同時(shí)還支持動(dòng)態(tài)效果,但是不支持配置滾動(dòng)到距離頂部的距離,會(huì)出現(xiàn)遮罩現(xiàn)象,但是很適合做會(huì)到頂部的功能

滾動(dòng)高亮按鈕

當(dāng)用戶滾動(dòng)內(nèi)容區(qū)時(shí),高亮距離按鈕組件最近的那個(gè)元素所對(duì)應(yīng)的按鈕。 通過(guò)監(jiān)聽(tīng)滾動(dòng)事件,獲取當(dāng)前選中的tab的offsetTop值和當(dāng)前頁(yè)面的scrollTop值,判斷向上或者向下滾動(dòng),做出不同的處理,具體如下:

//頁(yè)面滾動(dòng)要做的事情
handleScroll() {
  let scrollTop =
    window.pageYOffset ||
    document.documentElement.scrollTop ||
    document.body.scrollTop;
   this.scrollTop = scrollTop;
   
   if (!this.isScroll) return;
   /**
    * scrollTop 頁(yè)面的滾動(dòng)條的高度
    * offsetTop 當(dāng)前選中的tab元素的offsetTop
    * offsetHeight 當(dāng)前選中元素的高度
    */
   let offsetTop = this.getOffsetTop(this.$refs[this.activeName]);
   let offsetHeight = this.$refs[this.activeName].offsetHeight;
   let actuaTop = scrollTop + this.fixedHeight;
   let length = this.tabList.length;
   /**
    * 頁(yè)面滾動(dòng)中根據(jù)相應(yīng)位置變換選中tab
    */
   if (actuaTop < offsetTop && this.index > 0) {
    this.index = this.index - 1;
    this.activeName = this.tabList[this.index].key;
   } else if (this.index < length && actuaTop > offsetTop + offsetHeight) {
    this.index = this.index + 1;
    this.activeName = this.tabList[this.index].key;
 }
}

性能優(yōu)化

頁(yè)面中讀取屬性會(huì)導(dǎo)致頁(yè)面reflow(下次會(huì)對(duì)導(dǎo)致頁(yè)面reflow和repaint 的操作做一個(gè)總結(jié)),過(guò)度的reflow會(huì)導(dǎo)致頁(yè)面性能下降,所以我們應(yīng)該盡量減少reflow的次數(shù),以便給用戶更好的體驗(yàn)。 如果產(chǎn)品可以接受效果有延遲,就可以使用節(jié)流函數(shù)控制在一定時(shí)間內(nèi)只執(zhí)行一次函數(shù)(節(jié)流函數(shù)可以使用lodash.js 封裝好的 throttle 方法)

總結(jié)

寫(xiě)到這里,需求中的三個(gè)功能都已經(jīng)實(shí)現(xiàn),也許還存在更好的方案,但是通過(guò)這次實(shí)現(xiàn)這三個(gè)需求,如果大家有其他更好的方法,歡迎留言補(bǔ)充,但我也從中學(xué)習(xí)到了一些東西 1.position:sticky的用法和使用條件

2.scrollTop、offsetTop等元素的相關(guān)屬性、getBoundingClientRect()用法和 scrollTo、scrollIntoView的用法 3.錨點(diǎn)時(shí)間和滾動(dòng)高亮事件導(dǎo)致的沖突處理等

以上所述是小編給大家介紹的vue實(shí)現(xiàn)吸頂、錨點(diǎn)和滾動(dòng)高亮按鈕效果,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)創(chuàng)新互聯(lián)網(wǎng)站的支持!
如果你覺(jué)得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!


網(wǎng)頁(yè)名稱:vue實(shí)現(xiàn)吸頂、錨點(diǎn)和滾動(dòng)高亮按鈕效果
轉(zhuǎn)載注明:http://weahome.cn/article/ggsjes.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部