欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

React虛擬列表的實(shí)現(xiàn)代碼

 更新時(shí)間:2023年08月27日 09:22:24   作者:Kakarotto  
最近看了vueuse的useVirtualList的實(shí)現(xiàn)方式,發(fā)現(xiàn)虛擬滾動(dòng)效果不錯(cuò),就嘗試著同樣的寫法改成react版本,虛擬列表主要包含三部分組成,offset,viewcapacity,overscan,本文就給大家介紹一下React虛擬列表的實(shí)現(xiàn),需要的朋友可以參考下

最近看了vueuse的useVirtualList的實(shí)現(xiàn)方式,發(fā)現(xiàn)虛擬滾動(dòng)效果不錯(cuò),就嘗試著同樣的寫法改成react版本。

虛擬列表主要包含三部分組成,示意圖如下:

定義一個(gè) useVirtualList 的 hook 需要接收兩個(gè)參數(shù),一是接收所有的數(shù)據(jù),二是定義一些配置參數(shù);需要返回的四個(gè)參數(shù),一是當(dāng)前加載的列表,二是獲取容器的 ref 和容器滾動(dòng)事件,三是快速定位的方法,四是內(nèi)層容器的樣式?;镜拇a結(jié)構(gòu)如下:

export type UseVirtualListOptions = {
	// 如果是每條數(shù)據(jù)的高度相同為number 否則通過方法計(jì)算高度
	itemHeight: number | ((index: number) => number);
	overscan?: number;
};
export const useVirtualList = (
	list: AllItems[],
	options: UseVirtualListOptions
): UseVirtualListReturn => {
	const containerRef = useRef(null);
	const [currentList, setCurrentList] = useState<CurrentList[]>([]);
	const [wrapperStyle, setWrapperStyle] = useState({
		width: "100%",
		height: "0px",
		marginTop: "0px",
	});
	return {
		list: currentList,
		containerProps: {
			ref: containerRef,
			onScroll: () => {},
		},
		scrollTop,
		wrapperStyle,
	};
};

因?yàn)樾枰萜鞯母叨群蜐L動(dòng)距離來計(jì)算加載的內(nèi)容,因此需要在 dom 掛載完成后執(zhí)行。

1、計(jì)算容器的的高度能顯示的數(shù)據(jù)

如果 itemHeight 類型為 number,則可以通過容器高度 / itemHeight 來計(jì)算容器能顯示多少條數(shù)據(jù),否則需要把上一次開始索引作為起始值,預(yù)估滾動(dòng)之后能夠在可視區(qū)域顯示多少數(shù)據(jù)。

const getViewCapacity = (containerHeight: number) => {
	if (typeof itemHeight === "number") {
		return Math.ceil(containerHeight / itemHeight);
	}
	let sum = 0;
	let capacity = 0;
	const { start = 0 } = state;
	console.log(start);
	for (let i = start; i < list.length; i++) {
		const height = itemHeight(i);
		sum += height;
		if (sum >= containerHeight) {
			capacity = i;
			break;
		}
	}
	return capacity - start;
};

2、計(jì)算滾動(dòng)偏移的數(shù)據(jù)量

如果 itemHeight 類型為 number,則可以通過容器高度 / itemHeight 來計(jì)算滾動(dòng)偏移的數(shù)據(jù)量,否則需要對(duì)滾動(dòng)偏移的 dom 高度進(jìn)行累加,計(jì)算出偏移的數(shù)據(jù)量。

const getOffset = (scrollTop: number) => {
	if (typeof itemHeight === "number") {
		return Math.floor(scrollTop / itemHeight) + 1;
	}
	let sum = 0;
	let offset = 0;
	for (let i = 0; i < list.length; i++) {
		const height = itemHeight(i);
		sum += height;
		if (sum >= scrollTop) {
			offset = i;
			break;
		}
	}
	return offset + 1;
};

3、計(jì)算滾動(dòng)時(shí)需要加載的數(shù)據(jù)

需要加載的數(shù)據(jù)開始位置 = 滾動(dòng)偏移量 - 預(yù)加載的數(shù)量
需要加載的數(shù)據(jù)結(jié)束位置 = 滾動(dòng)偏移量 + 可視區(qū)域顯示數(shù)據(jù)量 + 預(yù)加載數(shù)據(jù)量
如果開始位置 < 0 則從 0 開始,結(jié)束位置大于總數(shù)組長(zhǎng)度,則結(jié)束位置為總數(shù)據(jù)的長(zhǎng)度

const from = offset - overscan;
const to = offset + viewCapacity + overscan;
state.start = from < 0 ? 0 : from;
state.end = to > list.length ? list.length : to;
setCurrentList(() => {
	return list.slice(state.start, state.end).map((ele, index) => ({
		data: ele,
		index: index + state.start,
	}));
});

