* multi-file paste support

* allow media files (image/videos) to be attachments
* insert trailing node if file placeholder is at the end of the editor
This commit is contained in:
Philipinho 2025-03-15 17:17:42 +00:00
parent d021d0a38f
commit 9da49fd6a5
7 changed files with 122 additions and 44 deletions

View File

@ -17,8 +17,11 @@ export const uploadAttachmentAction = handleAttachmentUpload({
throw err; throw err;
} }
}, },
validateFn: (file) => { validateFn: (file, allowMedia: boolean) => {
if (file.type.includes("image/") || file.type.includes("video/")) { if (
(file.type.includes("image/") || file.type.includes("video/")) &&
!allowMedia
) {
return false; return false;
} }
if (file.size > getFileUploadSizeLimit()) { if (file.size > getFileUploadSizeLimit()) {

View File

@ -40,10 +40,8 @@ export const handlePaste = (
if (event.clipboardData?.files.length) { if (event.clipboardData?.files.length) {
event.preventDefault(); event.preventDefault();
const [file] = Array.from(event.clipboardData.files); for (const file of event.clipboardData.files) {
const pos = view.state.selection.from; const pos = view.state.selection.from;
if (file) {
uploadImageAction(file, view, pos, pageId); uploadImageAction(file, view, pos, pageId);
uploadVideoAction(file, view, pos, pageId); uploadVideoAction(file, view, pos, pageId);
uploadAttachmentAction(file, view, pos, pageId); uploadAttachmentAction(file, view, pos, pageId);
@ -61,13 +59,13 @@ export const handleFileDrop = (
) => { ) => {
if (!moved && event.dataTransfer?.files.length) { if (!moved && event.dataTransfer?.files.length) {
event.preventDefault(); event.preventDefault();
const [file] = Array.from(event.dataTransfer.files);
const coordinates = view.posAtCoords({ for (const file of event.dataTransfer.files) {
left: event.clientX, const coordinates = view.posAtCoords({
top: event.clientY, left: event.clientX,
}); top: event.clientY,
// here we deduct 1 from the pos or else the image will create an extra node });
if (file) {
uploadImageAction(file, view, coordinates?.pos ?? 0 - 1, pageId); uploadImageAction(file, view, coordinates?.pos ?? 0 - 1, pageId);
uploadVideoAction(file, view, coordinates?.pos ?? 0 - 1, pageId); uploadVideoAction(file, view, coordinates?.pos ?? 0 - 1, pageId);
uploadAttachmentAction(file, view, coordinates?.pos ?? 0 - 1, pageId); uploadAttachmentAction(file, view, coordinates?.pos ?? 0 - 1, pageId);

View File

@ -38,7 +38,8 @@ import {
LoomIcon, LoomIcon,
MiroIcon, MiroIcon,
TypeformIcon, TypeformIcon,
VimeoIcon, YoutubeIcon VimeoIcon,
YoutubeIcon,
} from "@/components/icons"; } from "@/components/icons";
const CommandGroups: SlashMenuGroupedItemsType = { const CommandGroups: SlashMenuGroupedItemsType = {
@ -221,13 +222,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
if (input.files?.length) { if (input.files?.length) {
const file = input.files[0]; const file = input.files[0];
const pos = editor.view.state.selection.from; const pos = editor.view.state.selection.from;
if (file.type.includes("image/*")) { uploadAttachmentAction(file, editor.view, pos, pageId, true);
uploadImageAction(file, editor.view, pos, pageId);
} else if (file.type.includes("video/*")) {
uploadVideoAction(file, editor.view, pos, pageId);
} else {
uploadAttachmentAction(file, editor.view, pos, pageId);
}
} }
}; };
input.click(); input.click();
@ -368,7 +363,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["airtable"], searchTerms: ["airtable"],
icon: AirtableIcon, icon: AirtableIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'airtable' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "airtable" })
.run();
}, },
}, },
{ {
@ -377,7 +377,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["loom"], searchTerms: ["loom"],
icon: LoomIcon, icon: LoomIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'loom' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "loom" })
.run();
}, },
}, },
{ {
@ -386,7 +391,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["figma"], searchTerms: ["figma"],
icon: FigmaIcon, icon: FigmaIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'figma' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "figma" })
.run();
}, },
}, },
{ {
@ -395,7 +405,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["typeform"], searchTerms: ["typeform"],
icon: TypeformIcon, icon: TypeformIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'typeform' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "typeform" })
.run();
}, },
}, },
{ {
@ -404,7 +419,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["miro"], searchTerms: ["miro"],
icon: MiroIcon, icon: MiroIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'miro' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "miro" })
.run();
}, },
}, },
{ {
@ -413,7 +433,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["youtube", "yt"], searchTerms: ["youtube", "yt"],
icon: YoutubeIcon, icon: YoutubeIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'youtube' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "youtube" })
.run();
}, },
}, },
{ {
@ -422,7 +447,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["vimeo"], searchTerms: ["vimeo"],
icon: VimeoIcon, icon: VimeoIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'vimeo' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "vimeo" })
.run();
}, },
}, },
{ {
@ -431,7 +461,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["framer"], searchTerms: ["framer"],
icon: FramerIcon, icon: FramerIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'framer' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "framer" })
.run();
}, },
}, },
{ {
@ -440,7 +475,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["google drive", "gdrive"], searchTerms: ["google drive", "gdrive"],
icon: GoogleDriveIcon, icon: GoogleDriveIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'gdrive' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "gdrive" })
.run();
}, },
}, },
{ {
@ -449,7 +489,12 @@ const CommandGroups: SlashMenuGroupedItemsType = {
searchTerms: ["google sheets", "gsheets"], searchTerms: ["google sheets", "gsheets"],
icon: GoogleSheetsIcon, icon: GoogleSheetsIcon,
command: ({ editor, range }: CommandProps) => { command: ({ editor, range }: CommandProps) => {
editor.chain().focus().deleteRange(range).setEmbed({ provider: 'gsheets' }).run(); editor
.chain()
.focus()
.deleteRange(range)
.setEmbed({ provider: "gsheets" })
.run();
}, },
}, },
], ],

