[dyad] Added image file size analysis - wrote 3 file(s)

This commit is contained in:
[dyad]
2026-01-20 14:24:21 +01:00
parent daeda47c4c
commit 92aca8ba3f
3 changed files with 56 additions and 6 deletions

View File

@@ -18,6 +18,7 @@ export interface HeadlineNode {
export interface ImageAltData { export interface ImageAltData {
src: string; src: string;
alt: string; alt: string;
size: number | null;
} }
export async function extractMetaData(url: string, keyword?: string) { export async function extractMetaData(url: string, keyword?: string) {
@@ -127,7 +128,7 @@ export async function extractMetaData(url: string, keyword?: string) {
keywordCount = matches ? matches.length : 0; keywordCount = matches ? matches.length : 0;
} }
const imageAltData: ImageAltData[] = []; const imageSrcs: { src: string; alt: string }[] = [];
$("img").each((i, el) => { $("img").each((i, el) => {
const src = $(el).attr("src"); const src = $(el).attr("src");
const alt = $(el).attr("alt") || ""; const alt = $(el).attr("alt") || "";
@@ -135,7 +136,7 @@ export async function extractMetaData(url: string, keyword?: string) {
if (src) { if (src) {
try { try {
const absoluteSrc = new URL(src, formattedUrl).href; const absoluteSrc = new URL(src, formattedUrl).href;
imageAltData.push({ imageSrcs.push({
src: absoluteSrc, src: absoluteSrc,
alt: alt.trim(), alt: alt.trim(),
}); });
@@ -145,6 +146,25 @@ export async function extractMetaData(url: string, keyword?: string) {
} }
}); });
const imageSizePromises = imageSrcs.map(async (img) => {
try {
// Use a HEAD request for efficiency
const res = await fetch(img.src, { method: "HEAD" });
if (res.ok) {
const contentLength = res.headers.get("content-length");
return {
...img,
size: contentLength ? parseInt(contentLength, 10) : null,
};
}
return { ...img, size: null };
} catch (error) {
return { ...img, size: null };
}
});
const imageAltData: ImageAltData[] = await Promise.all(imageSizePromises);
return { return {
data: { data: {
title, title,

View File

@@ -6,6 +6,7 @@ import { Checkbox } from "@/components/ui/checkbox";
import { Label } from "@/components/ui/label"; import { Label } from "@/components/ui/label";
import { Badge } from "@/components/ui/badge"; import { Badge } from "@/components/ui/badge";
import { ImageOff } from "lucide-react"; import { ImageOff } from "lucide-react";
import { formatBytes } from "@/lib/utils";
interface ImageAltDisplayProps { interface ImageAltDisplayProps {
images: ImageAltData[]; images: ImageAltData[];
@@ -25,6 +26,13 @@ export function ImageAltDisplay({ images }: ImageAltDisplayProps) {
setImageErrors((prev) => ({ ...prev, [src]: true })); setImageErrors((prev) => ({ ...prev, [src]: true }));
}; };
const getSizeBadgeVariant = (size: number | null) => {
if (size === null) return "outline";
if (size > 500000) return "destructive"; // > 500KB
if (size > 100000) return "secondary"; // > 100KB
return "outline";
};
return ( return (
<div className="space-y-4"> <div className="space-y-4">
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4 p-4 bg-muted/50 rounded-lg"> <div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4 p-4 bg-muted/50 rounded-lg">
@@ -67,10 +75,20 @@ export function ImageAltDisplay({ images }: ImageAltDisplayProps) {
/> />
)} )}
</div> </div>
<div className="flex-grow"> <div className="flex-grow min-w-0">
<p className="text-sm text-muted-foreground break-all"> <div className="flex justify-between items-start gap-2">
{image.src} <p className="text-sm text-muted-foreground break-all flex-grow">
</p> {image.src}
</p>
{image.size !== null && (
<Badge
variant={getSizeBadgeVariant(image.size)}
className="flex-shrink-0"
>
{formatBytes(image.size)}
</Badge>
)}
</div>
<div className="mt-2"> <div className="mt-2">
{image.alt ? ( {image.alt ? (
<p className="text-sm text-foreground bg-muted/50 p-2 rounded-md"> <p className="text-sm text-foreground bg-muted/50 p-2 rounded-md">

View File

@@ -4,3 +4,15 @@ import { twMerge } from "tailwind-merge"
export function cn(...inputs: ClassValue[]) { export function cn(...inputs: ClassValue[]) {
return twMerge(clsx(inputs)) return twMerge(clsx(inputs))
} }
export function formatBytes(bytes: number, decimals = 2) {
if (!+bytes) return "0 Bytes";
const k = 1024;
const dm = decimals < 0 ? 0 : decimals;
const sizes = ["Bytes", "KB", "MB", "GB", "TB"];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`;
}