在大多數電商場景中,頁面都會有類目切換加上商品列表的部分,頁面大概會長這樣


每次寫類似場景的時候,都需要去為類目商品列表寫很多邏輯,為了提高開發效率我決定將這一部分抽離成組件。
實現
1.樣式
所有tab欄的樣式和商品列表的樣式都提供插槽,供業務自己定制
2.變量
isTabFixed: false,//是否吸頂
tab: 1,//當前tab
page: 1,//當前頁數
listStatus: {
finished: false,//是否已是最后一頁
loading: false,//是否加載中
},
items: [],//商品數組
tabMap: [],//tab列表數組
cache:{},//緩存
listName: '',//商品列表名稱
tabName: '',//tab列表名稱
apiName: '',//api方法名稱
queryName: '',//api請求參數名稱
3.緩存設計
為了減少消耗,已經請求過的商品列表我都將他們緩存下來
_addCache(proList) {
cache[this.tab] = {
finished: this.listStatus.finished,
page: this.page,
list: proList,
};
},
4.請求數據
_getList(type) {
let data = {};
data[this.queryName] = this.tab;
data.page = this.page;
this.$http[this.apiName](data)
.then((res) => {
this.listStatus.finished = !res.has_more;//更新是否是最后一頁的狀態
this._handleData(res.items);//處理得到的商品列表
})
.catch((err) => {
note(err.message || '出錯啦');
});
},
_handleData(proList) {
if (this.page === 1) {//表示是tab切換時請求的數據,所以直接將items的指向切換
this.items = proList;
} else {//因為是翻頁,所以需要把數據拼接
this.items = this.items.concat(proList);
}
this.$store.setData(this.listName, this.items);//把數據更新給父組件
this._addCache(this.items);//把數據加入緩存
},
5.操作
邏輯部分主要分兩塊:一是列表翻頁,二是tab相關
列表翻頁
這部分的邏輯比較簡單,主要分兩點
_loadmore() {
this.page = this.page + 1;
this._getList();
},
其實對于手機列表的上拉翻頁操作,還有很多的點要去注意,比如如何去避免連續請求等,由于我將這些交給了另一個專注列表渲染的組件,這里就不需要再去考慮這些操作。
tab相關
tab切換
tab切換的時候主要是兩點
changeTab(id) {
this.tab = id;
this.$store.setData(this.tabName, this.tab);//將tab的指向同步給父組件
this._scrollToTab();//視圖回到頂部
if (cache[this.tab]) {
const target = cache[this.tab];
this.listStatus.finished = target.finished;
this.page = target.page;
this.items = target.list;
this.$store.setData(this.listName, this.items);//將商品列表同步給父組件
} else {
this.page = 1;
this._getList();
}
},
//視圖回到頂部
_scrollToTab() {
let scrollTop = document.documentElement.scrollTop || document.body.scrollTop;
let productsTop = this.$refs.products.$el.offsetTop;//商品列表距離頂部的距離
let topHeight = this.$refs.tabNav.offsetHeight;//tab欄的高度
if (scrollTop > productsTop) {
window.scrollTo(0, productsTop - topHeight);
}
},
吸頂
在吸頂的時候,需要對tab欄的樣式做一些小小的變動,所以需要一個變量來知道是否吸頂了
handleNavFixed() {
this.stickyTop = this.$store.getData('stickyTop') || 0;//吸頂的高度
window.onscroll = (e) => {
let top = document.documentElement.scrollTop || document.body.scrollTop;
let tabHeight = this.$refs.tabNav.offsetTop - top - 1;
if (tabHeight <= this.stickyTop && !this.isTabFixed) {//結合判斷為了避免重復計算
this.isTabFixed = true;
}
if (tabHeight >= this.stickyTop && this.isTabFixed) {
this.isTabFixed = false;
}
};
},
6.組件相關
作為一個組件,不同點在于需要做到的是可用性和通用性。 對于組件如何更好的得到父組件的值并把更新的值傳回父組件方面,是我在開發過程中的一個卡點,最后我用了一套基于vue的組件通行機制。這種通行機制的實現網上很多,這里就不詳細說了。通行機制主要有兩個功能
由于將tab列表都作為插槽傳入,所以初始數據并不需要關心,需要關心的只是更新數據。
對于不同的頁面,tab列表的名稱,tab定位的名稱,商品列表的名稱,接口的名稱,請求接口的參數都可以會不一樣,所以我在這里將這些項作為參數,在初始化這個組件的時候需要傳入
//組件部分
init(data = {}) {
this.listStatus.finished = !data.hasMore;
this.tabName = data.tabName;
this.listName = data.listName;
this.apiName = data.apiName;
this.queryName = data.queryName;
this.handleNavFixed();//判斷是否吸頂
},
//調用組件
this.$bus.emit('tab-list.init', {
tabName: 'tab',
listName: 'items',
apiName: 'homeList',
queryName: 'tab_id',
hasMore: this.hasMore,
});
總結
以上所述是小編給大家介紹的基于vue的tab-list類目切換商品列表組件的示例代碼,希望對大家有所幫助!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。