因公司后臺(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è)咨詢
需求
元素吸頂實(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 屬性
這種定位方式很簡(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)注明出處,謝謝!