mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-01-06 07:43:40 +08:00
ad6b26ba97
May show bad option label names on chrome/safari. This was an easy first pass without loads of extra custom UI since we're using native datalists.
97 lines
3.2 KiB
TypeScript
97 lines
3.2 KiB
TypeScript
import {EditorContainerUiElement} from "../core";
|
|
import {el} from "../../../utils/dom";
|
|
import {EditorFormField} from "../forms";
|
|
import {CustomHeadingNode} from "../../../nodes/custom-heading";
|
|
import {$getAllNodesOfType} from "../../../utils/nodes";
|
|
import {$isHeadingNode} from "@lexical/rich-text";
|
|
import {uniqueIdSmall} from "../../../../services/util";
|
|
|
|
export class LinkField extends EditorContainerUiElement {
|
|
protected input: EditorFormField;
|
|
protected headerMap = new Map<string, CustomHeadingNode>();
|
|
|
|
constructor(input: EditorFormField) {
|
|
super([input]);
|
|
|
|
this.input = input;
|
|
}
|
|
|
|
buildDOM(): HTMLElement {
|
|
const listId = 'editor-form-datalist-' + this.input.getName() + '-' + Date.now();
|
|
const inputOuterDOM = this.input.getDOMElement();
|
|
const inputFieldDOM = inputOuterDOM.querySelector('input');
|
|
inputFieldDOM?.setAttribute('list', listId);
|
|
inputFieldDOM?.setAttribute('autocomplete', 'off');
|
|
const datalist = el('datalist', {id: listId});
|
|
|
|
const container = el('div', {
|
|
class: 'editor-link-field-container',
|
|
}, [inputOuterDOM, datalist]);
|
|
|
|
inputFieldDOM?.addEventListener('focusin', () => {
|
|
this.updateDataList(datalist);
|
|
});
|
|
|
|
inputFieldDOM?.addEventListener('input', () => {
|
|
const value = inputFieldDOM.value;
|
|
const header = this.headerMap.get(value);
|
|
if (header) {
|
|
this.updateFormFromHeader(header);
|
|
}
|
|
});
|
|
|
|
return container;
|
|
}
|
|
|
|
updateFormFromHeader(header: CustomHeadingNode) {
|
|
this.getHeaderIdAndText(header).then(({id, text}) => {
|
|
console.log('updating form', id, text);
|
|
const modal = this.getContext().manager.getActiveModal('link');
|
|
if (modal) {
|
|
modal.getForm().setValues({
|
|
url: `#${id}`,
|
|
text: text,
|
|
title: text,
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
getHeaderIdAndText(header: CustomHeadingNode): Promise<{id: string, text: string}> {
|
|
return new Promise((res) => {
|
|
this.getContext().editor.update(() => {
|
|
let id = header.getId();
|
|
console.log('header', id, header.__id);
|
|
if (!id) {
|
|
id = 'header-' + uniqueIdSmall();
|
|
header.setId(id);
|
|
}
|
|
|
|
const text = header.getTextContent();
|
|
res({id, text});
|
|
});
|
|
});
|
|
}
|
|
|
|
updateDataList(listEl: HTMLElement) {
|
|
this.getContext().editor.getEditorState().read(() => {
|
|
const headers = $getAllNodesOfType($isHeadingNode) as CustomHeadingNode[];
|
|
|
|
this.headerMap.clear();
|
|
const listEls: HTMLElement[] = [];
|
|
|
|
for (const header of headers) {
|
|
const key = 'header-' + header.getKey();
|
|
this.headerMap.set(key, header);
|
|
listEls.push(el('option', {
|
|
value: key,
|
|
label: header.getTextContent().substring(0, 54),
|
|
}));
|
|
}
|
|
|
|
listEl.innerHTML = '';
|
|
listEl.append(...listEls);
|
|
});
|
|
}
|
|
}
|