mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-14 13:25:12 +00:00
383 lines
10 KiB
TypeScript
383 lines
10 KiB
TypeScript
import type { TableBlockModel, TableCell } from '@blocksuite/affine-model';
|
|
import { generateFractionalIndexingKeyBetween } from '@blocksuite/affine-shared/utils';
|
|
import { nanoid, Text } from '@blocksuite/store';
|
|
import { computed, type ReadonlySignal, signal } from '@preact/signals-core';
|
|
|
|
import type { TableAreaSelection } from './selection-schema';
|
|
|
|
export class TableDataManager {
|
|
constructor(private readonly model: TableBlockModel) {}
|
|
readonly readonly$: ReadonlySignal<boolean> = computed(() => {
|
|
return this.model.doc.readonly;
|
|
});
|
|
readonly ui = {
|
|
columnIndicatorIndex$: signal<number>(),
|
|
rowIndicatorIndex$: signal<number>(),
|
|
};
|
|
readonly hoverColumnIndex$ = signal<number>();
|
|
readonly hoverRowIndex$ = signal<number>();
|
|
readonly hoverDragHandleColumnId$ = signal<string>();
|
|
readonly widthAdjustColumnId$ = signal<string>();
|
|
readonly virtualColumnCount$ = signal<number>(0);
|
|
readonly virtualRowCount$ = signal<number>(0);
|
|
readonly virtualWidth$ = signal<
|
|
{ columnId: string; width: number } | undefined
|
|
>();
|
|
readonly cellCountTips$ = computed(
|
|
() =>
|
|
`${this.virtualRowCount$.value + this.rows$.value.length} x ${this.virtualColumnCount$.value + this.columns$.value.length}`
|
|
);
|
|
readonly rows$ = computed(() => {
|
|
return Object.values(this.model.props.rows$.value).sort((a, b) =>
|
|
a.order > b.order ? 1 : -1
|
|
);
|
|
});
|
|
|
|
readonly columns$ = computed(() => {
|
|
return Object.values(this.model.props.columns$.value).sort((a, b) =>
|
|
a.order > b.order ? 1 : -1
|
|
);
|
|
});
|
|
|
|
readonly uiRows$ = computed(() => {
|
|
const virtualRowCount = this.virtualRowCount$.value;
|
|
const rows = this.rows$.value;
|
|
if (virtualRowCount === 0) {
|
|
return rows;
|
|
}
|
|
if (virtualRowCount > 0) {
|
|
return [
|
|
...rows,
|
|
...Array.from({ length: virtualRowCount }, (_, i) => ({
|
|
rowId: `${i}`,
|
|
backgroundColor: undefined,
|
|
})),
|
|
];
|
|
}
|
|
return rows.slice(0, rows.length + virtualRowCount);
|
|
});
|
|
|
|
readonly uiColumns$ = computed(() => {
|
|
const virtualColumnCount = this.virtualColumnCount$.value;
|
|
const columns = this.columns$.value;
|
|
if (virtualColumnCount === 0) {
|
|
return columns;
|
|
}
|
|
if (virtualColumnCount > 0) {
|
|
return [
|
|
...columns,
|
|
...Array.from({ length: virtualColumnCount }, (_, i) => ({
|
|
columnId: `${i}`,
|
|
backgroundColor: undefined,
|
|
width: undefined,
|
|
})),
|
|
];
|
|
}
|
|
return columns.slice(0, columns.length + virtualColumnCount);
|
|
});
|
|
|
|
getCell(rowId: string, columnId: string): TableCell | undefined {
|
|
return this.model.props.cells$.value[`${rowId}:${columnId}`];
|
|
}
|
|
|
|
addRow(after?: number) {
|
|
const order = this.getOrder(this.rows$.value, after);
|
|
const rowId = nanoid();
|
|
this.model.doc.transact(() => {
|
|
this.model.props.rows[rowId] = {
|
|
rowId,
|
|
order,
|
|
};
|
|
|
|
this.columns$.value.forEach(column => {
|
|
this.model.props.cells[`${rowId}:${column.columnId}`] = {
|
|
text: new Text(),
|
|
};
|
|
});
|
|
});
|
|
return rowId;
|
|
}
|
|
addNRow(count: number) {
|
|
if (count === 0) {
|
|
return;
|
|
}
|
|
if (count > 0) {
|
|
this.model.doc.transact(() => {
|
|
for (let i = 0; i < count; i++) {
|
|
this.addRow(this.rows$.value.length - 1);
|
|
}
|
|
});
|
|
} else {
|
|
const rows = this.rows$.value;
|
|
const rowCount = rows.length;
|
|
this.model.doc.transact(() => {
|
|
rows.slice(rowCount + count, rowCount).forEach(row => {
|
|
this.deleteRow(row.rowId);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
addNColumn(count: number) {
|
|
if (count === 0) {
|
|
return;
|
|
}
|
|
if (count > 0) {
|
|
this.model.doc.transact(() => {
|
|
for (let i = 0; i < count; i++) {
|
|
this.addColumn(this.columns$.value.length - 1);
|
|
}
|
|
});
|
|
} else {
|
|
const columns = this.columns$.value;
|
|
const columnCount = columns.length;
|
|
this.model.doc.transact(() => {
|
|
columns.slice(columnCount + count, columnCount).forEach(column => {
|
|
this.deleteColumn(column.columnId);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
private getOrder<T extends { order: string }>(array: T[], after?: number) {
|
|
after = after != null ? (after < 0 ? undefined : after) : undefined;
|
|
const prevOrder = after == null ? null : array[after]?.order;
|
|
const nextOrder = after == null ? array[0]?.order : array[after + 1]?.order;
|
|
const order = generateFractionalIndexingKeyBetween(
|
|
prevOrder ?? null,
|
|
nextOrder ?? null
|
|
);
|
|
return order;
|
|
}
|
|
|
|
addColumn(after?: number) {
|
|
const order = this.getOrder(this.columns$.value, after);
|
|
const columnId = nanoid();
|
|
this.model.doc.transact(() => {
|
|
this.model.props.columns[columnId] = {
|
|
columnId,
|
|
order,
|
|
};
|
|
this.rows$.value.forEach(row => {
|
|
this.model.props.cells[`${row.rowId}:${columnId}`] = {
|
|
text: new Text(),
|
|
};
|
|
});
|
|
});
|
|
return columnId;
|
|
}
|
|
|
|
deleteRow(rowId: string) {
|
|
this.model.doc.transact(() => {
|
|
Object.keys(this.model.props.rows).forEach(id => {
|
|
if (id === rowId) {
|
|
delete this.model.props.rows[id];
|
|
}
|
|
});
|
|
Object.keys(this.model.props.cells).forEach(id => {
|
|
if (id.startsWith(rowId)) {
|
|
delete this.model.props.cells[id];
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
deleteColumn(columnId: string) {
|
|
this.model.doc.transact(() => {
|
|
Object.keys(this.model.props.columns).forEach(id => {
|
|
if (id === columnId) {
|
|
delete this.model.props.columns[id];
|
|
}
|
|
});
|
|
Object.keys(this.model.props.cells).forEach(id => {
|
|
if (id.endsWith(`:${columnId}`)) {
|
|
delete this.model.props.cells[id];
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
updateRowOrder(rowId: string, newOrder: string) {
|
|
this.model.doc.transact(() => {
|
|
if (this.model.props.rows[rowId]) {
|
|
this.model.props.rows[rowId].order = newOrder;
|
|
}
|
|
});
|
|
}
|
|
|
|
updateColumnOrder(columnId: string, newOrder: string) {
|
|
this.model.doc.transact(() => {
|
|
if (this.model.props.columns[columnId]) {
|
|
this.model.props.columns[columnId].order = newOrder;
|
|
}
|
|
});
|
|
}
|
|
|
|
setRowBackgroundColor(rowId: string, color?: string) {
|
|
this.model.doc.transact(() => {
|
|
if (this.model.props.rows[rowId]) {
|
|
this.model.props.rows[rowId].backgroundColor = color;
|
|
}
|
|
});
|
|
}
|
|
|
|
setColumnBackgroundColor(columnId: string, color?: string) {
|
|
this.model.doc.transact(() => {
|
|
if (this.model.props.columns[columnId]) {
|
|
this.model.props.columns[columnId].backgroundColor = color;
|
|
}
|
|
});
|
|
}
|
|
|
|
setColumnWidth(columnId: string, width: number) {
|
|
this.model.doc.transact(() => {
|
|
if (this.model.props.columns[columnId]) {
|
|
this.model.props.columns[columnId].width = width;
|
|
}
|
|
});
|
|
}
|
|
|
|
clearRow(rowId: string) {
|
|
this.model.doc.transact(() => {
|
|
Object.keys(this.model.props.cells).forEach(id => {
|
|
if (id.startsWith(rowId)) {
|
|
this.model.props.cells[id]?.text.replace(
|
|
0,
|
|
this.model.props.cells[id]?.text.length,
|
|
''
|
|
);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
clearColumn(columnId: string) {
|
|
this.model.doc.transact(() => {
|
|
Object.keys(this.model.props.cells).forEach(id => {
|
|
if (id.endsWith(`:${columnId}`)) {
|
|
this.model.props.cells[id]?.text.replace(
|
|
0,
|
|
this.model.props.cells[id]?.text.length,
|
|
''
|
|
);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
clearCellsBySelection(selection: TableAreaSelection) {
|
|
const columns = this.uiColumns$.value;
|
|
const rows = this.uiRows$.value;
|
|
const deleteCells: { rowId: string; columnId: string }[] = [];
|
|
for (let i = selection.rowStartIndex; i <= selection.rowEndIndex; i++) {
|
|
const row = rows[i];
|
|
if (!row) {
|
|
continue;
|
|
}
|
|
for (
|
|
let j = selection.columnStartIndex;
|
|
j <= selection.columnEndIndex;
|
|
j++
|
|
) {
|
|
const column = columns[j];
|
|
if (!column) {
|
|
continue;
|
|
}
|
|
deleteCells.push({ rowId: row.rowId, columnId: column.columnId });
|
|
}
|
|
}
|
|
this.clearCells(deleteCells);
|
|
}
|
|
|
|
clearCells(cells: { rowId: string; columnId: string }[]) {
|
|
this.model.doc.transact(() => {
|
|
cells.forEach(({ rowId, columnId }) => {
|
|
const text = this.model.props.cells[`${rowId}:${columnId}`]?.text;
|
|
if (text) {
|
|
text.replace(0, text.length, '');
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
insertColumn(after?: number) {
|
|
this.addColumn(after);
|
|
}
|
|
|
|
insertRow(after?: number) {
|
|
this.addRow(after);
|
|
}
|
|
|
|
moveColumn(from: number, after?: number) {
|
|
const columns = this.columns$.value;
|
|
const column = columns[from];
|
|
if (!column) return;
|
|
const order = this.getOrder(columns, after);
|
|
this.model.doc.transact(() => {
|
|
const realColumn = this.model.props.columns[column.columnId];
|
|
if (realColumn) {
|
|
realColumn.order = order;
|
|
}
|
|
});
|
|
}
|
|
|
|
moveRow(from: number, after?: number) {
|
|
const rows = this.rows$.value;
|
|
const row = rows[from];
|
|
if (!row) return;
|
|
const order = this.getOrder(rows, after);
|
|
this.model.doc.transact(() => {
|
|
const realRow = this.model.props.rows[row.rowId];
|
|
if (realRow) {
|
|
realRow.order = order;
|
|
}
|
|
});
|
|
}
|
|
|
|
duplicateColumn(index: number) {
|
|
const oldColumn = this.columns$.value[index];
|
|
if (!oldColumn) return;
|
|
const order = this.getOrder(this.columns$.value, index);
|
|
const newColumnId = nanoid();
|
|
this.model.doc.transact(() => {
|
|
this.model.props.columns[newColumnId] = {
|
|
...oldColumn,
|
|
columnId: newColumnId,
|
|
order,
|
|
};
|
|
this.rows$.value.forEach(row => {
|
|
this.model.props.cells[`${row.rowId}:${newColumnId}`] = {
|
|
text:
|
|
this.model.props.cells[
|
|
`${row.rowId}:${oldColumn.columnId}`
|
|
]?.text.clone() ?? new Text(),
|
|
};
|
|
});
|
|
});
|
|
return newColumnId;
|
|
}
|
|
|
|
duplicateRow(index: number) {
|
|
const oldRow = this.rows$.value[index];
|
|
if (!oldRow) return;
|
|
const order = this.getOrder(this.rows$.value, index);
|
|
const newRowId = nanoid();
|
|
this.model.doc.transact(() => {
|
|
this.model.props.rows[newRowId] = {
|
|
...oldRow,
|
|
rowId: newRowId,
|
|
order,
|
|
};
|
|
this.columns$.value.forEach(column => {
|
|
this.model.props.cells[`${newRowId}:${column.columnId}`] = {
|
|
text:
|
|
this.model.props.cells[
|
|
`${oldRow.rowId}:${column.columnId}`
|
|
]?.text.clone() ?? new Text(),
|
|
};
|
|
});
|
|
});
|
|
return newRowId;
|
|
}
|
|
}
|