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

React使用Canvas繪制大數(shù)據(jù)表格的實例代碼

 更新時間:2023年09月12日 11:36:51   作者:Kakarotto  
之前一直想用Canvas做表格渲染的,最近發(fā)現(xiàn)了一個很不錯的Canvas繪圖框架Leafer,api很友好就試著寫了一下,文中有詳細(xì)的代碼示例供大家參考,感興趣的小伙伴可以自己動手試試

之前一直想用Canvas做表格渲染的,最近發(fā)現(xiàn)了一個很不錯的Canvas繪圖框架Leafer,api很友好就試著寫了一下。

表格渲染主要分為四個部分,1、表頭渲染,2、表格渲染,3、滾動條渲染,4、滾動條與表格的聯(lián)動。

1、表頭渲染

表頭的通過 JSON 格式來設(shè)置的,主要包括每列的名稱、對應(yīng)的數(shù)據(jù)的鍵值、寬度、是否需要對數(shù)據(jù)進(jìn)行二次渲染。

首先需要解決的是表頭的正確渲染,這里分為兩種情況:

1、表格列都沒有設(shè)置寬度

2、表格列有設(shè)置寬度

1.1、表格列都沒有設(shè)置寬度

1.1.1、計算表格每列的寬度

這里已知的是表格的寬度,表格的列數(shù)及表格列的名稱,解決方案如下:

文本與表格寬度比率 = 表格寬度 / 表格列文本總寬度

每列寬度 = 每列表格文本寬度 * 文本與表格寬度比率

獲取文本寬度方法:

const getTextWidth = (leafer: Leafer, text: string) => {
	return leafer.canvas.measureText(text).width;
};

1.1.2、計算表格每列的開始坐標(biāo)

初始化表格列數(shù)據(jù)結(jié)構(gòu),給表格列添加 width 字符

const thList = columns.map((item) => {
    return {
      ...item,
      width: item.width
        ? item.width
        : Math.floor(getTextWidth(leafer, item.title) * widthRatio),
    };
});

循環(huán)遍歷表格列 渲染表頭

const group = new Group({ x, y, id: "tableHeader" });
thList.forEach((th, index) => {
	const midLength = thList.slice(0, index).reduce((acc, cur) => {
		return acc + cur.width;
	}, 0);
	const x = index === 0 ? 0 : midLength - index;
	const rect = new Rect({
		x,
		y: 0,
		width: th.width,
		height: initParams.headerHeight,
		fill: "#417A77",
		stroke: "#b4c9fb",
	});
	group.add(rect);
	const text = new Text({
		x,
		y,
		width: th.width,
		textAlign: "center",
		height: headerHeight,
		verticalAlign: "middle",
		fill: "#000000",
		text: th.title,
		fontSize,
	});
	group.add(text);
});

到這里為止 表頭就可以正常渲染出來了

1.2、表格列有設(shè)置寬度

與沒有設(shè)置表格列的渲染類似

文本與表格寬度比率 = (表格寬度 - 表格設(shè)置列的總寬度) / 表格列文本總寬度

沒有設(shè)置寬度的列寬度 = 每列表格文本寬度 * 文本與表格寬度比率

const noSetWidthColWidth = columns.reduce((acc, cur) => {
    if (cur.width) {
      return "";
    }
    return acc + cur.title;
}, "");
const textWidth = getTextWidth(leafer, noSetWidthColWidth);
const setColWidthSum = columns.reduce((acc, cur) => {
    if (cur.width) {
      return acc + cur.width;
    }
    return acc;
}, 0);
const widthRatio = (width - setColWidthSum) / textWidth;

渲染方式同上,最后掛載到 leafer 中完成渲染

2、滾動條渲染

在表格渲染之前要先解決表格滾動條和表格聯(lián)動的問題,根據(jù)滾動條滾動的距離計算表格顯示的內(nèi)容,因為是自繪制表格,所以滾動條部分不能利用瀏覽器的滾動條。

2.1、創(chuàng)建滾動條

滾動條的本質(zhì)還是一個 Rect,使 Rect 模擬滾動條的行為。

