2023-07-02 18:15:37 +08:00
|
|
|
import { EditorView, syntaxTree, ViewPlugin, ViewUpdate } from "../deps.ts";
|
2023-07-14 22:56:20 +08:00
|
|
|
import { Client } from "../client.ts";
|
2022-04-12 02:34:09 +08:00
|
|
|
|
2022-11-09 18:38:12 +08:00
|
|
|
// We use turndown to convert HTML to Markdown
|
|
|
|
import TurndownService from "https://cdn.skypack.dev/turndown@7.1.1";
|
|
|
|
|
|
|
|
// With tables and task notation as well
|
|
|
|
import {
|
|
|
|
tables,
|
|
|
|
taskListItems,
|
|
|
|
} from "https://cdn.skypack.dev/@joplin/turndown-plugin-gfm@1.0.45";
|
2023-05-24 02:53:53 +08:00
|
|
|
import { safeRun } from "../../common/util.ts";
|
2023-07-02 18:15:37 +08:00
|
|
|
import { lezerToParseTree } from "../../common/markdown_parser/parse_tree.ts";
|
|
|
|
import {
|
|
|
|
addParentPointers,
|
|
|
|
findParentMatching,
|
|
|
|
nodeAtPos,
|
|
|
|
} from "../../plug-api/lib/tree.ts";
|
2023-07-07 19:09:44 +08:00
|
|
|
import { folderName, resolve } from "../../common/path.ts";
|
2023-08-20 23:51:00 +08:00
|
|
|
import { maximumAttachmentSize } from "../constants.ts";
|
2023-07-07 19:09:44 +08:00
|
|
|
|
2022-11-09 18:38:12 +08:00
|
|
|
const turndownService = new TurndownService({
|
|
|
|
hr: "---",
|
|
|
|
codeBlockStyle: "fenced",
|
|
|
|
headingStyle: "atx",
|
|
|
|
emDelimiter: "*",
|
|
|
|
bulletListMarker: "*", // Duh!
|
|
|
|
strongDelimiter: "**",
|
|
|
|
linkStyle: "inlined",
|
|
|
|
});
|
|
|
|
turndownService.use(taskListItems);
|
|
|
|
turndownService.use(tables);
|
|
|
|
|
|
|
|
function striptHtmlComments(s: string): string {
|
|
|
|
return s.replace(/<!--[\s\S]*?-->/g, "");
|
|
|
|
}
|
|
|
|
|
2022-04-12 02:34:09 +08:00
|
|
|
const urlRegexp =
|
|
|
|
/^https?:\/\/[-a-zA-Z0-9@:%._\+~#=]{1,256}([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/;
|
2022-03-30 21:16:22 +08:00
|
|
|
|
2022-05-16 21:09:36 +08:00
|
|
|
// Known iOS Safari paste issue (unrelated to this implementation): https://voxpelli.com/2015/03/ios-safari-url-copy-paste-bug/
|
2022-03-30 21:16:22 +08:00
|
|
|
export const pasteLinkExtension = ViewPlugin.fromClass(
|
|
|
|
class {
|
|
|
|
update(update: ViewUpdate): void {
|
|
|
|
update.transactions.forEach((tr) => {
|
|
|
|
if (tr.isUserEvent("input.paste")) {
|
2022-10-16 01:02:56 +08:00
|
|
|
const pastedText: string[] = [];
|
2022-03-30 21:16:22 +08:00
|
|
|
let from = 0;
|
|
|
|
let to = 0;
|
2022-10-16 01:02:56 +08:00
|
|
|
tr.changes.iterChanges((fromA, _toA, _fromB, toB, inserted) => {
|
2022-03-30 21:16:22 +08:00
|
|
|
pastedText.push(inserted.sliceString(0));
|
|
|
|
from = fromA;
|
|
|
|
to = toB;
|
|
|
|
});
|
2022-10-16 01:02:56 +08:00
|
|
|
const pastedString = pastedText.join("");
|
2022-03-30 21:16:22 +08:00
|
|
|
if (pastedString.match(urlRegexp)) {
|
2022-10-16 01:02:56 +08:00
|
|
|
const selection = update.startState.selection.main;
|
2022-03-30 21:16:22 +08:00
|
|
|
if (!selection.empty) {
|
|
|
|
setTimeout(() => {
|
|
|
|
update.view.dispatch({
|
|
|
|
changes: [
|
|
|
|
{
|
|
|
|
from: from,
|
|
|
|
to: to,
|
2022-10-10 20:50:21 +08:00
|
|
|
insert: `[${
|
|
|
|
update.startState.sliceDoc(
|
|
|
|
selection.from,
|
|
|
|
selection.to,
|
|
|
|
)
|
|
|
|
}](${pastedString})`,
|
2022-03-30 21:16:22 +08:00
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2022-10-10 20:50:21 +08:00
|
|
|
},
|
2022-03-30 21:16:22 +08:00
|
|
|
);
|
2022-09-05 17:47:30 +08:00
|
|
|
|
2023-07-14 22:56:20 +08:00
|
|
|
export function attachmentExtension(editor: Client) {
|
2022-11-23 18:53:11 +08:00
|
|
|
let shiftDown = false;
|
2022-09-05 17:47:30 +08:00
|
|
|
return EditorView.domEventHandlers({
|
2022-09-05 19:11:03 +08:00
|
|
|
dragover: (event) => {
|
|
|
|
event.preventDefault();
|
|
|
|
},
|
2022-11-23 18:53:11 +08:00
|
|
|
keydown: (event) => {
|
|
|
|
if (event.key === "Shift") {
|
|
|
|
shiftDown = true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
keyup: (event) => {
|
|
|
|
if (event.key === "Shift") {
|
|
|
|
shiftDown = false;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
2022-09-05 19:11:03 +08:00
|
|
|
drop: (event: DragEvent) => {
|
|
|
|
// TODO: This doesn't take into account the target cursor position,
|
|
|
|
// it just drops the attachment wherever the cursor was last.
|
|
|
|
if (event.dataTransfer) {
|
2022-10-16 01:02:56 +08:00
|
|
|
const payload = [...event.dataTransfer.files];
|
2022-09-05 19:24:37 +08:00
|
|
|
if (!payload.length) {
|
|
|
|
return;
|
|
|
|
}
|
2022-09-05 19:11:03 +08:00
|
|
|
safeRun(async () => {
|
|
|
|
await processFileTransfer(payload);
|
|
|
|
});
|
2022-09-05 17:47:30 +08:00
|
|
|
}
|
2022-09-05 19:11:03 +08:00
|
|
|
},
|
|
|
|
paste: (event: ClipboardEvent) => {
|
2022-10-16 01:02:56 +08:00
|
|
|
const payload = [...event.clipboardData!.items];
|
2022-11-09 18:38:12 +08:00
|
|
|
const richText = event.clipboardData?.getData("text/html");
|
2023-07-02 18:15:37 +08:00
|
|
|
|
2022-11-23 18:53:11 +08:00
|
|
|
// Only do rich text paste if shift is NOT down
|
|
|
|
if (richText && !shiftDown) {
|
2023-07-02 18:15:37 +08:00
|
|
|
// Are we in a fencede code block?
|
2023-07-27 17:41:44 +08:00
|
|
|
const editorText = editor.editorView.state.sliceDoc();
|
2023-07-02 18:15:37 +08:00
|
|
|
const tree = lezerToParseTree(
|
|
|
|
editorText,
|
2023-07-27 17:41:44 +08:00
|
|
|
syntaxTree(editor.editorView.state).topNode,
|
2023-07-02 18:15:37 +08:00
|
|
|
);
|
|
|
|
addParentPointers(tree);
|
|
|
|
const currentNode = nodeAtPos(
|
|
|
|
tree,
|
2023-07-27 17:41:44 +08:00
|
|
|
editor.editorView.state.selection.main.from,
|
2023-07-02 18:15:37 +08:00
|
|
|
);
|
|
|
|
if (currentNode) {
|
|
|
|
const fencedParentNode = findParentMatching(
|
|
|
|
currentNode,
|
|
|
|
(t) => t.type === "FencedCode",
|
|
|
|
);
|
|
|
|
if (fencedParentNode || currentNode.type === "FencedCode") {
|
|
|
|
console.log("Inside of fenced code block, not pasting rich text");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-09 18:38:12 +08:00
|
|
|
const markdown = striptHtmlComments(turndownService.turndown(richText))
|
|
|
|
.trim();
|
2023-07-27 17:41:44 +08:00
|
|
|
const view = editor.editorView;
|
2022-11-09 18:38:12 +08:00
|
|
|
const selection = view.state.selection.main;
|
|
|
|
view.dispatch({
|
|
|
|
changes: [
|
|
|
|
{
|
|
|
|
from: selection.from,
|
|
|
|
to: selection.to,
|
|
|
|
insert: markdown,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
selection: {
|
|
|
|
anchor: selection.from + markdown.length,
|
|
|
|
},
|
|
|
|
scrollIntoView: true,
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
}
|
2022-09-05 19:24:37 +08:00
|
|
|
if (!payload.length || payload.length === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
2022-09-05 19:11:03 +08:00
|
|
|
safeRun(async () => {
|
2022-09-05 19:24:37 +08:00
|
|
|
await processItemTransfer(payload);
|
2022-09-05 19:11:03 +08:00
|
|
|
});
|
2022-09-05 17:47:30 +08:00
|
|
|
},
|
|
|
|
});
|
2022-09-05 19:11:03 +08:00
|
|
|
|
2022-09-05 19:24:37 +08:00
|
|
|
async function processFileTransfer(payload: File[]) {
|
2022-10-16 01:02:56 +08:00
|
|
|
const data = await payload[0].arrayBuffer();
|
2023-05-24 02:53:53 +08:00
|
|
|
// data.byteLength > maximumAttachmentSize;
|
2022-09-05 19:24:37 +08:00
|
|
|
await saveFile(data!, payload[0].name, payload[0].type);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function processItemTransfer(payload: DataTransferItem[]) {
|
2022-10-16 01:02:56 +08:00
|
|
|
const file = payload.find((item) => item.kind === "file");
|
2022-09-05 19:11:03 +08:00
|
|
|
if (!file) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const fileType = file.type;
|
2022-10-16 01:02:56 +08:00
|
|
|
const ext = fileType.split("/")[1];
|
|
|
|
const fileName = new Date()
|
2022-09-05 19:24:37 +08:00
|
|
|
.toISOString()
|
|
|
|
.split(".")[0]
|
|
|
|
.replace("T", "_")
|
|
|
|
.replaceAll(":", "-");
|
2022-10-16 01:02:56 +08:00
|
|
|
const data = await file!.getAsFile()?.arrayBuffer();
|
2022-09-05 19:24:37 +08:00
|
|
|
await saveFile(data!, `${fileName}.${ext}`, fileType);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function saveFile(
|
|
|
|
data: ArrayBuffer,
|
|
|
|
suggestedName: string,
|
2022-10-10 20:50:21 +08:00
|
|
|
mimeType: string,
|
2022-09-05 19:24:37 +08:00
|
|
|
) {
|
2022-09-05 19:11:03 +08:00
|
|
|
if (data!.byteLength > maximumAttachmentSize) {
|
|
|
|
editor.flashNotification(
|
|
|
|
`Attachment is too large, maximum is ${
|
|
|
|
maximumAttachmentSize / 1024 / 1024
|
|
|
|
}MB`,
|
2022-10-10 20:50:21 +08:00
|
|
|
"error",
|
2022-09-05 19:11:03 +08:00
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2022-09-05 19:24:37 +08:00
|
|
|
|
2023-01-18 23:46:24 +08:00
|
|
|
const finalFileName = await editor.prompt(
|
2022-09-05 19:11:03 +08:00
|
|
|
"File name for pasted attachment",
|
2022-10-10 20:50:21 +08:00
|
|
|
suggestedName,
|
2022-09-05 19:11:03 +08:00
|
|
|
);
|
|
|
|
if (!finalFileName) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-02 17:25:32 +08:00
|
|
|
await editor.space.writeAttachment(
|
2023-12-20 01:59:12 +08:00
|
|
|
resolve(folderName(editor.currentPage!), finalFileName),
|
2023-07-02 17:25:32 +08:00
|
|
|
new Uint8Array(data),
|
|
|
|
);
|
|
|
|
let attachmentMarkdown = `[${finalFileName}](${encodeURI(finalFileName)})`;
|
2022-09-05 19:24:37 +08:00
|
|
|
if (mimeType.startsWith("image/")) {
|
2023-07-02 17:25:32 +08:00
|
|
|
attachmentMarkdown = `![](${encodeURI(finalFileName)})`;
|
2022-09-05 19:11:03 +08:00
|
|
|
}
|
2023-07-27 17:41:44 +08:00
|
|
|
editor.editorView.dispatch({
|
2022-09-05 19:11:03 +08:00
|
|
|
changes: [
|
|
|
|
{
|
|
|
|
insert: attachmentMarkdown,
|
2023-07-27 17:41:44 +08:00
|
|
|
from: editor.editorView.state.selection.main.from,
|
2022-09-05 19:11:03 +08:00
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
2022-09-05 17:47:30 +08:00
|
|
|
}
|