mirror of
https://github.com/Mail-0/Zero.git
synced 2026-03-03 02:27:00 +00:00
102 lines
3.0 KiB
TypeScript
102 lines
3.0 KiB
TypeScript
import { PopoverContent, Popover, PopoverTrigger } from '@/components/ui/popover';
|
|
import { Button } from '@/components/ui/button';
|
|
import { Check, Trash } from 'lucide-react';
|
|
import { useEffect, useRef } from 'react';
|
|
import { useEditor } from 'novel';
|
|
import { cn } from '@/lib/utils';
|
|
|
|
export function isValidUrl(url: string) {
|
|
try {
|
|
new URL(url);
|
|
return true;
|
|
} catch (_e) {
|
|
return false;
|
|
}
|
|
}
|
|
export function getUrlFromString(str: string) {
|
|
if (isValidUrl(str)) return str;
|
|
try {
|
|
if (str.includes('.') && !str.includes(' ')) {
|
|
return new URL(`https://${str}`).toString();
|
|
}
|
|
} catch (_e) {
|
|
return null;
|
|
}
|
|
}
|
|
interface LinkSelectorProps {
|
|
open: boolean;
|
|
onOpenChange: (open: boolean) => void;
|
|
}
|
|
|
|
export const LinkSelector = ({ open, onOpenChange }: LinkSelectorProps) => {
|
|
const inputRef = useRef<HTMLInputElement>(null);
|
|
const { editor } = useEditor();
|
|
|
|
// Autofocus on input by default
|
|
useEffect(() => {
|
|
inputRef.current?.focus();
|
|
});
|
|
if (!editor) return null;
|
|
|
|
return (
|
|
<Popover modal={true} open={open} onOpenChange={onOpenChange}>
|
|
<PopoverTrigger asChild>
|
|
<Button size="sm" variant="ghost" className="gap-2 rounded-none border-none">
|
|
<p className="text-base">↗</p>
|
|
<p
|
|
className={cn('underline decoration-stone-400 underline-offset-4', {
|
|
'text-blue-500': editor.isActive('link'),
|
|
})}
|
|
>
|
|
Link
|
|
</p>
|
|
</Button>
|
|
</PopoverTrigger>
|
|
<PopoverContent align="start" className="w-60 p-0" sideOffset={10}>
|
|
<form
|
|
onSubmit={(e) => {
|
|
const target = e.currentTarget as HTMLFormElement;
|
|
e.preventDefault();
|
|
const input = target[0] as HTMLInputElement;
|
|
const url = getUrlFromString(input.value);
|
|
if (url) {
|
|
editor.chain().focus().setLink({ href: url }).run();
|
|
onOpenChange(false);
|
|
}
|
|
}}
|
|
className="flex p-1"
|
|
>
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
placeholder="Paste a link"
|
|
className="bg-background flex-1 p-1 text-sm outline-none"
|
|
defaultValue={editor.getAttributes('link').href || ''}
|
|
/>
|
|
{editor.getAttributes('link').href ? (
|
|
<Button
|
|
size="icon"
|
|
variant="outline"
|
|
type="button"
|
|
className="flex h-8 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();
|
|
if (inputRef.current) {
|
|
inputRef.current.value = '';
|
|
}
|
|
onOpenChange(false);
|
|
}}
|
|
>
|
|
<Trash className="h-4 w-4" />
|
|
</Button>
|
|
) : (
|
|
<Button size="icon" className="h-8">
|
|
<Check className="h-4 w-4" />
|
|
</Button>
|
|
)}
|
|
</form>
|
|
</PopoverContent>
|
|
</Popover>
|
|
);
|
|
};
|