2024-02-09 04:00:45 +08:00
|
|
|
import { nodeAtPos } from "$lib/tree.ts";
|
2023-08-28 23:12:15 +08:00
|
|
|
import { editor, events, markdown } from "$sb/syscalls.ts";
|
2024-02-24 20:16:04 +08:00
|
|
|
import { extractYoutubeVideoId } from "./embed.ts";
|
2022-07-30 19:39:40 +08:00
|
|
|
|
|
|
|
type UnfurlOption = {
|
|
|
|
id: string;
|
|
|
|
name: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export async function unfurlCommand() {
|
2022-10-14 21:11:33 +08:00
|
|
|
const mdTree = await markdown.parseMarkdown(await editor.getText());
|
2024-02-24 20:16:04 +08:00
|
|
|
const cursorPos = await editor.getCursor();
|
|
|
|
let nakedUrlNode = nodeAtPos(mdTree, cursorPos);
|
|
|
|
if (nakedUrlNode?.type !== "NakedURL") {
|
|
|
|
nakedUrlNode = nodeAtPos(mdTree, cursorPos - 1);
|
|
|
|
}
|
|
|
|
if (nakedUrlNode?.type !== "NakedURL") {
|
|
|
|
await editor.flashNotification("No URL found under cursor", "error");
|
|
|
|
return;
|
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
const url = nakedUrlNode!.children![0].text!;
|
2022-07-30 19:39:40 +08:00
|
|
|
console.log("Got URL to unfurl", url);
|
2022-10-14 21:11:33 +08:00
|
|
|
const optionResponses = await events.dispatchEvent("unfurl:options", url);
|
|
|
|
const options: UnfurlOption[] = [];
|
|
|
|
for (const resp of optionResponses) {
|
2022-07-30 19:39:40 +08:00
|
|
|
options.push(...resp);
|
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
const selectedUnfurl: any = await editor.filterBox(
|
2022-07-30 19:39:40 +08:00
|
|
|
"Unfurl",
|
|
|
|
options,
|
2022-10-12 17:47:13 +08:00
|
|
|
"Select the unfurl strategy of your choice",
|
2022-07-30 19:39:40 +08:00
|
|
|
);
|
|
|
|
if (!selectedUnfurl) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
2023-05-24 02:53:53 +08:00
|
|
|
const replacement = await events.dispatchEvent(
|
|
|
|
`unfurl:${selectedUnfurl.id}`,
|
2022-10-12 17:47:13 +08:00
|
|
|
url,
|
2022-07-30 19:39:40 +08:00
|
|
|
);
|
2023-05-24 02:53:53 +08:00
|
|
|
if (replacement.length === 0) {
|
|
|
|
throw new Error("Unfurl failed");
|
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
await editor.replaceRange(
|
|
|
|
nakedUrlNode?.from!,
|
|
|
|
nakedUrlNode?.to!,
|
2023-05-24 02:53:53 +08:00
|
|
|
replacement[0],
|
2022-10-14 21:11:33 +08:00
|
|
|
);
|
2022-07-30 19:39:40 +08:00
|
|
|
} catch (e: any) {
|
2022-10-14 21:11:33 +08:00
|
|
|
await editor.flashNotification(e.message, "error");
|
2022-07-30 19:39:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-14 21:11:33 +08:00
|
|
|
export function titleUnfurlOptions(): UnfurlOption[] {
|
2022-07-30 19:39:40 +08:00
|
|
|
return [
|
|
|
|
{
|
|
|
|
id: "title-unfurl",
|
|
|
|
name: "Extract title",
|
|
|
|
},
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
const titleRegex = /<title[^>]*>\s*([^<]+)\s*<\/title\s*>/i;
|
|
|
|
|
|
|
|
export async function titleUnfurl(url: string): Promise<string> {
|
2022-10-14 21:11:33 +08:00
|
|
|
const response = await fetch(url);
|
2022-07-30 19:39:40 +08:00
|
|
|
if (response.status < 200 || response.status >= 300) {
|
|
|
|
console.error("Unfurl failed", await response.text());
|
2023-10-03 20:16:33 +08:00
|
|
|
throw new Error(`Failed to fetch: ${response.statusText}`);
|
2022-07-30 19:39:40 +08:00
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
const body = await response.text();
|
|
|
|
const match = titleRegex.exec(body);
|
2022-07-30 19:39:40 +08:00
|
|
|
if (match) {
|
|
|
|
return `[${match[1]}](${url})`;
|
|
|
|
} else {
|
|
|
|
throw new Error("No title found");
|
|
|
|
}
|
|
|
|
}
|
2024-02-24 20:16:04 +08:00
|
|
|
|
|
|
|
export function youtubeUnfurlOptions(url: string): UnfurlOption[] {
|
|
|
|
if (extractYoutubeVideoId(url)) {
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
id: "youtube-unfurl",
|
|
|
|
name: "Embed video",
|
|
|
|
},
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function youtubeUnfurl(url: string): string {
|
|
|
|
return "```embed\nurl: " + url + "\n```";
|
|
|
|
}
|