mirror of
https://github.com/docmost/docmost
synced 2025-03-28 21:13:28 +00:00
* 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:
parent
d021d0a38f
commit
9da49fd6a5
@ -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()) {
|
||||||
|
@ -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);
|
||||||
|
@ -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();
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -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(
|
||||||
|
@ -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);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -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);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user