mirror of
https://github.com/HKUDS/nanobot.git
synced 2026-06-15 23:34:00 +00:00
* feat(webui): refine output timeline and composer queue * feat(webui): add provider model picker * fix(webui): polish model settings and heartbeat checks * chore: keep heartbeat changes out of webui pr * refactor(webui): isolate settings routes * fix(providers): align minimax anthropic test * fix(providers): keep minimax anthropic base sdk-compatible * fix(providers): normalize anthropic base urls
174 lines
5.2 KiB
TypeScript
174 lines
5.2 KiB
TypeScript
import { useState, type ReactNode } from "react";
|
|
import { FileIcon, ImageIcon, PlaySquare } from "lucide-react";
|
|
import { useTranslation } from "react-i18next";
|
|
|
|
import { cn } from "@/lib/utils";
|
|
import type { UIMediaAttachment } from "@/lib/types";
|
|
|
|
interface AttachmentTileProps {
|
|
attachment: UIMediaAttachment;
|
|
className?: string;
|
|
inline?: boolean;
|
|
variant?: "default" | "compact";
|
|
}
|
|
|
|
export function AttachmentTile({ attachment, className, inline = false, variant = "default" }: AttachmentTileProps) {
|
|
const { t } = useTranslation();
|
|
const [failed, setFailed] = useState(false);
|
|
const hasUrl = typeof attachment.url === "string" && attachment.url.length > 0;
|
|
const label = attachmentLabel(attachment, t);
|
|
|
|
if (attachment.kind === "image" && hasUrl && !failed) {
|
|
return (
|
|
<AttachmentFrame
|
|
attachment={attachment}
|
|
className={className}
|
|
inline={inline}
|
|
variant={variant}
|
|
>
|
|
<a
|
|
href={attachment.url}
|
|
target="_blank"
|
|
rel="noreferrer noopener"
|
|
className="block bg-muted/20"
|
|
aria-label={attachment.name ? `Open ${attachment.name}` : t("lightbox.open", { defaultValue: "Open image" })}
|
|
>
|
|
<img
|
|
src={attachment.url}
|
|
alt={attachment.name ?? ""}
|
|
loading="lazy"
|
|
decoding="async"
|
|
draggable={false}
|
|
onError={() => setFailed(true)}
|
|
className={cn(
|
|
"block h-auto max-w-full bg-background object-contain",
|
|
variant === "compact" ? "max-h-40" : "max-h-[34rem]",
|
|
)}
|
|
/>
|
|
</a>
|
|
</AttachmentFrame>
|
|
);
|
|
}
|
|
|
|
if (attachment.kind === "video" && hasUrl) {
|
|
return (
|
|
<AttachmentFrame
|
|
attachment={attachment}
|
|
className={className}
|
|
inline={inline}
|
|
variant={variant}
|
|
>
|
|
<video
|
|
src={attachment.url}
|
|
controls
|
|
preload="auto"
|
|
className={cn(
|
|
"block w-full bg-black",
|
|
variant === "compact" ? "max-h-40" : "max-h-[26rem]",
|
|
)}
|
|
aria-label={attachment.name ? `${t("message.videoAttachment", { defaultValue: "Video attachment" })}: ${attachment.name}` : t("message.videoAttachment", { defaultValue: "Video attachment" })}
|
|
/>
|
|
</AttachmentFrame>
|
|
);
|
|
}
|
|
|
|
const Icon = attachment.kind === "video"
|
|
? PlaySquare
|
|
: attachment.kind === "image"
|
|
? ImageIcon
|
|
: FileIcon;
|
|
const body = (
|
|
<>
|
|
<Icon className="h-4 w-4 flex-none" aria-hidden />
|
|
<span className="min-w-0 truncate">{attachment.name ?? label}</span>
|
|
</>
|
|
);
|
|
|
|
if (hasUrl && !failed) {
|
|
return (
|
|
<a
|
|
href={attachment.url}
|
|
download={attachment.name ?? label}
|
|
title={attachment.name ?? undefined}
|
|
aria-label={label}
|
|
className={cn(
|
|
"flex max-w-[18rem] items-center gap-2 rounded-[14px]",
|
|
"border border-border/60 bg-muted/40 px-3 py-2 text-xs text-muted-foreground",
|
|
"transition-colors hover:bg-muted/55 hover:text-foreground",
|
|
variant === "compact" && "max-w-[14rem] rounded-xl px-2.5 py-1.5 text-[11.5px]",
|
|
className,
|
|
)}
|
|
>
|
|
{body}
|
|
</a>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
"flex max-w-[18rem] items-center gap-2 rounded-[14px]",
|
|
"border border-border/60 bg-muted/35 px-3 py-2 text-xs text-muted-foreground",
|
|
variant === "compact" && "max-w-[14rem] rounded-xl px-2.5 py-1.5 text-[11.5px]",
|
|
className,
|
|
)}
|
|
title={attachment.name ?? undefined}
|
|
aria-label={label}
|
|
>
|
|
{body}
|
|
<span className="sr-only">
|
|
{t("message.attachmentUnavailable", { defaultValue: "Attachment unavailable" })}
|
|
</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function AttachmentFrame({
|
|
attachment,
|
|
children,
|
|
className,
|
|
inline = false,
|
|
variant = "default",
|
|
}: {
|
|
attachment: UIMediaAttachment;
|
|
children: ReactNode;
|
|
className?: string;
|
|
inline?: boolean;
|
|
variant?: "default" | "compact";
|
|
}) {
|
|
const frameClassName = cn(
|
|
"not-prose my-3 block w-fit max-w-full overflow-hidden rounded-[14px]",
|
|
"border border-border/60 bg-muted/40",
|
|
attachment.kind === "image" && "bg-background/85",
|
|
attachment.kind === "video" ? "w-[min(100%,32rem)]" : "",
|
|
variant === "compact" && "my-1 rounded-xl shadow-none",
|
|
variant === "compact" && attachment.kind === "video" && "w-[min(100%,20rem)]",
|
|
className,
|
|
);
|
|
const bodyClassName = "block max-w-full";
|
|
const body = inline ? (
|
|
<span className={bodyClassName}>{children}</span>
|
|
) : (
|
|
<div className={bodyClassName}>{children}</div>
|
|
);
|
|
return inline ? (
|
|
<span className={frameClassName}>
|
|
{body}
|
|
</span>
|
|
) : (
|
|
<figure className={frameClassName}>
|
|
{body}
|
|
</figure>
|
|
);
|
|
}
|
|
|
|
function attachmentLabel(attachment: UIMediaAttachment, t: ReturnType<typeof useTranslation>["t"]): string {
|
|
if (attachment.kind === "video") {
|
|
return t("message.videoAttachment", { defaultValue: "Video attachment" });
|
|
}
|
|
if (attachment.kind === "image") {
|
|
return t("message.imageAttachment", { defaultValue: "Image attachment" });
|
|
}
|
|
return t("message.fileAttachment", { defaultValue: "File attachment" });
|
|
}
|