m/cup
1
0
mirror of https://github.com/sergi0g/cup.git synced 2025-11-10 14:13:49 -05:00
Files
cup/web/src/components/Image.tsx
2025-05-09 13:19:42 +03:00

232 lines
8.2 KiB
TypeScript

import { useState } from "react";
import {
Dialog,
DialogBackdrop,
DialogPanel,
DialogTitle,
} from "@headlessui/react";
import { WithTooltip } from "./ui/Tooltip";
import type { Image } from "../types";
import { theme } from "../theme";
import { CodeBlock } from "./CodeBlock";
import {
Box,
CircleArrowUp,
CircleCheck,
HelpCircle,
Timer,
TriangleAlert,
X,
} from "lucide-react";
import Badge from "./Badge";
import { getDescription } from "../utils";
const clickable_registries = [
"registry-1.docker.io",
"ghcr.io",
"quay.io",
"gcr.io",
]; // Not all registries redirect to an info page when visiting the image reference in a browser (e.g. Gitea and derivatives), so we only enable clicking those who do.
export default function Image({ data }: { data: Image }) {
const [open, setOpen] = useState(false);
const handleOpen = () => {
setOpen(true);
};
const handleClose = () => {
setOpen(false);
};
const new_reference =
data.result.info?.type == "version"
? data.reference.split(":")[0] + ":" + data.result.info.new_tag
: data.reference;
const info = getInfo(data);
let url: string | null = null;
if (data.url) {
url = data.url;
} else if (clickable_registries.includes(data.parts.registry)) {
switch (data.parts.registry) {
case "registry-1.docker.io":
url = `https://hub.docker.com/r/${data.parts.repository}`;
break;
default:
url = `https://${data.parts.registry}/${data.parts.repository}`;
break;
}
}
return (
<>
<button onClick={handleOpen} className="w-full">
<li
className={`flex items-center gap-4 break-all px-6 py-4 text-start hover:bg-${theme}-100 hover:dark:bg-${theme}-900/50 transition-colors duration-200`}
>
<Box className={`size-6 shrink-0 text-${theme}-500`} />
<span className="font-mono">{data.reference}</span>
<div className="ml-auto flex gap-2">
{data.result.info?.type === "version" ? (
<Badge
from={data.result.info.current_version}
to={data.result.info.new_version}
/>
) : null}
<WithTooltip
text={info.description}
className={`size-6 shrink-0 ${info.color}`}
>
<info.icon />
</WithTooltip>
</div>
</li>
</button>
<Dialog open={open} onClose={setOpen} className="relative z-10">
<DialogBackdrop
transition
className={`fixed inset-0 bg-${theme}-500 dark:bg-${theme}-950 !bg-opacity-75 transition-opacity data-[closed]:opacity-0 data-[enter]:duration-300 data-[leave]:duration-200 data-[enter]:ease-out data-[leave]:ease-in`}
/>
<div className="fixed inset-0 z-10 w-screen overflow-y-auto">
<div className="flex min-h-full items-end justify-center text-center sm:items-center sm:p-0">
<DialogPanel
transition
className={`relative transform overflow-hidden rounded-t-lg bg-white dark:border dark:border-${theme}-800 md:rounded-lg dark:bg-${theme}-900 w-full text-left shadow-xl transition-all data-[closed]:translate-y-4 data-[closed]:opacity-0 data-[enter]:duration-300 data-[leave]:duration-200 data-[enter]:ease-out data-[leave]:ease-in sm:my-8 sm:w-full sm:max-w-lg data-[closed]:sm:translate-y-0 data-[closed]:sm:scale-95 md:max-w-xl lg:max-w-2xl dark:text-white`}
>
<div
className={`flex flex-col gap-3 px-6 py-4 text-${theme}-600 dark:text-${theme}-400`}
>
<div className="mb-4 flex items-center gap-3">
<Box className={`size-6 shrink-0 text-${theme}-500`} />
<DialogTitle className="break-all font-mono text-black dark:text-white">
{url ? (
<>
<a
href={url}
target="_blank"
rel="noopener noreferrer"
className={`group w-fit hover:underline`}
>
<span>
{data.reference}
<svg
viewBox="0 0 12 12"
fill="none"
height="1cap"
xmlns="http://www.w3.org/2000/svg"
className="ml-1 inline transition-all duration-100 group-hover:rotate-45"
>
<path
d="M11 9.283V1H2.727v1.44h5.83L1 9.99 2.01 11l7.556-7.55v5.833H11Z"
fill="currentColor"
></path>
</svg>
</span>
</a>
</>
) : (
data.reference
)}
</DialogTitle>
<button onClick={handleClose} className="ml-auto">
<X
className={`size-6 shrink-0 text-${theme}-500 transition-colors duration-200 hover:text-black dark:hover:text-white`}
/>
</button>
</div>
<div className="flex items-center gap-3">
<info.icon className={`size-6 shrink-0 ${info.color}`} />
{info.description}
</div>
<div className="flex items-center gap-3">
<Timer className="size-6 shrink-0 text-gray-500" />
<span>
Checked in <b>{data.time}</b> ms
</span>
</div>
{data.result.error && (
<div className="break-before mt-4 flex items-center gap-3 overflow-hidden rounded-md bg-yellow-400/10 px-3 py-2">
<TriangleAlert className="size-6 shrink-0 text-yellow-500" />
{data.result.error}
</div>
)}
{data.result.has_update && (
<div className="mt-4 flex flex-col gap-1">
Pull command
<CodeBlock enableCopy>
docker pull {new_reference}
</CodeBlock>
</div>
)}
<div className="flex flex-col gap-1">
{data.result.info?.type == "digest" && (
<>
{data.result.info.local_digests.length > 1
? "Local digests"
: "Local digest"}
<CodeBlock enableCopy>
{data.result.info.local_digests.join("\n")}
</CodeBlock>
{data.result.info.remote_digest && (
<div className="flex flex-col gap-1">
Remote digest
<CodeBlock enableCopy>
{data.result.info.remote_digest}
</CodeBlock>
</div>
)}
</>
)}
</div>
</div>
</DialogPanel>
</div>
</div>
</Dialog>
</>
);
}
function getInfo(data: Image): {
color: string;
icon: typeof HelpCircle;
description: string;
} {
const description = getDescription(data);
switch (description) {
case "Unknown":
return {
color: "text-gray-500",
icon: HelpCircle,
description,
};
case "Up to date":
return {
color: "text-green-500",
icon: CircleCheck,
description,
};
case "Major update":
return {
color: "text-red-500",
icon: CircleArrowUp,
description,
};
case "Minor update":
return {
color: "text-yellow-500",
icon: CircleArrowUp,
description,
};
case "Patch update":
return {
color: "text-blue-500",
icon: CircleArrowUp,
description,
};
case "Digest update":
return {
color: "text-blue-500",
icon: CircleArrowUp,
description,
};
}
}