import { Brain, CheckSquare, Code, Heading1, Heading2, Heading3, ImageIcon, List, ListOrdered, MessageSquarePlus, Text, TextQuote, Twitter, Youtube, } from "lucide-react"; import { Command, createSuggestionItems, renderItems } from "novel"; import { uploadFn } from "./image-upload"; export const suggestionItems = createSuggestionItems([ { title: "Send Feedback", description: "Let us know how we can improve.", icon: , command: ({ editor, range }) => { editor.chain().focus().deleteRange(range).run(); window.open("/feedback", "_blank"); }, }, { title: "Text", description: "Just start typing with plain text.", searchTerms: ["p", "paragraph"], icon: , command: ({ editor, range }) => { editor .chain() .focus() .deleteRange(range) .toggleNode("paragraph", "paragraph") .run(); }, }, { title: "To-do List", description: "Track tasks with a to-do list.", searchTerms: ["todo", "task", "list", "check", "checkbox"], icon: , command: ({ editor, range }) => { editor.chain().focus().deleteRange(range).toggleTaskList().run(); }, }, { title: "Heading 1", description: "Big section heading.", searchTerms: ["title", "big", "large"], icon: , command: ({ editor, range }) => { editor .chain() .focus() .deleteRange(range) .setNode("heading", { level: 1 }) .run(); }, }, { title: "Heading 2", description: "Medium section heading.", searchTerms: ["subtitle", "medium"], icon: , command: ({ editor, range }) => { editor .chain() .focus() .deleteRange(range) .setNode("heading", { level: 2 }) .run(); }, }, { title: "Heading 3", description: "Small section heading.", searchTerms: ["subtitle", "small"], icon: , command: ({ editor, range }) => { editor .chain() .focus() .deleteRange(range) .setNode("heading", { level: 3 }) .run(); }, }, { title: "Bullet List", description: "Create a simple bullet list.", searchTerms: ["unordered", "point"], icon: , command: ({ editor, range }) => { editor.chain().focus().deleteRange(range).toggleBulletList().run(); }, }, { title: "Numbered List", description: "Create a list with numbering.", searchTerms: ["ordered"], icon: , command: ({ editor, range }) => { editor.chain().focus().deleteRange(range).toggleOrderedList().run(); }, }, { title: "Quote", description: "Capture a quote.", searchTerms: ["blockquote"], icon: , command: ({ editor, range }) => editor .chain() .focus() .deleteRange(range) .toggleNode("paragraph", "paragraph") .toggleBlockquote() .run(), }, { title: "Code", description: "Capture a code snippet.", searchTerms: ["codeblock"], icon: , command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleCodeBlock().run(), }, { title: "Image", description: "Upload an image from your computer.", searchTerms: ["photo", "picture", "media"], icon: , command: ({ editor, range }) => { editor.chain().focus().deleteRange(range).run(); // upload image const input = document.createElement("input"); input.type = "file"; input.accept = "image/*"; input.onchange = async () => { if (input.files?.length) { const file = input.files[0]; if (!file) return; const pos = editor.view.state.selection.from; uploadFn(file, editor.view, pos); } }; input.click(); }, }, { title: "Youtube", description: "Embed a Youtube video.", searchTerms: ["video", "youtube", "embed"], icon: , command: ({ editor, range }) => { const videoLink = prompt("Please enter Youtube Video Link"); //From https://regexr.com/3dj5t const ytRegex = new RegExp( /^((?:https?:)?\/\/)?((?:www|m)\.)?((?:youtube\.com|youtu.be))(\/(?:[\w\-]+\?v=|embed\/|v\/)?)([\w\-]+)(\S+)?$/, ); if (videoLink && ytRegex.test(videoLink)) { editor .chain() .focus() .deleteRange(range) .setYoutubeVideo({ src: videoLink, }) .run(); } else { if (videoLink !== null) { alert("Please enter a correct Youtube Video Link"); } } }, }, { title: "Twitter", description: "Embed a Tweet.", searchTerms: ["twitter", "embed"], icon: , command: ({ editor, range }) => { const tweetLink = prompt("Please enter Twitter Link"); const tweetRegex = new RegExp( /^https?:\/\/(www\.)?x\.com\/([a-zA-Z0-9_]{1,15})(\/status\/(\d+))?(\/\S*)?$/, ); if (tweetLink && tweetRegex.test(tweetLink)) { editor .chain() .focus() .deleteRange(range) .setTweet({ src: tweetLink, }) .run(); } else { if (tweetLink !== null) { alert("Please enter a correct Twitter Link"); } } }, }, ]); export const slashCommand = Command.configure({ suggestion: { items: () => suggestionItems, render: renderItems, }, });