const rect = new Rect({
	x: width - scrollBar.width,
	y: initParams.headerHeight,
	width: scrollBar.width - scrollBar.margin * 2,
	height: scrollBar.height,
	fill: "rgba(133,117,85, 0.8)",
	cornerRadius: 10,
	id: "scrollBar",
	zIndex: scrollBar.zIndex,
});

2.2、計算滾動條的高度、位置、樣式

2.2.1、計算滾動條的高度

根據(jù)數(shù)據(jù)量的大小,需要調(diào)整滾動條渲染的高度,計算方式如下:

每條數(shù)據(jù)對應(yīng)滾動條高度 = (表格總高度 - 表頭高度) / 數(shù)據(jù)長度

滾動條高度 = 滾動條最小高度 + 視圖內(nèi)顯示行數(shù) * 每條數(shù)據(jù)對應(yīng)滾動條高度

const computedScrollBarHeight = (
	leafer: Leafer,
	dataSource: Record<string, string>[],
	jumpIndex = 0
) => {
	const { height } = leafer;
	const { viewHeight, viewCapacity } = getViewInfo(leafer);
	const unitLength = (height - initParams.headerHeight) / dataSource.length;
	if (jumpIndex) {
      return initParams.scrollBar.height;
	}
	const targetHeight = initParams.scrollBar.height + viewCapacity * unitLength;
	// 小數(shù)據(jù)量做臨時處理
	return targetHeight < viewHeight ? Math.ceil(targetHeight) : viewHeight - 10;
};

2.2.2、滾動條的位置

滾動條的 X 軸位置 = 表格的寬度 - 滾動條區(qū)域的寬度

滾動條的 Y 軸滾動需要添加鼠標(biāo)滾輪和拖拽事件的監(jiān)聽,對滾動條拖拽事件的監(jiān)聽是通過監(jiān)聽滾動條本身,鼠標(biāo)滾輪的監(jiān)聽需要對表格本身添加監(jiān)聽事件

leafer.on(MoveEvent.MOVE, function (e) {
	setScroll(leafer, rect, e, dataSource, -0.1, scrollParams);
});
rect.on(DragEvent.DRAG, function (e) {
	setScroll(leafer, rect, e, dataSource, 1, scrollParams);
});

滾動條的最大滾動高度 = 表格的高度 - 滾動條高度

滾動條的渲染是從設(shè)置的坐標(biāo)點開始 + 滾動條的高度,保證滾動條在可視區(qū)域內(nèi),需要減去滾動條的高度。

當(dāng)滾動或拖拽計算值超過最大高度時,為最大高度;當(dāng)滾動或拖拽計算值小于表頭高度時,為表頭高度,其他情況為滾動條在 Y 軸方向的偏移值 + 鼠標(biāo)滾輪滾動的距離或拖拽的距離

const setScroll = (
	leafer: Leafer,
	rect: Rect,
	e: MoveEvent | DragEvent,
	dataSource: Record<string, string>[],
	val = 1,
	scrollInfo: ScrollInfo
) => {
	const {
		scrollMaxHeight,
		headerHeight,
		height,
		scrollBar,
		viewCapacity,
		unitLength,
	} = scrollInfo;
	leafer.children = leafer.children.filter((item) =>
		fixedGroup.includes(item.id ?? "")
	);
	/**
	 * 鼠標(biāo)滾輪的滾動向上滾動是正值,向下是負(fù)值
	 * 這與滾動條位置是相反的,需要在獲取滾動距離時 * -1
	 *  */
	rect.y =
		rect.y + e.moveY * val >= scrollMaxHeight
			? scrollMaxHeight
			: rect.y + e.moveY * val < headerHeight
			? headerHeight
			: rect.y + e.moveY * val;
};

2.2.3、滾動條的樣式

滾動條 = 滾動條本身 + 左右邊距

滾動條本身寬度 = 滾動條寬度 - 邊距 * 2

鼠標(biāo)移入移出滾動條時會有顯隱效果,通過對 Rect 添加移入移出事件來修改透明度

rect.on(PointerEvent.ENTER, (e) => {
	e.target.fill = "rgba(133,117,85, 1)";
});
rect.on(PointerEvent.LEAVE, (e) => {
	e.target.fill = "rgba(133,117,85, 0.8)";
});

2.3、滾動條是否顯示

