mirror of
https://github.com/gradio-app/gradio.git
synced 2025-01-12 10:34:32 +08:00
235 lines
5.1 KiB
Svelte
235 lines
5.1 KiB
Svelte
|
<script lang="ts">
|
||
|
import type { ViewUpdate } from "@codemirror/view";
|
||
|
import { createEventDispatcher, onMount } from "svelte";
|
||
|
import {
|
||
|
EditorView,
|
||
|
keymap,
|
||
|
placeholder as placeholderExt
|
||
|
} 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";
|
||
|
|
||
|
export let classNames = "";
|
||
|
export let value = "";
|
||
|
export let dark_mode: boolean;
|
||
|
|
||
|
export let basic = true;
|
||
|
export let language: string;
|
||
|
export let extensions: Extension[] = [];
|
||
|
|
||
|
export let useTab = true;
|
||
|
|
||
|
export let readonly = false;
|
||
|
export let placeholder: string | HTMLElement | null | undefined = undefined;
|
||
|
|
||
|
const dispatch = createEventDispatcher<{ change: string }>();
|
||
|
let lang_extension: Extension | undefined;
|
||
|
let element: HTMLDivElement;
|
||
|
let view: EditorView;
|
||
|
|
||
|
$: get_lang(language);
|
||
|
|
||
|
async function get_lang(val: string) {
|
||
|
const ext = await getLanguageExtension(val);
|
||
|
lang_extension = ext;
|
||
|
}
|
||
|
|
||
|
$: reconfigure(), lang_extension;
|
||
|
$: setDoc(value);
|
||
|
|
||
|
function setDoc(newDoc: string) {
|
||
|
if (view && newDoc !== view.state.doc.toString()) {
|
||
|
view.dispatch({
|
||
|
changes: {
|
||
|
from: 0,
|
||
|
to: view.state.doc.length,
|
||
|
insert: newDoc
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function createEditorView(): EditorView {
|
||
|
return new EditorView({
|
||
|
parent: element,
|
||
|
state: createEditorState(value)
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function handleChange(vu: ViewUpdate): void {
|
||
|
if (vu.docChanged) {
|
||
|
const doc = vu.state.doc;
|
||
|
const text = doc.toString();
|
||
|
value = text;
|
||
|
dispatch("change", text);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function getExtensions() {
|
||
|
const stateExtensions = [
|
||
|
...getBaseExtensions(
|
||
|
basic,
|
||
|
useTab,
|
||
|
placeholder,
|
||
|
readonly,
|
||
|
lang_extension
|
||
|
),
|
||
|
FontTheme,
|
||
|
...getTheme(),
|
||
|
...extensions
|
||
|
];
|
||
|
return stateExtensions;
|
||
|
}
|
||
|
|
||
|
const FontTheme = EditorView.theme({
|
||
|
"&": {
|
||
|
fontSize: "var(--text-sm)",
|
||
|
backgroundColor: "var(--color-border-secondary)"
|
||
|
},
|
||
|
".cm-content": {
|
||
|
paddingTop: "5px",
|
||
|
paddingBottom: "5px",
|
||
|
|
||
|
fontFamily: "var(--font-mono)",
|
||
|
minHeight: "100%"
|
||
|
},
|
||
|
".cm-gutter": {
|
||
|
minHeight: "231px"
|
||
|
},
|
||
|
".cm-gutters": {
|
||
|
marginRight: "1px",
|
||
|
borderRight: "1px solid var(--color-border-primary)",
|
||
|
backgroundColor: "transparent",
|
||
|
color: "var(--text-color-subdued)"
|
||
|
},
|
||
|
".cm-focused": {
|
||
|
outline: "none"
|
||
|
}
|
||
|
});
|
||
|
|
||
|
function createEditorState(value: string | null | undefined): EditorState {
|
||
|
return EditorState.create({
|
||
|
doc: value ?? undefined,
|
||
|
extensions: getExtensions()
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function getBaseExtensions(
|
||
|
basic: boolean,
|
||
|
useTab: boolean,
|
||
|
placeholder: string | HTMLElement | null | undefined,
|
||
|
readonly: boolean,
|
||
|
lang: Extension | null | undefined
|
||
|
): Extension[] {
|
||
|
const extensions: Extension[] = [
|
||
|
EditorView.editable.of(!readonly),
|
||
|
EditorState.readOnly.of(readonly)
|
||
|
];
|
||
|
|
||
|
if (basic) {
|
||
|
extensions.push(basicSetup);
|
||
|
}
|
||
|
if (useTab) {
|
||
|
extensions.push(keymap.of([indentWithTab]));
|
||
|
}
|
||
|
if (placeholder) {
|
||
|
extensions.push(placeholderExt(placeholder));
|
||
|
}
|
||
|
if (lang) {
|
||
|
extensions.push(lang);
|
||
|
}
|
||
|
|
||
|
extensions.push(EditorView.updateListener.of(handleChange));
|
||
|
return extensions;
|
||
|
}
|
||
|
|
||
|
function getTheme(): Extension[] {
|
||
|
const extensions: Extension[] = [];
|
||
|
|
||
|
if (dark_mode) {
|
||
|
extensions.push(basicDark);
|
||
|
} else {
|
||
|
extensions.push(basicLight);
|
||
|
}
|
||
|
return extensions;
|
||
|
}
|
||
|
|
||
|
function reconfigure(): void {
|
||
|
view?.dispatch({
|
||
|
effects: StateEffect.reconfigure.of(getExtensions())
|
||
|
});
|
||
|
}
|
||
|
|
||
|
onMount(() => {
|
||
|
view = createEditorView();
|
||
|
return () => view?.destroy();
|
||
|
});
|
||
|
|
||
|
let padding = "";
|
||
|
|
||
|
const label_margin_var = "--block-label-margin";
|
||
|
const label_padding_var = "--block-label-padding";
|
||
|
const label_font_var = "--block-label-text-size";
|
||
|
const label_line_height = "--line-sm";
|
||
|
|
||
|
const RE = /(?:([0-9\.]+[a-zA-Z%]+)\s*)/g;
|
||
|
function get_padding_for_label() {
|
||
|
const m = getComputedStyle(document.documentElement)
|
||
|
.getPropertyValue(label_margin_var)
|
||
|
.trim();
|
||
|
const p = getComputedStyle(document.documentElement).getPropertyValue(
|
||
|
label_padding_var
|
||
|
);
|
||
|
|
||
|
const f = getComputedStyle(document.documentElement)
|
||
|
.getPropertyValue(label_font_var)
|
||
|
.trim();
|
||
|
const l = getComputedStyle(document.documentElement)
|
||
|
.getPropertyValue(label_line_height)
|
||
|
.trim();
|
||
|
|
||
|
const x = p.match(RE);
|
||
|
if (!x) return;
|
||
|
|
||
|
let [top, , bottom] = x.map((s) => s.trim());
|
||
|
if (!bottom) bottom = top;
|
||
|
|
||
|
const _margin = !m || m == "0" ? "" : ` ${m} +`;
|
||
|
const _height = /[a-zA-Z%]/.test(l) ? l : `(${f} * ${l})`;
|
||
|
|
||
|
padding = `padding-top: calc(${top} + ${bottom} +${_margin} ${_height});`;
|
||
|
}
|
||
|
|
||
|
get_padding_for_label();
|
||
|
</script>
|
||
|
|
||
|
<div class="wrap">
|
||
|
<div
|
||
|
class="codemirror-wrapper {classNames}"
|
||
|
style={padding}
|
||
|
bind:this={element}
|
||
|
/>
|
||
|
</div>
|
||
|
|
||
|
<style>
|
||
|
.codemirror-wrapper {
|
||
|
min-height: 250px;
|
||
|
max-height: 480px;
|
||
|
overflow: auto;
|
||
|
}
|
||
|
|
||
|
/* 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>
|