mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-04-02 21:59:06 +08:00
Lexical: Started linking up cell properties form
This commit is contained in:
parent
efec752985
commit
8939f310db
@ -93,6 +93,6 @@ export function $createCustomParagraphNode() {
|
|||||||
return new CustomParagraphNode();
|
return new CustomParagraphNode();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function $isCustomParagraphNode(node: LexicalNode | null | undefined) {
|
export function $isCustomParagraphNode(node: LexicalNode | null | undefined): node is CustomParagraphNode {
|
||||||
return node instanceof CustomParagraphNode;
|
return node instanceof CustomParagraphNode;
|
||||||
}
|
}
|
90
resources/js/wysiwyg/nodes/custom-table-cell-node.ts
Normal file
90
resources/js/wysiwyg/nodes/custom-table-cell-node.ts
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
import {EditorConfig} from "lexical/LexicalEditor";
|
||||||
|
import {DOMExportOutput, LexicalEditor, LexicalNode, Spread} from "lexical";
|
||||||
|
|
||||||
|
import {SerializedTableCellNode, TableCellHeaderStates, TableCellNode} from "@lexical/table";
|
||||||
|
import {TableCellHeaderState} from "@lexical/table/LexicalTableCellNode";
|
||||||
|
|
||||||
|
export type SerializedCustomTableCellNode = Spread<{
|
||||||
|
styles: Record<string, string>,
|
||||||
|
}, SerializedTableCellNode>
|
||||||
|
|
||||||
|
export class CustomTableCellNode extends TableCellNode {
|
||||||
|
__styles: Map<string, string> = new Map;
|
||||||
|
|
||||||
|
static getType(): string {
|
||||||
|
return 'custom-table-cell';
|
||||||
|
}
|
||||||
|
|
||||||
|
static clone(node: CustomTableCellNode): CustomTableCellNode {
|
||||||
|
const cellNode = new CustomTableCellNode(
|
||||||
|
node.__headerState,
|
||||||
|
node.__colSpan,
|
||||||
|
node.__width,
|
||||||
|
node.__key,
|
||||||
|
);
|
||||||
|
cellNode.__rowSpan = node.__rowSpan;
|
||||||
|
cellNode.__styles = new Map(node.__styles);
|
||||||
|
return cellNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
getStyles(): Map<string, string> {
|
||||||
|
const self = this.getLatest();
|
||||||
|
return new Map(self.__styles);
|
||||||
|
}
|
||||||
|
|
||||||
|
setStyles(styles: Map<string, string>): void {
|
||||||
|
const self = this.getWritable();
|
||||||
|
self.__styles = new Map(styles);
|
||||||
|
}
|
||||||
|
|
||||||
|
updateTag(tag: string): void {
|
||||||
|
const isHeader = tag.toLowerCase() === 'th';
|
||||||
|
const state = isHeader ? TableCellHeaderStates.ROW : TableCellHeaderStates.NO_STATUS;
|
||||||
|
const self = this.getWritable();
|
||||||
|
self.__headerState = state;
|
||||||
|
}
|
||||||
|
|
||||||
|
createDOM(config: EditorConfig): HTMLElement {
|
||||||
|
const element = super.createDOM(config);
|
||||||
|
|
||||||
|
for (const [name, value] of this.__styles.entries()) {
|
||||||
|
element.style.setProperty(name, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return element;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO - Import DOM
|
||||||
|
|
||||||
|
updateDOM(prevNode: CustomTableCellNode): boolean {
|
||||||
|
return super.updateDOM(prevNode)
|
||||||
|
|| this.__styles !== prevNode.__styles;
|
||||||
|
}
|
||||||
|
|
||||||
|
exportDOM(editor: LexicalEditor): DOMExportOutput {
|
||||||
|
const element = this.createDOM(editor._config);
|
||||||
|
return {
|
||||||
|
element
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
exportJSON(): SerializedCustomTableCellNode {
|
||||||
|
return {
|
||||||
|
...super.exportJSON(),
|
||||||
|
type: 'custom-table-cell',
|
||||||
|
styles: Object.fromEntries(this.__styles),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function $createCustomTableCellNode(
|
||||||
|
headerState: TableCellHeaderState,
|
||||||
|
colSpan = 1,
|
||||||
|
width?: number,
|
||||||
|
): CustomTableCellNode {
|
||||||
|
return new CustomTableCellNode(headerState, colSpan, width);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function $isCustomTableCellNode(node: LexicalNode | null | undefined): node is CustomTableCellNode {
|
||||||
|
return node instanceof CustomTableCellNode;
|
||||||
|
}
|
@ -20,6 +20,7 @@ import {DiagramNode} from "./diagram";
|
|||||||
import {EditorUiContext} from "../ui/framework/core";
|
import {EditorUiContext} from "../ui/framework/core";
|
||||||
import {MediaNode} from "./media";
|
import {MediaNode} from "./media";
|
||||||
import {CustomListItemNode} from "./custom-list-item";
|
import {CustomListItemNode} from "./custom-list-item";
|
||||||
|
import {CustomTableCellNode} from "./custom-table-cell-node";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load the nodes for lexical.
|
* Load the nodes for lexical.
|
||||||
@ -33,7 +34,7 @@ export function getNodesForPageEditor(): (KlassConstructor<typeof LexicalNode> |
|
|||||||
CustomListItemNode,
|
CustomListItemNode,
|
||||||
CustomTableNode,
|
CustomTableNode,
|
||||||
TableRowNode,
|
TableRowNode,
|
||||||
TableCellNode,
|
CustomTableCellNode,
|
||||||
ImageNode,
|
ImageNode,
|
||||||
HorizontalRuleNode,
|
HorizontalRuleNode,
|
||||||
DetailsNode, SummaryNode,
|
DetailsNode, SummaryNode,
|
||||||
@ -59,7 +60,19 @@ export function getNodesForPageEditor(): (KlassConstructor<typeof LexicalNode> |
|
|||||||
with: (node: ListItemNode) => {
|
with: (node: ListItemNode) => {
|
||||||
return new CustomListItemNode(node.__value, node.__checked);
|
return new CustomListItemNode(node.__value, node.__checked);
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
replace: TableCellNode,
|
||||||
|
with: (node: TableCellNode) => {
|
||||||
|
const cell = new CustomTableCellNode(
|
||||||
|
node.__headerState,
|
||||||
|
node.__colSpan,
|
||||||
|
node.__width,
|
||||||
|
);
|
||||||
|
cell.__rowSpan = node.__rowSpan;
|
||||||
|
return cell;
|
||||||
}
|
}
|
||||||
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
|
|
||||||
## Main Todo
|
## Main Todo
|
||||||
|
|
||||||
- Alignments: Use existing classes for blocks
|
- Alignments: Use existing classes for blocks (including table cells)
|
||||||
- Alignments: Handle inline block content (image, video)
|
- Alignments: Handle inline block content (image, video)
|
||||||
- Image paste upload
|
- Image paste upload
|
||||||
- Keyboard shortcuts support
|
- Keyboard shortcuts support
|
||||||
|
@ -11,18 +11,19 @@ import {EditorUiContext} from "../../framework/core";
|
|||||||
import {$getSelection, BaseSelection} from "lexical";
|
import {$getSelection, BaseSelection} from "lexical";
|
||||||
import {$isCustomTableNode} from "../../../nodes/custom-table";
|
import {$isCustomTableNode} from "../../../nodes/custom-table";
|
||||||
import {
|
import {
|
||||||
$createTableRowNode,
|
|
||||||
$deleteTableColumn__EXPERIMENTAL,
|
$deleteTableColumn__EXPERIMENTAL,
|
||||||
$deleteTableRow__EXPERIMENTAL,
|
$deleteTableRow__EXPERIMENTAL,
|
||||||
$insertTableColumn__EXPERIMENTAL,
|
$insertTableColumn__EXPERIMENTAL,
|
||||||
$insertTableRow__EXPERIMENTAL, $isTableCellNode,
|
$insertTableRow__EXPERIMENTAL,
|
||||||
$isTableNode, $isTableRowNode, $isTableSelection, $unmergeCell, TableCellNode, TableNode,
|
$isTableNode, $isTableRowNode, $isTableSelection, $unmergeCell, TableCellNode,
|
||||||
} from "@lexical/table";
|
} from "@lexical/table";
|
||||||
import {$getNodeFromSelection, $selectionContainsNodeType} from "../../../utils/selection";
|
import {$getNodeFromSelection, $selectionContainsNodeType} from "../../../utils/selection";
|
||||||
import {$getParentOfType} from "../../../utils/nodes";
|
import {$getParentOfType} from "../../../utils/nodes";
|
||||||
|
import {$isCustomTableCellNode} from "../../../nodes/custom-table-cell-node";
|
||||||
|
import {showCellPropertiesForm} from "../forms/tables";
|
||||||
|
|
||||||
const neverActive = (): boolean => false;
|
const neverActive = (): boolean => false;
|
||||||
const cellNotSelected = (selection: BaseSelection|null) => !$selectionContainsNodeType(selection, $isTableCellNode);
|
const cellNotSelected = (selection: BaseSelection|null) => !$selectionContainsNodeType(selection, $isCustomTableCellNode);
|
||||||
|
|
||||||
export const table: EditorBasicButtonDefinition = {
|
export const table: EditorBasicButtonDefinition = {
|
||||||
label: 'Table',
|
label: 'Table',
|
||||||
@ -34,8 +35,8 @@ export const tableProperties: EditorButtonDefinition = {
|
|||||||
icon: tableIcon,
|
icon: tableIcon,
|
||||||
action(context: EditorUiContext) {
|
action(context: EditorUiContext) {
|
||||||
context.editor.getEditorState().read(() => {
|
context.editor.getEditorState().read(() => {
|
||||||
const cell = $getNodeFromSelection($getSelection(), $isTableCellNode);
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
if (!$isTableCellNode(cell)) {
|
if (!$isCustomTableCellNode(cell)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -54,8 +55,8 @@ export const clearTableFormatting: EditorButtonDefinition = {
|
|||||||
format: 'long',
|
format: 'long',
|
||||||
action(context: EditorUiContext) {
|
action(context: EditorUiContext) {
|
||||||
context.editor.getEditorState().read(() => {
|
context.editor.getEditorState().read(() => {
|
||||||
const cell = $getNodeFromSelection($getSelection(), $isTableCellNode);
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
if (!$isTableCellNode(cell)) {
|
if (!$isCustomTableCellNode(cell)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -72,8 +73,8 @@ export const resizeTableToContents: EditorButtonDefinition = {
|
|||||||
format: 'long',
|
format: 'long',
|
||||||
action(context: EditorUiContext) {
|
action(context: EditorUiContext) {
|
||||||
context.editor.getEditorState().read(() => {
|
context.editor.getEditorState().read(() => {
|
||||||
const cell = $getNodeFromSelection($getSelection(), $isTableCellNode);
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
if (!$isTableCellNode(cell)) {
|
if (!$isCustomTableCellNode(cell)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -159,8 +160,8 @@ export const rowProperties: EditorButtonDefinition = {
|
|||||||
format: 'long',
|
format: 'long',
|
||||||
action(context: EditorUiContext) {
|
action(context: EditorUiContext) {
|
||||||
context.editor.getEditorState().read(() => {
|
context.editor.getEditorState().read(() => {
|
||||||
const cell = $getNodeFromSelection($getSelection(), $isTableCellNode);
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
if (!$isTableCellNode(cell)) {
|
if (!$isCustomTableCellNode(cell)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,11 +314,9 @@ export const cellProperties: EditorButtonDefinition = {
|
|||||||
label: 'Cell properties',
|
label: 'Cell properties',
|
||||||
action(context: EditorUiContext) {
|
action(context: EditorUiContext) {
|
||||||
context.editor.getEditorState().read(() => {
|
context.editor.getEditorState().read(() => {
|
||||||
const cell = $getNodeFromSelection($getSelection(), $isTableCellNode);
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
if ($isTableCellNode(cell)) {
|
if ($isCustomTableCellNode(cell)) {
|
||||||
|
showCellPropertiesForm(cell, context);
|
||||||
const modalForm = context.manager.createModal('cell_properties');
|
|
||||||
modalForm.show({});
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
@ -349,7 +348,7 @@ export const splitCell: EditorButtonDefinition = {
|
|||||||
},
|
},
|
||||||
isActive: neverActive,
|
isActive: neverActive,
|
||||||
isDisabled(selection) {
|
isDisabled(selection) {
|
||||||
const cell = $getNodeFromSelection(selection, $isTableCellNode) as TableCellNode|null;
|
const cell = $getNodeFromSelection(selection, $isCustomTableCellNode) as TableCellNode|null;
|
||||||
if (cell) {
|
if (cell) {
|
||||||
const merged = cell.getRowSpan() > 1 || cell.getColSpan() > 1;
|
const merged = cell.getRowSpan() > 1 || cell.getColSpan() > 1;
|
||||||
return !merged;
|
return !merged;
|
||||||
|
@ -5,6 +5,11 @@ import {
|
|||||||
EditorSelectFormFieldDefinition
|
EditorSelectFormFieldDefinition
|
||||||
} from "../../framework/forms";
|
} from "../../framework/forms";
|
||||||
import {EditorUiContext} from "../../framework/core";
|
import {EditorUiContext} from "../../framework/core";
|
||||||
|
import {$isCustomTableCellNode, CustomTableCellNode} from "../../../nodes/custom-table-cell-node";
|
||||||
|
import {EditorFormModal} from "../../framework/modals";
|
||||||
|
import {$getNodeFromSelection} from "../../../utils/selection";
|
||||||
|
import {$getSelection, ElementFormatType} from "lexical";
|
||||||
|
import {TableCellHeaderStates} from "@lexical/table";
|
||||||
|
|
||||||
const borderStyleInput: EditorSelectFormFieldDefinition = {
|
const borderStyleInput: EditorSelectFormFieldDefinition = {
|
||||||
label: 'Border style',
|
label: 'Border style',
|
||||||
@ -49,10 +54,46 @@ const alignmentInput: EditorSelectFormFieldDefinition = {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export function showCellPropertiesForm(cell: CustomTableCellNode, context: EditorUiContext): EditorFormModal {
|
||||||
|
const styles = cell.getStyles();
|
||||||
|
const modalForm = context.manager.createModal('cell_properties');
|
||||||
|
modalForm.show({
|
||||||
|
width: '', // TODO
|
||||||
|
height: styles.get('height') || '',
|
||||||
|
type: cell.getTag(),
|
||||||
|
h_align: '', // TODO
|
||||||
|
v_align: styles.get('vertical-align') || '',
|
||||||
|
border_width: styles.get('border-width') || '',
|
||||||
|
border_style: styles.get('border-style') || '',
|
||||||
|
border_color: styles.get('border-color') || '',
|
||||||
|
background_color: styles.get('background-color') || '',
|
||||||
|
});
|
||||||
|
return modalForm;
|
||||||
|
}
|
||||||
|
|
||||||
export const cellProperties: EditorFormDefinition = {
|
export const cellProperties: EditorFormDefinition = {
|
||||||
submitText: 'Save',
|
submitText: 'Save',
|
||||||
async action(formData, context: EditorUiContext) {
|
async action(formData, context: EditorUiContext) {
|
||||||
// TODO
|
// TODO - Set for cell selection range
|
||||||
|
context.editor.update(() => {
|
||||||
|
const cell = $getNodeFromSelection($getSelection(), $isCustomTableCellNode);
|
||||||
|
if ($isCustomTableCellNode(cell)) {
|
||||||
|
// TODO - Set width
|
||||||
|
cell.setFormat((formData.get('h_align')?.toString() || '') as ElementFormatType);
|
||||||
|
cell.updateTag(formData.get('type')?.toString() || '');
|
||||||
|
|
||||||
|
const styles = cell.getStyles();
|
||||||
|
styles.set('height', formData.get('height')?.toString() || '');
|
||||||
|
styles.set('vertical-align', formData.get('v_align')?.toString() || '');
|
||||||
|
styles.set('border-width', formData.get('border_width')?.toString() || '');
|
||||||
|
styles.set('border-style', formData.get('border_style')?.toString() || '');
|
||||||
|
styles.set('border-color', formData.get('border_color')?.toString() || '');
|
||||||
|
styles.set('background-color', formData.get('background_color')?.toString() || '');
|
||||||
|
|
||||||
|
cell.setStyles(styles);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
fields: [
|
fields: [
|
||||||
@ -60,31 +101,31 @@ export const cellProperties: EditorFormDefinition = {
|
|||||||
build() {
|
build() {
|
||||||
const generalFields: EditorFormFieldDefinition[] = [
|
const generalFields: EditorFormFieldDefinition[] = [
|
||||||
{
|
{
|
||||||
label: 'Width',
|
label: 'Width', // Colgroup width
|
||||||
name: 'width',
|
name: 'width',
|
||||||
type: 'text',
|
type: 'text',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Height',
|
label: 'Height', // inline-style: height
|
||||||
name: 'height',
|
name: 'height',
|
||||||
type: 'text',
|
type: 'text',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Cell type',
|
label: 'Cell type', // element
|
||||||
name: 'type',
|
name: 'type',
|
||||||
type: 'select',
|
type: 'select',
|
||||||
valuesByLabel: {
|
valuesByLabel: {
|
||||||
'Cell': 'cell',
|
'Cell': 'td',
|
||||||
'Header cell': 'header',
|
'Header cell': 'th',
|
||||||
}
|
}
|
||||||
} as EditorSelectFormFieldDefinition,
|
} as EditorSelectFormFieldDefinition,
|
||||||
{
|
{
|
||||||
...alignmentInput,
|
...alignmentInput, // class: 'align-right/left/center'
|
||||||
label: 'Horizontal align',
|
label: 'Horizontal align',
|
||||||
name: 'h_align',
|
name: 'h_align',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Vertical align',
|
label: 'Vertical align', // inline-style: vertical-align
|
||||||
name: 'v_align',
|
name: 'v_align',
|
||||||
type: 'select',
|
type: 'select',
|
||||||
valuesByLabel: {
|
valuesByLabel: {
|
||||||
@ -98,13 +139,13 @@ export const cellProperties: EditorFormDefinition = {
|
|||||||
|
|
||||||
const advancedFields: EditorFormFieldDefinition[] = [
|
const advancedFields: EditorFormFieldDefinition[] = [
|
||||||
{
|
{
|
||||||
label: 'Border width',
|
label: 'Border width', // inline-style: border-width
|
||||||
name: 'border_width',
|
name: 'border_width',
|
||||||
type: 'text',
|
type: 'text',
|
||||||
},
|
},
|
||||||
borderStyleInput,
|
borderStyleInput, // inline-style: border-style
|
||||||
borderColorInput,
|
borderColorInput, // inline-style: border-color
|
||||||
backgroundColorInput,
|
backgroundColorInput, // inline-style: background-color
|
||||||
];
|
];
|
||||||
|
|
||||||
return new EditorFormTabs([
|
return new EditorFormTabs([
|
||||||
@ -170,7 +211,6 @@ export const rowProperties: EditorFormDefinition = {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
export const tableProperties: EditorFormDefinition = {
|
export const tableProperties: EditorFormDefinition = {
|
||||||
submitText: 'Save',
|
submitText: 'Save',
|
||||||
async action(formData, context: EditorUiContext) {
|
async action(formData, context: EditorUiContext) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user