561aa6891f
Big bang migration to Deno 🤯
139 lines
3.5 KiB
TypeScript
139 lines
3.5 KiB
TypeScript
import {
|
|
getSelection,
|
|
getText,
|
|
insertAtCursor,
|
|
moveCursor,
|
|
replaceRange,
|
|
setSelection,
|
|
} from "$sb/silverbullet-syscall/editor.ts";
|
|
|
|
export async function quoteSelection() {
|
|
let text = await getText();
|
|
const selection = await getSelection();
|
|
let from = selection.from;
|
|
while (from >= 0 && text[from] !== "\n") {
|
|
from--;
|
|
}
|
|
from++;
|
|
if (text[from] === ">" && text[from + 1] === " ") {
|
|
// Already quoted, we have to unquote
|
|
text = text.slice(from + 2, selection.to);
|
|
text = text.replaceAll("\n> ", "\n");
|
|
} else {
|
|
text = text.slice(from, selection.to);
|
|
text = `> ${text.replaceAll("\n", "\n> ")}`;
|
|
}
|
|
await replaceRange(from, selection.to, text);
|
|
}
|
|
|
|
export async function listifySelection() {
|
|
let text = await getText();
|
|
const selection = await getSelection();
|
|
let from = selection.from;
|
|
while (from >= 0 && text[from] !== "\n") {
|
|
from--;
|
|
}
|
|
from++;
|
|
text = text.slice(from, selection.to);
|
|
text = `* ${text.replaceAll(/\n(?!\n)/g, "\n* ")}`;
|
|
await replaceRange(from, selection.to, text);
|
|
}
|
|
|
|
export async function numberListifySelection() {
|
|
let text = await getText();
|
|
const selection = await getSelection();
|
|
let from = selection.from;
|
|
while (from >= 0 && text[from] !== "\n") {
|
|
from--;
|
|
}
|
|
from++;
|
|
text = text.slice(from, selection.to);
|
|
let counter = 1;
|
|
text = `1. ${
|
|
text.replaceAll(/\n(?!\n)/g, () => {
|
|
counter++;
|
|
return `\n${counter}. `;
|
|
})
|
|
}`;
|
|
await replaceRange(from, selection.to, text);
|
|
}
|
|
|
|
export async function linkSelection() {
|
|
const text = await getText();
|
|
const selection = await getSelection();
|
|
const textSelection = text.slice(selection.from, selection.to);
|
|
let linkedText = `[]()`;
|
|
let pos = 1;
|
|
if (textSelection.length > 0) {
|
|
try {
|
|
new URL(textSelection);
|
|
linkedText = `[](${textSelection})`;
|
|
} catch {
|
|
linkedText = `[${textSelection}]()`;
|
|
pos = linkedText.length - 1;
|
|
}
|
|
}
|
|
await replaceRange(selection.from, selection.to, linkedText);
|
|
await moveCursor(selection.from + pos);
|
|
}
|
|
|
|
export function wrapSelection(cmdDef: any) {
|
|
return insertMarker(cmdDef.wrapper);
|
|
}
|
|
|
|
async function insertMarker(marker: string) {
|
|
let text = await getText();
|
|
const selection = await getSelection();
|
|
if (selection.from === selection.to) {
|
|
// empty selection
|
|
if (markerAt(selection.from)) {
|
|
// Already there, skipping ahead
|
|
await moveCursor(selection.from + marker.length);
|
|
} else {
|
|
// Not there, inserting
|
|
await insertAtCursor(marker + marker);
|
|
await moveCursor(selection.from + marker.length);
|
|
}
|
|
} else {
|
|
let from = selection.from;
|
|
let to = selection.to;
|
|
let hasMarker = markerAt(from);
|
|
if (!markerAt(from)) {
|
|
// Maybe just before the cursor? We'll accept that
|
|
from = selection.from - marker.length;
|
|
to = selection.to + marker.length;
|
|
hasMarker = markerAt(from);
|
|
}
|
|
|
|
if (!hasMarker) {
|
|
// Adding
|
|
await replaceRange(
|
|
selection.from,
|
|
selection.to,
|
|
marker + text.slice(selection.from, selection.to) + marker,
|
|
);
|
|
await setSelection(
|
|
selection.from + marker.length,
|
|
selection.to + marker.length,
|
|
);
|
|
} else {
|
|
// Removing
|
|
await replaceRange(
|
|
from,
|
|
to,
|
|
text.substring(from + marker.length, to - marker.length),
|
|
);
|
|
await setSelection(from, to - marker.length * 2);
|
|
}
|
|
}
|
|
|
|
function markerAt(pos: number) {
|
|
for (var i = 0; i < marker.length; i++) {
|
|
if (text[pos + i] !== marker[i]) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
}
|