openchemlib
Version:
Manipulate molecules
1,341 lines (1,325 loc) • 3.58 MB
JavaScript
// 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