當(dāng)數(shù)據(jù)長度小于可視區(qū)域內(nèi)的行數(shù)時,此時不需要出滾動條,在初始化表格調(diào)用滾動條方法添加判斷。

export const drawCanvasTable = (
	leafer: Leafer,
	columns: Column[],
	dataSource: Record<string, string>[],
	jumpIndex = 0
) => {
	// ...
	dataSource.length > viewCapacity &&
		initScrollBar(leafer, dataSource, jumpIndex);
};

3、表格渲染

3.1、初始化渲染

表格的渲染類似于表頭的渲染,表格的渲染是按照行來渲染,每行的列坐標(biāo)、寬度是和表頭一樣的,可以在表格渲染的部分保存一份。

thList.forEach((th, index) => {
	const midLength = thList.slice(0, index).reduce((acc, cur) => {
		return acc + cur.width;
	}, 0);
	const x = index === 0 ? 0 : midLength - index;
	tableHeaderInfo[th.dataIndex] = {
		x,
		width: th.width,
	};
	// ...省略渲染部分...
});

3.2、獲取渲染的范圍

表格渲染內(nèi)容的起始位置是通過滾動條位置來計算的,并通過滾動條位置的變化來重新渲染表格。

滾動距離等于最大滾動距離時,渲染的起始位置為數(shù)據(jù)總長度 - 視圖可顯示的行數(shù)。

滾動距離小于最大滾動距離時:

滾動條偏移范圍內(nèi)需要渲染的數(shù)據(jù)單位長度 = (表格高度 - 表格頭高度 - 滾動條高度) / (數(shù)據(jù)長度 - 視圖可顯示行數(shù))

渲染的起始位置 = (滾動條位置 - 表格頭高度) / 滾動條偏移范圍內(nèi)需要渲染的數(shù)據(jù)單位長度

const setScroll = (
	leafer: Leafer,
	rect: Rect,
	e: MoveEvent | DragEvent,
	dataSource: Record<string, string>[],
	val = 1,
	scrollInfo: ScrollInfo
) => {
	// ...計算滾動條位置代碼...
	from =
     rect.y === scrollMaxHeight
       ? dataSource.length - viewCapacity
       : Math.ceil((rect.y - headerHeight) / unitLength);
	initTableBody();
};

當(dāng)數(shù)據(jù)大于表格視圖行數(shù)時,表格結(jié)束范圍 = 起始位置 + 表格視圖可以顯示的最大行數(shù),如果計算值大于數(shù)據(jù)最大長度,則為數(shù)據(jù)長度,否則表格的結(jié)束范圍 = 數(shù)據(jù)長度

const computedViewBoundary = (
	i: number,
	start: number,
	viewCapacity: number,
	dataSource: Record<string, string>[]
) => {
	if (dataSource.length > viewCapacity) {
     return (i < start + viewCapacity && start + viewCapacity <= dataSource.length);
	} else {
     return i < dataSource.length;
	}
};

3.3、計算表格 Y 軸方向的偏移

因為計算視圖內(nèi)可以顯示的表格行時,會有小數(shù)的存在,這里是采用向下取整,這樣顯示的行數(shù)總高度會超過表格的可視區(qū)域高度,這時候需要對表格進(jìn)行部分偏移,使其在滾動到底部時能夠正常顯示

  • 1、數(shù)據(jù)長度小于可視區(qū)域行數(shù)時,不需要偏移
  • 2、數(shù)據(jù)長度大于可視區(qū)域行數(shù)時
    • 2.1 開始位置小于需要隨滾動條渲染的數(shù)據(jù)長度時,不需要偏移
    • 2.2 開始位置大于等于需要隨滾動條渲染的數(shù)據(jù)長度時:
      • 2.2.1 如果表格行高可以被視圖高度整除,不需要偏移
      • 2.2.2 如果表格行高不可以被視圖高度整除,偏移值 = 表格頭高度 - (表格行高 - 視圖高度 % 行高)
const computedTableOffset = (
	viewCapacity: number,
	headerHeight: number,
	viewHeight: number,
	rowHeight: number
) => {
	return globalDataSource.length > viewCapacity
     ? from < Math.floor(globalDataSource.length - viewCapacity)
	     ? headerHeight
		: headerHeight -
	         (viewHeight % rowHeight ? rowHeight - (viewHeight % rowHeight) : 0)
	     : headerHeight;
};

