import React, { useMemo } from 'react'; import _CodeMirror from '@uiw/react-codemirror'; import { EditorView, Decoration } from '@codemirror/view'; import { StateField, StateEffect } from '@codemirror/state'; import { javascript } from '@codemirror/lang-javascript'; import strudelTheme from '../themes/strudel-theme'; import './style.css'; import { useCallback } from 'react'; import { autocompletion } from '@codemirror/autocomplete'; import { strudelAutocomplete } from './Autocomplete'; import { vim } from '@replit/codemirror-vim'; import { emacs } from '@replit/codemirror-emacs'; export const setFlash = StateEffect.define(); const flashField = StateField.define({ create() { return Decoration.none; }, update(flash, tr) { try { for (let e of tr.effects) { if (e.is(setFlash)) { if (e.value) { const mark = Decoration.mark({ attributes: { style: `background-color: #FFCA2880` } }); flash = Decoration.set([mark.range(0, tr.newDoc.length)]); } else { flash = Decoration.set([]); } } } return flash; } catch (err) { console.warn('flash error', err); return flash; } }, provide: (f) => EditorView.decorations.from(f), }); export const flash = (view) => { view.dispatch({ effects: setFlash.of(true) }); setTimeout(() => { view.dispatch({ effects: setFlash.of(false) }); }, 200); }; export const setHighlights = StateEffect.define(); const highlightField = StateField.define({ create() { return Decoration.none; }, update(highlights, tr) { try { for (let e of tr.effects) { if (e.is(setHighlights)) { const { haps } = e.value; const marks = haps .map((hap) => (hap.context.locations || []).map(({ start, end }) => { // const color = hap.context.color || e.value.color || '#FFCA28'; let from = tr.newDoc.line(start.line).from + start.column; let to = tr.newDoc.line(end.line).from + end.column; const l = tr.newDoc.length; if (from > l || to > l) { return; // dont mark outside of range, as it will throw an error } //const mark = Decoration.mark({ attributes: { style: `outline: 2px solid ${color};` } }); const mark = Decoration.mark({ attributes: { class: `outline outline-2 outline-foreground` } }); return mark.range(from, to); }), ) .flat() .filter(Boolean) || []; highlights = Decoration.set(marks, true); } } return highlights; } catch (err) { // console.warn('highlighting error', err); return Decoration.set([]); } }, provide: (f) => EditorView.decorations.from(f), }); const staticExtensions = [ javascript(), highlightField, flashField, // javascriptLanguage.data.of({ autocomplete: strudelAutocomplete }), // autocompletion({ override: [strudelAutocomplete] }), autocompletion({ override: [] }), // wait for https://github.com/uiwjs/react-codemirror/pull/458 ]; export default function CodeMirror({ value, onChange, onViewChanged, onSelectionChange, theme, keybindings, fontSize = 18, fontFamily = 'monospace', options, editorDidMount, }) { const handleOnChange = useCallback( (value) => { onChange?.(value); }, [onChange], ); const handleOnCreateEditor = useCallback( (view) => { onViewChanged?.(view); }, [onViewChanged], ); const handleOnUpdate = useCallback( (viewUpdate) => { if (viewUpdate.selectionSet && onSelectionChange) { onSelectionChange?.(viewUpdate.state.selection); } }, [onSelectionChange], ); const extensions = useMemo(() => { let bindings = { vim, emacs, }; if (bindings[keybindings]) { return [...staticExtensions, bindings[keybindings]()]; } return staticExtensions; }, [keybindings]); return (
<_CodeMirror value={value} theme={theme || strudelTheme} onChange={handleOnChange} onCreateEditor={handleOnCreateEditor} onUpdate={handleOnUpdate} extensions={extensions} />
); } let parenMark; export const markParens = (editor, data) => { const v = editor.getDoc().getValue(); const marked = getCurrentParenArea(v, data); parenMark?.clear(); parenMark = editor.getDoc().markText(...marked, { css: 'background-color: #00007720' }); // }; // returns { line, ch } from absolute character offset export function offsetToPosition(offset, code) { const lines = code.split('\n'); let line = 0; let ch = 0; for (let i = 0; i < offset; i++) { if (ch === lines[line].length) { line++; ch = 0; } else { ch++; } } return { line, ch }; } // returns absolute character offset from { line, ch } export function positionToOffset(position, code) { const lines = code.split('\n'); if (position.line > lines.length) { // throw new Error('positionToOffset: position.line > lines.length'); return 0; } let offset = 0; for (let i = 0; i < position.line; i++) { offset += lines[i].length + 1; } offset += position.ch; return offset; } // given code and caret position, the functions returns the indices of the parens we are in export function getCurrentParenArea(code, caretPosition) { const caret = positionToOffset(caretPosition, code); let open, i, begin, end; // walk left i = caret; open = 0; while (i > 0) { if (code[i - 1] === '(') { open--; } else if (code[i - 1] === ')') { open++; } if (open === -1) { break; } i--; } begin = i; // walk right i = caret; open = 0; while (i < code.length) { if (code[i] === '(') { open--; } else if (code[i] === ')') { open++; } if (open === 1) { break; } i++; } end = i; return [begin, end].map((o) => offsetToPosition(o, code)); } /* export const markEvent = (editor) => (time, event) => { const locs = event.context.locations; if (!locs || !editor) { return; } const col = event.context?.color || '#FFCA28'; // mark active event const marks = locs.map(({ start, end }) => editor.getDoc().markText( { line: start.line - 1, ch: start.column }, { line: end.line - 1, ch: end.column }, //{ css: 'background-color: #FFCA28; color: black' } // background-color is now used by parent marking { css: 'outline: 1px solid ' + col + '; box-sizing:border-box' }, //{ css: `background-color: ${col};border-radius:5px` }, ), ); //Tone.Transport.schedule(() => { // problem: this can be cleared by scheduler... setTimeout(() => { marks.forEach((mark) => mark.clear()); // }, '+' + event.duration * 0.5); }, event.duration * 1000); }; */