4、計(jì)算列表容器的高度和滾動(dòng)的高度

列表容器的高度 = 數(shù)據(jù)長(zhǎng)度 * 預(yù)估每條數(shù)據(jù)的高度 - 滾動(dòng)偏移的高度

因?yàn)榧虞d的數(shù)據(jù)不是每次都從 0 開始,但每次渲染是從頂部開始的,所以滾動(dòng)到高度大于數(shù)據(jù)的高度時(shí),數(shù)據(jù)位于可視區(qū)域的上方,此時(shí)需要設(shè)置 marginTop,讓加載的數(shù)據(jù)顯示在可視區(qū)域的頂部,但如果總高度不變,數(shù)據(jù)全部加載完成后,底部會(huì)有大面積留白。

const computedWrapperStype = () => {
	const offsetTop = getDistanceTop(state.start);
	setWrapperStyle({
		width: "100%",
		height: `${totalHeight() - offsetTop}px`,
		marginTop: `${offsetTop}px`,
	});
};

5、添加快速定位功能

當(dāng)在輸入框數(shù)據(jù)要定位的索引值時(shí),可以根據(jù)索引值計(jì)算出可視區(qū)域上方需要偏移的高度,然后重新調(diào)用計(jì)算需要加載數(shù)據(jù)開始、結(jié)束位置的方法,即可實(shí)現(xiàn)快速定位。

const scrollTop = (index: number) => {
	if (containerRef.current) {
		(containerRef.current as HTMLDivElement).scrollTop = getDistanceTop(index);
		calculateRange();
	}
};

6、源碼地址

https://stackblitz.com/edit/vitejs-vite-oy7obm?file=src%2FApp.tsx

到此這篇關(guān)于React虛擬列表的實(shí)現(xiàn)代碼的文章就介紹到這了,更多相關(guān)React虛擬列表內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 一文帶你搞懂React的函數(shù)組件

    一文帶你搞懂React的函數(shù)組件

    React中函數(shù)式組件的基本意義是,組件實(shí)際上是一個(gè)函數(shù),不是類,下面就來給大家介紹一下關(guān)于React中函數(shù)組件的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • react中useState改變值不渲染的解決方式

    react中useState改變值不渲染的解決方式

    這篇文章主要介紹了react中useState改變值不渲染的解決方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • react-native ListView下拉刷新上拉加載實(shí)現(xiàn)代碼

    react-native ListView下拉刷新上拉加載實(shí)現(xiàn)代碼

    本篇文章主要介紹了react-native ListView下拉刷新上拉加載實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • Redis數(shù)據(jù)結(jié)構(gòu)面試高頻問題解析

    Redis數(shù)據(jù)結(jié)構(gòu)面試高頻問題解析

    這篇文章主要為大家介紹了Redis數(shù)據(jù)結(jié)構(gòu)高頻面試問題解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-06-06
  • React中用@符號(hào)編寫文件路徑實(shí)現(xiàn)方法介紹

    React中用@符號(hào)編寫文件路徑實(shí)現(xiàn)方法介紹

    在Vue中,我們導(dǎo)入文件時(shí),文件路徑中可以使用@符號(hào)指代src目錄,極大的簡(jiǎn)化了我們對(duì)路徑的書寫。但是react中,要想實(shí)現(xiàn)這種方式書寫文件路徑,需要寫配置文件來實(shí)現(xiàn)
    2022-09-09
  • 簡(jiǎn)析React Native startReactApplication 方法

    簡(jiǎn)析React Native startReactApplication 方法

    這篇文章主要介紹了React Native startReactApplication 方法簡(jiǎn)析,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-09-09
  • ReactDOM.render在react源碼中執(zhí)行原理

    ReactDOM.render在react源碼中執(zhí)行原理

    這篇文章主要為大家介紹了ReactDOM.render在react源碼中執(zhí)行原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • react+ant design實(shí)現(xiàn)Table的增、刪、改的示例代碼

    react+ant design實(shí)現(xiàn)Table的增、刪、改的示例代碼

    這篇文章主要介紹了react+ant design實(shí)現(xiàn)Table的增、刪、改的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-12-12
  • React+Koa實(shí)現(xiàn)文件上傳的示例

    React+Koa實(shí)現(xiàn)文件上傳的示例

    這篇文章主要介紹了React+Koa實(shí)現(xiàn)文件上傳的示例,幫助大家更好的理解和學(xué)習(xí)使用React,感興趣的朋友可以了解下
    2021-04-04
  • react+typescript中使用echarts的實(shí)現(xiàn)步驟

    react+typescript中使用echarts的實(shí)現(xiàn)步驟

    本文主要介紹了react+typescript中使用echarts的實(shí)現(xiàn)步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-08-08

最新評(píng)論