Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: 编辑表格新增拖拽填充功能 #1858

Merged
merged 14 commits into from
Nov 21, 2022
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
.drag-copy-mask {
serializedowen marked this conversation as resolved.
Show resolved Hide resolved
position: absolute;
top: -4px;
left: -4px;
z-index: 9;
background-color: rgba(#c2d5fe, 0.5);
cursor: crosshair;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,207 @@
import React, { useState, useEffect } from 'react';
import { S2Event, DataCell, InteractionStateName } from '@antv/s2';
import type { Point } from '@antv/g-canvas';
import { throttle, pick, get } from 'lodash';
import { useSpreadSheetRef } from '../../../../utils/SpreadSheetContext';
import './drag-copy-mask.less';

type DragCopyProps = {
onCopyFinished?: () => void;
};

export function DragCopyMask({ onCopyFinished }: DragCopyProps) {
const spreadsheet = useSpreadSheetRef();

const [startCell, setstartCell] = useState<DataCell>();
const [maskPosition, setMaskPosition] = useState({ right: 0, bottom: 0 });
const [dragPoint, setDragPoint] = useState<Point>();

const isInCell = (point: Point, cellTarget) => {
const cellMeta = pick(cellTarget.getMeta(), [
'x',
'y',
'width',
'height',
'fieldValue',
]);
const scrollOffset = spreadsheet.facet.getScrollOffset();
// 点位偏移
const pointX = point.x;
const pointY = point.y;
const scrollOffsetY =
scrollOffset?.scrollY -
(spreadsheet.getColumnNodes()[0] || { height: 0 }).height;
const cellMaxX = cellMeta.x - scrollOffset.scrollX + cellMeta.width + 4;
const cellMinX = cellMeta.x - scrollOffset.scrollX;
const cellMaxY = cellMeta.y - scrollOffsetY + cellMeta.height + 4;
const cellMinY = cellMeta.y - scrollOffsetY;

return (
cellMaxX >= pointX &&
cellMinX < pointX &&
cellMaxY >= pointY &&
cellMinY < pointY
);
};

/** 判断当前位置是否在表格可视区域内 */
const judgePointInView = (point: Point) => {
const rect = spreadsheet.getCanvasElement().getBoundingClientRect();
const { frozenRow } = (spreadsheet.facet as any).frozenGroupInfo;
const viewMinX = rect.x;
const viewMaxX = rect.x + rect.width;
const viewMinY = rect.y + frozenRow.height; // 减去列头高度
const viewMaxY = rect.y + rect.height;
return (
point.x <= viewMaxX &&
point.x >= viewMinX &&
point.y <= viewMaxY &&
point.y >= viewMinY
);
};

const getCurrentHoverCell = (event: MouseEvent) => {
const rect = spreadsheet.getCanvasElement().getBoundingClientRect();
const allCelles = spreadsheet.interaction.getPanelGroupAllDataCells();

return allCelles.find((v) =>
isInCell({ y: event.y - rect.y, x: event.x - rect.x }, v),
);
};

const getSelectedCellRange = (start, endCell) => {
const startCellMeta = start.getMeta();
const endCellMeta = endCell?.getMeta();
const minX = Math.min(startCellMeta.colIndex, endCellMeta.colIndex);
const maxX = Math.max(startCellMeta.colIndex, endCellMeta.colIndex);
const maxY = Math.max(startCellMeta.rowIndex, endCellMeta.rowIndex);
const minY = Math.min(startCellMeta.rowIndex, endCellMeta.rowIndex);
const allCelles = spreadsheet.interaction.getPanelGroupAllDataCells();
return allCelles.filter((item) => {
const itemMeta = item.getMeta();
return (
itemMeta.rowIndex <= maxY &&
itemMeta.rowIndex >= minY &&
itemMeta.colIndex <= maxX &&
itemMeta.colIndex >= minX
);
});
};

// 利用闭包记录最近一次hover位置,防止鼠标拖拽越界报错
const lastHoverPoint = { x: 0, y: 0 };

const dragMove = throttle((event: MouseEvent) => {
if (!startCell) {
return;
}

let targetCell = getCurrentHoverCell(event);
let newX = event.x - dragPoint.x;
let newY = event.y - dragPoint.y;

// 判断鼠标是否在表格可视区域,不在的话使用最近一次hover位置
if (!judgePointInView(event)) {
targetCell = getCurrentHoverCell(lastHoverPoint as MouseEvent);
newX = lastHoverPoint.x - dragPoint.x;
newY = lastHoverPoint.y - dragPoint.y;
} else {
lastHoverPoint.x = event.x;
lastHoverPoint.y = event.y;
}

setMaskPosition({ right: newX, bottom: newY });
const selectedRange = getSelectedCellRange(startCell, targetCell);

// 更改选中状态
spreadsheet.interaction.changeState({
cells: selectedRange.map((v) => v.getMeta() as any),
stateName: InteractionStateName.PREPARE_SELECT,
force: true,
});
}, 10);

const dragMouseUp = (event: MouseEvent) => {
let targetCell = getCurrentHoverCell(event);

if (!startCell) {
return;
}
if (!targetCell) {
targetCell = getCurrentHoverCell(lastHoverPoint as MouseEvent);
}
const source = spreadsheet.dataSet.originData;

const selectedRange = getSelectedCellRange(startCell, targetCell);
const { fieldValue } = startCell.getMeta();
const changedCells = selectedRange.map((item) => {
const { rowIndex, valueField } = item.getMeta();
if (
source[rowIndex] &&
typeof source[rowIndex][valueField] !== undefined
) {
source[rowIndex][valueField] = fieldValue;
}
return item;
});

// 更改选中状态
spreadsheet.interaction.changeState({
cells: changedCells.map((v) => v.getMeta() as any),
stateName: InteractionStateName.PREPARE_SELECT,
force: true,
});
spreadsheet.render(true);
setMaskPosition({ right: 0, bottom: 0 });
spreadsheet.off(S2Event.GLOBAL_MOUSE_MOVE, dragMove);
spreadsheet.off(S2Event.GLOBAL_MOUSE_UP, dragMouseUp);
setstartCell(undefined);
onCopyFinished?.();
};

useEffect(() => {
if (startCell) {
spreadsheet.on(S2Event.GLOBAL_MOUSE_MOVE, dragMove);
spreadsheet.on(S2Event.GLOBAL_MOUSE_UP, dragMouseUp);
}
}, [startCell]);

const dragMouseDown = (event: MouseEvent) => {
if (get(event, 'target.id') !== 'spreadsheet-drag-copy-point') {
return;
}

const rect = (event.target as HTMLElement).getBoundingClientRect();
const { top, left } = get(event, 'target.style', {});
const allCelles = spreadsheet.interaction.getPanelGroupAllDataCells();
const targetCell = allCelles.find((v) =>
isInCell({ y: parseFloat(top), x: parseFloat(left) }, v),
);
setDragPoint({ x: rect.x, y: rect.y });
setstartCell(targetCell as DataCell);
};

useEffect(() => {
const pointElement = document.getElementById('spreadsheet-drag-copy-point');
if (pointElement && spreadsheet) {
pointElement.addEventListener('mousedown', dragMouseDown);
}
return () => {
pointElement.removeEventListener('mousedown', dragMouseDown);
};
}, [spreadsheet]);

return (
<div
className="drag-copy-mask"
serializedowen marked this conversation as resolved.
Show resolved Hide resolved
style={{
right: maskPosition.right > 0 ? maskPosition.right : 0,
bottom: maskPosition.bottom > 0 ? maskPosition.bottom : 0,
left: maskPosition.right <= 0 ? maskPosition.right : 0,
top: maskPosition.bottom <= 0 ? maskPosition.bottom : 0,
width: Math.abs(maskPosition.right),
height: Math.abs(maskPosition.bottom),
}}
></div>
);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
.drag-copy-point {
position: absolute;
z-index: 9;
width: 8px;
height: 8px;
background-color: #c2d5fe;
cursor: crosshair;
user-select: none;
-webkit-user-select: none;
serializedowen marked this conversation as resolved.
Show resolved Hide resolved
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
import React, { useState, useEffect } from 'react';
import { DataCell, S2Event } from '@antv/s2';
import type { ScrollOffset } from '@antv/s2';
import { isEqual, pick } from 'lodash';
import type { Event as CanvasEvent } from '@antv/g-canvas';
import { useS2Event } from '../../../../hooks';
import { useSpreadSheetRef } from '../../../../utils/SpreadSheetContext';
import './drag-copy-point.less';
import { DragCopyMask } from './drag-copy-mask';

export type DragCopyProps = {
onChange?: (val) => void;
};

export function DragCopyPoint(props: DragCopyProps) {
const spreadsheet = useSpreadSheetRef();

const [scroll, setScroll] = useState<
ScrollOffset & { width?: number; overflow?: boolean }
>({
scrollX: -999,
scrollY: -999,
width: 8,
overflow: true,
});
const [position, setPosition] = useState({ left: -999, top: -999 });
const [cell, setCell] = useState<DataCell>();

const handleScroll = () => {
if (spreadsheet) {
const newScroll = spreadsheet.facet.getScrollOffset();
const { frozenCol, frozenRow } = (spreadsheet.facet as any)
.frozenGroupInfo;
const rect = spreadsheet.getCanvasElement().getBoundingClientRect();
const cellMeta = cell?.getMeta();

if (!isEqual(newScroll, scroll)) {
// 超出可视区域隐藏point
if (cellMeta) {
const { verticalBorderWidth: vWidth, horizontalBorderWidth: hWidth } =
cell.getStyle().cell;

// 确定点位
const pointX = cellMeta.width + cellMeta.x;
const pointY = cellMeta.height + cellMeta.y;
// 计算点位的偏移量
const pointWidth =
pointX - newScroll.scrollX - rect.width + (vWidth + hWidth) * 2;
let overflow = true;
if (
frozenCol.width >= pointX - newScroll.scrollX - hWidth * 2 ||
frozenRow.height >= pointY - newScroll.scrollY - vWidth * 2 ||
rect.width <= pointX - newScroll.scrollX - hWidth * 2 ||
rect.height <= pointY - newScroll.scrollY + frozenRow.height
) {
overflow = true;
} else {
overflow = false;
}
setScroll({
...newScroll,
overflow,
width: 8 - (pointWidth > 0 ? pointWidth : 0),
});
} else {
setScroll({ scrollX: -999, scrollY: -999, overflow: true });
}
}
}
};

const fixPostiton = (event: CanvasEvent) => {
const eventCell = event.target.cfg.parent;
const isEventCellSelected =
spreadsheet.interaction.isSelectedCell(eventCell);
// 如果点击单元格时,单元格取消选中,隐藏拖拽点
if (isEventCellSelected) {
setCell(eventCell);
} else {
setCell(undefined);
}
};

useEffect(() => {
handleScroll();
if (spreadsheet) {
spreadsheet.off(S2Event.GLOBAL_SCROLL, handleScroll);
spreadsheet.on(S2Event.GLOBAL_SCROLL, handleScroll);
}
return () => {
spreadsheet?.off(S2Event.GLOBAL_SCROLL, handleScroll);
};
}, [cell]);

/** 单元格实例更改,选中态去除 */
useEffect(() => {
if (cell) {
setCell(undefined);
}
}, [spreadsheet?.dataSet.sortParams, spreadsheet?.dataSet.filterParams]);

useEffect(() => {
if (!spreadsheet?.container.cfg.container) {
return;
}

if (spreadsheet && cell) {
const cellMeta = pick(cell.getMeta(), [
'x',
'y',
'width',
'height',
'fieldValue',
]);
cellMeta.x -= scroll?.scrollX;
cellMeta.y -=
scroll?.scrollY -
(spreadsheet.getColumnNodes()[0] || { height: 0 }).height;
setPosition({
left: cellMeta.x + cellMeta.width - 4,
top: cellMeta.y + cellMeta.height - 4,
});
}
}, [scroll, cell]);

/** 多选时隐藏拖拽点 */
const batchSelected = () => {
setCell(undefined);
};

useS2Event(S2Event.COL_CELL_CLICK, batchSelected, spreadsheet);
useS2Event(S2Event.ROW_CELL_CLICK, batchSelected, spreadsheet);
useS2Event(S2Event.CORNER_CELL_CLICK, batchSelected, spreadsheet);
useS2Event(S2Event.DATA_CELL_BRUSH_SELECTION, batchSelected, spreadsheet);

useS2Event(S2Event.DATA_CELL_CLICK, fixPostiton, spreadsheet);

return (
<div
id="spreadsheet-drag-copy-point"
className="drag-copy-point"
serializedowen marked this conversation as resolved.
Show resolved Hide resolved
style={{
display: scroll.overflow ? 'none' : 'block',
left: position.left,
top: position.top,
width: scroll.width,
}}
>
<DragCopyMask onCopyFinished={batchSelected} />
<div></div>
serializedowen marked this conversation as resolved.
Show resolved Hide resolved
</div>
);
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,14 @@ import React from 'react';
import { BaseSheet } from '../base-sheet';
import type { SheetComponentsProps } from '../interface';
import { EditCell } from './edit-cell';
import { DragCopyPoint } from './drag-copy';

export const EditableSheet: React.FC<SheetComponentsProps> = React.memo(
(props) => {
return (
<BaseSheet {...props} sheetType={'table'}>
<EditCell onChange={() => {}} />
<DragCopyPoint />
</BaseSheet>
);
},
Expand Down