CommandLink alias syntax

pull/138/head
Zef Hemel 2022-11-29 09:11:23 +01:00
parent 55791cc88e
commit 6a047e1ef4
8 changed files with 172 additions and 22 deletions

View File

@ -23,10 +23,12 @@ import {
export const pageLinkRegex = /^\[\[([^\]\|]+)(\|([^\]]+))?\]\]/; export const pageLinkRegex = /^\[\[([^\]\|]+)(\|([^\]]+))?\]\]/;
const WikiLink: MarkdownConfig = { const WikiLink: MarkdownConfig = {
defineNodes: ["WikiLink", "WikiLinkPage", "WikiLinkAlias", { defineNodes: [
name: "WikiLinkMark", { name: "WikiLink", style: ct.WikiLinkTag },
style: t.processingInstruction, { name: "WikiLinkPage", style: ct.WikiLinkPageTag },
}], { name: "WikiLinkAlias", style: ct.WikiLinkPageTag },
{ name: "WikiLinkMark", style: t.processingInstruction },
],
parseInline: [ parseInline: [
{ {
name: "WikiLink", name: "WikiLink",
@ -38,8 +40,8 @@ const WikiLink: MarkdownConfig = {
) { ) {
return -1; return -1;
} }
const [_fullMatch, page, pipePart, label] = match; const [fullMatch, page, pipePart, label] = match;
const endPos = pos + match[0].length; const endPos = pos + fullMatch.length;
let aliasElts: any[] = []; let aliasElts: any[] = [];
if (pipePart) { if (pipePart) {
const pipeStartPos = pos + 2 + page.length; const pipeStartPos = pos + 2 + page.length;
@ -66,16 +68,14 @@ const WikiLink: MarkdownConfig = {
], ],
}; };
const commandLinkRegex = /^\{\[([^\]]+)\]\}/; export const commandLinkRegex = /^\{\[([^\]\|]+)(\|([^\]]+))?\]\}/;
const CommandLink: MarkdownConfig = { const CommandLink: MarkdownConfig = {
defineNodes: [ defineNodes: [
{ name: "CommandLink", style: { "CommandLink/...": ct.CommandLinkTag } }, { name: "CommandLink", style: { "CommandLink/...": ct.CommandLinkTag } },
{ name: "CommandLinkName", style: ct.CommandLinkNameTag }, { name: "CommandLinkName", style: ct.CommandLinkNameTag },
{ { name: "CommandLinkAlias", style: ct.CommandLinkNameTag },
name: "CommandLinkMark", { name: "CommandLinkMark", style: t.processingInstruction },
style: t.processingInstruction,
},
], ],
parseInline: [ parseInline: [
{ {
@ -88,14 +88,37 @@ const CommandLink: MarkdownConfig = {
) { ) {
return -1; return -1;
} }
const endPos = pos + match[0].length; const [fullMatch, command, pipePart, label] = match;
const endPos = pos + fullMatch.length;
let aliasElts: any[] = [];
if (pipePart) {
const pipeStartPos = pos + 2 + command.length;
aliasElts = [
cx.elt("CommandLinkMark", pipeStartPos, pipeStartPos + 1),
cx.elt(
"CommandLinkAlias",
pipeStartPos + 1,
pipeStartPos + 1 + label.length,
),
];
}
return cx.addElement( return cx.addElement(
cx.elt("CommandLink", pos, endPos, [ cx.elt("CommandLink", pos, endPos, [
cx.elt("CommandLinkMark", pos, pos + 2), cx.elt("CommandLinkMark", pos, pos + 2),
cx.elt("CommandLinkName", pos + 2, endPos - 2), cx.elt("CommandLinkName", pos + 2, pos + 2 + command.length),
...aliasElts,
cx.elt("CommandLinkMark", endPos - 2, endPos), cx.elt("CommandLinkMark", endPos - 2, endPos),
]), ]),
); );
// return cx.addElement(
// cx.elt("CommandLink", pos, endPos, [
// cx.elt("CommandLinkMark", pos, pos + 2),
// cx.elt("CommandLinkName", pos + 2, endPos - 2),
// cx.elt("CommandLinkMark", endPos - 2, endPos),
// ]),
// );
}, },
after: "Emphasis", after: "Emphasis",
}, },
@ -234,9 +257,9 @@ export default function buildMarkdown(mdExtensions: MDExt[]): Language {
{ {
props: [ props: [
styleTags({ styleTags({
WikiLink: ct.WikiLinkTag, // WikiLink: ct.WikiLinkTag,
WikiLinkPage: ct.WikiLinkPageTag, // WikiLinkPage: ct.WikiLinkPageTag,
WikiLinkAlias: ct.WikiLinkPageTag, // WikiLinkAlias: ct.WikiLinkPageTag,
// CommandLink: ct.CommandLinkTag, // CommandLink: ct.CommandLinkTag,
// CommandLinkName: ct.CommandLinkNameTag, // CommandLinkName: ct.CommandLinkNameTag,
Task: ct.TaskTag, Task: ct.TaskTag,

View File

@ -10,6 +10,7 @@ import { listBulletPlugin } from "./list.ts";
import { tablePlugin } from "./table.ts"; import { tablePlugin } from "./table.ts";
import { taskListPlugin } from "./task.ts"; import { taskListPlugin } from "./task.ts";
import { cleanWikiLinkPlugin } from "./wiki_link.ts"; import { cleanWikiLinkPlugin } from "./wiki_link.ts";
import { cleanCommandLinkPlugin } from "./command_link.ts";
export function cleanModePlugins(editor: Editor) { export function cleanModePlugins(editor: Editor) {
return [ return [
@ -36,5 +37,6 @@ export function cleanModePlugins(editor: Editor) {
listBulletPlugin, listBulletPlugin,
tablePlugin, tablePlugin,
cleanWikiLinkPlugin(editor), cleanWikiLinkPlugin(editor),
cleanCommandLinkPlugin(editor),
] as Extension[]; ] as Extension[];
} }

View File

@ -0,0 +1,99 @@
import { commandLinkRegex, pageLinkRegex } from "../../common/parser.ts";
import { ClickEvent } from "../../plug-api/app_event.ts";
import {
Decoration,
DecorationSet,
EditorView,
ViewPlugin,
ViewUpdate,
} from "../deps.ts";
import { Editor } from "../editor.tsx";
import {
ButtonWidget,
invisibleDecoration,
isCursorInRange,
iterateTreeInVisibleRanges,
} from "./util.ts";
/**
* Plugin to hide path prefix when the cursor is not inside.
*/
export function cleanCommandLinkPlugin(editor: Editor) {
return ViewPlugin.fromClass(
class {
decorations: DecorationSet;
constructor(view: EditorView) {
this.decorations = this.compute(view);
}
update(update: ViewUpdate) {
if (
update.docChanged || update.viewportChanged || update.selectionSet
) {
this.decorations = this.compute(update.view);
}
}
compute(view: EditorView): DecorationSet {
const widgets: any[] = [];
// let parentRange: [number, number];
iterateTreeInVisibleRanges(view, {
enter: ({ type, from, to }) => {
if (type.name !== "CommandLink") {
return;
}
if (isCursorInRange(view.state, [from, to])) {
return;
}
const text = view.state.sliceDoc(from, to);
const match = commandLinkRegex.exec(text);
if (!match) return;
const [_fullMatch, command, _pipePart, alias] = match;
// Hide the whole thing
widgets.push(
invisibleDecoration.range(
from,
to,
),
);
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
return view.dispatch({
selection: { anchor: from + 2 },
});
}
// Dispatch click event to navigate there without moving the cursor
const clickEvent: ClickEvent = {
page: editor.currentPage!,
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);
}
},
{
decorations: (v) => v.decorations,
},
);
}

View File

@ -25,7 +25,7 @@ const typesWithMarks = [
"InlineCode", "InlineCode",
"Highlight", "Highlight",
"Strikethrough", "Strikethrough",
"CommandLink", // "CommandLink",
]; ];
/** /**
* The elements which are used as marks. * The elements which are used as marks.
@ -35,7 +35,7 @@ const markTypes = [
"CodeMark", "CodeMark",
"HighlightMark", "HighlightMark",
"StrikethroughMark", "StrikethroughMark",
"CommandLinkMark", // "CommandLinkMark",
]; ];
/** /**

View File

@ -35,6 +35,29 @@ export class LinkWidget extends WidgetType {
} }
} }
export class ButtonWidget extends WidgetType {
constructor(
readonly text: string,
readonly title: string,
readonly cssClass: string,
readonly callback: (e: MouseEvent) => void,
) {
super();
}
toDOM(): HTMLElement {
const anchor = document.createElement("button");
anchor.className = this.cssClass;
anchor.textContent = this.text;
anchor.addEventListener("click", (e) => {
e.preventDefault();
e.stopPropagation();
this.callback(e);
});
anchor.setAttribute("title", this.title);
return anchor;
}
}
/** /**
* Check if two ranges overlap * Check if two ranges overlap
* Based on the visual diagram on https://stackoverflow.com/a/25369187 * Based on the visual diagram on https://stackoverflow.com/a/25369187

View File

@ -6,7 +6,6 @@ import {
EditorView, EditorView,
ViewPlugin, ViewPlugin,
ViewUpdate, ViewUpdate,
WidgetType,
} from "../deps.ts"; } from "../deps.ts";
import { Editor } from "../editor.tsx"; import { Editor } from "../editor.tsx";
import { import {
@ -41,7 +40,6 @@ export function cleanWikiLinkPlugin(editor: Editor) {
if (type.name !== "WikiLink") { if (type.name !== "WikiLink") {
return; return;
} }
// Adding 2 on each side due to [[ and ]] that are outside the WikiLinkPage node
if (isCursorInRange(view.state, [from, to])) { if (isCursorInRange(view.state, [from, to])) {
return; return;
} }

View File

@ -200,6 +200,11 @@
color: #959595; color: #959595;
} }
.sb-command-button {
font-family: "iA-Mono";
font-size: 1em;
}
.sb-command-link.sb-meta { .sb-command-link.sb-meta {
color: #959595; color: #959595;
} }

View File

@ -5,8 +5,8 @@ release.
## 0.2.2 ## 0.2.2
* New page link aliasing syntax (Obsidian compatible) is here: `[[page link|alias]]` e.g. [[CHANGELOG|this is a link to this changelog]]. * New page link aliasing syntax (Obsidian compatible) is here: `[[page link|alias]]` e.g. [[CHANGELOG|this is a link to this changelog]]. Also supported for command links: `{[Plugs: Add|add a plug]}`
* Less "floppy" behavior when clicking links (wiki and regular): just navigates there right away. Note: use `Alt-click` to move cursor inside of a link. * Less "floppy" behavior when clicking links (regular, wiki and command): just navigates there right away. Note: use `Alt-click` to move the cursor inside of a link.
* Added `invokeFunction` `silverbullet` CLI sub-command to run arbitrary plug functions from the CLI. * Added `invokeFunction` `silverbullet` CLI sub-command to run arbitrary plug functions from the CLI.
--- ---