2022-11-13 02:12:18 +08:00
|
|
|
import {
|
|
|
|
Decoration,
|
|
|
|
DecorationSet,
|
|
|
|
EditorView,
|
|
|
|
ViewPlugin,
|
|
|
|
ViewUpdate,
|
|
|
|
} from "../deps.ts";
|
|
|
|
import {
|
2022-11-15 02:36:35 +08:00
|
|
|
invisibleDecoration,
|
2022-11-13 02:12:18 +08:00
|
|
|
isCursorInRange,
|
|
|
|
iterateTreeInVisibleRanges,
|
|
|
|
} from "./util.ts";
|
|
|
|
|
2022-11-15 02:36:35 +08:00
|
|
|
class BlockquotePlugin {
|
|
|
|
decorations: DecorationSet = Decoration.none;
|
2022-11-13 02:12:18 +08:00
|
|
|
constructor(view: EditorView) {
|
2022-11-15 02:36:35 +08:00
|
|
|
this.decorations = this.decorateLists(view);
|
2022-11-13 02:12:18 +08:00
|
|
|
}
|
|
|
|
update(update: ViewUpdate) {
|
2022-11-15 02:36:35 +08:00
|
|
|
if (update.docChanged || update.viewportChanged || update.selectionSet) {
|
|
|
|
this.decorations = this.decorateLists(update.view);
|
2022-11-13 02:12:18 +08:00
|
|
|
}
|
|
|
|
}
|
2022-11-15 02:36:35 +08:00
|
|
|
private decorateLists(view: EditorView) {
|
2022-11-13 02:12:18 +08:00
|
|
|
const widgets: any[] = [];
|
|
|
|
iterateTreeInVisibleRanges(view, {
|
2022-11-15 02:36:35 +08:00
|
|
|
enter: ({ type, from, to }) => {
|
|
|
|
if (isCursorInRange(view.state, [from, to])) return;
|
|
|
|
if (type.name === "QuoteMark") {
|
|
|
|
widgets.push(invisibleDecoration.range(from, to));
|
|
|
|
widgets.push(
|
|
|
|
Decoration.line({ class: "sb-blockquote-outside" }).range(from),
|
|
|
|
);
|
2022-11-13 02:12:18 +08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
return Decoration.set(widgets, true);
|
|
|
|
}
|
|
|
|
}
|
2022-11-15 02:36:35 +08:00
|
|
|
export const blockquotePlugin = ViewPlugin.fromClass(
|
|
|
|
BlockquotePlugin,
|
|
|
|
{
|
|
|
|
decorations: (v) => v.decorations,
|
|
|
|
},
|
|
|
|
);
|