View File

@ -1,6 +1,10 @@
import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state"; import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state";
import { Decoration, DecorationSet } from "@tiptap/pm/view"; import { Decoration, DecorationSet } from "@tiptap/pm/view";
import { MediaUploadOptions, UploadFn } from "../media-utils"; import {
insertTrailingNode,
MediaUploadOptions,
UploadFn,
} from "../media-utils";
import { IAttachment } from "../types"; import { IAttachment } from "../types";
const uploadKey = new PluginKey("attachment-upload"); const uploadKey = new PluginKey("attachment-upload");
@ -33,7 +37,8 @@ export const AttachmentUploadPlugin = ({
placeholder.appendChild(uploadingText); placeholder.appendChild(uploadingText);
const deco = Decoration.widget(pos + 1, placeholder, { const realPos = pos + 1;
const deco = Decoration.widget(realPos, placeholder, {
id, id,
}); });
set = set.add(tr.doc, [deco]); set = set.add(tr.doc, [deco]);
@ -64,8 +69,8 @@ function findPlaceholder(state: EditorState, id: {}) {
export const handleAttachmentUpload = export const handleAttachmentUpload =
({ validateFn, onUpload }: MediaUploadOptions): UploadFn => ({ validateFn, onUpload }: MediaUploadOptions): UploadFn =>
async (file, view, pos, pageId) => { async (file, view, pos, pageId, allowMedia) => {
const validated = validateFn?.(file); const validated = validateFn?.(file, allowMedia);
// @ts-ignore // @ts-ignore
if (!validated) return; if (!validated) return;
// A fresh object to act as the ID for this upload // A fresh object to act as the ID for this upload
@ -82,6 +87,8 @@ export const handleAttachmentUpload =
fileName: file.name, fileName: file.name,
}, },
}); });
insertTrailingNode(tr, pos, view);
view.dispatch(tr); view.dispatch(tr);
await onUpload(file, pageId).then( await onUpload(file, pageId).then(

View File

@ -1,6 +1,6 @@
import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state"; import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state";
import { Decoration, DecorationSet } from "@tiptap/pm/view"; import { Decoration, DecorationSet } from "@tiptap/pm/view";
import { MediaUploadOptions, UploadFn } from "../media-utils"; import { insertTrailingNode, MediaUploadOptions, UploadFn } from "../media-utils";
import { IAttachment } from "../types"; import { IAttachment } from "../types";
const uploadKey = new PluginKey("image-upload"); const uploadKey = new PluginKey("image-upload");
@ -69,13 +69,13 @@ export const handleImageUpload =
// A fresh object to act as the ID for this upload // A fresh object to act as the ID for this upload
const id = {}; const id = {};
// Replace the selection with a placeholder
const tr = view.state.tr;
if (!tr.selection.empty) tr.deleteSelection();
const reader = new FileReader(); const reader = new FileReader();
reader.readAsDataURL(file); reader.readAsDataURL(file);
reader.onload = () => { reader.onload = () => {
const tr = view.state.tr;
// Replace the selection with a placeholder
if (!tr.selection.empty) tr.deleteSelection();
tr.setMeta(uploadKey, { tr.setMeta(uploadKey, {
add: { add: {
id, id,
@ -83,6 +83,8 @@ export const handleImageUpload =
src: reader.result, src: reader.result,
}, },
}); });
insertTrailingNode(tr, pos, view);
view.dispatch(tr); view.dispatch(tr);
}; };

View File

@ -1,13 +1,29 @@
import type { EditorView } from "@tiptap/pm/view"; import type { EditorView } from "@tiptap/pm/view";
import { Transaction } from "@tiptap/pm/state";
export type UploadFn = ( export type UploadFn = (
file: File, file: File,
view: EditorView, view: EditorView,
pos: number, pos: number,
pageId: string, pageId: string,
// only applicable to file attachments
allowMedia?: boolean,
) => void; ) => void;
export interface MediaUploadOptions { export interface MediaUploadOptions {
validateFn?: (file: File) => void; validateFn?: (file: File, allowMedia?: boolean) => void;
onUpload: (file: File, pageId: string) => Promise<any>; onUpload: (file: File, pageId: string) => Promise<any>;
} }
export function insertTrailingNode(
tr: Transaction,
pos: number,
view: EditorView,
) {
// create trailing node after decoration
// if decoration is at the last node
const currentDocSize = view.state.doc.content.size;
if (pos + 1 === currentDocSize) {
tr.insert(currentDocSize, view.state.schema.nodes.paragraph.create());
}
}

View File

@ -1,6 +1,10 @@
import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state"; import { type EditorState, Plugin, PluginKey } from "@tiptap/pm/state";
import { Decoration, DecorationSet } from "@tiptap/pm/view"; import { Decoration, DecorationSet } from "@tiptap/pm/view";
import { MediaUploadOptions, UploadFn } from "../media-utils"; import {
insertTrailingNode,
MediaUploadOptions,
UploadFn,
} from "../media-utils";
import { IAttachment } from "../types"; import { IAttachment } from "../types";
const uploadKey = new PluginKey("video-upload"); const uploadKey = new PluginKey("video-upload");
@ -70,12 +74,13 @@ export const handleVideoUpload =
const id = {}; const id = {};
// Replace the selection with a placeholder // Replace the selection with a placeholder
const tr = view.state.tr;
if (!tr.selection.empty) tr.deleteSelection();
const reader = new FileReader(); const reader = new FileReader();
reader.readAsDataURL(file); reader.readAsDataURL(file);
reader.onload = () => { reader.onload = () => {
const tr = view.state.tr;
if (!tr.selection.empty) tr.deleteSelection();
tr.setMeta(uploadKey, { tr.setMeta(uploadKey, {
add: { add: {
id, id,
@ -83,6 +88,8 @@ export const handleVideoUpload =
src: reader.result, src: reader.result,
}, },
}); });
insertTrailingNode(tr, pos, view);
view.dispatch(tr); view.dispatch(tr);
}; };