mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-15 05:37:32 +00:00
close: BS-3446 <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **Refactor** - Migrated all styling from vanilla-extract to Emotion for improved CSS-in-JS consistency across database, table, and data view components. - Updated import paths for style modules to reflect the new Emotion-based file naming. - Removed unused or redundant style exports and updated selector syntax where necessary. - **Chores** - Updated dependencies: removed vanilla-extract and added Emotion in relevant package files. - **Style** - No visual changes to existing components; all style definitions remain consistent with previous designs. - **New Features** - Introduced new reusable CSS classes for data view and table components using Emotion. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
210 lines
7.2 KiB
TypeScript
210 lines
7.2 KiB
TypeScript
import { CaptionedBlockComponent } from '@blocksuite/affine-components/caption';
|
|
import type { TableBlockModel } from '@blocksuite/affine-model';
|
|
import { EDGELESS_TOP_CONTENTEDITABLE_SELECTOR } from '@blocksuite/affine-shared/consts';
|
|
import { DocModeProvider } from '@blocksuite/affine-shared/services';
|
|
import { VirtualPaddingController } from '@blocksuite/affine-shared/utils';
|
|
import { IS_MOBILE } from '@blocksuite/global/env';
|
|
import type { BlockComponent } from '@blocksuite/std';
|
|
import { RANGE_SYNC_EXCLUDE_ATTR } from '@blocksuite/std/inline';
|
|
import { signal } from '@preact/signals-core';
|
|
import { html, nothing } from 'lit';
|
|
import { ref } from 'lit/directives/ref.js';
|
|
import { repeat } from 'lit/directives/repeat.js';
|
|
import { styleMap } from 'lit/directives/style-map.js';
|
|
|
|
import { SelectionController } from './selection-controller';
|
|
import {
|
|
rowStyle,
|
|
table,
|
|
tableContainer,
|
|
tableWrapper,
|
|
} from './table-block-css';
|
|
import { TableDataManager } from './table-data-manager';
|
|
|
|
export const TableBlockComponentName = 'affine-table';
|
|
export class TableBlockComponent extends CaptionedBlockComponent<TableBlockModel> {
|
|
private _dataManager: TableDataManager | null = null;
|
|
|
|
get dataManager(): TableDataManager {
|
|
if (!this._dataManager) {
|
|
this._dataManager = new TableDataManager(this.model);
|
|
}
|
|
return this._dataManager;
|
|
}
|
|
|
|
selectionController = new SelectionController(this);
|
|
|
|
override connectedCallback() {
|
|
super.connectedCallback();
|
|
this.setAttribute(RANGE_SYNC_EXCLUDE_ATTR, 'true');
|
|
this.style.position = 'relative';
|
|
}
|
|
|
|
override get topContenteditableElement() {
|
|
if (this.std.get(DocModeProvider).getEditorMode() === 'edgeless') {
|
|
return this.closest<BlockComponent>(
|
|
EDGELESS_TOP_CONTENTEDITABLE_SELECTOR
|
|
);
|
|
}
|
|
return this.rootComponent;
|
|
}
|
|
|
|
private readonly virtualPaddingController: VirtualPaddingController =
|
|
new VirtualPaddingController(this);
|
|
|
|
table$ = signal<HTMLTableElement>();
|
|
|
|
public getScale(): number {
|
|
const table = this.table$.value;
|
|
if (!table) return 1;
|
|
return table.getBoundingClientRect().width / table.offsetWidth;
|
|
}
|
|
|
|
private readonly getRootRect = () => {
|
|
const table = this.table$.value;
|
|
if (!table) return;
|
|
return table.getBoundingClientRect();
|
|
};
|
|
|
|
private readonly getRowRect = (rowId: string) => {
|
|
const row = this.querySelector(`tr[data-row-id="${rowId}"]`);
|
|
const rootRect = this.getRootRect();
|
|
if (!row || !rootRect) return;
|
|
const rect = row.getBoundingClientRect();
|
|
const scale = this.getScale();
|
|
return {
|
|
top: (rect.top - rootRect.top) / scale,
|
|
left: (rect.left - rootRect.left) / scale,
|
|
width: rect.width / scale,
|
|
height: rect.height / scale,
|
|
};
|
|
};
|
|
|
|
private readonly getColumnRect = (columnId: string) => {
|
|
const columns = this.querySelectorAll(`td[data-column-id="${columnId}"]`);
|
|
const rootRect = this.getRootRect();
|
|
if (!rootRect) return;
|
|
const firstRect = columns.item(0)?.getBoundingClientRect();
|
|
const lastRect = columns.item(columns.length - 1)?.getBoundingClientRect();
|
|
if (!firstRect || !lastRect) return;
|
|
const scale = this.getScale();
|
|
return {
|
|
top: (firstRect.top - rootRect.top) / scale,
|
|
left: (firstRect.left - rootRect.left) / scale,
|
|
width: firstRect.width / scale,
|
|
height: (lastRect.bottom - firstRect.top) / scale,
|
|
};
|
|
};
|
|
|
|
private readonly getAreaRect = (
|
|
rowStartIndex: number,
|
|
rowEndIndex: number,
|
|
columnStartIndex: number,
|
|
columnEndIndex: number
|
|
) => {
|
|
const rootRect = this.getRootRect();
|
|
const rows = this.querySelectorAll('tr');
|
|
const startRow = rows.item(rowStartIndex);
|
|
const endRow = rows.item(rowEndIndex);
|
|
if (!startRow || !endRow || !rootRect) return;
|
|
|
|
const startCells = startRow.querySelectorAll('td');
|
|
const endCells = endRow.querySelectorAll('td');
|
|
const startCell = startCells.item(columnStartIndex);
|
|
const endCell = endCells.item(columnEndIndex);
|
|
if (!startCell || !endCell) return;
|
|
|
|
const startRect = startCell.getBoundingClientRect();
|
|
const endRect = endCell.getBoundingClientRect();
|
|
const scale = this.getScale();
|
|
|
|
return {
|
|
top: (startRect.top - rootRect.top) / scale,
|
|
left: (startRect.left - rootRect.left) / scale,
|
|
width: (endRect.right - startRect.left) / scale,
|
|
height: (endRect.bottom - startRect.top) / scale,
|
|
};
|
|
};
|
|
|
|
override renderBlock() {
|
|
const rows = this.dataManager.uiRows$.value;
|
|
const columns = this.dataManager.uiColumns$.value;
|
|
const virtualPadding = this.virtualPaddingController.virtualPadding$.value;
|
|
return html`
|
|
<div
|
|
contenteditable="false"
|
|
class=${tableContainer}
|
|
style=${styleMap({
|
|
marginLeft: `-${virtualPadding + 10}px`,
|
|
marginRight: `-${virtualPadding}px`,
|
|
position: 'relative',
|
|
})}
|
|
>
|
|
<div
|
|
style=${styleMap({
|
|
paddingLeft: `${virtualPadding}px`,
|
|
paddingRight: `${virtualPadding}px`,
|
|
width: 'max-content',
|
|
})}
|
|
>
|
|
<table class=${tableWrapper} ${ref(this.table$)}>
|
|
<tbody class=${table}>
|
|
${repeat(
|
|
rows,
|
|
row => row.rowId,
|
|
(row, rowIndex) => {
|
|
return html`
|
|
<tr class=${rowStyle} data-row-id=${row.rowId}>
|
|
${repeat(
|
|
columns,
|
|
column => column.columnId,
|
|
(column, columnIndex) => {
|
|
const cell = this.dataManager.getCell(
|
|
row.rowId,
|
|
column.columnId
|
|
);
|
|
return html`
|
|
<affine-table-cell
|
|
style="display: contents;"
|
|
.rowIndex=${rowIndex}
|
|
.columnIndex=${columnIndex}
|
|
.row=${row}
|
|
.column=${column}
|
|
.text=${cell?.text}
|
|
.dataManager=${this.dataManager}
|
|
.selectionController=${this.selectionController}
|
|
></affine-table-cell>
|
|
`;
|
|
}
|
|
)}
|
|
</tr>
|
|
`;
|
|
}
|
|
)}
|
|
</tbody>
|
|
${IS_MOBILE || this.dataManager.readonly$.value
|
|
? nothing
|
|
: html`<affine-table-add-button
|
|
style="display: contents;"
|
|
.dataManager=${this.dataManager}
|
|
></affine-table-add-button>`}
|
|
${html`<affine-table-selection-layer
|
|
style="display: contents;"
|
|
.selectionController=${this.selectionController}
|
|
.getRowRect=${this.getRowRect}
|
|
.getColumnRect=${this.getColumnRect}
|
|
.getAreaRect=${this.getAreaRect}
|
|
></affine-table-selection-layer>`}
|
|
</table>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
}
|
|
|
|
declare global {
|
|
interface HTMLElementTagNameMap {
|
|
[TableBlockComponentName]: TableBlockComponent;
|
|
}
|
|
}
|