2023-08-15 09:34:46 +00:00
|
|
|
import { Editor } from "@tiptap/core";
|
|
|
|
import { Check, Trash } from "lucide-react";
|
2023-08-17 16:26:32 +00:00
|
|
|
import { Dispatch, FC, SetStateAction, useCallback, useEffect, useRef } from "react";
|
2023-08-15 09:34:46 +00:00
|
|
|
import { cn } from "../utils";
|
2023-08-17 14:22:58 +00:00
|
|
|
import isValidHttpUrl from "./utils/link-validator";
|
2023-08-15 09:34:46 +00:00
|
|
|
interface LinkSelectorProps {
|
|
|
|
editor: Editor;
|
|
|
|
isOpen: boolean;
|
|
|
|
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
|
|
|
}
|
|
|
|
|
2023-08-17 16:26:32 +00:00
|
|
|
|
2023-08-15 09:34:46 +00:00
|
|
|
export const LinkSelector: FC<LinkSelectorProps> = ({ editor, isOpen, setIsOpen }) => {
|
|
|
|
const inputRef = useRef<HTMLInputElement>(null);
|
|
|
|
|
2023-08-17 16:26:32 +00:00
|
|
|
const onLinkSubmit = useCallback(() => {
|
|
|
|
const input = inputRef.current;
|
|
|
|
const url = input?.value;
|
|
|
|
if (url && isValidHttpUrl(url)) {
|
|
|
|
editor.chain().focus().setLink({ href: url }).run();
|
|
|
|
setIsOpen(false);
|
|
|
|
}
|
|
|
|
}, [editor, inputRef, setIsOpen]);
|
|
|
|
|
2023-08-15 09:34:46 +00:00
|
|
|
useEffect(() => {
|
|
|
|
inputRef.current && inputRef.current?.focus();
|
|
|
|
});
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="relative">
|
|
|
|
<button
|
2023-08-16 12:55:11 +00:00
|
|
|
type="button"
|
|
|
|
className={cn(
|
|
|
|
"flex h-full items-center space-x-2 px-3 py-1.5 text-sm font-medium text-custom-text-300 hover:bg-custom-background-100 active:bg-custom-background-100",
|
|
|
|
{ "bg-custom-background-100": isOpen }
|
|
|
|
)}
|
2023-08-15 09:34:46 +00:00
|
|
|
onClick={() => {
|
|
|
|
setIsOpen(!isOpen);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<p className="text-base">↗</p>
|
|
|
|
<p
|
|
|
|
className={cn("underline underline-offset-4", {
|
|
|
|
"text-custom-text-100": editor.isActive("link"),
|
|
|
|
})}
|
|
|
|
>
|
|
|
|
Link
|
|
|
|
</p>
|
|
|
|
</button>
|
|
|
|
{isOpen && (
|
2023-08-17 14:22:58 +00:00
|
|
|
<div
|
2023-08-15 09:34:46 +00:00
|
|
|
className="fixed top-full z-[99999] mt-1 flex w-60 overflow-hidden rounded border border-custom-border-300 bg-custom-background-100 dow-xl animate-in fade-in slide-in-from-top-1"
|
2023-08-17 16:26:32 +00:00
|
|
|
onKeyDown={(e) => {
|
|
|
|
if (e.key === "Enter") {
|
|
|
|
e.preventDefault(); onLinkSubmit();
|
|
|
|
}
|
|
|
|
}}
|
2023-08-15 09:34:46 +00:00
|
|
|
>
|
|
|
|
<input
|
|
|
|
ref={inputRef}
|
|
|
|
type="url"
|
|
|
|
placeholder="Paste a link"
|
2023-08-16 12:55:11 +00:00
|
|
|
className="flex-1 bg-custom-background-100 border-r border-custom-border-300 p-1 text-sm outline-none placeholder:text-custom-text-400"
|
2023-08-15 09:34:46 +00:00
|
|
|
defaultValue={editor.getAttributes("link").href || ""}
|
|
|
|
/>
|
|
|
|
{editor.getAttributes("link").href ? (
|
|
|
|
<button
|
2023-08-17 14:22:58 +00:00
|
|
|
type="button"
|
2023-08-15 09:34:46 +00:00
|
|
|
className="flex items-center rounded-sm p-1 text-red-600 transition-all hover:bg-red-100 dark:hover:bg-red-800"
|
|
|
|
onClick={() => {
|
|
|
|
editor.chain().focus().unsetLink().run();
|
|
|
|
setIsOpen(false);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Trash className="h-4 w-4" />
|
|
|
|
</button>
|
|
|
|
) : (
|
2023-08-17 14:22:58 +00:00
|
|
|
<button className="flex items-center rounded-sm p-1 text-custom-text-300 transition-all hover:bg-custom-background-90" type="button"
|
|
|
|
onClick={() => {
|
2023-08-17 16:26:32 +00:00
|
|
|
onLinkSubmit();
|
2023-08-17 14:22:58 +00:00
|
|
|
}}
|
|
|
|
>
|
2023-08-15 09:34:46 +00:00
|
|
|
<Check className="h-4 w-4" />
|
|
|
|
</button>
|
|
|
|
)}
|
2023-08-17 14:22:58 +00:00
|
|
|
</div>
|
2023-08-15 09:34:46 +00:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|