mirror of
https://github.com/upscayl/upscayl.git
synced 2025-01-19 01:24:09 +01:00
357 lines
11 KiB
TypeScript
357 lines
11 KiB
TypeScript
import React, { useEffect, useState } from "react";
|
|
import Select from "react-select";
|
|
import ReactTooltip from "react-tooltip";
|
|
import { themeChange } from "theme-change";
|
|
|
|
interface IProps {
|
|
progress: string;
|
|
selectImageHandler: () => Promise<void>;
|
|
selectFolderHandler: () => Promise<void>;
|
|
handleModelChange: (e: any) => void;
|
|
handleDrop: (e: any) => void;
|
|
outputHandler: () => Promise<void>;
|
|
upscaylHandler: () => Promise<void>;
|
|
batchMode: boolean;
|
|
setBatchMode: React.Dispatch<React.SetStateAction<boolean>>;
|
|
imagePath: string;
|
|
outputPath: string;
|
|
doubleUpscayl: boolean;
|
|
setDoubleUpscayl: React.Dispatch<React.SetStateAction<boolean>>;
|
|
model: string;
|
|
setModel: React.Dispatch<React.SetStateAction<string>>;
|
|
isVideo: boolean;
|
|
setIsVideo: React.Dispatch<React.SetStateAction<boolean>>;
|
|
saveImageAs: string;
|
|
setSaveImageAs: React.Dispatch<React.SetStateAction<string>>;
|
|
gpuId: string;
|
|
setGpuId: React.Dispatch<React.SetStateAction<string>>;
|
|
dimensions: {
|
|
width: number | null;
|
|
height: number | null;
|
|
};
|
|
}
|
|
|
|
function LeftPaneImageSteps({
|
|
progress,
|
|
selectImageHandler,
|
|
selectFolderHandler,
|
|
handleModelChange,
|
|
handleDrop,
|
|
outputHandler,
|
|
upscaylHandler,
|
|
batchMode,
|
|
setBatchMode,
|
|
imagePath,
|
|
outputPath,
|
|
doubleUpscayl,
|
|
setDoubleUpscayl,
|
|
model,
|
|
setModel,
|
|
isVideo,
|
|
setIsVideo,
|
|
gpuId,
|
|
setGpuId,
|
|
saveImageAs,
|
|
setSaveImageAs,
|
|
dimensions,
|
|
}: IProps) {
|
|
const [currentModel, setCurrentModel] = useState<{
|
|
label: string;
|
|
value: string;
|
|
}>({
|
|
label: null,
|
|
value: null,
|
|
});
|
|
|
|
useEffect(() => {
|
|
themeChange(false);
|
|
|
|
if (!localStorage.getItem("saveImageAs")) {
|
|
localStorage.setItem("saveImageAs", "png");
|
|
} else {
|
|
const currentlySavedImageFormat = localStorage.getItem("saveImageAs");
|
|
setSaveImageAs(currentlySavedImageFormat);
|
|
}
|
|
|
|
if (!localStorage.getItem("model")) {
|
|
setCurrentModel(modelOptions[0]);
|
|
setModel(modelOptions[0].value);
|
|
localStorage.setItem("model", JSON.stringify(modelOptions[0]));
|
|
} else {
|
|
const currentlySavedModel = JSON.parse(
|
|
localStorage.getItem("model")
|
|
) as typeof modelOptions[0];
|
|
setCurrentModel(currentlySavedModel);
|
|
setModel(currentlySavedModel.value);
|
|
}
|
|
|
|
if (!localStorage.getItem("gpuId")) {
|
|
localStorage.setItem("gpuId", "");
|
|
} else {
|
|
const currentlySavedGpuId = localStorage.getItem("gpuId");
|
|
setGpuId(currentlySavedGpuId);
|
|
}
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
console.log("Current Model: ", currentModel);
|
|
}, [currentModel]);
|
|
|
|
const setExportType = (format: string) => {
|
|
setSaveImageAs(format);
|
|
localStorage.setItem("saveImageAs", format);
|
|
};
|
|
|
|
const handleBatchMode = () => {
|
|
setBatchMode((oldValue) => !oldValue);
|
|
};
|
|
|
|
const handleGpuIdChange = (e) => {
|
|
setGpuId(e.target.value);
|
|
localStorage.setItem("gpuId", e.target.value);
|
|
};
|
|
|
|
const customStyles = {
|
|
option: (provided, state) => ({
|
|
...provided,
|
|
borderBottom: "1px dotted pink",
|
|
color: state.isSelected ? "red" : "blue",
|
|
padding: 20,
|
|
}),
|
|
control: () => ({
|
|
// none of react-select's styles are passed to <Control />
|
|
width: 200,
|
|
}),
|
|
singleValue: (provided, state) => {
|
|
const opacity = state.isDisabled ? 0.5 : 1;
|
|
const transition = "opacity 300ms";
|
|
|
|
return { ...provided, opacity, transition };
|
|
},
|
|
};
|
|
|
|
const modelOptions = [
|
|
{ label: "General Photo (Real-ESRGAN)", value: "realesrgan-x4plus" },
|
|
{ label: "General Photo (Remacri)", value: "remacri" },
|
|
{ label: "General Photo (Ultramix Balanced)", value: "ultramix_balanced" },
|
|
{ label: "General Photo (Ultrasharp)", value: "ultrasharp" },
|
|
{ label: "Digital Art", value: "realesrgan-x4plus-anime" },
|
|
{ label: "Sharpen Image", value: "models-DF2K" },
|
|
];
|
|
|
|
const availableThemes = [
|
|
{ label: "light", value: "light" },
|
|
{ label: "dark", value: "dark" },
|
|
{ label: "cupcake", value: "cupcake" },
|
|
{ label: "bumblebee", value: "bumblebee" },
|
|
{ label: "emerald", value: "emerald" },
|
|
{ label: "corporate", value: "corporate" },
|
|
{ label: "synthwave", value: "synthwave" },
|
|
{ label: "retro", value: "retro" },
|
|
{ label: "cyberpunk", value: "cyberpunk" },
|
|
{ label: "valentine", value: "valentine" },
|
|
{ label: "halloween", value: "halloween" },
|
|
{ label: "garden", value: "garden" },
|
|
{ label: "forest", value: "forest" },
|
|
{ label: "aqua", value: "aqua" },
|
|
{ label: "lofi", value: "lofi" },
|
|
{ label: "pastel", value: "pastel" },
|
|
{ label: "fantasy", value: "fantasy" },
|
|
{ label: "wireframe", value: "wireframe" },
|
|
{ label: "black", value: "black" },
|
|
{ label: "luxury", value: "luxury" },
|
|
{ label: "dracula", value: "dracula" },
|
|
{ label: "cmyk", value: "cmyk" },
|
|
{ label: "autumn", value: "autumn" },
|
|
{ label: "business", value: "business" },
|
|
{ label: "acid", value: "acid" },
|
|
{ label: "lemonade", value: "lemonade" },
|
|
{ label: "night", value: "night" },
|
|
{ label: "coffee", value: "coffee" },
|
|
{ label: "winter", value: "winter" },
|
|
];
|
|
|
|
useEffect(() => {}, [imagePath]);
|
|
|
|
return (
|
|
<div className="animate-step-in animate flex h-screen flex-col gap-7 overflow-y-auto p-5 overflow-x-hidden">
|
|
{/* BATCH OPTION */}
|
|
<div className="flex flex-row items-center gap-2">
|
|
<input
|
|
type="checkbox"
|
|
className="toggle"
|
|
onClick={handleBatchMode}></input>
|
|
<p
|
|
className="mr-1 inline-block cursor-help text-sm"
|
|
data-tip="This will let you upscale all files in a folder at once">
|
|
Batch Upscale
|
|
</p>
|
|
</div>
|
|
|
|
{/* STEP 1 */}
|
|
<div data-tip={imagePath}>
|
|
<p className="step-heading">Step 1</p>
|
|
<button
|
|
className="btn-primary btn"
|
|
onClick={!batchMode ? selectImageHandler : selectFolderHandler}>
|
|
Select {batchMode ? "Folder" : "Image"}
|
|
</button>
|
|
</div>
|
|
|
|
{/* STEP 2 */}
|
|
<div className="animate-step-in">
|
|
<p className="step-heading">Step 2</p>
|
|
<p className="mb-2 text-sm">Select Upscaling Type</p>
|
|
|
|
<Select
|
|
options={modelOptions}
|
|
components={{
|
|
IndicatorSeparator: () => null,
|
|
DropdownIndicator: () => null,
|
|
}}
|
|
onChange={(e) => {
|
|
handleModelChange(e);
|
|
setCurrentModel({ label: e.label, value: e.value });
|
|
}}
|
|
className="react-select-container"
|
|
classNamePrefix="react-select"
|
|
value={currentModel}
|
|
/>
|
|
|
|
{model !== "models-DF2K" && !batchMode && (
|
|
<div className="mt-4 flex items-center gap-1">
|
|
<input
|
|
type="checkbox"
|
|
className="checkbox"
|
|
checked={doubleUpscayl}
|
|
onChange={(e) => {
|
|
if (e.target.checked) {
|
|
setDoubleUpscayl(true);
|
|
} else {
|
|
setDoubleUpscayl(false);
|
|
}
|
|
}}
|
|
/>
|
|
<p
|
|
className="cursor-pointer text-sm"
|
|
onClick={(e) => {
|
|
setDoubleUpscayl(!doubleUpscayl);
|
|
}}>
|
|
Double Upscayl
|
|
</p>
|
|
<button
|
|
className="badge-info badge cursor-help"
|
|
data-tip="Enable this option to get an 8x upscayl. Note that this may not always work properly with all images, for example, images with really large resolutions.">
|
|
i
|
|
</button>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* STEP 3 */}
|
|
<div className="animate-step-in" data-tip={outputPath}>
|
|
<p className="step-heading">Step 3</p>
|
|
<p className="mb-2 text-sm">
|
|
Defaults to {!batchMode ? "Image's" : "Folder's"} path
|
|
</p>
|
|
<button className="btn-primary btn" onClick={outputHandler}>
|
|
Set Output Folder
|
|
</button>
|
|
</div>
|
|
|
|
{/* STEP 4 */}
|
|
<div className="animate-step-in">
|
|
<p className="step-heading">Step 4</p>
|
|
{dimensions.width && dimensions.height && (
|
|
<p className="mb-2 text-sm">
|
|
Upscale from{" "}
|
|
<span className="font-bold">
|
|
{dimensions.width}x{dimensions.height}
|
|
</span>{" "}
|
|
to{" "}
|
|
<span className="font-bold">
|
|
{doubleUpscayl ? dimensions.width * 8 : dimensions.width * 4}x
|
|
{doubleUpscayl ? dimensions.height * 8 : dimensions.height * 4}
|
|
</span>
|
|
</p>
|
|
)}
|
|
<button
|
|
className="btn-accent btn"
|
|
onClick={upscaylHandler}
|
|
disabled={progress.length > 0}>
|
|
{progress.length > 0 ? "Upscayling⏳" : "Upscayl"}
|
|
</button>
|
|
<div className="rounded-btn collapse mt-5">
|
|
<input type="checkbox" className="peer" />
|
|
<div className="collapse-title bg-neutral text-neutral-content peer-checked:bg-primary peer-checked:text-primary-content">
|
|
Advanced Options
|
|
</div>
|
|
<div className="collapse-content flex flex-col gap-4 bg-neutral text-neutral-content peer-checked:bg-base-300 peer-checked:py-4 peer-checked:text-base-content">
|
|
<div className="flex flex-col gap-2">
|
|
<p>Save Image As:</p>
|
|
<div className="flex flex-wrap gap-2">
|
|
<button
|
|
className={`btn-primary btn ${
|
|
saveImageAs === "png" && "btn-accent"
|
|
}`}
|
|
onClick={() => setExportType("png")}>
|
|
PNG
|
|
</button>
|
|
<button
|
|
className={`btn-primary btn ${
|
|
saveImageAs === "jpg" && "btn-accent"
|
|
}`}
|
|
onClick={() => setExportType("jpg")}>
|
|
JPG
|
|
</button>
|
|
<button
|
|
className={`btn-primary btn ${
|
|
saveImageAs === "webp" && "btn-accent"
|
|
}`}
|
|
onClick={() => setExportType("webp")}>
|
|
WEBP
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-2">
|
|
<p>Upscayl Theme:</p>
|
|
<select data-choose-theme className="select-primary select">
|
|
<option value="dark">Default</option>
|
|
{availableThemes.map((theme) => {
|
|
return (
|
|
<option value={theme.value} key={theme.value}>
|
|
{theme.label.toLocaleUpperCase()}
|
|
</option>
|
|
);
|
|
})}
|
|
</select>
|
|
</div>
|
|
<div className="flex flex-col gap-2">
|
|
<p>GPU ID:</p>
|
|
<input
|
|
type="text"
|
|
placeholder="Type here"
|
|
className="input w-full max-w-xs"
|
|
value={gpuId}
|
|
onChange={handleGpuIdChange}
|
|
/>
|
|
</div>
|
|
<div className="flex flex-col gap-2">
|
|
<p>If you like what we do :)</p>
|
|
<a
|
|
href="https://buymeacoffee.com/fossisthefuture"
|
|
target="_blank">
|
|
<button className="btn-primary btn">Donate</button>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<ReactTooltip class="max-w-sm" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default LeftPaneImageSteps;
|