2023-03-11 01:52:17 +08:00
|
|
|
<script lang="ts">
|
|
|
|
import { createEventDispatcher, onMount } from "svelte";
|
|
|
|
import {
|
|
|
|
EditorView,
|
2024-02-01 06:47:18 +08:00
|
|
|
ViewUpdate,
|
2023-03-11 01:52:17 +08:00
|
|
|
keymap,
|
2023-10-31 12:46:02 +08:00
|
|
|
placeholder as placeholderExt
|
2023-03-11 01:52:17 +08:00
|
|
|
} from "@codemirror/view";
|
|
|
|
import { StateEffect, EditorState, type Extension } from "@codemirror/state";
|
|
|
|
import { indentWithTab } from "@codemirror/commands";
|
|
|
|
|
|
|
|
import { basicDark } from "cm6-theme-basic-dark";
|
|
|
|
import { basicLight } from "cm6-theme-basic-light";
|
|
|
|
import { basicSetup } from "./extensions";
|
|
|
|
import { getLanguageExtension } from "./language";
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
export let class_names = "";
|
2023-03-11 01:52:17 +08:00
|
|
|
export let value = "";
|
|
|
|
export let dark_mode: boolean;
|
|
|
|
export let basic = true;
|
|
|
|
export let language: string;
|
2023-08-04 06:01:18 +08:00
|
|
|
export let lines = 5;
|
2023-03-11 01:52:17 +08:00
|
|
|
export let extensions: Extension[] = [];
|
2024-02-01 06:47:18 +08:00
|
|
|
export let use_tab = true;
|
2023-03-11 01:52:17 +08:00
|
|
|
export let readonly = false;
|
|
|
|
export let placeholder: string | HTMLElement | null | undefined = undefined;
|
|
|
|
|
2023-11-10 08:20:54 +08:00
|
|
|
const dispatch = createEventDispatcher<{
|
|
|
|
change: string;
|
|
|
|
blur: undefined;
|
|
|
|
focus: undefined;
|
|
|
|
}>();
|
2023-03-11 01:52:17 +08:00
|
|
|
let lang_extension: Extension | undefined;
|
|
|
|
let element: HTMLDivElement;
|
|
|
|
let view: EditorView;
|
|
|
|
|
|
|
|
$: get_lang(language);
|
|
|
|
|
2023-08-04 06:01:18 +08:00
|
|
|
async function get_lang(val: string): Promise<void> {
|
2023-03-11 01:52:17 +08:00
|
|
|
const ext = await getLanguageExtension(val);
|
|
|
|
lang_extension = ext;
|
|
|
|
}
|
|
|
|
|
|
|
|
$: reconfigure(), lang_extension;
|
2024-02-01 06:47:18 +08:00
|
|
|
$: set_doc(value);
|
|
|
|
$: update_lines();
|
2023-03-11 01:52:17 +08:00
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function set_doc(new_doc: string): void {
|
|
|
|
if (view && new_doc !== view.state.doc.toString()) {
|
2023-03-11 01:52:17 +08:00
|
|
|
view.dispatch({
|
|
|
|
changes: {
|
|
|
|
from: 0,
|
|
|
|
to: view.state.doc.length,
|
2024-02-01 06:47:18 +08:00
|
|
|
insert: new_doc
|
2023-10-31 12:46:02 +08:00
|
|
|
}
|
2023-03-11 01:52:17 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function update_lines(): void {
|
2023-04-06 22:52:57 +08:00
|
|
|
if (view) {
|
2024-02-01 06:47:18 +08:00
|
|
|
view.requestMeasure({ read: update_gutters });
|
2023-04-06 22:52:57 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function create_editor_view(): EditorView {
|
2023-11-10 08:20:54 +08:00
|
|
|
const editorView = new EditorView({
|
2023-03-11 01:52:17 +08:00
|
|
|
parent: element,
|
2024-02-01 06:47:18 +08:00
|
|
|
state: create_editor_state(value)
|
2023-03-11 01:52:17 +08:00
|
|
|
});
|
2024-02-01 06:47:18 +08:00
|
|
|
editorView.dom.addEventListener("focus", handle_focus, true);
|
|
|
|
editorView.dom.addEventListener("blur", handle_blur, true);
|
2023-11-10 08:20:54 +08:00
|
|
|
return editorView;
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function handle_focus(): void {
|
2023-11-10 08:20:54 +08:00
|
|
|
dispatch("focus");
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function handle_blur(): void {
|
2023-11-10 08:20:54 +08:00
|
|
|
dispatch("blur");
|
2023-03-11 01:52:17 +08:00
|
|
|
}
|
|
|
|
|
2023-08-04 06:01:18 +08:00
|
|
|
function getGutterLineHeight(_view: EditorView): string | null {
|
|
|
|
let elements = _view.dom.querySelectorAll<HTMLElement>(".cm-gutterElement");
|
2023-04-06 22:52:57 +08:00
|
|
|
if (elements.length === 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
for (var i = 0; i < elements.length; i++) {
|
|
|
|
let node = elements[i];
|
|
|
|
let height = getComputedStyle(node)?.height ?? "0px";
|
|
|
|
if (height != "0px") {
|
|
|
|
return height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function update_gutters(_view: EditorView): any {
|
2023-08-04 06:01:18 +08:00
|
|
|
let gutters = _view.dom.querySelectorAll<HTMLElement>(".cm-gutter");
|
2023-04-06 22:52:57 +08:00
|
|
|
let _lines = lines + 1;
|
2023-08-04 06:01:18 +08:00
|
|
|
let lineHeight = getGutterLineHeight(_view);
|
2023-04-06 22:52:57 +08:00
|
|
|
if (!lineHeight) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
for (var i = 0; i < gutters.length; i++) {
|
|
|
|
let node = gutters[i];
|
|
|
|
node.style.minHeight = `calc(${lineHeight} * ${_lines})`;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function handle_change(vu: ViewUpdate): void {
|
2023-03-11 01:52:17 +08:00
|
|
|
if (vu.docChanged) {
|
|
|
|
const doc = vu.state.doc;
|
|
|
|
const text = doc.toString();
|
|
|
|
value = text;
|
|
|
|
dispatch("change", text);
|
|
|
|
}
|
2024-02-01 06:47:18 +08:00
|
|
|
view.requestMeasure({ read: update_gutters });
|
2023-03-11 01:52:17 +08:00
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function get_extensions(): Extension[] {
|
2023-03-11 01:52:17 +08:00
|
|
|
const stateExtensions = [
|
2024-02-01 06:47:18 +08:00
|
|
|
...get_base_extensions(
|
2023-03-11 01:52:17 +08:00
|
|
|
basic,
|
2024-02-01 06:47:18 +08:00
|
|
|
use_tab,
|
2023-03-11 01:52:17 +08:00
|
|
|
placeholder,
|
|
|
|
readonly,
|
|
|
|
lang_extension
|
|
|
|
),
|
|
|
|
FontTheme,
|
2024-02-01 06:47:18 +08:00
|
|
|
...get_theme(),
|
2023-10-31 12:46:02 +08:00
|
|
|
...extensions
|
2023-03-11 01:52:17 +08:00
|
|
|
];
|
|
|
|
return stateExtensions;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FontTheme = EditorView.theme({
|
|
|
|
"&": {
|
|
|
|
fontSize: "var(--text-sm)",
|
2023-10-31 12:46:02 +08:00
|
|
|
backgroundColor: "var(--border-color-secondary)"
|
2023-03-11 01:52:17 +08:00
|
|
|
},
|
|
|
|
".cm-content": {
|
|
|
|
paddingTop: "5px",
|
|
|
|
paddingBottom: "5px",
|
2023-05-03 22:52:52 +08:00
|
|
|
color: "var(--body-text-color)",
|
2023-03-11 01:52:17 +08:00
|
|
|
fontFamily: "var(--font-mono)",
|
2023-10-31 12:46:02 +08:00
|
|
|
minHeight: "100%"
|
2023-03-11 01:52:17 +08:00
|
|
|
},
|
|
|
|
".cm-gutters": {
|
|
|
|
marginRight: "1px",
|
2023-03-17 22:41:53 +08:00
|
|
|
borderRight: "1px solid var(--border-color-primary)",
|
2023-03-11 01:52:17 +08:00
|
|
|
backgroundColor: "transparent",
|
2023-10-31 12:46:02 +08:00
|
|
|
color: "var(--body-text-color-subdued)"
|
2023-03-11 01:52:17 +08:00
|
|
|
},
|
|
|
|
".cm-focused": {
|
2023-10-31 12:46:02 +08:00
|
|
|
outline: "none"
|
2023-05-03 22:52:52 +08:00
|
|
|
},
|
|
|
|
".cm-scroller": {
|
2023-10-31 12:46:02 +08:00
|
|
|
height: "auto"
|
2023-05-03 22:52:52 +08:00
|
|
|
},
|
|
|
|
".cm-cursor": {
|
2023-10-31 12:46:02 +08:00
|
|
|
borderLeftColor: "var(--body-text-color)"
|
|
|
|
}
|
2023-03-11 01:52:17 +08:00
|
|
|
});
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function create_editor_state(_value: string | null | undefined): EditorState {
|
2023-03-11 01:52:17 +08:00
|
|
|
return EditorState.create({
|
2023-08-04 06:01:18 +08:00
|
|
|
doc: _value ?? undefined,
|
2024-02-01 06:47:18 +08:00
|
|
|
extensions: get_extensions()
|
2023-03-11 01:52:17 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function get_base_extensions(
|
2023-03-11 01:52:17 +08:00
|
|
|
basic: boolean,
|
2024-02-01 06:47:18 +08:00
|
|
|
use_tab: boolean,
|
2023-03-11 01:52:17 +08:00
|
|
|
placeholder: string | HTMLElement | null | undefined,
|
|
|
|
readonly: boolean,
|
|
|
|
lang: Extension | null | undefined
|
|
|
|
): Extension[] {
|
|
|
|
const extensions: Extension[] = [
|
|
|
|
EditorView.editable.of(!readonly),
|
2023-09-22 20:12:26 +08:00
|
|
|
EditorState.readOnly.of(readonly),
|
2023-10-31 12:46:02 +08:00
|
|
|
EditorView.contentAttributes.of({ "aria-label": "Code input container" })
|
2023-03-11 01:52:17 +08:00
|
|
|
];
|
|
|
|
|
|
|
|
if (basic) {
|
|
|
|
extensions.push(basicSetup);
|
|
|
|
}
|
2024-02-01 06:47:18 +08:00
|
|
|
if (use_tab) {
|
2023-03-11 01:52:17 +08:00
|
|
|
extensions.push(keymap.of([indentWithTab]));
|
|
|
|
}
|
|
|
|
if (placeholder) {
|
|
|
|
extensions.push(placeholderExt(placeholder));
|
|
|
|
}
|
|
|
|
if (lang) {
|
|
|
|
extensions.push(lang);
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
extensions.push(EditorView.updateListener.of(handle_change));
|
2023-03-11 01:52:17 +08:00
|
|
|
return extensions;
|
|
|
|
}
|
|
|
|
|
2024-02-01 06:47:18 +08:00
|
|
|
function get_theme(): Extension[] {
|
2023-03-11 01:52:17 +08:00
|
|
|
const extensions: Extension[] = [];
|
|
|
|
|
|
|
|
if (dark_mode) {
|
|
|
|
extensions.push(basicDark);
|
|
|
|
} else {
|
|
|
|
extensions.push(basicLight);
|
|
|
|
}
|
|
|
|
return extensions;
|
|
|
|
}
|
|
|
|
|
|
|
|
function reconfigure(): void {
|
|
|
|
view?.dispatch({
|
2024-02-01 06:47:18 +08:00
|
|
|
effects: StateEffect.reconfigure.of(get_extensions())
|
2023-03-11 01:52:17 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
onMount(() => {
|
2024-02-01 06:47:18 +08:00
|
|
|
view = create_editor_view();
|
2023-03-11 01:52:17 +08:00
|
|
|
return () => view?.destroy();
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<div class="wrap">
|
2024-02-01 06:47:18 +08:00
|
|
|
<div class="codemirror-wrapper {class_names}" bind:this={element} />
|
2023-03-11 01:52:17 +08:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<style>
|
2023-04-06 22:52:57 +08:00
|
|
|
.wrap {
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
|
|
|
flex-flow: column;
|
|
|
|
margin: 0;
|
|
|
|
padding: 0;
|
|
|
|
height: 100%;
|
|
|
|
}
|
2023-03-11 01:52:17 +08:00
|
|
|
.codemirror-wrapper {
|
2023-04-06 22:52:57 +08:00
|
|
|
height: 100%;
|
2023-03-11 01:52:17 +08:00
|
|
|
overflow: auto;
|
|
|
|
}
|
|
|
|
|
2023-04-06 22:52:57 +08:00
|
|
|
:global(.cm-editor) {
|
|
|
|
height: 100%;
|
|
|
|
}
|
|
|
|
|
2023-03-11 01:52:17 +08:00
|
|
|
/* Dunno why this doesn't work through the theme API -- don't remove*/
|
|
|
|
:global(.cm-selectionBackground) {
|
|
|
|
background-color: #b9d2ff30 !important;
|
|
|
|
}
|
|
|
|
|
|
|
|
:global(.cm-focused) {
|
|
|
|
outline: none !important;
|
|
|
|
}
|
|
|
|
</style>
|