UNPKG

openchemlib

Version:
1,341 lines (1,325 loc) 3.58 MB
// lib/canvas_editor/clipboard_handler.js var ClipboardHandler = class { copyMolecule(molecule) { const data = molecule.getIDCodeAndCoordinates(); navigator.clipboard.writeText(`${data.idCode} ${data.coordinates}`); } pasteMolecule() { return null; } }; // lib/canvas_editor/utils.js var chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; var lookup = new Uint8Array(256); for (let i = 0; i < chars.length; i++) { lookup[chars.charCodeAt(i)] = i; } function decodeBase64(base64) { let bufferLength = base64.length * 0.75; let len = base64.length; let i; let p = 0; let encoded1; let encoded2; let encoded3; let encoded4; if (base64.at(-1) === "=") { bufferLength--; if (base64.at(-2) === "=") { bufferLength--; } } const arraybuffer = new ArrayBuffer(bufferLength); const bytes = new Uint8Array(arraybuffer); for (i = 0; i < len; i += 4) { encoded1 = lookup[base64.charCodeAt(i)]; encoded2 = lookup[base64.charCodeAt(i + 1)]; encoded3 = lookup[base64.charCodeAt(i + 2)]; encoded4 = lookup[base64.charCodeAt(i + 3)]; bytes[p++] = encoded1 << 2 | encoded2 >> 4; bytes[p++] = (encoded2 & 15) << 4 | encoded3 >> 2; bytes[p++] = (encoded3 & 3) << 6 | encoded4 & 63; } return arraybuffer; } function toHex(v) { return v.toString(16).padStart(2, "0"); } var hidpiScaleFactor = globalThis.devicePixelRatio || 1; // lib/canvas_editor/draw_context.js var DrawContext = class { /** * * @param {CanvasRenderingContext2D} ctx */ constructor(ctx2) { this.ctx = ctx2; this.ctx.textAlign = "left"; this.ctx.textBaseline = "top"; this.currentFontSize = 12; this.currentFont = "12px sans-serif"; this.ctx.font = this.currentFont; this.currentColor = "#000000"; this.currentLineWidth = 1; this.canvasCache = /* @__PURE__ */ new Map(); } clearRect(x, y, w, h) { this.ctx.clearRect(x, y, w, h); } getBackgroundRGB() { return 16777215; } getForegroundRGB() { return 0; } getSelectionBackgroundRGB() { return 12310268; } getLineWidth() { return this.currentLineWidth; } setRGB(rgb) { const r = rgb >>> 16 & 255; const g = rgb >>> 8 & 255; const b = rgb >>> 0 & 255; this.currentColor = `#${toHex(r)}${toHex(g)}${toHex(b)}`; this.ctx.fillStyle = this.currentColor; this.ctx.strokeStyle = this.currentColor; } setFont(size, isBold, isItalic) { this.currentFontSize = size; this.currentFont = `${isBold ? "bold" : ""} ${isItalic ? "italic" : ""} ${size}px sans-serif`; this.ctx.font = this.currentFont; } getFontSize() { return this.currentFontSize; } getBounds(s) { const metrics = this.ctx.measureText(s); return { x: metrics.actualBoundingBoxLeft, y: metrics.actualBoundingBoxAscent, width: metrics.actualBoundingBoxRight, height: metrics.actualBoundingBoxAscent }; } drawString(x, y, s) { this.ctx.fillText(s, x, y); } drawCenteredString(x, y, s) { this.ctx.textAlign = "center"; this.ctx.textBaseline = "middle"; this.ctx.fillText(s, x, y); this.ctx.textAlign = "left"; this.ctx.textBaseline = "top"; } setLineWidth(lineWidth) { this.currentLineWidth = lineWidth; this.ctx.lineWidth = lineWidth; } fillRectangle(x, y, w, h) { this.ctx.fillRect(x, y, w, h); } fillCircle(x, y, d) { const r = d / 2; this.ctx.beginPath(); this.ctx.arc(x + r, y + r, r, 0, 2 * Math.PI); this.ctx.fill(); } drawLine(x1, y1, x2, y2) { this.ctx.beginPath(); this.ctx.moveTo(x1, y1); this.ctx.lineTo(x2, y2); this.ctx.stroke(); } drawPolygon(p) { this.ctx.beginPath(); this.ctx.moveTo(p.getX(0), p.getY(0)); for (let i = 1; i < p.getSize(); i++) { this.ctx.lineTo(p.getX(i), p.getY(i)); } this.ctx.stroke(); } drawRectangle(x, y, w, h) { this.ctx.strokeRect(x, y, w, h); } fillPolygon(p) { this.ctx.beginPath(); this.ctx.moveTo(p.getX(0), p.getY(0)); for (let i = 1; i < p.getSize(); i++) { this.ctx.lineTo(p.getX(i), p.getY(i)); } this.ctx.fill(); } drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh) { if (arguments.length !== 9) { throw new Error( `drawImage call with ${arguments.length} arguments unimplemented` ); } let fullScaleCanvas = this.canvasCache.get(image); if (!fullScaleCanvas) { fullScaleCanvas = document.createElement("canvas"); const imageData = image.imageData; fullScaleCanvas.width = imageData.width; fullScaleCanvas.height = imageData.height; const fullScaleContext = fullScaleCanvas.getContext("2d"); fullScaleContext.globalAlpha = 0; fullScaleContext.putImageData(imageData, 0, 0); this.canvasCache.set(image, fullScaleCanvas); } this.ctx.drawImage(fullScaleCanvas, sx, sy, sw, sh, dx, dy, dw, dh); } isDarkBackground() { return false; } }; // lib/canvas_editor/editor_area.js var EditorArea = class { constructor(canvasElement, onChange) { this.canvasElement = canvasElement; this.changeListener = onChange; this.drawContext = new DrawContext(this.canvasElement.getContext("2d")); } getBackgroundRGB() { return 16777215; } getCanvasWidth() { return this.canvasElement.width; } getCanvasHeight() { return this.canvasElement.height; } getDrawContext() { return this.drawContext; } onChange(what, isUserEvent) { this.changeListener?.({ what, isUserEvent }); } getClipboardHandler() { return new ClipboardHandler(); } }; // lib/canvas_editor/editor_stylesheet.js var styles = ` /* We can customize editor styles here. */ `; var stylesheet; function getEditorStylesheet() { if (stylesheet) { return stylesheet; } const sheet = new CSSStyleSheet(); sheet.replaceSync(styles); stylesheet = sheet; return sheet; } // lib/canvas_editor/events.js function addPointerListeners(canvasElement, drawArea, JavaEditorArea) { let pointerDownId = -1; function fireMouseEvent(what, ev, clickCount = 0) { if (ev.button > 0) { return; } drawArea.fireMouseEvent( what, ev.button + 1, clickCount, Math.round(ev.offsetX * hidpiScaleFactor), Math.round(ev.offsetY * hidpiScaleFactor), ev.shiftKey, ev.ctrlKey, ev.altKey, ev.button === 2 ); } canvasElement.addEventListener("pointerdown", (ev) => { if (pointerDownId === -1) { pointerDownId = ev.pointerId; fireMouseEvent(JavaEditorArea.MOUSE_EVENT_PRESSED, ev); } }); function handlePointerUp(ev) { if (pointerDownId === ev.pointerId) { pointerDownId = -1; fireMouseEvent(JavaEditorArea.MOUSE_EVENT_RELEASED, ev); } } document.addEventListener("pointerup", handlePointerUp); canvasElement.addEventListener("click", (ev) => { fireMouseEvent(JavaEditorArea.MOUSE_EVENT_CLICKED, ev, ev.detail); }); canvasElement.addEventListener("pointerenter", (ev) => { fireMouseEvent(JavaEditorArea.MOUSE_EVENT_ENTERED, ev); }); canvasElement.addEventListener("pointerleave", (ev) => { fireMouseEvent(JavaEditorArea.MOUSE_EVENT_EXITED, ev); }); canvasElement.addEventListener("pointermove", (ev) => { if (pointerDownId !== -1) { if (pointerDownId === ev.pointerId) { fireMouseEvent(JavaEditorArea.MOUSE_EVENT_DRAGGED, ev); } } else { fireMouseEvent(JavaEditorArea.MOUSE_EVENT_MOVED, ev); } }); return () => { document.removeEventListener("pointerup", handlePointerUp); }; } function addKeyboardListeners(parentElement, canvasElement, editorArea, JavaEditorArea, Molecule2) { const isMac = typeof navigator !== "undefined" && navigator.platform === "MacIntel"; const isMenuKey = (ev) => isMac && ev.metaKey || !isMac && ev.ctrlKey; function fireKeyEvent(what, ev) { const key = getKeyFromEvent(ev, JavaEditorArea); if (key === null) return; editorArea.fireKeyEvent( what, key, ev.altKey, ev.ctrlKey, ev.shiftKey, isMenuKey(ev) ); } canvasElement.addEventListener("keydown", (ev) => { if (isMenuKey(ev) && ev.key === "c") return; if (isMenuKey(ev) && ev.key === "v") return; fireKeyEvent(JavaEditorArea.KEY_EVENT_PRESSED, ev); }); canvasElement.addEventListener("keyup", (ev) => { fireKeyEvent(JavaEditorArea.KEY_EVENT_RELEASED, ev); }); parentElement.addEventListener("paste", (ev) => { const textData = ev.clipboardData.getData("text"); const molecule = Molecule2.fromText(textData); if (molecule && molecule.getAllAtoms() > 0) { editorArea.addPastedOrDropped(molecule); } }); return () => { }; } function getKeyFromEvent(ev, JavaEditorArea) { switch (ev.key) { case "Control": return JavaEditorArea.KEY_CTRL; case "Alt": return JavaEditorArea.KEY_ALT; case "Shift": return JavaEditorArea.KEY_SHIFT; case "Delete": case "Backspace": return JavaEditorArea.KEY_DELETE; // Backspace is currently unused by the Java code, so we remap it. // return JavaEditorArea.KEY_BACKSPACE; case "F1": return JavaEditorArea.KEY_HELP; case "Escape": return JavaEditorArea.KEY_ESCAPE; case "Enter": return JavaEditorArea.KEY_ENTER; default: if (ev.key.length === 1) { return ev.key.codePointAt(0); } else { return null; } } } // lib/canvas_editor/toolbar.js var Toolbar = class { constructor(canvasElement) { this.canvasElement = canvasElement; this.drawContext = new DrawContext(this.canvasElement.getContext("2d")); } setDimensions(width, height) { this.canvasElement.width = width; this.canvasElement.style.width = `${width / hidpiScaleFactor}px`; this.canvasElement.height = height; this.canvasElement.style.height = `${height / hidpiScaleFactor}px`; } getDrawContext() { return this.drawContext; } getBackgroundRGB() { return 16777215; } getForegroundRGB() { return 0; } }; // lib/canvas_editor/cursors_24px.js var cursors = { "chain.png": "url()", "eraser.png": "url()", "fist.png": "url()", "hand.png": "url()", "handPlus.png": "url()", "invisible.png": "url()", "lasso.png": "url()", "lassoPlus.png": "url()", "pointingHand.png": "url()", "rect.png": "url()", "rectPlus.png": "url()", "zoom.png": "url()" }; // lib/canvas_editor/cursor_manager_24.js var computedCursors = /* @__PURE__ */ Object.create(null); var scaleFactor = 3 / 4; var CursorManager = class { constructor(JavaEditorArea) { this.HOTSPOT_32 = JavaEditorArea.HOTSPOT_32; this.IMAGE_NAME_32 = JavaEditorArea.IMAGE_NAME_32; this.cPointerCursor = JavaEditorArea.cPointerCursor; this.cTextCursor = JavaEditorArea.cTextCursor; } getCursor(cursor) { if (computedCursors[cursor]) { return computedCursors[cursor]; } if (this.IMAGE_NAME_32[cursor]) { return this.buildCursor(cursor); } switch (cursor) { case this.cPointerCursor: return "default"; case this.cTextCursor: return "text"; default: throw new Error(`Unknown cursor: ${cursor}`); } } buildCursor(cursor) { const cursorName = this.IMAGE_NAME_32[cursor]; const cursorUrl = cursors[cursorName]; const builtCursor = `${cursorUrl} ${this.HOTSPOT_32[cursor * 2] * scaleFactor} ${this.HOTSPOT_32[cursor * 2 + 1] * scaleFactor}, default`; computedCursors[cursor] = builtCursor; return builtCursor; } }; // lib/canvas_editor/editor_dialog.js var EditorDialog = class { /** * * @param {string} title * @param {HTMLElement} rootElement */ constructor(title, rootElement) { this.title = title; this.rootElement = rootElement; this.elements = []; this.dialogElement = null; } setLayout(hLayout, vLayout) { this.hLayout = generateLayout(hLayout); this.vLayout = generateLayout(vLayout); } add(component, x, y, x2, y2) { this.elements.push({ component, x, y, x2, y2 }); } createTextField(width, height) { return new TextField(width, height); } createLabel(text) { return new Label(text); } createComboBox() { return new ComboBox(); } createCheckBox(text) { return new CheckBox(text); } setEventConsumer(consumer) { this.consumer = consumer; } showMessage(message) { window.alert(message); } showDialog() { const dialog = document.createElement("dialog"); const rootElementBounds = this.rootElement.getBoundingClientRect(); Object.assign(dialog.style, { position: "absolute", marginBlock: 0, left: `${rootElementBounds.left}px`, right: `${document.body.parentElement.clientWidth - rootElementBounds.right}px`, top: `${this.rootElement.offsetTop + 30}px` }); this.dialogElement = dialog; this.rootElement.getRootNode().append(dialog); const grid = document.createElement("div"); grid.style.display = "grid"; grid.style.gridTemplateColumns = this.hLayout; grid.style.gridTemplateRows = this.vLayout; dialog.append(grid); for (const { component, x, y, x2, y2 } of this.elements) { const div = document.createElement("div"); if (x2 === void 0) { div.style.gridColumn = `${x + 1} / ${x + 2}`; div.style.gridRow = `${y + 1} / ${y + 2}`; } else { div.style.gridColumn = `${x + 1} / ${x2 + 2}`; div.style.gridRow = `${y + 1} / ${y2 + 2}`; } div.append(component.getElement()); grid.append(div); } const buttons = document.createElement("div"); buttons.style.display = "flex"; buttons.style.flexDirection = "row-reverse"; buttons.style.gap = "15px"; const okButton = document.createElement("button"); okButton.textContent = "OK"; okButton.addEventListener("click", () => { this.consumer.fireOk(); }); buttons.append(okButton); const cancelButton = document.createElement("button"); cancelButton.textContent = "Cancel"; cancelButton.addEventListener("click", () => { this.consumer.fireCancel(); }); buttons.append(cancelButton); dialog.append(buttons); dialog.showModal(); dialog.addEventListener("cancel", () => { this.consumer.fireCancel(); }); } disposeDialog() { if (this.dialogElement !== null) { this.dialogElement.remove(); this.dialogElement = null; } } }; var Component = class { setEventHandler(eventHandler) { this.eventHandler = eventHandler; } fireEvent(what, value) { this.eventHandler(what, value); } }; var Label = class extends Component { constructor(text) { super(); this.element = document.createElement("label"); this.setText(text); } setText(text) { this.element.textContent = text; } getElement() { return this.element; } }; var TextField = class extends Component { constructor() { super(); this.element = document.createElement("input"); this.element.type = "text"; } setText(text) { this.element.value = text; } getText() { return this.element.value; } getElement() { return this.element; } }; var ComboBox = class extends Component { constructor() { super(); this.element = document.createElement("select"); this.element.addEventListener("change", () => { this.fireEvent(2, this.element.selectedIndex); }); } setEnabled(enabled) { this.element.disabled = !enabled; } addItem(item) { const option = document.createElement("option"); option.textContent = item; this.element.append(option); } getSelectedIndex() { return this.element.selectedIndex; } setSelectedIndex(index) { this.element.selectedIndex = index; } setSelectedItem(item) { const options = this.element.options; for (let i = 0; i < options.length; i++) { if (options[i].textContent === item) { this.element.selectedIndex = i; } } } getSelectedItem() { return this.element.options[this.element.selectedIndex].textContent; } removeAllItems() { this.element.innerHTML = ""; } getElement() { return this.element; } }; var CheckBox = class extends Component { constructor(text) { super(); const label = document.createElement("label"); const input = document.createElement("input"); input.type = "checkbox"; input.addEventListener("change", () => { this.fireEvent(3, input.checked ? 1 : 0); }); label.append(input); label.append(text); this.element = label; this.checkBox = input; } setEnabled(enabled) { this.checkBox.disabled = !enabled; } isSelected() { return this.checkBox.checked; } setSelected(selected) { this.checkBox.checked = selected; } getElement() { return this.element; } }; function generateLayout(layout) { return layout.map((dim) => { if (dim > 0) { return `${dim}px`; } else { return "auto"; } }).join(" "); } // lib/canvas_editor/editor_image.js var EditorImage = class { /** * * @param {ImageData} imageData */ constructor(imageData) { this.imageData = imageData; this.dataView = new DataView(imageData.data.buffer); } getWidth() { return this.imageData.width; } getHeight() { return this.imageData.height; } getRGB(x, y) { const color = this.dataView.getInt32( (y * this.imageData.width + x) * 4, false ); const alpha = color & 255; return alpha << 24 | color >>> 8; } setRGB(x, y, argb) { const alpha = argb >>> 24 & 255; const rgb = argb << 8 | alpha; this.dataView.setInt32((y * this.imageData.width + x) * 4, rgb, false); } toDataURL() { const canvas = document.createElement("canvas"); const ctx2 = canvas.getContext("2d"); canvas.width = this.imageData.width; canvas.height = this.imageData.height; ctx2.putImageData(this.imageData, 0, 0); return canvas.toDataURL("image/png"); } }; // lib/canvas_editor/ui_helper.js var UIHelper = class { /** * * @param {HTMLCanvasElement} canvasElement * @param {HTMLElement} dialogRoot * @param JavaEditorArea */ constructor(canvasElement, dialogRoot, JavaEditorArea) { this.canvasElement = canvasElement; this.dialogRoot = dialogRoot; this.JavaEditorArea = JavaEditorArea; } register(javaUiHelper) { this.javaUiHelper = javaUiHelper; this.cursorManager = new CursorManager(this.JavaEditorArea, javaUiHelper); } grabFocus() { this.canvasElement.focus({ preventScroll: true }); } setCursor(cursor) { this.canvasElement.style.cursor = this.cursorManager.getCursor(cursor); } showHelpDialog() { } createImage(width, height) { const imageData = new ImageData(width, height); return new EditorImage(imageData); } createImageFromBase64(width, height, base64) { base64 = base64.replaceAll( /%\d+%/g, (match) => "A".repeat(Number(match.slice(1, -1))) ); const decoded = decodeBase64(base64); const typedArray = new Uint8ClampedArray(decoded); const imageData = new ImageData(typedArray, width, height); return new EditorImage(imageData); } createDialog(title) { return new EditorDialog(title, this.dialogRoot); } runLater(fn) { if (typeof requestAnimationFrame === "function") { requestAnimationFrame(fn); } else if (typeof setImmediate === "function") { setImmediate(fn); } else { setTimeout(fn, 0); } } }; // lib/canvas_editor/create_editor.js function createEditor(parentElement, options, onChange, JavaEditorArea, JavaEditorToolbar, JavaUIHelper, Molecule2, Reaction2) { const { readOnly = false, initialMode = "molecule", initialFragment = false } = options; const rootElement = document.createElement("div"); rootElement.dataset.openchemlibCanvasEditor = "true"; Object.assign(rootElement.style, { width: "100%", height: "100%", display: "flex", flexDirection: "row", alignItems: "start", backgroundColor: "white", // Prevent side effects of pointer events, like scrolling the page or text // selection. touchAction: "none", userSelect: "none", webkitUserSelect: "none" }); const shadowRoot = rootElement.attachShadow({ mode: "open" }); shadowRoot.adoptedStyleSheets = [getEditorStylesheet()]; let toolbarCanvas = null; if (!readOnly) { toolbarCanvas = document.createElement("canvas"); shadowRoot.append(toolbarCanvas); } const editorContainer = document.createElement("div"); Object.assign(editorContainer.style, { width: "100%", height: "100%" }); shadowRoot.append(editorContainer); const editorCanvas = document.createElement("canvas"); editorCanvas.tabIndex = 0; Object.assign(editorCanvas.style, { outline: "none" }); editorContainer.append(editorCanvas); parentElement.append(rootElement); const uiHelper = new JavaUIHelper( new UIHelper(editorCanvas, editorContainer, JavaEditorArea) ); const editorArea = new JavaEditorArea( computeMode(initialMode, JavaEditorArea), new EditorArea(editorCanvas, onChange), uiHelper ); if (initialFragment) { if (initialMode === "molecule") { const fragmentMolecule = new Molecule2(0, 0); fragmentMolecule.setFragment(true); editorArea.setMolecule(fragmentMolecule); } else { const fragmentReaction = Reaction2.create(); fragmentReaction.setFragment(true); editorArea.setReaction(fragmentReaction); } } uiHelper.setEditorArea(editorArea); const toolbar = readOnly ? null : new JavaEditorToolbar(editorArea, new Toolbar(toolbarCanvas), uiHelper); function updateCanvasDimensions(containerSize2) { editorCanvas.style.width = `${containerSize2.width}px`; editorCanvas.width = Math.floor(containerSize2.width * hidpiScaleFactor); editorCanvas.style.height = `${containerSize2.height}px`; editorCanvas.height = Math.floor(containerSize2.height * hidpiScaleFactor); if (containerSize2.width > 0 && containerSize2.height > 0) { editorArea.repaint(); } } const containerSize = editorContainer.getBoundingClientRect(); updateCanvasDimensions(containerSize); const resizeObserver = new ResizeObserver(([entry]) => { updateCanvasDimensions(entry.contentRect); }); resizeObserver.observe(editorContainer); let removePointerListeners = null; let removeKeyboardListeners = null; let removeToolbarPointerListeners = null; if (!readOnly) { removePointerListeners = addPointerListeners( editorCanvas, editorArea, JavaEditorArea ); removeKeyboardListeners = addKeyboardListeners( editorContainer, editorCanvas, editorArea, JavaEditorArea, Molecule2 ); removeToolbarPointerListeners = addPointerListeners( toolbarCanvas, toolbar, JavaEditorArea ); } function destroy() { rootElement.remove(); resizeObserver.disconnect(); removePointerListeners?.(); removeKeyboardListeners?.(); removeToolbarPointerListeners?.(); } return { editorArea, toolbar, uiHelper, destroy }; } function computeMode(initialMode, JavaEditorArea) { switch (initialMode) { case "molecule": return 0; case "reaction": return JavaEditorArea.MODE_REACTION | JavaEditorArea.MODE_MULTIPLE_FRAGMENTS; default: throw new Error(`Invalid initial mode: ${initialMode}`); } } // lib/canvas_editor/init/canvas_editor.js function initCanvasEditor(JavaEditorArea, JavaEditorToolbar, JavaUIHelper, Molecule2, Reaction2) { class CanvasEditor2 { #editorArea; // Can be useful for debugging. /* eslint-disable no-unused-private-class-members */ #toolbar; #uiHelper; /* eslint-enable no-unused-private-class-members */ #onChange; #changeEventMap; #destroy; constructor(parentElement, options = {}) { const { editorArea, toolbar, uiHelper, destroy } = createEditor( parentElement, options, (event) => this.#handleChange(event), JavaEditorArea, JavaEditorToolbar, JavaUIHelper, Molecule2, Reaction2 ); this.#editorArea = editorArea; this.#toolbar = toolbar; this.#uiHelper = uiHelper; this.#onChange = null; this.#changeEventMap = { [JavaEditorArea.EDITOR_EVENT_MOLECULE_CHANGED]: "molecule", [JavaEditorArea.EDITOR_EVENT_SELECTION_CHANGED]: "selection", [JavaEditorArea.EDITOR_EVENT_HIGHLIGHT_ATOM_CHANGED]: "highlight-atom", [JavaEditorArea.EDITOR_EVENT_HIGHLIGHT_BOND_CHANGED]: "highlight-bond" }; this.#destroy = destroy; } getMode() { this.#checkNotDestroyed(); const mode = this.#editorArea.getMode(); const isReaction = mode & JavaEditorArea.MODE_REACTION !== 0; return isReaction ? "reaction" : "molecule"; } setMolecule(molecule) { this.#checkNotDestroyed(); this.#editorArea.setMolecule(molecule); } getMolecule() { this.#checkNotDestroyed(); return this.#editorArea.getMolecule(); } setReaction(reaction) { this.#checkNotDestroyed(); this.#editorArea.setReaction(reaction); } getReaction() { this.#checkNotDestroyed(); return this.#editorArea.getReaction(); } setOnChangeListener(onChange) { this.#checkNotDestroyed(); this.#onChange = onChange; } removeOnChangeListener() { this.#checkNotDestroyed(); this.#onChange = null; } clearAll() { this.#checkNotDestroyed(); this.#editorArea.clearAll(); } destroy() { this.#checkNotDestroyed(); this.#destroy(); this.#editorArea = null; this.#toolbar = null; this.#uiHelper = null; this.#onChange = null; this.#destroy = null; } get isDestroyed() { return !this.#editorArea; } moleculeChanged() { this.#checkNotDestroyed(); this.#editorArea.moleculeChanged(); } #checkNotDestroyed() { if (this.isDestroyed) { throw new Error("CanvasEditor has been destroyed"); } } /** * @param {{ what: number; isUserEvent: boolean; }} event */ #handleChange(event) { if (!this.#onChange) return; const { what, isUserEvent } = event; this.#onChange({ type: this.#changeEventMap[what], isUserEvent }); } } return CanvasEditor2; } // lib/canvas_editor/init/canvas_editor_element.js function initCanvasEditorElement(CanvasEditor2, Molecule2, Reaction2, ReactionEncoder2) { class CanvasEditorElement extends HTMLElement { /** @type {{MOLECULE: 'molecule', REACTION: 'reaction'}} */ static MODE = Object.freeze( /* @__PURE__ */ Object.create({ MOLECULE: "molecule", REACTION: "reaction" }) ); static observedAttributes = Object.freeze([ "idcode", "fragment", "mode", "readonly" ]); /** * @type {{mode: 'molecule' | 'reaction', fragment: boolean, idcode: string, readonly: boolean}} */ #state = { idcode: "", fragment: false, mode: CanvasEditorElement.MODE_MOLECULE, readonly: false }; get idcode() { return this.#state.idcode; } set idcode(value) { this.#state.idcode = String(value); this.setAttribute("idcode", this.#state.idcode); } get fragment() { return this.#state.fragment; } set fragment(value) { this.#state.fragment = Boolean(value); if (this.#state.fragment) { this.setAttribute("fragment", ""); } else { this.removeAttribute("fragment"); } } get mode() { return this.#state.mode; } set mode(value) { this.#state.mode = String(value); this.setAttribute("mode", this.#state.mode); } get readonly() { return this.#state.readonly; } set readonly(value) { this.#state.readonly = Boolean(value); if (this.#state.readonly) { this.setAttribute("readonly", ""); } else { this.removeAttribute("readonly"); } } /* --- custom element api --- */ /** * @param {Molecule} molecule * @this {CanvasEditorElement} */ setMolecule(molecule) { this.fragment = molecule.isFragment(); this.idcode = `${molecule.getIDCode()} ${molecule.getIDCoordinates()}`; this.#editor.setMolecule(molecule); } /** * @return {Molecule} * @this {CanvasEditorElement} */ getMolecule() { return this.#editor.getMolecule(); } /** * @param {Reaction} reaction * @this {CanvasEditorElement} */ setReaction(reaction) { this.fragment = reaction.isFragment(); this.idcode = ReactionEncoder2.encode(reaction, { keepAbsoluteCoordinates: true, mode: ReactionEncoder2.INCLUDE_MAPPING | ReactionEncoder2.INCLUDE_COORDS | ReactionEncoder2.RETAIN_REACTANT_AND_PRODUCT_ORDER }) ?? ""; this.#editor.setReaction(reaction); } /** * @return {Reaction} * @this {CanvasEditorElement} */ getReaction() { return this.#editor.getReaction(); } /** * @this {CanvasEditorElement} */ clearAll() { this.#editor.clearAll(); this.idcode = ""; } /** * @this {CanvasEditorElement} */ moleculeChanged() { this.#editor.moleculeChanged(); } /* --- internals --- */ /** @type {CanvasEditor} */ #editor; /** * @this {CanvasEditorElement} */ #initEditor() { if (this.#editor) return; this.#editor = new CanvasEditor2(this, { readOnly: this.readonly, initialMode: this.mode }); this.#editor.setOnChangeListener(this.#handleChange); requestIdleCallback(() => this.#initIdCode()); } /** * @this {CanvasEditorElement} */ #initIdCode() { switch (this.mode) { case CanvasEditorElement.MODE.MOLECULE: { return this.#initMolecule(); } case CanvasEditorElement.MODE.REACTION: { return this.#initReaction(); } default: throw new Error(`Mode ${this.mode} is not supported`); } } /** * @param {string} idcodeAndCoordinates */ #moleculeFromIdCode(idcodeAndCoordinates) { const index = idcodeAndCoordinates.indexOf(" "); if (index === -1) { return Molecule2.fromIDCode(idcodeAndCoordinates); } const idcode = idcodeAndCoordinates.slice(0, index); const coordinates = idcodeAndCoordinates.slice(index + 1); return Molecule2.fromIDCode(idcode, coordinates); } /** * @this {CanvasEditorElement} */ #initMolecule() { const molecule = this.#moleculeFromIdCode(this.idcode); molecule.setFragment(this.fragment); this.#editor.setMolecule(molecule); } /** * @this {CanvasEditorElement} */ #initReaction() { const reaction = ReactionEncoder2.decode(this.idcode, { ensureCoordinates: true }) ?? Reaction2.create(); reaction.setFragment(this.fragment); this.#editor.setReaction(reaction); } #ignoreAttributeChange = false; /** * @param {() => void} fn * @this {CanvasEditorElement} */ #wrapIgnoreAttributeChange(fn) { this.#ignoreAttributeChange = true; try { fn(); } finally { this.#ignoreAttributeChange = false; } } /** * @param {{ * type: 'molecule' | 'selection' | 'highlight-atom' | 'highlight-bond'; * isUserEvent: boolean; * }} editorEventOnChange */ #handleChange = (editorEventOnChange) => { const idcode = this.idcode; const fragment = this.fragment; this.#wrapIgnoreAttributeChange(() => { if (editorEventOnChange.type !== "molecule") return; switch (this.mode) { case CanvasEditorElement.MODE.MOLECULE: { const molecule = this.getMolecule(); this.idcode = `${molecule.getIDCode()} ${molecule.getIDCoordinates()}`; this.fragment = molecule.isFragment(); break; } case CanvasEditorElement.MODE.REACTION: { const reaction = this.getReaction(); this.idcode = ReactionEncoder2.encode(reaction, { keepAbsoluteCoordinates: true, mode: ReactionEncoder2.INCLUDE_MAPPING | ReactionEncoder2.INCLUDE_COORDS | ReactionEncoder2.RETAIN_REACTANT_AND_PRODUCT_ORDER }); this.fragment = reaction.isFragment(); break; } default: throw new Error(`Unsupported mode ${this.mode}`); } }); const domEvent = new CustomEvent("change", { detail: editorEventOnChange, bubbles: true }); this.dispatchEvent(domEvent); if (editorEventOnChange.mode !== "molecule") return; if (this.idcode !== idcode) { const idcodeChangeEvent = new CustomEvent("idcode-changed", { detail: this.idcode, bubbles: true }); this.dispatchEvent(idcodeChangeEvent); } if (this.fragment !== fragment) { const fragmentChangeEvent = new CustomEvent("fragment-changed", { detail: this.fragment, bubbles: true }); this.dispatchEvent(fragmentChangeEvent); } }; #destroyEditor() { if (!this.#editor) return; this.#editor.destroy(); this.#editor = void 0; } #resetEditor() { this.#destroyEditor(); this.#initEditor(); } /* --- lifecycle hooks --- */ /** * Custom element added to page. */ connectedCallback() { this.#state = { idcode: this.getAttribute("idcode") || "", fragment: this.hasAttribute("fragment"), mode: this.getAttribute("mode") || CanvasEditorElement.MODE.MOLECULE, readonly: this.hasAttribute("readonly") }; this.#initEditor(); } /** * Custom element removed from page. */ disconnectedCallback() { this.#destroyEditor(); } /** * Custom element moved to new page. */ adoptedCallback() { this.connectedCallback(); } /** * Attribute ${name} has changed from ${oldValue} to ${newValue} * * Sync attribute changes to internal state. * propagate changes to editor. */ attributeChangedCallback(name, oldValue, newValue) { if (!this.#editor) return; if (this.#ignoreAttributeChange) return; const mutatorHandler = (() => { switch (name) { case "idcode": { this.#state.idcode = String(newValue); return () => this.#initIdCode(); } case "fragment": { this.#state.fragment = newValue !== null; return () => this.#initIdCode(); } case "mode": { this.#state.mode = String(newValue); return () => this.#resetEditor(); } case "readonly": { this.#state.readonly = newValue !== null; return () => this.#resetEditor(); } default: throw new Error("unsupported attribute change"); } })(); mutatorHandler(); } } return CanvasEditorElement; } // lib/canvas_editor/init/index.js function init(OCL) { const { GenericEditorArea: JavaEditorArea, GenericEditorToolbar: JavaEditorToolbar, GenericUIHelper: JavaUIHelper, Molecule: Molecule2, Reaction: Reaction2, ReactionEncoder: ReactionEncoder2 } = OCL; const CanvasEditor2 = initCanvasEditor( JavaEditorArea, JavaEditorToolbar, JavaUIHelper, Molecule2, Reaction2 ); function registerCustomElement2() { const constructor = customElements.get("openchemlib-editor"); if (constructor) return constructor; const CanvasEditorElement = initCanvasEditorElement( CanvasEditor2, Molecule2, Reaction2, ReactionEncoder2 ); customElements.define("openchemlib-editor", CanvasEditorElement); const style = document.createElement("style"); style.id = "openchemlib-editor-default-style"; style.innerHTML = ` /* dynamicaly added from openchemlib registerCustomElement with low priority */ openchemlib-editor:defined { display: block; height: 400px; width: 600px; } `; document.head.prepend(style); return CanvasEditorElement; } OCL.CanvasEditor = CanvasEditor2; OCL.registerCustomElement = registerCustomElement2; delete OCL.GenericEditorArea; delete OCL.GenericEditorToolbar; delete OCL.GenericUIHelper; } // lib/extend/index.js function extendOCL(OCL) { const { ConformerGenerator: ConformerGenerator2, ForceFieldMMFF94: ForceFieldMMFF942, Molecule: Molecule2 } = OCL; ConformerGenerator2.prototype.molecules = function* molecules() { let nextConformer; while ((nextConformer = this.getNextConformerAsMolecule()) !== null) { yield nextConformer; } }; const defaultMinimiseOptions = { maxIts: 4e3, gradTol: 1e-4, funcTol: 1e-6 }; const _minimise = ForceFieldMMFF942.prototype._minimise; delete ForceFieldMMFF942.prototype._minimise; ForceFieldMMFF942.prototype.minimise = function minimise(options) { options = { ...defaultMinimiseOptions, ...options }; return _minimise.call( this, options.maxIts, options.gradTol, options.funcTol ); }; function parseMoleculeFromText(text) { if (!text) { return null; } if (text.includes("V2000") || text.includes("V3000")) { return Molecule2.fromMolfile(text); } try { return Molecule2.fromSmiles(text); } catch { } try { return Molecule2.fromIDCode(text); } catch { } return null; } Molecule2.fromText = function fromText(text) { const molecule = parseMoleculeFromText(text); if (molecule && molecule.getAllAtoms() > 0) { return molecule; } return null; }; Molecule2.prototype.getOCL = function getOCL() { ret