mirror of
https://github.com/eliasstepanik/strudel-docker.git
synced 2026-01-11 21:58:31 +00:00
251 lines
6.9 KiB
JavaScript
251 lines
6.9 KiB
JavaScript
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 (
|
|
<div style={{ fontSize, fontFamily }} className="w-full">
|
|
<_CodeMirror
|
|
value={value}
|
|
theme={theme || strudelTheme}
|
|
onChange={handleOnChange}
|
|
onCreateEditor={handleOnCreateEditor}
|
|
onUpdate={handleOnUpdate}
|
|
extensions={extensions}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
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);
|
|
}; */
|