Fixes #164: Rewrote all CM view plugins to statefields
parent
453c613ef4
commit
c8c4271aeb
|
@ -1,4 +1,4 @@
|
||||||
import { nodeAtPos, ParseTree, renderToText } from "$sb/lib/tree.ts";
|
import { nodeAtPos, ParseTree } from "$sb/lib/tree.ts";
|
||||||
import { replaceAsync } from "$sb/lib/util.ts";
|
import { replaceAsync } from "$sb/lib/util.ts";
|
||||||
import { markdown } from "$sb/silverbullet-syscall/mod.ts";
|
import { markdown } from "$sb/silverbullet-syscall/mod.ts";
|
||||||
|
|
||||||
|
|
|
@ -1,23 +1,17 @@
|
||||||
|
import { Decoration, EditorState, syntaxTree } from "../deps.ts";
|
||||||
import {
|
import {
|
||||||
Decoration,
|
decoratorStateField,
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import {
|
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
function hideNodes(view: EditorView) {
|
function hideNodes(state: EditorState) {
|
||||||
const widgets: any[] = [];
|
const widgets: any[] = [];
|
||||||
iterateTreeInVisibleRanges(view, {
|
syntaxTree(state).iterate({
|
||||||
enter(node) {
|
enter(node) {
|
||||||
if (
|
if (
|
||||||
node.name === "HorizontalRule" &&
|
node.name === "HorizontalRule" &&
|
||||||
!isCursorInRange(view.state, [node.from, node.to])
|
!isCursorInRange(state, [node.from, node.to])
|
||||||
) {
|
) {
|
||||||
widgets.push(invisibleDecoration.range(node.from, node.to));
|
widgets.push(invisibleDecoration.range(node.from, node.to));
|
||||||
widgets.push(
|
widgets.push(
|
||||||
|
@ -29,7 +23,7 @@ function hideNodes(view: EditorView) {
|
||||||
|
|
||||||
if (
|
if (
|
||||||
node.name === "Image" &&
|
node.name === "Image" &&
|
||||||
!isCursorInRange(view.state, [node.from, node.to])
|
!isCursorInRange(state, [node.from, node.to])
|
||||||
) {
|
) {
|
||||||
widgets.push(invisibleDecoration.range(node.from, node.to));
|
widgets.push(invisibleDecoration.range(node.from, node.to));
|
||||||
}
|
}
|
||||||
|
@ -38,7 +32,7 @@ function hideNodes(view: EditorView) {
|
||||||
node.name === "FrontMatterMarker"
|
node.name === "FrontMatterMarker"
|
||||||
) {
|
) {
|
||||||
const parent = node.node.parent!;
|
const parent = node.node.parent!;
|
||||||
if (!isCursorInRange(view.state, [parent.from, parent.to])) {
|
if (!isCursorInRange(state, [parent.from, parent.to])) {
|
||||||
widgets.push(
|
widgets.push(
|
||||||
Decoration.line({
|
Decoration.line({
|
||||||
class: "sb-line-frontmatter-outside",
|
class: "sb-line-frontmatter-outside",
|
||||||
|
@ -54,7 +48,7 @@ function hideNodes(view: EditorView) {
|
||||||
// Hide ONLY if CodeMark is not insine backticks (InlineCode) and the cursor is placed outside
|
// Hide ONLY if CodeMark is not insine backticks (InlineCode) and the cursor is placed outside
|
||||||
if (
|
if (
|
||||||
parent.node.name !== "InlineCode" &&
|
parent.node.name !== "InlineCode" &&
|
||||||
!isCursorInRange(view.state, [parent.from, parent.to])
|
!isCursorInRange(state, [parent.from, parent.to])
|
||||||
) {
|
) {
|
||||||
widgets.push(
|
widgets.push(
|
||||||
Decoration.line({
|
Decoration.line({
|
||||||
|
@ -68,19 +62,6 @@ function hideNodes(view: EditorView) {
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
export const cleanBlockPlugin = ViewPlugin.fromClass(
|
export function cleanBlockPlugin() {
|
||||||
class {
|
return decoratorStateField(hideNodes);
|
||||||
decorations: DecorationSet;
|
}
|
||||||
|
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = hideNodes(view);
|
|
||||||
}
|
|
||||||
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.selectionSet) {
|
|
||||||
this.decorations = hideNodes(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{ decorations: (v) => v.decorations },
|
|
||||||
);
|
|
||||||
|
|
|
@ -1,45 +1,26 @@
|
||||||
|
import { Decoration, EditorState, syntaxTree } from "../deps.ts";
|
||||||
import {
|
import {
|
||||||
Decoration,
|
decoratorStateField,
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import {
|
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
class BlockquotePlugin {
|
function decorateBlockQuote(state: EditorState) {
|
||||||
decorations: DecorationSet = Decoration.none;
|
const widgets: any[] = [];
|
||||||
constructor(view: EditorView) {
|
syntaxTree(state).iterate({
|
||||||
this.decorations = this.decorateLists(view);
|
enter: ({ type, from, to }) => {
|
||||||
}
|
if (isCursorInRange(state, [from, to])) return;
|
||||||
update(update: ViewUpdate) {
|
if (type.name === "QuoteMark") {
|
||||||
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
widgets.push(invisibleDecoration.range(from, to));
|
||||||
this.decorations = this.decorateLists(update.view);
|
widgets.push(
|
||||||
}
|
Decoration.line({ class: "sb-blockquote-outside" }).range(from),
|
||||||
}
|
);
|
||||||
private decorateLists(view: EditorView) {
|
}
|
||||||
const widgets: any[] = [];
|
},
|
||||||
iterateTreeInVisibleRanges(view, {
|
});
|
||||||
enter: ({ type, from, to }) => {
|
return Decoration.set(widgets, true);
|
||||||
if (isCursorInRange(view.state, [from, to])) return;
|
}
|
||||||
if (type.name === "QuoteMark") {
|
|
||||||
widgets.push(invisibleDecoration.range(from, to));
|
export function blockquotePlugin() {
|
||||||
widgets.push(
|
return decoratorStateField(decorateBlockQuote);
|
||||||
Decoration.line({ class: "sb-blockquote-outside" }).range(from),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return Decoration.set(widgets, true);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
export const blockquotePlugin = ViewPlugin.fromClass(
|
|
||||||
BlockquotePlugin,
|
|
||||||
{
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ import type { Extension } from "../deps.ts";
|
||||||
import { Editor } from "../editor.tsx";
|
import { Editor } from "../editor.tsx";
|
||||||
import { blockquotePlugin } from "./block_quote.ts";
|
import { blockquotePlugin } from "./block_quote.ts";
|
||||||
import { directivePlugin } from "./directive.ts";
|
import { directivePlugin } from "./directive.ts";
|
||||||
import { hideHeaderMarkPlugin, hideMarks } from "./hide_mark.ts";
|
import { hideHeaderMarkPlugin, hideMarksPlugin } from "./hide_mark.ts";
|
||||||
import { cleanBlockPlugin } from "./block.ts";
|
import { cleanBlockPlugin } from "./block.ts";
|
||||||
import { linkPlugin } from "./link.ts";
|
import { linkPlugin } from "./link.ts";
|
||||||
import { listBulletPlugin } from "./list.ts";
|
import { listBulletPlugin } from "./list.ts";
|
||||||
|
@ -15,11 +15,11 @@ import { cleanCommandLinkPlugin } from "./command_link.ts";
|
||||||
export function cleanModePlugins(editor: Editor) {
|
export function cleanModePlugins(editor: Editor) {
|
||||||
return [
|
return [
|
||||||
linkPlugin(editor),
|
linkPlugin(editor),
|
||||||
directivePlugin,
|
directivePlugin(),
|
||||||
blockquotePlugin,
|
blockquotePlugin(),
|
||||||
hideMarks(),
|
hideMarksPlugin(),
|
||||||
hideHeaderMarkPlugin,
|
hideHeaderMarkPlugin(),
|
||||||
cleanBlockPlugin,
|
cleanBlockPlugin(),
|
||||||
taskListPlugin({
|
taskListPlugin({
|
||||||
// TODO: Move this logic elsewhere?
|
// TODO: Move this logic elsewhere?
|
||||||
onCheckboxClick: (pos) => {
|
onCheckboxClick: (pos) => {
|
||||||
|
@ -34,8 +34,8 @@ export function cleanModePlugins(editor: Editor) {
|
||||||
editor.dispatchAppEvent("page:click", clickEvent);
|
editor.dispatchAppEvent("page:click", clickEvent);
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
listBulletPlugin,
|
listBulletPlugin(),
|
||||||
tablePlugin,
|
tablePlugin(editor),
|
||||||
cleanWikiLinkPlugin(editor),
|
cleanWikiLinkPlugin(editor),
|
||||||
cleanCommandLinkPlugin(editor),
|
cleanCommandLinkPlugin(editor),
|
||||||
] as Extension[];
|
] as Extension[];
|
||||||
|
|
|
@ -1,99 +1,75 @@
|
||||||
import { commandLinkRegex, pageLinkRegex } from "../../common/parser.ts";
|
import { commandLinkRegex } from "../../common/parser.ts";
|
||||||
import { ClickEvent } from "../../plug-api/app_event.ts";
|
import { ClickEvent } from "$sb/app_event.ts";
|
||||||
import {
|
import { Decoration, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { Editor } from "../editor.tsx";
|
import { Editor } from "../editor.tsx";
|
||||||
import {
|
import {
|
||||||
ButtonWidget,
|
ButtonWidget,
|
||||||
|
decoratorStateField,
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Plugin to hide path prefix when the cursor is not inside.
|
* Plugin to hide path prefix when the cursor is not inside.
|
||||||
*/
|
*/
|
||||||
export function cleanCommandLinkPlugin(editor: Editor) {
|
export function cleanCommandLinkPlugin(editor: Editor) {
|
||||||
return ViewPlugin.fromClass(
|
return decoratorStateField((state) => {
|
||||||
class {
|
const widgets: any[] = [];
|
||||||
decorations: DecorationSet;
|
// let parentRange: [number, number];
|
||||||
constructor(view: EditorView) {
|
syntaxTree(state).iterate({
|
||||||
this.decorations = this.compute(view);
|
enter: ({ type, from, to }) => {
|
||||||
}
|
if (type.name !== "CommandLink") {
|
||||||
update(update: ViewUpdate) {
|
return;
|
||||||
if (
|
}
|
||||||
update.docChanged || update.viewportChanged || update.selectionSet
|
if (isCursorInRange(state, [from, to])) {
|
||||||
) {
|
return;
|
||||||
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 text = state.sliceDoc(from, to);
|
||||||
const match = commandLinkRegex.exec(text);
|
const match = commandLinkRegex.exec(text);
|
||||||
if (!match) return;
|
if (!match) return;
|
||||||
const [_fullMatch, command, _pipePart, alias] = match;
|
const [_fullMatch, command, _pipePart, alias] = match;
|
||||||
|
|
||||||
// Hide the whole thing
|
// Hide the whole thing
|
||||||
widgets.push(
|
widgets.push(
|
||||||
invisibleDecoration.range(
|
invisibleDecoration.range(
|
||||||
from,
|
from,
|
||||||
to,
|
to,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
const linkText = alias || command;
|
const linkText = alias || command;
|
||||||
// And replace it with a widget
|
// And replace it with a widget
|
||||||
widgets.push(
|
widgets.push(
|
||||||
Decoration.widget({
|
Decoration.widget({
|
||||||
widget: new ButtonWidget(
|
widget: new ButtonWidget(
|
||||||
linkText,
|
linkText,
|
||||||
`Run command: ${command}`,
|
`Run command: ${command}`,
|
||||||
"sb-command-button",
|
"sb-command-button",
|
||||||
(e) => {
|
(e) => {
|
||||||
if (e.altKey) {
|
if (e.altKey) {
|
||||||
// Move cursor into the link
|
// Move cursor into the link
|
||||||
return view.dispatch({
|
return editor.editorView!.dispatch({
|
||||||
selection: { anchor: from + 2 },
|
selection: { anchor: from + 2 },
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// Dispatch click event to navigate there without moving the cursor
|
// Dispatch click event to navigate there without moving the cursor
|
||||||
const clickEvent: ClickEvent = {
|
const clickEvent: ClickEvent = {
|
||||||
page: editor.currentPage!,
|
page: editor.currentPage!,
|
||||||
ctrlKey: e.ctrlKey,
|
ctrlKey: e.ctrlKey,
|
||||||
metaKey: e.metaKey,
|
metaKey: e.metaKey,
|
||||||
altKey: e.altKey,
|
altKey: e.altKey,
|
||||||
pos: from,
|
pos: from,
|
||||||
};
|
};
|
||||||
editor.dispatchAppEvent("page:click", clickEvent).catch(
|
editor.dispatchAppEvent("page:click", clickEvent).catch(
|
||||||
console.error,
|
console.error,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
}).range(from),
|
}).range(from),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
});
|
||||||
},
|
|
||||||
{
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,70 +1,44 @@
|
||||||
import {
|
import { Decoration, EditorState, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
import { decoratorStateField, isCursorInRange } from "./util.ts";
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
syntaxTree,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { isCursorInRange } from "./util.ts";
|
|
||||||
|
|
||||||
function getDirectives(view: EditorView) {
|
function getDirectives(state: EditorState) {
|
||||||
const widgets: any[] = [];
|
const widgets: any[] = [];
|
||||||
|
|
||||||
for (const { from, to } of view.visibleRanges) {
|
syntaxTree(state).iterate({
|
||||||
syntaxTree(view.state).iterate({
|
enter: ({ type, from, to }) => {
|
||||||
from,
|
if (type.name !== "CommentBlock") {
|
||||||
to,
|
return;
|
||||||
enter: ({ type, from, to }) => {
|
}
|
||||||
if (type.name !== "CommentBlock") {
|
const text = state.sliceDoc(from, to);
|
||||||
return;
|
if (/<!--\s*#/.exec(text)) {
|
||||||
}
|
// Open directive
|
||||||
const text = view.state.sliceDoc(from, to);
|
widgets.push(
|
||||||
if (/<!--\s*#/.exec(text)) {
|
Decoration.line({
|
||||||
// Open directive
|
class: "sb-directive-start",
|
||||||
widgets.push(
|
}).range(from),
|
||||||
Decoration.line({
|
);
|
||||||
class: "sb-directive-start",
|
} else if (/<!--\s*\//.exec(text)) {
|
||||||
}).range(from),
|
widgets.push(
|
||||||
);
|
Decoration.line({
|
||||||
} else if (/<!--\s*\//.exec(text)) {
|
class: "sb-directive-end",
|
||||||
widgets.push(
|
}).range(from),
|
||||||
Decoration.line({
|
);
|
||||||
class: "sb-directive-end",
|
} else {
|
||||||
}).range(from),
|
return;
|
||||||
);
|
}
|
||||||
} else {
|
if (!isCursorInRange(state, [from, to])) {
|
||||||
return;
|
widgets.push(
|
||||||
}
|
Decoration.line({
|
||||||
if (!isCursorInRange(view.state, [from, to])) {
|
class: "sb-directive-outside",
|
||||||
widgets.push(
|
}).range(from),
|
||||||
Decoration.line({
|
);
|
||||||
class: "sb-directive-outside",
|
}
|
||||||
}).range(from),
|
},
|
||||||
);
|
});
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
export const directivePlugin = ViewPlugin.fromClass(
|
export function directivePlugin() {
|
||||||
class {
|
return decoratorStateField(getDirectives);
|
||||||
decorations: DecorationSet = Decoration.none;
|
}
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = getDirectives(view);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (
|
|
||||||
update.docChanged ||
|
|
||||||
update.viewportChanged ||
|
|
||||||
update.selectionSet
|
|
||||||
) {
|
|
||||||
this.decorations = getDirectives(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{ decorations: (v) => v.decorations },
|
|
||||||
);
|
|
||||||
|
|
|
@ -2,18 +2,12 @@
|
||||||
// Original author: Pranav Karawale
|
// Original author: Pranav Karawale
|
||||||
// License: Apache License 2.0.
|
// License: Apache License 2.0.
|
||||||
|
|
||||||
import {
|
import { Decoration, EditorState, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import {
|
import {
|
||||||
checkRangeOverlap,
|
checkRangeOverlap,
|
||||||
|
decoratorStateField,
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,22 +31,16 @@ const markTypes = [
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Plugin to hide marks when the they are not in the editor selection.
|
* Ixora hide marks plugin.
|
||||||
|
*
|
||||||
|
* This plugin allows to:
|
||||||
|
* - Hide marks when they are not in the editor selection.
|
||||||
*/
|
*/
|
||||||
class HideMarkPlugin {
|
export function hideMarksPlugin() {
|
||||||
decorations: DecorationSet;
|
return decoratorStateField((state: EditorState) => {
|
||||||
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[] = [];
|
const widgets: any[] = [];
|
||||||
let parentRange: [number, number];
|
let parentRange: [number, number];
|
||||||
iterateTreeInVisibleRanges(view, {
|
syntaxTree(state).iterate({
|
||||||
enter: ({ type, from, to, node }) => {
|
enter: ({ type, from, to, node }) => {
|
||||||
if (typesWithMarks.includes(type.name)) {
|
if (typesWithMarks.includes(type.name)) {
|
||||||
// There can be a possibility that the current node is a
|
// There can be a possibility that the current node is a
|
||||||
|
@ -64,7 +52,7 @@ class HideMarkPlugin {
|
||||||
) {
|
) {
|
||||||
return;
|
return;
|
||||||
} else parentRange = [from, to];
|
} else parentRange = [from, to];
|
||||||
if (isCursorInRange(view.state, [from, to])) return;
|
if (isCursorInRange(state, [from, to])) return;
|
||||||
const innerTree = node.toTree();
|
const innerTree = node.toTree();
|
||||||
innerTree.iterate({
|
innerTree.iterate({
|
||||||
enter({ type, from: markFrom, to: markTo }) {
|
enter({ type, from: markFrom, to: markTo }) {
|
||||||
|
@ -83,78 +71,36 @@ class HideMarkPlugin {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Ixora hide marks plugin.
|
|
||||||
*
|
|
||||||
* This plugin allows to:
|
|
||||||
* - Hide marks when they are not in the editor selection.
|
|
||||||
*/
|
|
||||||
export const hideMarks = () => [
|
|
||||||
ViewPlugin.fromClass(HideMarkPlugin, {
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
}),
|
|
||||||
];
|
|
||||||
|
|
||||||
// HEADINGS
|
// HEADINGS
|
||||||
|
|
||||||
class HideHeaderMarkPlugin {
|
export function hideHeaderMarkPlugin() {
|
||||||
decorations: DecorationSet;
|
return decoratorStateField((state) => {
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = this.hideHeaderMark(view);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
|
||||||
this.decorations = this.hideHeaderMark(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Function to decide if to insert a decoration to hide the header mark
|
|
||||||
* @param view - Editor view
|
|
||||||
* @returns The `Decoration`s that hide the header marks
|
|
||||||
*/
|
|
||||||
private hideHeaderMark(view: EditorView) {
|
|
||||||
const widgets: any[] = [];
|
const widgets: any[] = [];
|
||||||
const ranges = view.state.selection.ranges;
|
syntaxTree(state).iterate({
|
||||||
iterateTreeInVisibleRanges(view, {
|
|
||||||
enter: ({ type, from, to }) => {
|
enter: ({ type, from, to }) => {
|
||||||
|
if (!type.name.startsWith("ATXHeading")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
// Get the active line
|
// Get the active line
|
||||||
const line = view.lineBlockAt(from);
|
const line = state.sliceDoc(from, to);
|
||||||
// If any cursor overlaps with the heading line, skip
|
if (isCursorInRange(state, [from, to])) {
|
||||||
const cursorOverlaps = ranges.some(({ from, to }) =>
|
|
||||||
checkRangeOverlap([from, to], [line.from, line.to])
|
|
||||||
);
|
|
||||||
if (cursorOverlaps && type.name === "HeaderMark") {
|
|
||||||
widgets.push(
|
widgets.push(
|
||||||
Decoration.line({ class: "sb-header-inside" }).range(from),
|
Decoration.line({ class: "sb-header-inside" }).range(from),
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
} else if (cursorOverlaps) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (
|
|
||||||
type.name === "HeaderMark" &&
|
|
||||||
// Setext heading's horizontal lines are not hidden.
|
|
||||||
/[#]/.test(view.state.sliceDoc(from, to))
|
|
||||||
) {
|
|
||||||
const dec = Decoration.replace({});
|
|
||||||
widgets.push(dec.range(from, to + 1));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
widgets.push(
|
||||||
|
invisibleDecoration.range(
|
||||||
|
from,
|
||||||
|
from + line.indexOf(" ") + 1,
|
||||||
|
),
|
||||||
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Plugin to hide the header mark.
|
|
||||||
*
|
|
||||||
* The header mark will not be hidden when:
|
|
||||||
* - The cursor is on the active line
|
|
||||||
* - The mark is on a line which is in the current selection
|
|
||||||
*/
|
|
||||||
export const hideHeaderMarkPlugin = ViewPlugin.fromClass(HideHeaderMarkPlugin, {
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
});
|
|
||||||
|
|
|
@ -1,14 +1,11 @@
|
||||||
import {
|
import {
|
||||||
Decoration,
|
Decoration,
|
||||||
DecorationSet,
|
EditorState,
|
||||||
EditorView,
|
|
||||||
Range,
|
Range,
|
||||||
syntaxTree,
|
syntaxTree,
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
WidgetType,
|
WidgetType,
|
||||||
} from "../deps.ts";
|
} from "../deps.ts";
|
||||||
import { invisibleDecoration, isCursorInRange } from "./util.ts";
|
import { decoratorStateField } from "./util.ts";
|
||||||
|
|
||||||
class InlineImageWidget extends WidgetType {
|
class InlineImageWidget extends WidgetType {
|
||||||
constructor(readonly url: string, readonly title: string) {
|
constructor(readonly url: string, readonly title: string) {
|
||||||
|
@ -35,21 +32,19 @@ class InlineImageWidget extends WidgetType {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const inlineImages = (view: EditorView) => {
|
export function inlineImagesPlugin() {
|
||||||
const widgets: Range<Decoration>[] = [];
|
return decoratorStateField((state: EditorState) => {
|
||||||
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
|
const widgets: Range<Decoration>[] = [];
|
||||||
|
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
|
||||||
|
|
||||||
for (const { from, to } of view.visibleRanges) {
|
syntaxTree(state).iterate({
|
||||||
syntaxTree(view.state).iterate({
|
|
||||||
from,
|
|
||||||
to,
|
|
||||||
enter: (node) => {
|
enter: (node) => {
|
||||||
if (node.name !== "Image") {
|
if (node.name !== "Image") {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const imageRexexResult = imageRegex.exec(
|
const imageRexexResult = imageRegex.exec(
|
||||||
view.state.sliceDoc(node.from, node.to),
|
state.sliceDoc(node.from, node.to),
|
||||||
);
|
);
|
||||||
if (imageRexexResult === null || !imageRexexResult.groups) {
|
if (imageRexexResult === null || !imageRexexResult.groups) {
|
||||||
return;
|
return;
|
||||||
|
@ -64,27 +59,7 @@ const inlineImages = (view: EditorView) => {
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
|
||||||
|
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
};
|
});
|
||||||
|
}
|
||||||
export const inlineImagesPlugin = () =>
|
|
||||||
ViewPlugin.fromClass(
|
|
||||||
class {
|
|
||||||
decorations: DecorationSet;
|
|
||||||
|
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = inlineImages(view);
|
|
||||||
}
|
|
||||||
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged) {
|
|
||||||
this.decorations = inlineImages(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
|
@ -1,12 +1,5 @@
|
||||||
import {
|
import { Decoration, EditorState, Range, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
import { decoratorStateField } from "./util.ts";
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
Range,
|
|
||||||
syntaxTree,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
|
|
||||||
interface WrapElement {
|
interface WrapElement {
|
||||||
selector: string;
|
selector: string;
|
||||||
|
@ -14,16 +7,12 @@ interface WrapElement {
|
||||||
nesting?: boolean;
|
nesting?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
function wrapLines(view: EditorView, wrapElements: WrapElement[]) {
|
export function lineWrapper(wrapElements: WrapElement[]) {
|
||||||
let widgets: Range<Decoration>[] = [];
|
return decoratorStateField((state: EditorState) => {
|
||||||
const elementStack: string[] = [];
|
const widgets: Range<Decoration>[] = [];
|
||||||
const doc = view.state.doc;
|
const elementStack: string[] = [];
|
||||||
// Disabling the visible ranges for now, because it may be a bit buggy.
|
const doc = state.doc;
|
||||||
// RISK: this may actually become slow for large documents.
|
syntaxTree(state).iterate({
|
||||||
for (const { from, to } of view.visibleRanges) {
|
|
||||||
syntaxTree(view.state).iterate({
|
|
||||||
from,
|
|
||||||
to,
|
|
||||||
enter: ({ type, from, to }) => {
|
enter: ({ type, from, to }) => {
|
||||||
for (const wrapElement of wrapElements) {
|
for (const wrapElement of wrapElements) {
|
||||||
if (type.name == wrapElement.selector) {
|
if (type.name == wrapElement.selector) {
|
||||||
|
@ -55,30 +44,7 @@ function wrapLines(view: EditorView, wrapElements: WrapElement[]) {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
|
||||||
// Widgets have to be sorted by `from` in ascending order
|
return Decoration.set(widgets, true);
|
||||||
widgets = widgets.sort((a, b) => {
|
|
||||||
return a.from < b.from ? -1 : 1;
|
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const lineWrapper = (wrapElements: WrapElement[]) =>
|
|
||||||
ViewPlugin.fromClass(
|
|
||||||
class {
|
|
||||||
decorations: DecorationSet;
|
|
||||||
|
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = wrapLines(view, wrapElements);
|
|
||||||
}
|
|
||||||
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.viewportChanged) {
|
|
||||||
this.decorations = wrapLines(update.view, wrapElements);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
|
@ -1,104 +1,79 @@
|
||||||
import { ClickEvent } from "../../plug-api/app_event.ts";
|
import { ClickEvent } from "../../plug-api/app_event.ts";
|
||||||
import {
|
import { Decoration, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { Editor } from "../editor.tsx";
|
import { Editor } from "../editor.tsx";
|
||||||
import {
|
import {
|
||||||
|
decoratorStateField,
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
import { LinkWidget } from "./util.ts";
|
import { LinkWidget } from "./util.ts";
|
||||||
|
|
||||||
export function linkPlugin(editor: Editor) {
|
export function linkPlugin(editor: Editor) {
|
||||||
return ViewPlugin.fromClass(
|
return decoratorStateField((state) => {
|
||||||
class {
|
const widgets: any[] = [];
|
||||||
decorations: DecorationSet = Decoration.none;
|
|
||||||
constructor(readonly view: EditorView) {
|
|
||||||
this.decorations = this.calculateDecorations();
|
|
||||||
}
|
|
||||||
calculateDecorations() {
|
|
||||||
const widgets: any[] = [];
|
|
||||||
const view = this.view;
|
|
||||||
|
|
||||||
iterateTreeInVisibleRanges(this.view, {
|
syntaxTree(state).iterate({
|
||||||
enter: ({ type, from, to }) => {
|
enter: ({ type, from, to }) => {
|
||||||
if (type.name !== "Link") {
|
if (type.name !== "Link") {
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
// Adding 2 on each side due to [[ and ]] that are outside the WikiLinkPage node
|
|
||||||
if (isCursorInRange(view.state, [from, to])) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = view.state.sliceDoc(from, to);
|
|
||||||
// Links are of the form [hell](https://example.com)
|
|
||||||
const [anchorPart, linkPart] = text.split("]("); // Not pretty
|
|
||||||
if (!linkPart) {
|
|
||||||
// Invalid link
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const cleanAnchor = anchorPart.substring(1); // cut off the initial [
|
|
||||||
const cleanLink = linkPart.substring(0, linkPart.length - 1); // cut off the final )
|
|
||||||
|
|
||||||
// Hide the whole thing
|
|
||||||
widgets.push(
|
|
||||||
invisibleDecoration.range(
|
|
||||||
from,
|
|
||||||
to,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
widgets.push(
|
|
||||||
Decoration.widget({
|
|
||||||
widget: new LinkWidget(
|
|
||||||
{
|
|
||||||
text: cleanAnchor,
|
|
||||||
title: `Click to visit ${cleanLink}`,
|
|
||||||
cssClass: "sb-link",
|
|
||||||
href: cleanLink,
|
|
||||||
callback: (e) => {
|
|
||||||
if (e.altKey) {
|
|
||||||
// Move cursor into the link, approximate location
|
|
||||||
return view.dispatch({
|
|
||||||
selection: { anchor: from + 1 },
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// 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);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (
|
|
||||||
update.docChanged ||
|
|
||||||
update.viewportChanged ||
|
|
||||||
update.selectionSet
|
|
||||||
) {
|
|
||||||
this.decorations = this.calculateDecorations();
|
|
||||||
}
|
}
|
||||||
}
|
// Adding 2 on each side due to [[ and ]] that are outside the WikiLinkPage node
|
||||||
},
|
if (isCursorInRange(state, [from, to])) {
|
||||||
{ decorations: (v) => v.decorations },
|
return;
|
||||||
);
|
}
|
||||||
|
|
||||||
|
const text = state.sliceDoc(from, to);
|
||||||
|
// Links are of the form [hell](https://example.com)
|
||||||
|
const [anchorPart, linkPart] = text.split("]("); // Not pretty
|
||||||
|
if (!linkPart) {
|
||||||
|
// Invalid link
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const cleanAnchor = anchorPart.substring(1); // cut off the initial [
|
||||||
|
const cleanLink = linkPart.substring(0, linkPart.length - 1); // cut off the final )
|
||||||
|
|
||||||
|
// Hide the whole thing
|
||||||
|
widgets.push(
|
||||||
|
invisibleDecoration.range(
|
||||||
|
from,
|
||||||
|
to,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
widgets.push(
|
||||||
|
Decoration.widget({
|
||||||
|
widget: new LinkWidget(
|
||||||
|
{
|
||||||
|
text: cleanAnchor,
|
||||||
|
title: `Click to visit ${cleanLink}`,
|
||||||
|
cssClass: "sb-link",
|
||||||
|
href: cleanLink,
|
||||||
|
callback: (e) => {
|
||||||
|
if (e.altKey) {
|
||||||
|
// Move cursor into the link, approximate location
|
||||||
|
return editor.editorView!.dispatch({
|
||||||
|
selection: { anchor: from + 1 },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 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);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,38 +2,19 @@
|
||||||
// Original author: Pranav Karawale
|
// Original author: Pranav Karawale
|
||||||
// License: Apache License 2.0.
|
// License: Apache License 2.0.
|
||||||
|
|
||||||
import {
|
import { Decoration, syntaxTree, WidgetType } from "../deps.ts";
|
||||||
Decoration,
|
import { decoratorStateField, isCursorInRange } from "./util.ts";
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
WidgetType,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { isCursorInRange, iterateTreeInVisibleRanges } from "./util.ts";
|
|
||||||
|
|
||||||
const bulletListMarkerRE = /^[-+*]/;
|
const bulletListMarkerRE = /^[-+*]/;
|
||||||
|
|
||||||
/**
|
export function listBulletPlugin() {
|
||||||
* Plugin to add custom list bullet mark.
|
return decoratorStateField((state) => {
|
||||||
*/
|
|
||||||
class ListBulletPlugin {
|
|
||||||
decorations: DecorationSet = Decoration.none;
|
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = this.decorateLists(view);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
|
||||||
this.decorations = this.decorateLists(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
private decorateLists(view: EditorView) {
|
|
||||||
const widgets: any[] = [];
|
const widgets: any[] = [];
|
||||||
iterateTreeInVisibleRanges(view, {
|
syntaxTree(state).iterate({
|
||||||
enter: ({ type, from, to }) => {
|
enter: ({ type, from, to }) => {
|
||||||
if (isCursorInRange(view.state, [from, to])) return;
|
if (isCursorInRange(state, [from, to])) return;
|
||||||
if (type.name === "ListMark") {
|
if (type.name === "ListMark") {
|
||||||
const listMark = view.state.sliceDoc(from, to);
|
const listMark = state.sliceDoc(from, to);
|
||||||
if (bulletListMarkerRE.test(listMark)) {
|
if (bulletListMarkerRE.test(listMark)) {
|
||||||
const dec = Decoration.replace({
|
const dec = Decoration.replace({
|
||||||
widget: new ListBulletWidget(listMark),
|
widget: new ListBulletWidget(listMark),
|
||||||
|
@ -44,11 +25,8 @@ class ListBulletPlugin {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
export const listBulletPlugin = ViewPlugin.fromClass(ListBulletPlugin, {
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
});
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Widget to render list bullet mark.
|
* Widget to render list bullet mark.
|
||||||
|
|
|
@ -1,21 +1,20 @@
|
||||||
import {
|
import {
|
||||||
Decoration,
|
Decoration,
|
||||||
DecorationSet,
|
EditorState,
|
||||||
EditorView,
|
EditorView,
|
||||||
ViewPlugin,
|
syntaxTree,
|
||||||
ViewUpdate,
|
|
||||||
WidgetType,
|
WidgetType,
|
||||||
} from "../deps.ts";
|
} from "../deps.ts";
|
||||||
import {
|
import {
|
||||||
editorLines,
|
decoratorStateField,
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
import { renderMarkdownToHtml } from "../../plugs/markdown/markdown_render.ts";
|
import { renderMarkdownToHtml } from "../../plugs/markdown/markdown_render.ts";
|
||||||
import { ParseTree } from "$sb/lib/tree.ts";
|
import { ParseTree } from "$sb/lib/tree.ts";
|
||||||
import { lezerToParseTree } from "../../common/parse_tree.ts";
|
import { lezerToParseTree } from "../../common/parse_tree.ts";
|
||||||
|
import type { Editor } from "../editor.tsx";
|
||||||
|
|
||||||
class TableViewWidget extends WidgetType {
|
class TableViewWidget extends WidgetType {
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -49,25 +48,27 @@ class TableViewWidget extends WidgetType {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class TablePlugin {
|
export function tablePlugin(editor: Editor) {
|
||||||
decorations: DecorationSet = Decoration.none;
|
return decoratorStateField((state: EditorState) => {
|
||||||
constructor(view: EditorView) {
|
|
||||||
this.decorations = this.decorateLists(view);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
|
||||||
this.decorations = this.decorateLists(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
private decorateLists(view: EditorView) {
|
|
||||||
const widgets: any[] = [];
|
const widgets: any[] = [];
|
||||||
iterateTreeInVisibleRanges(view, {
|
syntaxTree(state).iterate({
|
||||||
enter: (node) => {
|
enter: (node) => {
|
||||||
const { from, to, name } = node;
|
const { from, to, name } = node;
|
||||||
if (name !== "Table") return;
|
if (name !== "Table") return;
|
||||||
if (isCursorInRange(view.state, [from, to])) return;
|
if (isCursorInRange(state, [from, to])) return;
|
||||||
|
|
||||||
const lines = editorLines(view, from, to);
|
const tableText = state.sliceDoc(from, to);
|
||||||
|
const lineStrings = tableText.split("\n");
|
||||||
|
|
||||||
|
const lines: { from: number; to: number }[] = [];
|
||||||
|
let fromIt = from;
|
||||||
|
for (const line of lineStrings) {
|
||||||
|
lines.push({
|
||||||
|
from: fromIt,
|
||||||
|
to: fromIt + line.length,
|
||||||
|
});
|
||||||
|
fromIt += line.length + 1;
|
||||||
|
}
|
||||||
|
|
||||||
const firstLine = lines[0], lastLine = lines[lines.length - 1];
|
const firstLine = lines[0], lastLine = lines[lines.length - 1];
|
||||||
|
|
||||||
|
@ -84,12 +85,12 @@ class TablePlugin {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
const text = view.state.sliceDoc(0, to);
|
const text = state.sliceDoc(0, to);
|
||||||
widgets.push(
|
widgets.push(
|
||||||
Decoration.widget({
|
Decoration.widget({
|
||||||
widget: new TableViewWidget(
|
widget: new TableViewWidget(
|
||||||
from,
|
from,
|
||||||
view,
|
editor.editorView!,
|
||||||
lezerToParseTree(text, node.node),
|
lezerToParseTree(text, node.node),
|
||||||
),
|
),
|
||||||
}).range(from),
|
}).range(from),
|
||||||
|
@ -97,11 +98,5 @@ class TablePlugin {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return Decoration.set(widgets, true);
|
return Decoration.set(widgets, true);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
export const tablePlugin = ViewPlugin.fromClass(
|
|
||||||
TablePlugin,
|
|
||||||
{
|
|
||||||
decorations: (v) => v.decorations,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
|
@ -1,73 +1,5 @@
|
||||||
import {
|
import { Decoration, NodeType, syntaxTree, WidgetType } from "../deps.ts";
|
||||||
Decoration,
|
import { decoratorStateField, isCursorInRange } from "./util.ts";
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
NodeType,
|
|
||||||
SyntaxNodeRef,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
WidgetType,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { isCursorInRange, iterateTreeInVisibleRanges } from "./util.ts";
|
|
||||||
|
|
||||||
// TODO: Find a nicer way to inject this on task handler into the class
|
|
||||||
function TaskListsPluginFactory(onCheckboxClick: (pos: number) => void) {
|
|
||||||
return class TaskListsPlugin {
|
|
||||||
decorations: DecorationSet = Decoration.none;
|
|
||||||
constructor(
|
|
||||||
view: EditorView,
|
|
||||||
) {
|
|
||||||
this.decorations = this.addCheckboxes(view);
|
|
||||||
}
|
|
||||||
update(update: ViewUpdate) {
|
|
||||||
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
|
||||||
this.decorations = this.addCheckboxes(update.view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
addCheckboxes(view: EditorView) {
|
|
||||||
const widgets: any[] = [];
|
|
||||||
iterateTreeInVisibleRanges(view, {
|
|
||||||
enter: this.iterateTree(view, widgets),
|
|
||||||
});
|
|
||||||
return Decoration.set(widgets, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
private iterateTree(view: EditorView, widgets: any[]) {
|
|
||||||
return ({ type, from, to, node }: SyntaxNodeRef) => {
|
|
||||||
if (type.name !== "Task") return;
|
|
||||||
let checked = false;
|
|
||||||
// Iterate inside the task node to find the checkbox
|
|
||||||
node.toTree().iterate({
|
|
||||||
enter: (ref) => iterateInner(ref.type, ref.from, ref.to),
|
|
||||||
});
|
|
||||||
if (checked) {
|
|
||||||
widgets.push(
|
|
||||||
Decoration.mark({
|
|
||||||
tagName: "span",
|
|
||||||
class: "cm-task-checked",
|
|
||||||
}).range(from, to),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function iterateInner(type: NodeType, nfrom: number, nto: number) {
|
|
||||||
if (type.name !== "TaskMarker") return;
|
|
||||||
if (isCursorInRange(view.state, [from + nfrom, from + nto])) return;
|
|
||||||
const checkbox = view.state.sliceDoc(from + nfrom, from + nto);
|
|
||||||
// Checkbox is checked if it has a 'x' in between the []
|
|
||||||
if ("xX".includes(checkbox[1])) checked = true;
|
|
||||||
const dec = Decoration.replace({
|
|
||||||
widget: new CheckboxWidget(
|
|
||||||
checked,
|
|
||||||
from + nfrom + 1,
|
|
||||||
onCheckboxClick,
|
|
||||||
),
|
|
||||||
});
|
|
||||||
widgets.push(dec.range(from + nfrom, from + nto));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Widget to render checkbox for a task list item.
|
* Widget to render checkbox for a task list item.
|
||||||
|
@ -80,7 +12,7 @@ class CheckboxWidget extends WidgetType {
|
||||||
) {
|
) {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
toDOM(_view: EditorView): HTMLElement {
|
toDOM(): HTMLElement {
|
||||||
const wrap = document.createElement("span");
|
const wrap = document.createElement("span");
|
||||||
wrap.classList.add("sb-checkbox");
|
wrap.classList.add("sb-checkbox");
|
||||||
const checkbox = document.createElement("input");
|
const checkbox = document.createElement("input");
|
||||||
|
@ -100,7 +32,42 @@ class CheckboxWidget extends WidgetType {
|
||||||
export function taskListPlugin(
|
export function taskListPlugin(
|
||||||
{ onCheckboxClick }: { onCheckboxClick: (pos: number) => void },
|
{ onCheckboxClick }: { onCheckboxClick: (pos: number) => void },
|
||||||
) {
|
) {
|
||||||
return ViewPlugin.fromClass(TaskListsPluginFactory(onCheckboxClick), {
|
return decoratorStateField((state) => {
|
||||||
decorations: (v) => v.decorations,
|
const widgets: any[] = [];
|
||||||
|
syntaxTree(state).iterate({
|
||||||
|
enter({ type, from, to, node }) {
|
||||||
|
if (type.name !== "Task") return;
|
||||||
|
let checked = false;
|
||||||
|
// Iterate inside the task node to find the checkbox
|
||||||
|
node.toTree().iterate({
|
||||||
|
enter: (ref) => iterateInner(ref.type, ref.from, ref.to),
|
||||||
|
});
|
||||||
|
if (checked) {
|
||||||
|
widgets.push(
|
||||||
|
Decoration.mark({
|
||||||
|
tagName: "span",
|
||||||
|
class: "cm-task-checked",
|
||||||
|
}).range(from, to),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function iterateInner(type: NodeType, nfrom: number, nto: number) {
|
||||||
|
if (type.name !== "TaskMarker") return;
|
||||||
|
if (isCursorInRange(state, [from + nfrom, from + nto])) return;
|
||||||
|
const checkbox = state.sliceDoc(from + nfrom, from + nto);
|
||||||
|
// Checkbox is checked if it has a 'x' in between the []
|
||||||
|
if ("xX".includes(checkbox[1])) checked = true;
|
||||||
|
const dec = Decoration.replace({
|
||||||
|
widget: new CheckboxWidget(
|
||||||
|
checked,
|
||||||
|
from + nfrom + 1,
|
||||||
|
onCheckboxClick,
|
||||||
|
),
|
||||||
|
});
|
||||||
|
widgets.push(dec.range(from + nfrom, from + nto));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return Decoration.set(widgets, true);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,11 +3,12 @@
|
||||||
// License: Apache License 2.0.
|
// License: Apache License 2.0.
|
||||||
import {
|
import {
|
||||||
Decoration,
|
Decoration,
|
||||||
|
DecorationSet,
|
||||||
EditorState,
|
EditorState,
|
||||||
EditorView,
|
EditorView,
|
||||||
foldedRanges,
|
foldedRanges,
|
||||||
SyntaxNodeRef,
|
StateField,
|
||||||
syntaxTree,
|
Transaction,
|
||||||
WidgetType,
|
WidgetType,
|
||||||
} from "../deps.ts";
|
} from "../deps.ts";
|
||||||
|
|
||||||
|
@ -39,6 +40,25 @@ export class LinkWidget extends WidgetType {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function decoratorStateField(
|
||||||
|
stateToDecoratorMapper: (state: EditorState) => DecorationSet,
|
||||||
|
) {
|
||||||
|
return StateField.define<DecorationSet>({
|
||||||
|
create(state: EditorState) {
|
||||||
|
return stateToDecoratorMapper(state);
|
||||||
|
},
|
||||||
|
|
||||||
|
update(value: DecorationSet, tr: Transaction) {
|
||||||
|
// if (tr.docChanged || tr.selection) {
|
||||||
|
return stateToDecoratorMapper(tr.state);
|
||||||
|
// }
|
||||||
|
// return value;
|
||||||
|
},
|
||||||
|
|
||||||
|
provide: (f) => EditorView.decorations.from(f),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
export class ButtonWidget extends WidgetType {
|
export class ButtonWidget extends WidgetType {
|
||||||
constructor(
|
constructor(
|
||||||
readonly text: string,
|
readonly text: string,
|
||||||
|
@ -106,19 +126,6 @@ export function isCursorInRange(state: EditorState, range: [number, number]) {
|
||||||
*/
|
*/
|
||||||
export const invisibleDecoration = Decoration.replace({});
|
export const invisibleDecoration = Decoration.replace({});
|
||||||
|
|
||||||
export function iterateTreeInVisibleRanges(
|
|
||||||
view: EditorView,
|
|
||||||
iterateFns: {
|
|
||||||
enter(node: SyntaxNodeRef): boolean | void;
|
|
||||||
leave?(node: SyntaxNodeRef): void;
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
// for (const { from, to } of view.visibleRanges) {
|
|
||||||
// syntaxTree(view.state).iterate({ ...iterateFns, from, to });
|
|
||||||
// }
|
|
||||||
syntaxTree(view.state).iterate(iterateFns);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the lines of the editor that are in the given range and not folded.
|
* Returns the lines of the editor that are in the given range and not folded.
|
||||||
* This function is of use when you need to get the lines of a particular
|
* This function is of use when you need to get the lines of a particular
|
||||||
|
|
|
@ -1,17 +1,11 @@
|
||||||
import { pageLinkRegex } from "../../common/parser.ts";
|
import { pageLinkRegex } from "../../common/parser.ts";
|
||||||
import { ClickEvent } from "../../plug-api/app_event.ts";
|
import { ClickEvent } from "../../plug-api/app_event.ts";
|
||||||
import {
|
import { Decoration, syntaxTree } from "../deps.ts";
|
||||||
Decoration,
|
|
||||||
DecorationSet,
|
|
||||||
EditorView,
|
|
||||||
ViewPlugin,
|
|
||||||
ViewUpdate,
|
|
||||||
} from "../deps.ts";
|
|
||||||
import { Editor } from "../editor.tsx";
|
import { Editor } from "../editor.tsx";
|
||||||
import {
|
import {
|
||||||
|
decoratorStateField,
|
||||||
invisibleDecoration,
|
invisibleDecoration,
|
||||||
isCursorInRange,
|
isCursorInRange,
|
||||||
iterateTreeInVisibleRanges,
|
|
||||||
LinkWidget,
|
LinkWidget,
|
||||||
} from "./util.ts";
|
} from "./util.ts";
|
||||||
|
|
||||||
|
@ -19,119 +13,99 @@ import {
|
||||||
* Plugin to hide path prefix when the cursor is not inside.
|
* Plugin to hide path prefix when the cursor is not inside.
|
||||||
*/
|
*/
|
||||||
export function cleanWikiLinkPlugin(editor: Editor) {
|
export function cleanWikiLinkPlugin(editor: Editor) {
|
||||||
return ViewPlugin.fromClass(
|
return decoratorStateField((state) => {
|
||||||
class {
|
const widgets: any[] = [];
|
||||||
decorations: DecorationSet;
|
// let parentRange: [number, number];
|
||||||
constructor(view: EditorView) {
|
syntaxTree(state).iterate({
|
||||||
this.decorations = this.compute(view);
|
enter: ({ type, from, to }) => {
|
||||||
}
|
if (type.name !== "WikiLink") {
|
||||||
update(update: ViewUpdate) {
|
return;
|
||||||
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 !== "WikiLink") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = view.state.sliceDoc(from, to);
|
const text = state.sliceDoc(from, to);
|
||||||
const match = pageLinkRegex.exec(text);
|
const match = pageLinkRegex.exec(text);
|
||||||
if (!match) return;
|
if (!match) return;
|
||||||
const [_fullMatch, page, pipePart, alias] = match;
|
const [_fullMatch, page, pipePart, alias] = match;
|
||||||
|
|
||||||
const allPages = editor.space.listPages();
|
const allPages = editor.space.listPages();
|
||||||
let pageExists = false;
|
let pageExists = false;
|
||||||
let cleanPage = page;
|
let cleanPage = page;
|
||||||
if (page.includes("@")) {
|
if (page.includes("@")) {
|
||||||
cleanPage = page.split("@")[0];
|
cleanPage = page.split("@")[0];
|
||||||
}
|
}
|
||||||
for (const pageMeta of allPages) {
|
for (const pageMeta of allPages) {
|
||||||
if (pageMeta.name === cleanPage) {
|
if (pageMeta.name === cleanPage) {
|
||||||
pageExists = true;
|
pageExists = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (cleanPage === "") {
|
if (cleanPage === "") {
|
||||||
// Empty page name, or local @anchor use
|
// Empty page name, or local @anchor use
|
||||||
pageExists = true;
|
pageExists = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isCursorInRange(view.state, [from, to])) {
|
if (isCursorInRange(state, [from, to])) {
|
||||||
// Only attach a CSS class, then get out
|
// Only attach a CSS class, then get out
|
||||||
if (!pageExists) {
|
if (!pageExists) {
|
||||||
widgets.push(
|
|
||||||
Decoration.mark({
|
|
||||||
class: "sb-wiki-link-page-missing",
|
|
||||||
}).range(from + 2, from + page.length + 2),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hide the whole thing
|
|
||||||
widgets.push(
|
widgets.push(
|
||||||
invisibleDecoration.range(
|
Decoration.mark({
|
||||||
from,
|
class: "sb-wiki-link-page-missing",
|
||||||
to,
|
}).range(from + 2, from + page.length + 2),
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let linkText = alias || page;
|
// Hide the whole thing
|
||||||
if (!pipePart && text.indexOf("/") !== -1) {
|
widgets.push(
|
||||||
// Let's use the last part of the path as the link text
|
invisibleDecoration.range(
|
||||||
linkText = page.split("/").pop()!;
|
from,
|
||||||
}
|
to,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
// And replace it with a widget
|
let linkText = alias || page;
|
||||||
widgets.push(
|
if (!pipePart && text.indexOf("/") !== -1) {
|
||||||
Decoration.widget({
|
// Let's use the last part of the path as the link text
|
||||||
widget: new LinkWidget(
|
linkText = page.split("/").pop()!;
|
||||||
{
|
}
|
||||||
text: linkText,
|
|
||||||
title: pageExists
|
// And replace it with a widget
|
||||||
? `Navigate to ${page}`
|
widgets.push(
|
||||||
: `Create ${page}`,
|
Decoration.widget({
|
||||||
href: `/${page.replaceAll(" ", "_")}`,
|
widget: new LinkWidget(
|
||||||
cssClass: pageExists
|
{
|
||||||
? "sb-wiki-link-page"
|
text: linkText,
|
||||||
: "sb-wiki-link-page-missing",
|
title: pageExists ? `Navigate to ${page}` : `Create ${page}`,
|
||||||
callback: (e) => {
|
href: `/${page.replaceAll(" ", "_")}`,
|
||||||
if (e.altKey) {
|
cssClass: pageExists
|
||||||
// Move cursor into the link
|
? "sb-wiki-link-page"
|
||||||
return view.dispatch({
|
: "sb-wiki-link-page-missing",
|
||||||
selection: { anchor: from + 2 },
|
callback: (e) => {
|
||||||
});
|
if (e.altKey) {
|
||||||
}
|
// Move cursor into the link
|
||||||
// Dispatch click event to navigate there without moving the cursor
|
return editor.editorView!.dispatch({
|
||||||
const clickEvent: ClickEvent = {
|
selection: { anchor: from + 2 },
|
||||||
page: editor.currentPage!,
|
});
|
||||||
ctrlKey: e.ctrlKey,
|
}
|
||||||
metaKey: e.metaKey,
|
// Dispatch click event to navigate there without moving the cursor
|
||||||
altKey: e.altKey,
|
const clickEvent: ClickEvent = {
|
||||||
pos: from,
|
page: editor.currentPage!,
|
||||||
};
|
ctrlKey: e.ctrlKey,
|
||||||
editor.dispatchAppEvent("page:click", clickEvent).catch(
|
metaKey: e.metaKey,
|
||||||
console.error,
|
altKey: e.altKey,
|
||||||
);
|
pos: from,
|
||||||
},
|
};
|
||||||
},
|
editor.dispatchAppEvent("page:click", clickEvent).catch(
|
||||||
),
|
console.error,
|
||||||
}).range(from),
|
);
|
||||||
);
|
},
|
||||||
},
|
},
|
||||||
});
|
),
|
||||||
return Decoration.set(widgets, true);
|
}).range(from),
|
||||||
}
|
);
|
||||||
},
|
},
|
||||||
{
|
});
|
||||||
decorations: (v) => v.decorations,
|
return Decoration.set(widgets, true);
|
||||||
},
|
});
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -734,7 +734,6 @@ export class Editor {
|
||||||
const editorView = this.editorView!;
|
const editorView = this.editorView!;
|
||||||
if (pageState) {
|
if (pageState) {
|
||||||
// Restore state
|
// Restore state
|
||||||
// console.log("Restoring selection state", pageState);
|
|
||||||
editorView.scrollDOM.scrollTop = pageState!.scrollTop;
|
editorView.scrollDOM.scrollTop = pageState!.scrollTop;
|
||||||
editorView.dispatch({
|
editorView.dispatch({
|
||||||
selection: pageState.selection,
|
selection: pageState.selection,
|
||||||
|
|
Loading…
Reference in New Issue