upscayl/renderer/components/sidebar/settings-tab/copy-metadata-toggle.tsx
Pedro Medina 86c144b1e3
feat: add "Copy Metadata" functionality to upscaled images (#1220)
feat: copy metadata
fix: better copy-metadata toggle
raw: add new locales for metadata
fix: log metadata on double upscayl
feat: suggest jpg when copying metadata
raw: add metadata suggestion locales
refactor: add bot suggestions
feat: JPEG format validation + ARIA accessibility
2025-06-07 16:34:41 -04:00

108 lines
3.8 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import React, { useState, useCallback } from "react";
import { translationAtom } from "@/atoms/translations-atom";
import { copyMetadataAtom } from "@/atoms/user-settings-atom";
import { useAtom, useAtomValue } from "jotai";
type CopyMetadataToggleProps = {
saveImageAs: string;
setExportType: (type: string) => void;
};
const CopyMetadataToggle = ({ saveImageAs, setExportType }: CopyMetadataToggleProps) => {
const [copyMetadata, setCopyMetadata] = useAtom(copyMetadataAtom);
const t = useAtomValue(translationAtom);
const [showSuggestModal, setShowSuggestModal] = useState(false);
const handleToggle = useCallback(() => {
if (!copyMetadata && !["jpg", "jpeg"].includes(saveImageAs.toLowerCase()) && setExportType) {
setShowSuggestModal(true);
} else {
setCopyMetadata(!copyMetadata);
localStorage.setItem("copyMetadata", (!copyMetadata).toString());
}
}, [copyMetadata, saveImageAs, setExportType, setCopyMetadata]);
const handleChangeToJpg = useCallback(() => {
setShowSuggestModal(false);
setCopyMetadata(true);
localStorage.setItem("copyMetadata", true.toString());
setExportType("jpg");
}, [setCopyMetadata, setExportType]);
const handleKeepFormat = useCallback(() => {
setShowSuggestModal(false);
setCopyMetadata(true);
localStorage.setItem("copyMetadata", true.toString());
}, [setCopyMetadata]);
return (
<div className="flex flex-col gap-2">
<p className="text-sm font-medium" id="copy-metadata-title">
{t("SETTINGS.COPY_METADATA.TITLE")}
</p>
<p className="text-xs text-base-content/80" id="copy-metadata-description">
{t("SETTINGS.COPY_METADATA.DESCRIPTION")}
</p>
<input
type="checkbox"
className="toggle"
checked={copyMetadata}
onChange={handleToggle}
id="copy-metadata-toggle"
aria-labelledby="copy-metadata-title"
aria-describedby="copy-metadata-description"
/>
{copyMetadata && !["jpg", "jpeg"].includes(saveImageAs.toLowerCase()) && (
<p className="text-xs text-warning">
{t("WARNING.METADATA_FORMAT.DESCRIPTION")}
</p>
)}
{showSuggestModal && (
<div
className="fixed inset-0 z-50 flex items-center justify-center bg-black/40"
onClick={() => setShowSuggestModal(false)}
onKeyDown={(e) => e.key === "Escape" && setShowSuggestModal(false)}
tabIndex={0}
>
<div
className="w-96 rounded-lg bg-base-100 p-6 shadow-xl"
onClick={(e) => e.stopPropagation()}
>
<div className="flex justify-between items-center mb-2">
<h3 className="text-lg font-semibold">
{t("SETTINGS.COPY_METADATA.SUGGEST_JPG_TITLE")}
</h3>
<button
className="btn btn-sm btn-ghost"
onClick={() => setShowSuggestModal(false)}
aria-label="Close"
>
×
</button>
</div>
<p className="mb-4 text-sm text-base-content/80">
{t("SETTINGS.COPY_METADATA.SUGGEST_JPG_DESCRIPTION")}
</p>
<div className="flex justify-end gap-2">
<button
className="btn btn-primary"
onClick={handleChangeToJpg}
>
{t("SETTINGS.COPY_METADATA.CHANGE_TO_JPG")}
</button>
<button
className="btn"
onClick={handleKeepFormat}
>
{t("SETTINGS.COPY_METADATA.KEEP_CURRENT_FORMAT").replace("{format}", saveImageAs.toUpperCase())}
</button>
</div>
</div>
</div>
)}
</div>
);
};
export default CopyMetadataToggle;