mirror of
https://github.com/AmruthPillai/Reactive-Resume.git
synced 2026-06-22 04:11:55 +10:00
409 lines
12 KiB
TypeScript
409 lines
12 KiB
TypeScript
import type { DragEndEvent, DragStartEvent } from "@dnd-kit/core";
|
|
import {
|
|
closestCenter,
|
|
DndContext,
|
|
DragOverlay,
|
|
KeyboardSensor,
|
|
PointerSensor,
|
|
useSensor,
|
|
useSensors,
|
|
} from "@dnd-kit/core";
|
|
import { rectSortingStrategy, SortableContext, sortableKeyboardCoordinates, useSortable } from "@dnd-kit/sortable";
|
|
import { CSS } from "@dnd-kit/utilities";
|
|
import { t } from "@lingui/core/macro";
|
|
import { Trans } from "@lingui/react/macro";
|
|
import { PencilSimpleIcon, XIcon } from "@phosphor-icons/react";
|
|
import { AnimatePresence, m } from "motion/react";
|
|
import * as React from "react";
|
|
import { createPortal } from "react-dom";
|
|
import { Badge } from "@reactive-resume/ui/components/badge";
|
|
import { Input } from "@reactive-resume/ui/components/input";
|
|
import { Kbd } from "@reactive-resume/ui/components/kbd";
|
|
import { cn } from "@reactive-resume/utils/style";
|
|
import { useControlledState } from "@/hooks/use-controlled-state";
|
|
|
|
const RETURN_KEY = "Enter";
|
|
const COMMA_KEY = ",";
|
|
const EMPTY_CHIPS: string[] = [];
|
|
|
|
type ChipItemProps = {
|
|
id: string;
|
|
chip: string;
|
|
index: number;
|
|
isEditing: boolean;
|
|
onEdit: (index: number) => void;
|
|
onRemove: (index: number) => void;
|
|
};
|
|
|
|
function ChipDragPreview({ chip }: { chip: string }) {
|
|
return (
|
|
<Badge
|
|
variant="outline"
|
|
className="h-6 max-w-44 cursor-grabbing select-none justify-start rounded-md border-ring bg-muted px-2 font-medium text-foreground text-xs shadow-lg ring-2 ring-ring/25 sm:max-w-52"
|
|
>
|
|
<span className="truncate">{chip}</span>
|
|
</Badge>
|
|
);
|
|
}
|
|
|
|
function ChipDragOverlay({ activeChip }: { activeChip: string | null }) {
|
|
const overlay = (
|
|
<DragOverlay dropAnimation={null}>{activeChip ? <ChipDragPreview chip={activeChip} /> : null}</DragOverlay>
|
|
);
|
|
|
|
if (typeof document === "undefined") return overlay;
|
|
|
|
return createPortal(overlay, document.body);
|
|
}
|
|
|
|
function ChipItem({ id, chip, index, isEditing, onEdit, onRemove }: ChipItemProps) {
|
|
const { attributes, listeners, setNodeRef, transform, transition, isDragging } = useSortable({ id });
|
|
|
|
const style = {
|
|
transition,
|
|
zIndex: isDragging ? 10 : undefined,
|
|
transform: CSS.Transform.toString(transform),
|
|
};
|
|
|
|
return (
|
|
<m.div
|
|
layout
|
|
initial={{ opacity: 0, scale: 0.92, y: -4 }}
|
|
animate={{ opacity: isDragging ? 0.62 : 1, scale: 1, y: 0 }}
|
|
exit={{ opacity: 0, scale: 0.92, y: -4 }}
|
|
transition={{ duration: 0.1, ease: "easeOut" }}
|
|
style={style}
|
|
ref={setNodeRef}
|
|
className="group/chip relative touch-none"
|
|
{...attributes}
|
|
{...listeners}
|
|
>
|
|
<Badge
|
|
variant="outline"
|
|
className={cn(
|
|
"h-6 max-w-full cursor-grab select-none justify-start gap-0 rounded-md border-border bg-muted/55 px-2 font-medium text-foreground text-xs transition-colors hover:border-foreground/20 hover:bg-muted active:cursor-grabbing",
|
|
isEditing && "border-primary bg-primary/10 ring-1 ring-primary/40",
|
|
isDragging && "border-ring bg-muted shadow-sm",
|
|
)}
|
|
>
|
|
<span className="max-w-32 truncate sm:max-w-44">{chip}</span>
|
|
<m.div
|
|
initial={false}
|
|
animate={isEditing ? { opacity: 1 } : { opacity: 0.66 }}
|
|
transition={{ duration: 0.12, ease: "easeOut" }}
|
|
className="ms-1.5 flex shrink-0 items-center gap-x-0.5 will-change-[opacity] group-focus-within/chip:opacity-100 group-hover/chip:opacity-100"
|
|
>
|
|
<button
|
|
type="button"
|
|
tabIndex={-1}
|
|
className="rounded-sm p-0.5 text-foreground/70 transition-colors hover:bg-secondary hover:text-foreground focus:outline-none"
|
|
aria-label={t({
|
|
comment:
|
|
"Screen reader label for button that edits a keyword chip. Variable is the current keyword text.",
|
|
message: `Edit ${chip}`,
|
|
})}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
onEdit(index);
|
|
}}
|
|
>
|
|
<PencilSimpleIcon className="size-3.5" />
|
|
</button>
|
|
<button
|
|
type="button"
|
|
tabIndex={-1}
|
|
className="rounded-sm p-0.5 text-foreground/70 transition-colors hover:bg-destructive/10 hover:text-destructive focus:outline-none"
|
|
aria-label={t({
|
|
comment:
|
|
"Screen reader label for button that removes a keyword chip. Variable is the current keyword text.",
|
|
message: `Remove ${chip}`,
|
|
})}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
onRemove(index);
|
|
}}
|
|
>
|
|
<XIcon className="size-3.5" />
|
|
</button>
|
|
</m.div>
|
|
</Badge>
|
|
</m.div>
|
|
);
|
|
}
|
|
|
|
type Props = Omit<React.ComponentProps<"div">, "value" | "onChange"> & {
|
|
value?: string[];
|
|
defaultValue?: string[];
|
|
onChange?: (value: string[]) => void;
|
|
hideDescription?: boolean;
|
|
};
|
|
|
|
export function ChipInput({
|
|
value,
|
|
defaultValue = EMPTY_CHIPS,
|
|
onChange,
|
|
className,
|
|
hideDescription = false,
|
|
...props
|
|
}: Props) {
|
|
const [chips, setChips] = useControlledState<string[]>({
|
|
value,
|
|
defaultValue,
|
|
onChange,
|
|
});
|
|
|
|
const [input, setInput] = React.useState("");
|
|
const [editingIndex, setEditingIndex] = React.useState<number | null>(null);
|
|
const [activeChip, setActiveChip] = React.useState<string | null>(null);
|
|
const inputRef = React.useRef<HTMLInputElement>(null);
|
|
const dndContextId = React.useId();
|
|
const isEditingKeyword = editingIndex !== null;
|
|
const hasChips = chips.length > 0;
|
|
|
|
const addChips = React.useCallback(
|
|
(values: string[]) => {
|
|
const nextValues = values.flatMap((chip) => {
|
|
const trimmed = chip.trim();
|
|
return trimmed ? [trimmed] : [];
|
|
});
|
|
if (nextValues.length === 0) return;
|
|
|
|
const newChips = Array.from(new Set([...chips, ...nextValues]));
|
|
setChips(newChips);
|
|
},
|
|
[chips, setChips],
|
|
);
|
|
|
|
const addChip = React.useCallback(
|
|
(chip: string) => {
|
|
addChips([chip]);
|
|
},
|
|
[addChips],
|
|
);
|
|
|
|
const updateChip = React.useCallback(
|
|
(index: number, newValue: string) => {
|
|
const trimmed = newValue.trim();
|
|
if (!trimmed || index < 0 || index >= chips.length) return;
|
|
|
|
const existingIndex = chips.findIndex((c, i) => c === trimmed && i !== index);
|
|
if (existingIndex !== -1) return;
|
|
|
|
const newChips = [...chips];
|
|
newChips[index] = trimmed;
|
|
setChips(newChips);
|
|
},
|
|
[chips, setChips],
|
|
);
|
|
|
|
const removeChip = React.useCallback(
|
|
(index: number) => {
|
|
if (index < 0 || index >= chips.length) return;
|
|
const newChips = chips.slice(0, index).concat(chips.slice(index + 1));
|
|
setChips(newChips);
|
|
|
|
if (editingIndex === index) {
|
|
setEditingIndex(null);
|
|
setInput("");
|
|
} else if (editingIndex !== null && editingIndex > index) {
|
|
setEditingIndex((current) => (current !== null && current > index ? current - 1 : current));
|
|
}
|
|
},
|
|
[chips, setChips, editingIndex],
|
|
);
|
|
|
|
const handleEdit = React.useCallback(
|
|
(index: number) => {
|
|
setEditingIndex(index);
|
|
setInput(chips[index]);
|
|
inputRef.current?.focus();
|
|
},
|
|
[chips],
|
|
);
|
|
|
|
const handleReorder = React.useCallback(
|
|
(newOrder: string[]) => {
|
|
if (editingIndex !== null) {
|
|
const editingChip = chips[editingIndex];
|
|
const newIndex = newOrder.indexOf(editingChip);
|
|
if (newIndex !== -1 && newIndex !== editingIndex) {
|
|
setEditingIndex(newIndex);
|
|
}
|
|
}
|
|
setChips(newOrder);
|
|
},
|
|
[chips, editingIndex, setChips],
|
|
);
|
|
|
|
const sensors = useSensors(
|
|
useSensor(PointerSensor, {
|
|
activationConstraint: { distance: 3 },
|
|
}),
|
|
useSensor(KeyboardSensor, {
|
|
coordinateGetter: sortableKeyboardCoordinates,
|
|
}),
|
|
);
|
|
|
|
const handleDragStart = React.useCallback((event: DragStartEvent) => {
|
|
setActiveChip(event.active.id as string);
|
|
}, []);
|
|
|
|
const handleDragCancel = React.useCallback(() => {
|
|
setActiveChip(null);
|
|
}, []);
|
|
|
|
const handleDragEnd = React.useCallback(
|
|
(event: DragEndEvent) => {
|
|
setActiveChip(null);
|
|
|
|
const { active, over } = event;
|
|
if (!over || active.id === over.id) return;
|
|
const oldIndex = chips.indexOf(active.id as string);
|
|
const newIndex = chips.indexOf(over.id as string);
|
|
if (oldIndex !== -1 && newIndex !== -1 && oldIndex !== newIndex) {
|
|
const newOrder = Array.from(chips);
|
|
const [removed] = newOrder.splice(oldIndex, 1);
|
|
newOrder.splice(newIndex, 0, removed);
|
|
handleReorder(newOrder);
|
|
}
|
|
},
|
|
[chips, handleReorder],
|
|
);
|
|
|
|
const handleInputChange = React.useCallback(
|
|
(e: React.ChangeEvent<HTMLInputElement>) => {
|
|
const newValue = e.target.value;
|
|
|
|
if (editingIndex !== null) {
|
|
if (newValue.includes(",")) {
|
|
updateChip(editingIndex, newValue.replace(",", ""));
|
|
setEditingIndex(null);
|
|
setInput("");
|
|
} else {
|
|
setInput(newValue);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (newValue.includes(",")) {
|
|
const parts = newValue.split(",");
|
|
addChips(parts.slice(0, -1));
|
|
setInput(parts[parts.length - 1]);
|
|
} else {
|
|
setInput(newValue);
|
|
}
|
|
},
|
|
[addChips, editingIndex, updateChip],
|
|
);
|
|
|
|
const handleKeyDown = React.useCallback(
|
|
(e: React.KeyboardEvent<HTMLInputElement>) => {
|
|
if (e.key === "Enter" || e.key === ",") {
|
|
e.preventDefault();
|
|
|
|
if (editingIndex !== null) {
|
|
if (input.trim()) {
|
|
updateChip(editingIndex, input);
|
|
}
|
|
setEditingIndex(null);
|
|
setInput("");
|
|
} else if (input.trim()) {
|
|
addChip(input);
|
|
setInput("");
|
|
}
|
|
} else if (e.key === "Escape" && editingIndex !== null) {
|
|
setEditingIndex(null);
|
|
setInput("");
|
|
}
|
|
},
|
|
[input, addChip, editingIndex, updateChip],
|
|
);
|
|
|
|
return (
|
|
<div className={cn("space-y-1.5", className)} {...props}>
|
|
<DndContext
|
|
id={dndContextId}
|
|
sensors={sensors}
|
|
collisionDetection={closestCenter}
|
|
onDragStart={handleDragStart}
|
|
onDragEnd={handleDragEnd}
|
|
onDragCancel={handleDragCancel}
|
|
>
|
|
<div
|
|
role="none"
|
|
onClick={() => inputRef.current?.focus()}
|
|
className="overflow-hidden rounded-lg border border-input bg-background/40 transition-colors focus-within:border-ring focus-within:ring-3 focus-within:ring-ring/50 dark:bg-input/20"
|
|
>
|
|
<div className="flex flex-col">
|
|
<div
|
|
className={cn("max-h-24 overflow-y-auto px-2 py-1.5", hasChips ? "border-border/70 border-b" : "hidden")}
|
|
>
|
|
<SortableContext items={chips} strategy={rectSortingStrategy}>
|
|
<m.div layout className="flex flex-wrap gap-1">
|
|
<AnimatePresence initial={false} mode="popLayout">
|
|
{chips.map((chip, idx) => (
|
|
<ChipItem
|
|
key={chip}
|
|
id={chip}
|
|
chip={chip}
|
|
index={idx}
|
|
isEditing={editingIndex === idx}
|
|
onEdit={handleEdit}
|
|
onRemove={removeChip}
|
|
/>
|
|
))}
|
|
</AnimatePresence>
|
|
</m.div>
|
|
</SortableContext>
|
|
</div>
|
|
<div className={cn("flex items-center gap-1.5 px-2", hasChips ? "py-1.5" : "py-0")}>
|
|
<Input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={input}
|
|
autoComplete="off"
|
|
aria-label={isEditingKeyword ? t`Edit keyword` : t`Add keyword`}
|
|
placeholder={isEditingKeyword ? t`Editing keyword...` : t`Add a keyword...`}
|
|
onKeyDown={handleKeyDown}
|
|
onChange={handleInputChange}
|
|
className="h-9 flex-1 border-none p-0 focus-visible:border-none focus-visible:ring-0 dark:bg-transparent"
|
|
/>
|
|
<AnimatePresence>
|
|
{chips.length > 0 && (
|
|
<m.span
|
|
layout
|
|
initial={{ opacity: 0, scale: 0.95 }}
|
|
animate={{
|
|
opacity: isEditingKeyword ? 1 : 0.8,
|
|
scale: 1,
|
|
}}
|
|
exit={{ opacity: 0, scale: 0.95 }}
|
|
transition={{ duration: 0.12, ease: "easeOut" }}
|
|
className={cn(
|
|
"flex h-6 min-w-6 shrink-0 items-center justify-center rounded-md border px-1.5 font-medium text-[0.7rem] tabular-nums",
|
|
isEditingKeyword
|
|
? "border-primary/30 bg-primary/10 text-primary"
|
|
: "border-border bg-muted/50 text-foreground/80",
|
|
)}
|
|
>
|
|
{isEditingKeyword ? <Trans>Edit</Trans> : chips.length}
|
|
</m.span>
|
|
)}
|
|
</AnimatePresence>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<ChipDragOverlay activeChip={activeChip} />
|
|
</DndContext>
|
|
|
|
{!hideDescription && (
|
|
<p className="text-muted-foreground text-xs">
|
|
<Trans>
|
|
Press <Kbd>{RETURN_KEY}</Kbd> or <Kbd>{COMMA_KEY}</Kbd> to add or save the current keyword.
|
|
</Trans>
|
|
</p>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|