mirror of
https://github.com/shadcn-ui/ui.git
synced 2026-06-30 08:04:18 +00:00
* feat(v4): update home page * fix * fix: cards * feat(v4): charts page * feat: update pages * feat: colors * fix * feat: add docs * feat: mdx work * fix * fix * fix: sidebar * fix: lint * feat: updates * feat: update components * feat: fix docs * fix: responsive * feat: implement cmdk * fix: update navigation menu demo * fix: code style * fix: themes * feat: implement blocks page * fix: docs config * refactor * fix: outputFileTracingIncludes * fix * fix: output * fix * fix: registry * refactor: move docs * debug: docs * debug * revert * fix: mjs * deps: pin fumadocs * debug * fix: downgrade next * fix: index page * refactor: move mdx components * fix: remove copy button * fix * was it zod * yes it was * remove copy page * fix: color page * fix: colors page * fix: meta colors * fix: copy button * feat: sync registry * fix: registry build script * feat: update port * feat: clean up examples * fix * fix: mobile nav * fix: blur for mobile * fix: sidebar nav * feat: update examples * fix: build scripts * feat: update components * feat: restyle * fix: types * fix: styles * fix: margins * fix: screenshots * fix * feat: update theme * fix: charts nav * fix: image * feat: optimize images * fix: menu * fix: card * fix: border * check * feat: implement charts page * fix: charts * fix: og images * feat: extend touch * fix: static * fix: sizing * fix: mobile screenshots * fix: page nav * fix * feat: update favicon * fix: theme selector * fix: feedback * fix: sink * docs: update * fix: styles * chore: update registry * fix: command * fix * fix: minor updates * fix: typography on smaller devices * fix: format * fix: remove unused icon * feat: update favicon * fix: typography * docs: typography page * fix: steps
267 lines
8.0 KiB
TypeScript
267 lines
8.0 KiB
TypeScript
"use client"
|
|
|
|
import * as React from "react"
|
|
import { ArrowUpIcon, CheckIcon, PlusIcon } from "lucide-react"
|
|
|
|
import { cn } from "@/lib/utils"
|
|
import {
|
|
Avatar,
|
|
AvatarFallback,
|
|
AvatarImage,
|
|
} from "@/registry/new-york-v4/ui/avatar"
|
|
import { Button } from "@/registry/new-york-v4/ui/button"
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardFooter,
|
|
CardHeader,
|
|
} from "@/registry/new-york-v4/ui/card"
|
|
import {
|
|
Command,
|
|
CommandEmpty,
|
|
CommandGroup,
|
|
CommandInput,
|
|
CommandItem,
|
|
CommandList,
|
|
} from "@/registry/new-york-v4/ui/command"
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogDescription,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
} from "@/registry/new-york-v4/ui/dialog"
|
|
import { Input } from "@/registry/new-york-v4/ui/input"
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger,
|
|
} from "@/registry/new-york-v4/ui/tooltip"
|
|
|
|
const users = [
|
|
{
|
|
name: "Olivia Martin",
|
|
email: "m@example.com",
|
|
avatar: "/avatars/01.png",
|
|
},
|
|
{
|
|
name: "Isabella Nguyen",
|
|
email: "isabella.nguyen@email.com",
|
|
avatar: "/avatars/03.png",
|
|
},
|
|
{
|
|
name: "Emma Wilson",
|
|
email: "emma@example.com",
|
|
avatar: "/avatars/05.png",
|
|
},
|
|
{
|
|
name: "Jackson Lee",
|
|
email: "lee@example.com",
|
|
avatar: "/avatars/02.png",
|
|
},
|
|
{
|
|
name: "William Kim",
|
|
email: "will@email.com",
|
|
avatar: "/avatars/04.png",
|
|
},
|
|
] as const
|
|
|
|
type User = (typeof users)[number]
|
|
|
|
export function CardsChat() {
|
|
const [open, setOpen] = React.useState(false)
|
|
const [selectedUsers, setSelectedUsers] = React.useState<User[]>([])
|
|
|
|
const [messages, setMessages] = React.useState([
|
|
{
|
|
role: "agent",
|
|
content: "Hi, how can I help you today?",
|
|
},
|
|
{
|
|
role: "user",
|
|
content: "Hey, I'm having trouble with my account.",
|
|
},
|
|
{
|
|
role: "agent",
|
|
content: "What seems to be the problem?",
|
|
},
|
|
{
|
|
role: "user",
|
|
content: "I can't log in.",
|
|
},
|
|
])
|
|
const [input, setInput] = React.useState("")
|
|
const inputLength = input.trim().length
|
|
|
|
return (
|
|
<>
|
|
<Card>
|
|
<CardHeader className="flex flex-row items-center">
|
|
<div className="flex items-center gap-4">
|
|
<Avatar className="border">
|
|
<AvatarImage src="/avatars/01.png" alt="Image" />
|
|
<AvatarFallback>S</AvatarFallback>
|
|
</Avatar>
|
|
<div className="flex flex-col gap-0.5">
|
|
<p className="text-sm leading-none font-medium">Sofia Davis</p>
|
|
<p className="text-muted-foreground text-xs">m@example.com</p>
|
|
</div>
|
|
</div>
|
|
<TooltipProvider delayDuration={0}>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="icon"
|
|
variant="secondary"
|
|
className="ml-auto size-8 rounded-full"
|
|
onClick={() => setOpen(true)}
|
|
>
|
|
<PlusIcon />
|
|
<span className="sr-only">New message</span>
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent sideOffset={10}>New message</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<div className="flex flex-col gap-4">
|
|
{messages.map((message, index) => (
|
|
<div
|
|
key={index}
|
|
className={cn(
|
|
"flex w-max max-w-[75%] flex-col gap-2 rounded-lg px-3 py-2 text-sm",
|
|
message.role === "user"
|
|
? "bg-primary text-primary-foreground ml-auto"
|
|
: "bg-muted"
|
|
)}
|
|
>
|
|
{message.content}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</CardContent>
|
|
<CardFooter>
|
|
<form
|
|
onSubmit={(event) => {
|
|
event.preventDefault()
|
|
if (inputLength === 0) return
|
|
setMessages([
|
|
...messages,
|
|
{
|
|
role: "user",
|
|
content: input,
|
|
},
|
|
])
|
|
setInput("")
|
|
}}
|
|
className="relative w-full"
|
|
>
|
|
<Input
|
|
id="message"
|
|
placeholder="Type your message..."
|
|
className="flex-1 pr-10"
|
|
autoComplete="off"
|
|
value={input}
|
|
onChange={(event) => setInput(event.target.value)}
|
|
/>
|
|
<Button
|
|
type="submit"
|
|
size="icon"
|
|
className="absolute top-1/2 right-2 size-6 -translate-y-1/2 rounded-full"
|
|
disabled={inputLength === 0}
|
|
>
|
|
<ArrowUpIcon className="size-3.5" />
|
|
<span className="sr-only">Send</span>
|
|
</Button>
|
|
</form>
|
|
</CardFooter>
|
|
</Card>
|
|
<Dialog open={open} onOpenChange={setOpen}>
|
|
<DialogContent className="gap-0 p-0 outline-none">
|
|
<DialogHeader className="px-4 pt-5 pb-4">
|
|
<DialogTitle>New message</DialogTitle>
|
|
<DialogDescription>
|
|
Invite a user to this thread. This will create a new group
|
|
message.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
<Command className="overflow-hidden rounded-t-none border-t bg-transparent">
|
|
<CommandInput placeholder="Search user..." />
|
|
<CommandList>
|
|
<CommandEmpty>No users found.</CommandEmpty>
|
|
<CommandGroup>
|
|
{users.map((user) => (
|
|
<CommandItem
|
|
key={user.email}
|
|
data-active={selectedUsers.includes(user)}
|
|
className="data-[active=true]:opacity-50"
|
|
onSelect={() => {
|
|
if (selectedUsers.includes(user)) {
|
|
return setSelectedUsers(
|
|
selectedUsers.filter(
|
|
(selectedUser) => selectedUser !== user
|
|
)
|
|
)
|
|
}
|
|
|
|
return setSelectedUsers(
|
|
[...users].filter((u) =>
|
|
[...selectedUsers, user].includes(u)
|
|
)
|
|
)
|
|
}}
|
|
>
|
|
<Avatar className="border">
|
|
<AvatarImage src={user.avatar} alt="Image" />
|
|
<AvatarFallback>{user.name[0]}</AvatarFallback>
|
|
</Avatar>
|
|
<div className="ml-2">
|
|
<p className="text-sm leading-none font-medium">
|
|
{user.name}
|
|
</p>
|
|
<p className="text-muted-foreground text-sm">
|
|
{user.email}
|
|
</p>
|
|
</div>
|
|
{selectedUsers.includes(user) ? (
|
|
<CheckIcon className="text-primary ml-auto flex size-4" />
|
|
) : null}
|
|
</CommandItem>
|
|
))}
|
|
</CommandGroup>
|
|
</CommandList>
|
|
</Command>
|
|
<DialogFooter className="flex items-center border-t p-4 sm:justify-between">
|
|
{selectedUsers.length > 0 ? (
|
|
<div className="flex -space-x-2 overflow-hidden">
|
|
{selectedUsers.map((user) => (
|
|
<Avatar key={user.email} className="inline-block border">
|
|
<AvatarImage src={user.avatar} />
|
|
<AvatarFallback>{user.name[0]}</AvatarFallback>
|
|
</Avatar>
|
|
))}
|
|
</div>
|
|
) : (
|
|
<p className="text-muted-foreground text-sm">
|
|
Select users to add to this thread.
|
|
</p>
|
|
)}
|
|
<Button
|
|
disabled={selectedUsers.length < 2}
|
|
size="sm"
|
|
onClick={() => {
|
|
setOpen(false)
|
|
}}
|
|
>
|
|
Continue
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
</>
|
|
)
|
|
}
|