2023-08-28 23:12:15 +08:00
|
|
|
import { editor, space } from "$sb/syscalls.ts";
|
2023-07-28 21:20:56 +08:00
|
|
|
import { validatePageName } from "$sb/lib/page.ts";
|
|
|
|
import { getBackLinks } from "./page_links.ts";
|
2023-01-05 22:37:08 +08:00
|
|
|
|
2023-07-28 21:20:56 +08:00
|
|
|
export async function renamePageCommand(cmdDef: any) {
|
|
|
|
const oldName = await editor.getCurrentPage();
|
|
|
|
console.log("Old name is", oldName);
|
|
|
|
const newName = cmdDef.page ||
|
|
|
|
await editor.prompt(`Rename ${oldName} to:`, oldName);
|
|
|
|
if (!newName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
validatePageName(newName);
|
|
|
|
} catch (e: any) {
|
|
|
|
return editor.flashNotification(e.message, "error");
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("New name", newName);
|
|
|
|
|
|
|
|
if (newName.trim() === oldName.trim()) {
|
|
|
|
// Nothing to do here
|
|
|
|
console.log("Name unchanged, exiting");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await editor.save();
|
|
|
|
|
|
|
|
try {
|
|
|
|
console.log(
|
|
|
|
"Checking if target page already exists, this should result in a 'Not found' error",
|
|
|
|
);
|
|
|
|
try {
|
|
|
|
// This throws an error if the page does not exist, which we expect to be the case
|
|
|
|
await space.getPageMeta(newName);
|
|
|
|
// So when we get to this point, we error out
|
|
|
|
throw new Error(
|
|
|
|
`Page ${newName} already exists, cannot rename to existing page.`,
|
|
|
|
);
|
|
|
|
} catch (e: any) {
|
|
|
|
if (e.message === "Not found") {
|
|
|
|
// Expected not found error, so we can continue
|
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
2023-08-30 23:25:54 +08:00
|
|
|
const updatedReferences = await renamePage(oldName, newName, true);
|
2023-07-28 21:20:56 +08:00
|
|
|
|
|
|
|
await editor.flashNotification(
|
|
|
|
`Renamed page, and updated ${updatedReferences} references`,
|
|
|
|
);
|
|
|
|
} catch (e: any) {
|
|
|
|
await editor.flashNotification(e.message, "error");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 23:25:54 +08:00
|
|
|
async function renamePage(
|
|
|
|
oldName: string,
|
|
|
|
newName: string,
|
|
|
|
navigateThere = false,
|
|
|
|
): Promise<number> {
|
2023-07-28 21:20:56 +08:00
|
|
|
const text = await space.readPage(oldName);
|
|
|
|
|
|
|
|
console.log("Writing new page to space");
|
|
|
|
const newPageMeta = await space.writePage(newName, text);
|
|
|
|
|
2023-08-30 23:25:54 +08:00
|
|
|
if (navigateThere) {
|
|
|
|
console.log("Navigating to new page");
|
2024-01-24 18:58:33 +08:00
|
|
|
await editor.navigate({ page: newName, pos: 0 }, true);
|
2023-08-30 23:25:54 +08:00
|
|
|
}
|
|
|
|
|
2023-07-28 21:20:56 +08:00
|
|
|
const pagesToUpdate = await getBackLinks(oldName);
|
|
|
|
console.log("All pages containing backlinks", pagesToUpdate);
|
|
|
|
|
|
|
|
// Handling the edge case of a changing page name just in casing on a case insensitive FS
|
|
|
|
const oldPageMeta = await space.getPageMeta(oldName);
|
|
|
|
if (oldPageMeta.lastModified !== newPageMeta.lastModified) {
|
|
|
|
// If they're the same, let's assume it's the same file (case insensitive FS) and not delete, otherwise...
|
|
|
|
console.log("Deleting page from space");
|
|
|
|
await space.deletePage(oldName);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is the bit where we update all the links
|
|
|
|
const pageToUpdateSet = new Set<string>();
|
|
|
|
for (const pageToUpdate of pagesToUpdate) {
|
|
|
|
pageToUpdateSet.add(pageToUpdate.page);
|
|
|
|
}
|
|
|
|
|
|
|
|
let updatedReferences = 0;
|
|
|
|
|
|
|
|
for (const pageToUpdate of pageToUpdateSet) {
|
|
|
|
if (pageToUpdate === oldName) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
console.log("Now going to update links in", pageToUpdate);
|
|
|
|
const text = await space.readPage(pageToUpdate);
|
|
|
|
// console.log("Received text", text);
|
|
|
|
if (!text) {
|
|
|
|
// Page likely does not exist, but at least we can skip it
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-24 18:58:33 +08:00
|
|
|
// Replace all links found in place following the patterns [[Page]] and [[Page@pos]] as well as [[Page$anchor]]
|
2023-07-28 21:20:56 +08:00
|
|
|
const newText = text.replaceAll(`[[${oldName}]]`, () => {
|
2024-01-28 20:48:27 +08:00
|
|
|
// Plain link format
|
2023-07-28 21:20:56 +08:00
|
|
|
updatedReferences++;
|
|
|
|
return `[[${newName}]]`;
|
2024-01-28 20:48:27 +08:00
|
|
|
}).replaceAll(`[[${oldName}|`, () => {
|
|
|
|
// Aliased link format
|
|
|
|
updatedReferences++;
|
|
|
|
return `[[${newName}|`;
|
2023-07-28 21:20:56 +08:00
|
|
|
}).replaceAll(`[[${oldName}@`, () => {
|
2024-01-28 20:48:27 +08:00
|
|
|
// Link with position format
|
2023-07-28 21:20:56 +08:00
|
|
|
updatedReferences++;
|
|
|
|
return `[[${newName}@`;
|
2024-01-24 18:58:33 +08:00
|
|
|
}).replaceAll(`[[${oldName}$`, () => {
|
2024-01-28 20:48:27 +08:00
|
|
|
// Link with anchor format
|
2024-01-24 18:58:33 +08:00
|
|
|
updatedReferences++;
|
|
|
|
return `[[${newName}$`;
|
2024-01-28 20:48:27 +08:00
|
|
|
}).replaceAll(`[[${oldName}#`, () => {
|
|
|
|
// Link with header format
|
|
|
|
updatedReferences++;
|
|
|
|
return `[[${newName}#`;
|
2023-07-28 21:20:56 +08:00
|
|
|
});
|
|
|
|
if (text !== newText) {
|
|
|
|
console.log("Changes made, saving...");
|
|
|
|
await space.writePage(pageToUpdate, newText);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return updatedReferences;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function renamePrefixCommand() {
|
|
|
|
const oldPrefix = await editor.prompt("Prefix to rename:", "");
|
|
|
|
if (!oldPrefix) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const newPrefix = await editor.prompt("New prefix:", oldPrefix);
|
|
|
|
if (!newPrefix) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const allPages = await space.listPages();
|
|
|
|
const allAffectedPages = allPages.map((page) => page.name).filter((page) =>
|
|
|
|
page.startsWith(oldPrefix)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (
|
|
|
|
!(await editor.confirm(
|
|
|
|
`This will affect ${allAffectedPages.length} pages. Are you sure?`,
|
|
|
|
))
|
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const allNewNames = allAffectedPages.map((name) =>
|
|
|
|
// This may seem naive, but it's actually fine, because we're only renaming the first occurrence (which will be the prefix)
|
|
|
|
name.replace(oldPrefix, newPrefix)
|
|
|
|
);
|
|
|
|
|
|
|
|
try {
|
|
|
|
console.log("Pre-flight check to see if all new names are available");
|
|
|
|
await Promise.all(allNewNames.map(async (name) => {
|
|
|
|
try {
|
|
|
|
await space.getPageMeta(name);
|
|
|
|
// If we got here, the page exists, so we error out
|
|
|
|
throw Error(
|
|
|
|
`Target ${name} already exists, cannot perform batch rename when one of the target pages already exists.`,
|
|
|
|
);
|
|
|
|
} catch (e: any) {
|
|
|
|
if (e.message === "Not found") {
|
|
|
|
// Expected not found error, so we can continue
|
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
console.log("All new names are available, proceeding with rename");
|
|
|
|
for (let i = 0; i < allAffectedPages.length; i++) {
|
|
|
|
const oldName = allAffectedPages[i];
|
|
|
|
const newName = allNewNames[i];
|
|
|
|
console.log("Now renaming", oldName, "to", newName);
|
|
|
|
await renamePage(oldName, newName);
|
|
|
|
}
|
|
|
|
|
|
|
|
await editor.flashNotification("Batch rename complete", "info");
|
|
|
|
} catch (e: any) {
|
|
|
|
return editor.flashNotification(e.message, "error");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function extractToPageCommand() {
|
2023-01-23 01:53:14 +08:00
|
|
|
const newName = await editor.prompt(`New page title:`, "new page");
|
2023-01-05 22:37:08 +08:00
|
|
|
if (!newName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("New name", newName);
|
|
|
|
|
|
|
|
try {
|
|
|
|
// This throws an error if the page does not exist, which we expect to be the case
|
|
|
|
await space.getPageMeta(newName);
|
|
|
|
// So when we get to this point, we error out
|
|
|
|
throw new Error(
|
|
|
|
`Page ${newName} already exists, cannot rename to existing page.`,
|
|
|
|
);
|
|
|
|
} catch (e: any) {
|
2023-05-24 02:53:53 +08:00
|
|
|
if (e.message === "Not found") {
|
2023-01-05 22:37:08 +08:00
|
|
|
// Expected not found error, so we can continue
|
|
|
|
} else {
|
|
|
|
await editor.flashNotification(e.message, "error");
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let text = await editor.getText();
|
|
|
|
const selection = await editor.getSelection();
|
|
|
|
text = text.slice(selection.from, selection.to);
|
|
|
|
await editor.replaceRange(selection.from, selection.to, `[[${newName}]]`);
|
|
|
|
console.log("Writing new page to space");
|
|
|
|
await space.writePage(newName, text);
|
|
|
|
console.log("Navigating to new page");
|
2024-01-24 18:58:33 +08:00
|
|
|
await editor.navigate({ page: newName });
|
2023-01-05 22:37:08 +08:00
|
|
|
}
|