2024-02-29 22:23:05 +08:00
|
|
|
import { ClickEvent } from "../../plug-api/types.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { syntaxTree } from "@codemirror/language";
|
|
|
|
import { Decoration } from "@codemirror/view";
|
2023-07-14 22:56:20 +08:00
|
|
|
import { Client } from "../client.ts";
|
2022-11-29 16:11:23 +08:00
|
|
|
import {
|
|
|
|
ButtonWidget,
|
2022-12-09 23:09:53 +08:00
|
|
|
decoratorStateField,
|
2022-11-29 16:11:23 +08:00
|
|
|
invisibleDecoration,
|
|
|
|
isCursorInRange,
|
|
|
|
} from "./util.ts";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { commandLinkRegex } from "$common/command.ts";
|
2022-11-29 16:11:23 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Plugin to hide path prefix when the cursor is not inside.
|
|
|
|
*/
|
2023-07-14 22:56:20 +08:00
|
|
|
export function cleanCommandLinkPlugin(editor: Client) {
|
2022-12-09 23:09:53 +08:00
|
|
|
return decoratorStateField((state) => {
|
|
|
|
const widgets: any[] = [];
|
|
|
|
// let parentRange: [number, number];
|
|
|
|
syntaxTree(state).iterate({
|
|
|
|
enter: ({ type, from, to }) => {
|
|
|
|
if (type.name !== "CommandLink") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (isCursorInRange(state, [from, to])) {
|
|
|
|
return;
|
2022-11-29 16:11:23 +08:00
|
|
|
}
|
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
const text = state.sliceDoc(from, to);
|
|
|
|
const match = commandLinkRegex.exec(text);
|
|
|
|
if (!match) return;
|
|
|
|
const [_fullMatch, command, _pipePart, alias] = match;
|
2022-11-29 16:11:23 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
// Hide the whole thing
|
|
|
|
widgets.push(
|
|
|
|
invisibleDecoration.range(
|
|
|
|
from,
|
|
|
|
to,
|
|
|
|
),
|
|
|
|
);
|
2022-11-29 16:11:23 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
const linkText = alias || command;
|
|
|
|
// And replace it with a widget
|
|
|
|
widgets.push(
|
|
|
|
Decoration.widget({
|
|
|
|
widget: new ButtonWidget(
|
|
|
|
linkText,
|
|
|
|
`Run command: ${command}`,
|
|
|
|
"sb-command-button",
|
|
|
|
(e) => {
|
|
|
|
if (e.altKey) {
|
|
|
|
// Move cursor into the link
|
2023-07-27 17:41:44 +08:00
|
|
|
return editor.editorView.dispatch({
|
2022-12-09 23:09:53 +08:00
|
|
|
selection: { anchor: from + 2 },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// Dispatch click event to navigate there without moving the cursor
|
|
|
|
const clickEvent: ClickEvent = {
|
2024-01-24 21:44:39 +08:00
|
|
|
page: editor.currentPage,
|
2022-12-09 23:09:53 +08:00
|
|
|
ctrlKey: e.ctrlKey,
|
|
|
|
metaKey: e.metaKey,
|
|
|
|
altKey: e.altKey,
|
|
|
|
pos: from,
|
|
|
|
};
|
|
|
|
editor.dispatchAppEvent("page:click", clickEvent).catch(
|
|
|
|
console.error,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
}).range(from),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
return Decoration.set(widgets, true);
|
|
|
|
});
|
2022-11-29 16:11:23 +08:00
|
|
|
}
|