Files
yaak-mountain-loop/src-web/components/Editor/Editor.tsx
2023-03-01 09:05:00 -08:00

153 lines
4.5 KiB
TypeScript

import type { Transaction, TransactionSpec } from '@codemirror/state';
import { Compartment, EditorSelection, EditorState, Prec } from '@codemirror/state';
import classnames from 'classnames';
import { EditorView } from 'codemirror';
import type { HTMLAttributes } from 'react';
import { useEffect, useMemo, useRef, useState } from 'react';
import './Editor.css';
import { baseExtensions, getLanguageExtension, multiLineExtensions } from './extensions';
interface Props extends Omit<HTMLAttributes<HTMLDivElement>, 'onChange'> {
contentType: string;
valueKey?: string;
useTemplating?: boolean;
onChange?: (value: string) => void;
onSubmit?: () => void;
singleLine?: boolean;
}
export default function Editor({
contentType,
valueKey,
useTemplating,
defaultValue,
onChange,
onSubmit,
className,
singleLine,
...props
}: Props) {
const [cm, setCm] = useState<{ view: EditorView; langHolder: Compartment } | null>(null);
const ref = useRef<HTMLDivElement>(null);
const extensions = useMemo(
() => getExtensions({ onSubmit, singleLine, onChange, contentType, useTemplating }),
[contentType],
);
const newState = (langHolder: Compartment) => {
const langExt = getLanguageExtension({ contentType, useTemplating });
return EditorState.create({
doc: `${defaultValue ?? ''}`,
extensions: [...extensions, langHolder.of(langExt)],
});
};
// Create codemirror instance when ref initializes
useEffect(() => {
if (ref.current === null) return;
let view: EditorView | null = null;
try {
const langHolder = new Compartment();
view = new EditorView({
state: newState(langHolder),
parent: ref.current,
});
setCm({ view, langHolder });
} catch (e) {
console.log('Failed to initialize Codemirror', e);
}
return () => view?.destroy();
}, [ref.current]);
// Update value when valueKey changes
useEffect(() => {
if (cm === null) return;
cm.view.dispatch({
changes: { from: 0, to: cm.view.state.doc.length, insert: `${defaultValue ?? ''}` },
});
}, [valueKey]);
// Update language extension when contentType changes
useEffect(() => {
if (cm === null) return;
const ext = getLanguageExtension({ contentType, useTemplating });
cm.view.dispatch({ effects: cm.langHolder.reconfigure(ext) });
}, [contentType]);
return (
<div
ref={ref}
className={classnames(
className,
'cm-wrapper text-base',
singleLine ? 'cm-singleline' : 'cm-multiline',
)}
{...props}
/>
);
}
function getExtensions({
singleLine,
onChange,
onSubmit,
contentType,
useTemplating,
}: Pick<Props, 'singleLine' | 'onChange' | 'onSubmit' | 'contentType' | 'useTemplating'>) {
const ext = getLanguageExtension({ contentType, useTemplating });
return [
...(singleLine
? [
Prec.high(
EditorView.domEventHandlers({
keydown: (e) => {
// TODO: Figure out how to not have this not trigger on autocomplete selection
if (e.key === 'Enter') {
e.preventDefault();
onSubmit?.();
}
},
}),
),
EditorState.transactionFilter.of(
(tr: Transaction): TransactionSpec | TransactionSpec[] => {
if (!tr.isUserEvent('input.paste')) {
return tr;
}
// let addedNewline = false;
const trs: TransactionSpec[] = [];
tr.changes.iterChanges((fromA, toA, fromB, toB, inserted) => {
let insert = '';
for (const line of inserted) {
insert += line.replace('\n', '');
}
trs.push({
...tr,
selection: undefined,
changes: [{ from: fromB, to: toA, insert }],
});
});
// selection: EditorSelection.create([EditorSelection.cursor(8)], 1),
// console.log('TRS', trs, tr);
trs.push({
selection: EditorSelection.create([EditorSelection.cursor(8)], 1),
});
return trs;
// return addedNewline ? [] : tr;
},
),
]
: []),
...baseExtensions,
...(!singleLine ? [multiLineExtensions] : []),
...(ext ? [ext] : []),
EditorView.updateListener.of((update) => {
if (typeof onChange === 'function') {
onChange(update.state.doc.toString());
}
}),
];
}