supermemory/apps/web/components/space-selector.tsx
2026-03-05 19:26:05 -08:00

658 lines
19 KiB
TypeScript

"use client"
import { useState, useMemo } from "react"
import { cn } from "@lib/utils"
import { dmSans125ClassName, dmSansClassName } from "@/lib/fonts"
import { DEFAULT_PROJECT_ID } from "@lib/constants"
import {
ChevronsLeftRight,
Plus,
Trash2,
XIcon,
Loader2,
Globe,
Layers,
} from "lucide-react"
import type { ContainerTagListType } from "@lib/types"
import { AddSpaceModal } from "./add-space-modal"
import { SelectSpacesModal } from "./select-spaces-modal"
import { useProjectMutations } from "@/hooks/use-project-mutations"
import { useContainerTags } from "@/hooks/use-container-tags"
import { motion } from "motion/react"
import * as DialogPrimitive from "@radix-ui/react-dialog"
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuSeparator,
DropdownMenuTrigger,
} from "@ui/components/dropdown-menu"
import {
Dialog,
DialogContent,
DialogTitle,
DialogDescription,
} from "@repo/ui/components/dialog"
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@repo/ui/components/select"
import { Button } from "@repo/ui/components/button"
import { analytics } from "@/lib/analytics"
export interface SpaceSelectorProps {
selectedProjects: string[]
onValueChange: (containerTags: string[]) => void
variant?: "default" | "insideOut"
showChevron?: boolean
triggerClassName?: string
contentClassName?: string
showNewSpace?: boolean
enableDelete?: boolean
compact?: boolean
singleSelect?: boolean
}
const triggerVariants = {
default: "px-3 py-2 rounded-md hover:bg-white/5",
insideOut: "px-3 py-2 rounded-full bg-[#0D121A] shadow-inside-out",
}
export function SpaceSelector({
selectedProjects,
onValueChange,
variant = "default",
showChevron = false,
triggerClassName,
contentClassName,
showNewSpace = true,
enableDelete = false,
compact = false,
singleSelect = false,
}: SpaceSelectorProps) {
const [isOpen, setIsOpen] = useState(false)
const [showCreateDialog, setShowCreateDialog] = useState(false)
const [showSelectSpacesModal, setShowSelectSpacesModal] = useState(false)
const [deleteDialog, setDeleteDialog] = useState<{
open: boolean
project: { id: string; name: string; containerTag: string } | null
action: "move" | "delete"
targetProjectId: string
}>({
open: false,
project: null,
action: "move",
targetProjectId: "",
})
const { deleteProjectMutation } = useProjectMutations()
const { allProjects, novaProjects, isLoading } = useContainerTags()
const isNovaSpaces = selectedProjects.length === 0
const displayInfo = useMemo(() => {
if (isNovaSpaces) {
return { name: "Nova Spaces", emoji: null, isMultiple: false }
}
if (selectedProjects.length === 1) {
const containerTag = selectedProjects[0]
if (containerTag === DEFAULT_PROJECT_ID) {
return { name: "My Space", emoji: "📁", isMultiple: false }
}
const found = allProjects.find(
(p: ContainerTagListType) => p.containerTag === containerTag,
)
return {
name: found?.name || containerTag,
emoji: found?.emoji || "📁",
isMultiple: false,
}
}
return {
name: `${selectedProjects.length} spaces`,
emoji: null,
isMultiple: true,
}
}, [allProjects, selectedProjects, isNovaSpaces])
const handleSelectNovaSpaces = () => {
analytics.spaceSwitched({ space_id: "nova_spaces" })
onValueChange([]) // Empty array = "Nova Spaces" (all nova)
setIsOpen(false)
}
const handleSelectSingleSpace = (containerTag: string) => {
analytics.spaceSwitched({ space_id: containerTag })
onValueChange([containerTag])
setIsOpen(false)
}
const handleOpenSelectSpaces = () => {
setIsOpen(false)
setShowSelectSpacesModal(true)
}
const handleSelectSpacesApply = (selected: string[]) => {
if (selected.length > 0) {
analytics.spaceSwitched({
space_id:
selected.length === 1 ? (selected[0] ?? "unknown") : "multiple",
})
}
onValueChange(selected)
setShowSelectSpacesModal(false)
}
const handleNewSpace = () => {
setIsOpen(false)
setShowCreateDialog(true)
}
const handleDeleteClick = (
e: React.MouseEvent,
project: { id: string; name: string; containerTag: string },
) => {
e.stopPropagation()
e.preventDefault()
setDeleteDialog({
open: true,
project,
action: "move",
targetProjectId: "",
})
}
const handleDeleteConfirm = () => {
if (!deleteDialog.project) return
deleteProjectMutation.mutate(
{
projectId: deleteDialog.project.id,
action: deleteDialog.action,
targetProjectId:
deleteDialog.action === "move"
? deleteDialog.targetProjectId
: undefined,
},
{
onSuccess: () => {
setDeleteDialog({
open: false,
project: null,
action: "move",
targetProjectId: "",
})
setIsOpen(false)
},
},
)
}
const handleDeleteCancel = () => {
setDeleteDialog({
open: false,
project: null,
action: "move",
targetProjectId: "",
})
}
const availableTargetProjects = useMemo(() => {
const filtered = novaProjects.filter(
(p: ContainerTagListType) =>
p.id !== deleteDialog.project?.id &&
p.containerTag !== deleteDialog.project?.containerTag,
)
const defaultProject = novaProjects.find(
(p: ContainerTagListType) => p.containerTag === DEFAULT_PROJECT_ID,
)
const isDefaultProjectBeingDeleted =
deleteDialog.project?.containerTag === DEFAULT_PROJECT_ID
if (defaultProject && !isDefaultProjectBeingDeleted) {
const defaultProjectIncluded = filtered.some(
(p: ContainerTagListType) => p.containerTag === DEFAULT_PROJECT_ID,
)
if (!defaultProjectIncluded) {
return [defaultProject, ...filtered]
}
}
return filtered
}, [novaProjects, deleteDialog.project])
return (
<>
<DropdownMenu open={isOpen} onOpenChange={setIsOpen}>
<DropdownMenuTrigger asChild>
<button
type="button"
className={cn(
"flex items-center gap-2 cursor-pointer transition-colors focus:outline-none focus-visible:outline-none",
triggerVariants[variant],
dmSansClassName(),
triggerClassName,
)}
>
{isNovaSpaces ? (
<Globe className="size-4 text-white" />
) : displayInfo.isMultiple ? (
<Layers className="size-4 text-white" />
) : (
<span className="text-sm font-bold tracking-[-0.98px]">
{displayInfo.emoji}
</span>
)}
{!compact && (
<span className="text-sm font-medium text-white">
{isLoading ? "..." : displayInfo.name}
</span>
)}
{showChevron && (
<ChevronsLeftRight className="size-4 rotate-90 text-white/70" />
)}
</button>
</DropdownMenuTrigger>
<DropdownMenuContent
align="start"
className={cn(
"min-w-[200px] p-1.5 rounded-xl border border-[#2E3033] shadow-[0px_1.5px_20px_0px_rgba(0,0,0,0.65)]",
dmSansClassName(),
contentClassName,
)}
style={{
background: "linear-gradient(180deg, #0A0E14 0%, #05070A 100%)",
}}
>
<div className="flex flex-col gap-2">
<div className="flex flex-col">
{!singleSelect && (
<>
<DropdownMenuItem
onClick={handleSelectNovaSpaces}
className={cn(
"flex items-center gap-2 px-3 py-2.5 rounded-md cursor-pointer text-white text-sm font-medium",
isNovaSpaces
? "bg-[#293952]/40"
: "opacity-60 hover:opacity-100 hover:bg-[#293952]/40",
)}
>
<Globe className="size-4" />
<span className="flex-1">Nova Spaces</span>
</DropdownMenuItem>
<DropdownMenuSeparator className="bg-[#2E3033] my-1" />
</>
)}
<div className="px-3 py-1">
<span className="text-[10px] uppercase tracking-wider text-[#737373] font-medium">
My Spaces
</span>
</div>
<DropdownMenuItem
onClick={() => handleSelectSingleSpace(DEFAULT_PROJECT_ID)}
className={cn(
"flex items-center gap-2 px-3 py-2.5 rounded-md cursor-pointer text-white text-sm font-medium",
selectedProjects.length === 1 &&
selectedProjects[0] === DEFAULT_PROJECT_ID
? "bg-[#293952]/40"
: "opacity-60 hover:opacity-100 hover:bg-[#293952]/40",
)}
>
<span className="font-bold tracking-[-0.98px]">📁</span>
<span className="flex-1">My Space</span>
</DropdownMenuItem>
{novaProjects
.filter(
(p: ContainerTagListType) =>
p.containerTag !== DEFAULT_PROJECT_ID,
)
.map((project: ContainerTagListType) => (
<DropdownMenuItem
key={project.id}
onClick={() =>
handleSelectSingleSpace(project.containerTag)
}
className={cn(
"flex items-center gap-2 px-3 py-2.5 rounded-md cursor-pointer text-white text-sm font-medium group",
selectedProjects.length === 1 &&
selectedProjects[0] === project.containerTag
? "bg-[#293952]/40"
: "opacity-60 hover:opacity-100 hover:bg-[#293952]/40",
)}
>
<span className="font-bold tracking-[-0.98px]">
{project.emoji || "📁"}
</span>
<span className="truncate flex-1">
{project.name ?? project.containerTag}
</span>
{enableDelete && (
<button
type="button"
onClick={(e) =>
handleDeleteClick(e, {
id: project.id,
name: project.name,
containerTag: project.containerTag,
})
}
className="opacity-0 group-hover:opacity-100 transition-opacity p-1 rounded-full hover:bg-red-500/20"
>
<Trash2 className="size-3.5 text-red-500" />
</button>
)}
</DropdownMenuItem>
))}
</div>
<DropdownMenuSeparator className="bg-[#2E3033]" />
<button
type="button"
onClick={handleOpenSelectSpaces}
className="flex items-center justify-center gap-2 px-3 py-2 rounded-md cursor-pointer text-white text-sm font-medium border border-[#161F2C] hover:bg-[#0D121A]/80 transition-colors"
style={{
background: "linear-gradient(180deg, #0D121A 0%, #000000 100%)",
}}
>
<Layers className="size-4" />
<span>Select Space{!singleSelect && "s"}</span>
</button>
{showNewSpace && (
<button
type="button"
onClick={handleNewSpace}
className="flex items-center justify-center gap-2 px-3 py-2 rounded-md cursor-pointer text-white text-sm font-medium border border-[#161F2C] hover:bg-[#0D121A]/80 transition-colors"
style={{
background:
"linear-gradient(180deg, #0D121A 0%, #000000 100%)",
}}
>
<Plus className="size-4" />
<span>New Space</span>
</button>
)}
</div>
</DropdownMenuContent>
</DropdownMenu>
<AddSpaceModal
isOpen={showCreateDialog}
onClose={() => setShowCreateDialog(false)}
onCreated={(containerTag) => onValueChange([containerTag])}
/>
<SelectSpacesModal
isOpen={showSelectSpacesModal}
onClose={() => setShowSelectSpacesModal(false)}
selectedProjects={selectedProjects}
onApply={handleSelectSpacesApply}
projects={allProjects}
singleSelect={singleSelect}
/>
<Dialog
open={deleteDialog.open}
onOpenChange={(open: boolean) => {
if (!open) {
setDeleteDialog({
open: false,
project: null,
action: "move",
targetProjectId: "",
})
}
}}
>
<DialogContent
className={cn(
"w-[90%]! max-w-[500px]! border-none bg-[#1B1F24] flex flex-col p-4 gap-4 rounded-[22px]",
dmSansClassName(),
)}
style={{
boxShadow:
"0 2.842px 14.211px 0 rgba(0, 0, 0, 0.25), 0.711px 0.711px 0.711px 0 rgba(255, 255, 255, 0.10) inset",
}}
showCloseButton={false}
>
<div className="flex flex-col gap-4">
<div
id="delete-dialog-header"
className="flex justify-between items-start gap-4"
>
<div className="pl-1 space-y-1 flex-1">
<DialogTitle
className={cn(
"font-semibold text-[#fafafa]",
dmSans125ClassName(),
)}
>
Delete space
</DialogTitle>
<DialogDescription className="text-[#737373] font-medium text-[16px] leading-[1.35]">
What would you like to do with the documents and memories in{" "}
<span className="text-[#fafafa] font-medium">
"{deleteDialog.project?.name}"
</span>
?
</DialogDescription>
</div>
<DialogPrimitive.Close
className="bg-[#0D121A] w-7 h-7 flex items-center justify-center focus:ring-ring rounded-full transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 border border-[rgba(115,115,115,0.2)] shrink-0"
style={{
boxShadow:
"inset 1.313px 1.313px 3.938px 0px rgba(0,0,0,0.7)",
}}
>
<XIcon stroke="#737373" />
<span className="sr-only">Close</span>
</DialogPrimitive.Close>
</div>
<div id="delete-dialog-content" className="space-y-3">
<button
id="move-option"
type="button"
onClick={() =>
setDeleteDialog((prev) => ({ ...prev, action: "move" }))
}
className={cn(
"flex items-center gap-3 p-3 rounded-[12px] cursor-pointer transition-colors w-full text-left",
deleteDialog.action === "move"
? "bg-[#14161A] border border-[rgba(82,89,102,0.3)]"
: "bg-[#14161A]/50 border border-transparent hover:border-[rgba(82,89,102,0.2)]",
)}
style={{
boxShadow:
deleteDialog.action === "move"
? "0px 1px 2px 0px rgba(0,43,87,0.1), inset 0px 0px 0px 1px rgba(43,49,67,0.08), inset 0px 1px 1px 0px rgba(0,0,0,0.08)"
: "none",
}}
>
<div
className={cn(
"w-4 h-4 rounded-full border-2 flex items-center justify-center shrink-0",
deleteDialog.action === "move"
? "border-blue-500"
: "border-[#737373]",
)}
>
{deleteDialog.action === "move" && (
<div className="w-2 h-2 rounded-full bg-blue-500" />
)}
</div>
<span className="text-[#fafafa] text-sm font-medium">
Move to another space
</span>
</button>
{deleteDialog.action === "move" && (
<motion.div
initial={{ opacity: 0, height: 0 }}
animate={{ opacity: 1, height: "auto" }}
exit={{ opacity: 0, height: 0 }}
className="ml-7"
>
<Select
value={deleteDialog.targetProjectId}
onValueChange={(val: string) =>
setDeleteDialog((prev) => ({
...prev,
targetProjectId: val,
}))
}
>
<SelectTrigger
className={cn(
"bg-[#14161A] border border-[rgba(82,89,102,0.2)] rounded-[12px] text-[#fafafa] text-[14px] h-[45px]",
dmSansClassName(),
)}
style={{
boxShadow:
"0px 1px 2px 0px rgba(0,43,87,0.1), inset 0px 0px 0px 1px rgba(43,49,67,0.08), inset 0px 1px 1px 0px rgba(0,0,0,0.08), inset 0px 2px 4px 0px rgba(0,0,0,0.02)",
}}
>
<SelectValue placeholder="Select target space" />
</SelectTrigger>
<SelectContent
className={cn(
"bg-[#14161A] border border-[rgba(82,89,102,0.2)] rounded-[12px]",
dmSansClassName(),
)}
style={{
boxShadow:
"0px 1px 2px 0px rgba(0,43,87,0.1), inset 0px 0px 0px 1px rgba(43,49,67,0.08)",
}}
>
{availableTargetProjects.map(
(p: ContainerTagListType) => (
<SelectItem
key={p.id}
value={p.id}
className="text-[#fafafa] hover:bg-[#1B1F24] cursor-pointer rounded-md"
>
<span className="flex items-center gap-2">
<span>{p.emoji || "📁"}</span>
<span>
{p.containerTag === DEFAULT_PROJECT_ID
? "My Space"
: p.name}
</span>
</span>
</SelectItem>
),
)}
</SelectContent>
</Select>
</motion.div>
)}
<button
id="delete-option"
type="button"
onClick={() =>
setDeleteDialog((prev) => ({ ...prev, action: "delete" }))
}
className={cn(
"flex items-center gap-3 p-3 rounded-[12px] cursor-pointer transition-colors w-full text-left",
deleteDialog.action === "delete"
? "bg-[#14161A] border border-[rgba(220,38,38,0.3)]"
: "bg-[#14161A]/50 border border-transparent hover:border-[rgba(82,89,102,0.2)]",
)}
style={{
boxShadow:
deleteDialog.action === "delete"
? "0px 1px 2px 0px rgba(87,0,0,0.1), inset 0px 0px 0px 1px rgba(67,43,43,0.08), inset 0px 1px 1px 0px rgba(0,0,0,0.08)"
: "none",
}}
>
<div
className={cn(
"w-4 h-4 rounded-full border-2 flex items-center justify-center shrink-0",
deleteDialog.action === "delete"
? "border-red-500"
: "border-[#737373]",
)}
>
{deleteDialog.action === "delete" && (
<div className="w-2 h-2 rounded-full bg-red-500" />
)}
</div>
<span className="text-[#fafafa] text-sm font-medium">
Delete everything permanently
</span>
</button>
{deleteDialog.action === "delete" && (
<motion.p
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
className="text-xs text-red-400 ml-7"
>
All documents and memories will be permanently deleted.
</motion.p>
)}
</div>
<div
id="delete-dialog-footer"
className="flex items-center justify-end gap-[22px]"
>
<button
type="button"
onClick={handleDeleteCancel}
disabled={deleteProjectMutation.isPending}
className={cn(
"text-[#737373] font-medium text-[14px] cursor-pointer transition-colors hover:text-[#999]",
dmSansClassName(),
)}
>
Cancel
</button>
<Button
variant="insideOut"
onClick={handleDeleteConfirm}
disabled={
deleteProjectMutation.isPending ||
(deleteDialog.action === "move" &&
!deleteDialog.targetProjectId)
}
className={cn(
"px-4 py-[10px] rounded-full",
deleteDialog.action === "delete" &&
"bg-red-600 hover:bg-red-700 border-red-700",
)}
>
{deleteProjectMutation.isPending ? (
<>
<Loader2 className="size-4 animate-spin mr-2" />
{deleteDialog.action === "move"
? "Moving..."
: "Deleting..."}
</>
) : deleteDialog.action === "move" ? (
"Move & Delete"
) : (
"Delete Everything"
)}
</Button>
</div>
</div>
</DialogContent>
</Dialog>
</>
)
}