4、跳轉(zhuǎn)到指定位置

當(dāng)表格數(shù)據(jù)量大時,需要能夠快速定位到某條數(shù)據(jù),當(dāng)接收到需要跳轉(zhuǎn)到的行時,該數(shù)據(jù)為起始位置,重新執(zhí)行渲染表格一系列方法,因為在滾動條初始化時修改了滾動條的初始高度,所以在跳轉(zhuǎn)操作時不應(yīng)該修改表格行的高度

useEffect(() => {
	if (canvasDom.current) {
		const leafer = new Leafer({
			view: canvasDom.current,
			width: 500,
			height: 800,
			move: { dragOut: false },
			type: "user",
		});
		drawCanvasTable(leafer, columns, dataSource, jumpIndex);
	}
}, [columns, dataSource, jumpIndex]);
if (jumpIndex) {
	return initParams.scrollBar.height;
}

代碼地址:

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

以上就是React使用Canvas繪制大數(shù)據(jù)表格的詳細(xì)內(nèi)容,更多關(guān)于React Canvas繪制表格的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 在Ant Design Pro登錄功能中集成圖形驗證碼組件的方法步驟

    在Ant Design Pro登錄功能中集成圖形驗證碼組件的方法步驟

    這篇文章主要介紹了在Ant Design Pro登錄功能中集成圖形驗證碼組件的方法步驟,這里的登錄功能其實就是一個表單提交,實現(xiàn)起來也很簡單,具體實例代碼跟隨小編一起看看吧
    2021-05-05
  • 使用react-beautiful-dnd實現(xiàn)列表間拖拽踩坑

    使用react-beautiful-dnd實現(xiàn)列表間拖拽踩坑

    相比于react-dnd,react-beautiful-dnd更適用于列表之間拖拽的場景,本文主要介紹了使用react-beautiful-dnd實現(xiàn)列表間拖拽踩坑,感興趣的可以了解一下
    2021-05-05
  • 詳解React服務(wù)端渲染從入門到精通

    詳解React服務(wù)端渲染從入門到精通

    這篇文章主要介紹了詳解React服務(wù)端渲染從入門到精通,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-03-03
  • 使用react遍歷對象生成dom

    使用react遍歷對象生成dom

    這篇文章主要介紹了使用react遍歷對象生成dom問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • React特征學(xué)習(xí)Form數(shù)據(jù)管理示例詳解

    React特征學(xué)習(xí)Form數(shù)據(jù)管理示例詳解

    這篇文章主要為大家介紹了React特征學(xué)習(xí)Form數(shù)據(jù)管理示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • React獲取input值并提交的2種方法實例

    React獲取input值并提交的2種方法實例

    這篇文章主要給大家介紹了關(guān)于React獲取input值并提交的2種方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • React解析html 標(biāo)簽的方法

    React解析html 標(biāo)簽的方法

    在React中,解析HTML標(biāo)簽通常是使用JSX(JavaScript XML)語法的一部分,這篇文章主要介紹了React 用來解析html 標(biāo)簽的方法,本文通過示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-08-08
  • react拖拽組件react-sortable-hoc的使用

    react拖拽組件react-sortable-hoc的使用

    本文主要介紹了react拖拽組件react-sortable-hoc的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • 淺談react.js中實現(xiàn)tab吸頂效果的問題

    淺談react.js中實現(xiàn)tab吸頂效果的問題

    下面小編就為大家?guī)硪黄獪\談react.js中實現(xiàn)tab吸頂效果的問題。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • React使用setState更新數(shù)組的方法示例(追加新數(shù)據(jù))

    React使用setState更新數(shù)組的方法示例(追加新數(shù)據(jù))

    在?React?中,setState?是管理組件狀態(tài)的核心方法之一,然而,當(dāng)我們需要更新狀態(tài)中的數(shù)組時,如何高效且安全地操作變得尤為關(guān)鍵,本文將詳細(xì)解析以下代碼的實現(xiàn)邏輯,幫助你掌握在?React?中追加數(shù)組數(shù)據(jù)的最佳實踐,需要的朋友可以參考下
    2025-03-03

最新評論