1
0
mirror of https://github.com/upscayl/upscayl.git synced 2025-03-01 16:20:32 +01:00
upscayl/renderer/pages/index.tsx

911 lines
31 KiB
TypeScript
Raw Normal View History

2023-09-03 14:46:48 +05:30
"use client";
import { useState, useEffect, useCallback, useMemo } from "react";
2024-01-16 13:03:43 +05:30
import COMMAND from "../../common/commands";
import { ReactCompareSlider } from "react-compare-slider";
2022-11-12 02:09:28 +05:30
import Header from "../components/Header";
import Footer from "../components/Footer";
2023-07-22 16:37:53 +05:30
import ProgressBar from "../components/upscayl-tab/view/ProgressBar";
import RightPaneInfo from "../components/upscayl-tab/view/RightPaneInfo";
import ImageOptions from "../components/upscayl-tab/view/ImageOptions";
import LeftPaneImageSteps from "../components/upscayl-tab/config/LeftPaneImageSteps";
2023-03-18 22:38:50 +05:30
import Tabs from "../components/Tabs";
2023-07-22 16:37:53 +05:30
import SettingsTab from "../components/settings-tab";
2023-11-22 21:24:02 +05:30
import { useAtom, useAtomValue } from "jotai";
2023-04-08 12:23:32 +05:30
import { logAtom } from "../atoms/logAtom";
2023-04-09 11:18:53 +05:30
import { modelsListAtom } from "../atoms/modelsListAtom";
2023-09-03 14:46:48 +05:30
import {
batchModeAtom,
2024-01-23 14:14:32 +05:30
lensSizeAtom,
2023-11-22 21:27:44 +05:30
compressionAtom,
2023-09-03 14:46:48 +05:30
dontShowCloudModalAtom,
2023-11-22 21:24:02 +05:30
noImageProcessingAtom,
2024-04-09 23:41:24 +05:30
savedOutputPathAtom,
overwriteAtom,
2023-11-10 17:11:35 +05:30
progressAtom,
2023-09-03 14:46:48 +05:30
scaleAtom,
2024-01-23 14:14:32 +05:30
viewTypeAtom,
2024-02-14 12:02:52 +05:30
rememberOutputFolderAtom,
2024-04-17 21:48:45 +05:30
showSidebarAtom,
2024-04-20 21:14:42 +05:30
customWidthAtom,
useCustomWidthAtom,
tileSizeAtom,
2023-09-03 14:46:48 +05:30
} from "../atoms/userSettingsAtom";
2023-04-30 07:14:47 +05:30
import useLog from "../components/hooks/useLog";
import { UpscaylCloudModal } from "../components/UpscaylCloudModal";
2023-10-25 17:14:22 +05:30
import { featureFlags } from "@common/feature-flags";
2023-11-22 21:24:02 +05:30
import {
BatchUpscaylPayload,
DoubleUpscaylPayload,
ImageUpscaylPayload,
} from "@common/types/types";
2023-11-23 14:55:55 +05:30
import { NewsModal } from "@/components/NewsModal";
2023-11-26 13:34:03 +05:30
import { newsAtom, showNewsModalAtom } from "@/atoms/newsAtom";
import matter from "gray-matter";
import { ChevronLeftIcon, ChevronRightIcon } from "lucide-react";
2024-04-17 21:48:45 +05:30
import { cn } from "@/lib/utils";
2024-04-21 19:34:59 +05:30
import { useToast } from "@/components/ui/use-toast";
import { ToastAction } from "@/components/ui/toast";
2024-04-24 16:38:27 +05:30
import Logo from "@/components/icons/Logo";
import { sanitizePath } from "@common/sanitize-path";
import getDirectoryFromPath from "@common/get-directory-from-path";
import { translationAtom } from "@/atoms/translations-atom";
2023-03-31 16:03:48 +05:30
2022-11-12 02:09:28 +05:30
const Home = () => {
2023-11-23 14:23:09 +05:30
const allowedFileTypes = ["png", "jpg", "jpeg", "webp"];
const t = useAtomValue(translationAtom);
2023-11-22 21:24:02 +05:30
// LOCAL STATES
2023-08-14 16:26:11 +05:30
const [os, setOs] = useState<"linux" | "mac" | "win" | undefined>(undefined);
2023-11-23 14:23:09 +05:30
const [imagePath, setImagePath] = useState("");
const [upscaledImagePath, setUpscaledImagePath] = useState("");
2022-12-24 12:47:54 +05:30
const [model, setModel] = useState("realesrgan-x4plus");
2022-11-12 02:09:28 +05:30
const [version, setVersion] = useState("");
const [batchFolderPath, setBatchFolderPath] = useState("");
const [doubleUpscayl, setDoubleUpscayl] = useState(false);
const overwrite = useAtomValue(overwriteAtom);
2023-11-22 21:24:02 +05:30
const [upscaledBatchFolderPath, setUpscaledBatchFolderPath] = useState("");
2022-12-02 19:51:42 +05:30
const [doubleUpscaylCounter, setDoubleUpscaylCounter] = useState(0);
2022-12-16 21:50:46 +05:30
const [gpuId, setGpuId] = useState("");
const [saveImageAs, setSaveImageAs] = useState("png");
2024-01-23 14:14:32 +05:30
const [zoomAmount, setZoomAmount] = useState("100");
2022-12-21 16:02:45 +05:30
const [backgroundPosition, setBackgroundPosition] = useState("0% 0%");
2022-12-24 12:47:54 +05:30
const [dimensions, setDimensions] = useState({
width: null,
height: null,
});
2023-03-18 22:38:50 +05:30
const [selectedTab, setSelectedTab] = useState(0);
2023-11-22 21:24:02 +05:30
const [isLoading, setIsLoading] = useState(true);
const [showCloudModal, setShowCloudModal] = useState(false);
2024-04-17 21:48:45 +05:30
const [minSize, setMinSize] = useState(22);
2024-01-23 14:14:32 +05:30
const [cursorPosition, setCursorPosition] = useState({ x: 0, y: 0 });
2023-11-22 21:24:02 +05:30
// ATOMIC STATES
2024-04-09 23:41:24 +05:30
const [outputPath, setOutputPath] = useAtom(savedOutputPathAtom);
2023-11-22 21:27:44 +05:30
const [compression, setCompression] = useAtom(compressionAtom);
2023-11-22 21:24:02 +05:30
const [progress, setProgress] = useAtom(progressAtom);
const [batchMode, setBatchMode] = useAtom(batchModeAtom);
2023-04-08 12:23:32 +05:30
const [logData, setLogData] = useAtom(logAtom);
2023-04-09 11:18:53 +05:30
const [modelOptions, setModelOptions] = useAtom(modelsListAtom);
2023-04-14 16:01:37 +05:30
const [scale] = useAtom(scaleAtom);
2023-09-03 14:46:48 +05:30
const [dontShowCloudModal, setDontShowCloudModal] = useAtom(
2024-02-07 06:53:19 +05:30
dontShowCloudModalAtom,
2023-09-03 14:46:48 +05:30
);
2023-11-22 21:24:02 +05:30
const noImageProcessing = useAtomValue(noImageProcessingAtom);
2023-11-23 11:28:29 +05:30
const [news, setNews] = useAtom(newsAtom);
2023-11-26 13:34:03 +05:30
const [showNewsModal, setShowNewsModal] = useAtom(showNewsModalAtom);
2024-01-23 14:14:32 +05:30
const viewType = useAtomValue(viewTypeAtom);
const lensSize = useAtomValue(lensSizeAtom);
2024-02-14 12:02:52 +05:30
const rememberOutputFolder = useAtomValue(rememberOutputFolderAtom);
2024-04-17 21:48:45 +05:30
const [showSidebar, setShowSidebar] = useAtom(showSidebarAtom);
2024-04-20 21:14:42 +05:30
const customWidth = useAtomValue(customWidthAtom);
const useCustomWidth = useAtomValue(useCustomWidthAtom);
const tileSize = useAtomValue(tileSizeAtom);
2023-04-30 07:14:47 +05:30
const { logit } = useLog();
2024-04-21 19:34:59 +05:30
const { toast } = useToast();
2023-04-30 07:14:47 +05:30
const sanitizedImagePath = useMemo(
() => sanitizePath(imagePath),
[imagePath],
);
const sanitizedUpscaledImagePath = useMemo(
() => sanitizePath(upscaledImagePath),
[upscaledImagePath],
);
2024-01-23 14:14:32 +05:30
const handleMouseMoveCompare = (e: React.MouseEvent) => {
const { left, top, height, width } =
e.currentTarget.getBoundingClientRect();
const x = e.clientX - left;
const y = e.clientY - top;
setCursorPosition({ x, y });
const xZoom = ((e.pageX - left) / width) * 100;
const yZoom = ((e.pageY - top) / height) * 100;
setBackgroundPosition(`${xZoom}% ${yZoom}%`);
};
2023-11-23 14:23:09 +05:30
// SET CONFIG VARIABLES ON FIRST RUN
2022-11-12 02:09:28 +05:30
useEffect(() => {
2023-11-23 14:23:09 +05:30
// UPSCAYL VERSION
const upscaylVersion = navigator?.userAgent?.match(
2024-02-07 06:53:19 +05:30
/Upscayl\/([\d\.]+\d+)/,
2023-11-23 14:23:09 +05:30
)[1];
setVersion(upscaylVersion);
}, []);
2023-11-23 14:55:55 +05:30
// ELECTRON EVENT LISTENERS
2023-11-23 14:23:09 +05:30
useEffect(() => {
2023-04-08 12:23:32 +05:30
const handleErrors = (data: string) => {
2024-04-21 19:34:59 +05:30
if (data.includes("Invalid GPU")) {
toast({
title: t("ERRORS.GPU_ERROR.TITLE"),
description: t("ERRORS.GPU_ERROR.DESCRIPTION", { data }),
2024-04-21 19:34:59 +05:30
action: (
2024-04-22 13:25:44 +05:30
<div className="flex flex-col gap-2">
<ToastAction
altText={t("ERRORS.COPY_ERROR.TITLE")}
2024-04-22 13:25:44 +05:30
onClick={() => {
navigator.clipboard.writeText(data);
}}
>
{t("ERRORS.COPY_ERROR.TITLE")}
2024-04-22 13:25:44 +05:30
</ToastAction>
2024-04-25 01:58:03 +05:30
<a href="https://docs.upscayl.org/" target="_blank">
<ToastAction altText={t("ERRORS.OPEN_DOCS_TITLE")}>
{t("ERRORS.OPEN_DOCS_BUTTON_TITLE")}
</ToastAction>
2024-04-22 13:25:44 +05:30
</a>
</div>
2024-04-21 19:34:59 +05:30
),
});
2022-11-12 02:09:28 +05:30
resetImagePaths();
2024-04-21 19:34:59 +05:30
} else if (data.includes("write") || data.includes("read")) {
2022-11-12 02:09:28 +05:30
if (batchMode) return;
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.READ_WRITE_ERROR.TITLE"),
description: t("ERRORS.READ_WRITE_ERROR.DESCRIPTION", { data }),
2024-04-21 19:34:59 +05:30
action: (
2024-04-22 13:25:44 +05:30
<div className="flex flex-col gap-2">
<ToastAction
altText="Copy Error"
onClick={() => {
navigator.clipboard.writeText(data);
}}
>
{t("ERRORS.COPY_ERROR.TITLE")}
2024-04-22 13:25:44 +05:30
</ToastAction>
2024-04-25 01:58:03 +05:30
<a href="https://docs.upscayl.org/" target="_blank">
<ToastAction altText={t("ERRORS.OPEN_DOCS_TITLE")}>
{t("ERRORS.OPEN_DOCS_BUTTON_TITLE")}
</ToastAction>
2024-04-22 13:25:44 +05:30
</a>
</div>
2024-04-21 19:34:59 +05:30
),
});
resetImagePaths();
} else if (data.includes("tile size")) {
toast({
title: t("ERRORS.TILE_SIZE_ERROR.TITLE"),
description: t("ERRORS.TILE_SIZE_ERROR.DESCRIPTION", { data }),
2024-04-21 19:34:59 +05:30
});
2022-11-12 02:09:28 +05:30
resetImagePaths();
} else if (data.includes("uncaughtException")) {
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.EXCEPTION_ERROR.TITLE"),
description: t("ERRORS.EXCEPTION_ERROR.DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
2022-11-12 02:09:28 +05:30
resetImagePaths();
}
};
2023-11-23 14:23:09 +05:30
// OS
2023-08-14 16:25:30 +05:30
window.electron.on(
2023-09-10 23:12:18 +05:30
COMMAND.OS,
2023-08-14 16:25:30 +05:30
(_, data: "linux" | "mac" | "win" | undefined) => {
if (data) {
setOs(data);
}
2024-02-07 06:53:19 +05:30
},
2023-08-14 16:25:30 +05:30
);
2023-04-14 16:01:37 +05:30
// LOG
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.LOG, (_, data: string) => {
2024-04-25 16:13:16 +05:30
logit(`🎒 BACKEND REPORTED: `, data);
2023-04-14 16:01:37 +05:30
});
2023-11-23 14:23:09 +05:30
// SCALING AND CONVERTING
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.SCALING_AND_CONVERTING, (_, data: string) => {
setProgress(t("APP.PROGRESS.PROCESSING_TITLE"));
2023-09-09 17:48:00 +05:30
});
2023-11-23 14:23:09 +05:30
// UPSCAYL ERROR
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.UPSCAYL_ERROR, (_, data: string) => {
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.GENERIC_ERROR.TITLE"),
2024-04-21 19:34:59 +05:30
description: data,
});
2023-09-09 16:33:16 +05:30
resetImagePaths();
});
2022-12-16 21:50:46 +05:30
// UPSCAYL PROGRESS
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.UPSCAYL_PROGRESS, (_, data: string) => {
2022-11-12 02:09:28 +05:30
if (data.length > 0 && data.length < 10) {
setProgress(data);
2023-09-04 21:29:17 +05:30
} else if (data.includes("converting")) {
setProgress(t("APP.PROGRESS.SCALING_CONVERTING_TITLE"));
2023-10-21 19:44:42 +05:30
} else if (data.includes("Successful")) {
setProgress(t("APP.PROGRESS.SUCCESS_TITLE"));
2022-11-12 02:09:28 +05:30
}
handleErrors(data);
logit(`🚧 UPSCAYL_PROGRESS: `, data);
2022-11-12 02:09:28 +05:30
});
2022-12-16 21:50:46 +05:30
// FOLDER UPSCAYL PROGRESS
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.FOLDER_UPSCAYL_PROGRESS, (_, data: string) => {
2023-10-21 21:20:59 +05:30
if (data.includes("Successful")) {
setProgress(t("APP.PROGRESS.SUCCESS_TITLE"));
2023-10-21 21:20:59 +05:30
}
2022-11-12 02:09:28 +05:30
if (data.length > 0 && data.length < 10) {
setProgress(data);
}
handleErrors(data);
logit(`🚧 FOLDER_UPSCAYL_PROGRESS: `, data);
2022-11-12 02:09:28 +05:30
});
2022-12-16 21:50:46 +05:30
// DOUBLE UPSCAYL PROGRESS
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.DOUBLE_UPSCAYL_PROGRESS, (_, data: string) => {
2022-11-12 02:09:28 +05:30
if (data.length > 0 && data.length < 10) {
2022-12-02 19:51:42 +05:30
if (data === "0.00%") {
setDoubleUpscaylCounter(doubleUpscaylCounter + 1);
}
2022-11-12 02:09:28 +05:30
setProgress(data);
}
handleErrors(data);
logit(`🚧 DOUBLE_UPSCAYL_PROGRESS: `, data);
2022-11-12 02:09:28 +05:30
});
2022-12-16 21:50:46 +05:30
// UPSCAYL DONE
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.UPSCAYL_DONE, (_, data: string) => {
2022-11-12 02:09:28 +05:30
setProgress("");
2023-09-13 21:45:42 +05:30
setUpscaledImagePath(data);
2023-07-22 18:30:46 +05:30
logit("upscaledImagePath: ", data);
logit(`💯 UPSCAYL_DONE: `, data);
2022-11-12 02:09:28 +05:30
});
2022-12-16 21:50:46 +05:30
// FOLDER UPSCAYL DONE
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.FOLDER_UPSCAYL_DONE, (_, data: string) => {
2022-11-12 02:09:28 +05:30
setProgress("");
setUpscaledBatchFolderPath(data);
logit(`💯 FOLDER_UPSCAYL_DONE: `, data);
2022-11-12 02:09:28 +05:30
});
2022-12-16 21:50:46 +05:30
// DOUBLE UPSCAYL DONE
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.DOUBLE_UPSCAYL_DONE, (_, data: string) => {
2022-12-02 19:51:42 +05:30
setProgress("");
2023-09-11 09:27:16 +05:30
setTimeout(() => setUpscaledImagePath(data), 500);
2022-12-02 19:51:42 +05:30
setDoubleUpscaylCounter(0);
logit(`💯 DOUBLE_UPSCAYL_DONE: `, data);
2022-11-12 02:09:28 +05:30
});
2023-04-09 11:18:53 +05:30
// CUSTOM FOLDER LISTENER
2023-09-10 23:12:18 +05:30
window.electron.on(COMMAND.CUSTOM_MODEL_FILES_LIST, (_, data: string[]) => {
logit(`📜 CUSTOM_MODEL_FILES_LIST: `, data);
const newModelOptions = data.map((model) => {
return {
value: model,
label: model,
};
});
// Add newModelsList to modelOptions and remove duplicates
const combinedModelOptions = [...modelOptions, ...newModelOptions];
const uniqueModelOptions = combinedModelOptions.filter(
// Check if any model in the array appears more than once
(model, index, array) =>
2024-02-07 06:53:19 +05:30
array.findIndex((t) => t.value === model.value) === index,
2023-09-10 23:12:18 +05:30
);
setModelOptions(uniqueModelOptions);
});
2023-04-09 11:18:53 +05:30
}, []);
2023-11-23 14:23:09 +05:30
// FETCH CUSTOM MODELS FROM CUSTOM MODELS PATH
2023-04-09 11:18:53 +05:30
useEffect(() => {
const customModelsPath = JSON.parse(
2024-02-07 06:53:19 +05:30
localStorage.getItem("customModelsPath"),
2023-04-09 11:18:53 +05:30
);
if (customModelsPath !== null) {
2023-09-10 23:12:18 +05:30
window.electron.send(COMMAND.GET_MODELS_LIST, customModelsPath);
logit("🎯 GET_MODELS_LIST: ", customModelsPath);
2023-04-09 11:18:53 +05:30
}
2022-11-12 02:09:28 +05:30
}, []);
2023-11-23 14:23:09 +05:30
// FETCH NEWS
2023-11-23 11:28:29 +05:30
useEffect(() => {
// TODO: ADD AN ABOUT TAB
if (window && window.navigator.onLine === false) return;
2023-12-03 12:06:00 +05:30
try {
fetch("https://raw.githubusercontent.com/upscayl/upscayl/main/news.md", {
cache: "no-cache",
2023-11-23 14:23:09 +05:30
})
2023-12-03 12:06:00 +05:30
.then((res) => {
return res.text();
})
.then((result) => {
const newsData = result;
if (!newsData) {
console.log("📰 Could not fetch news data");
return;
2023-11-26 13:34:03 +05:30
}
2023-12-03 12:06:00 +05:30
const markdownData = matter(newsData);
if (!markdownData) return;
if (markdownData && markdownData.data.dontShow) {
2023-12-03 12:06:00 +05:30
return;
}
if (
markdownData &&
news &&
markdownData?.data?.version === news?.data?.version
) {
console.log("📰 News is up to date");
if (showNewsModal === false) {
setShowNewsModal(false);
}
} else if (markdownData) {
setNews(matter(newsData));
setShowNewsModal(true);
}
});
} catch (error) {
console.log("Could not fetch Upscayl News");
}
2023-11-26 13:34:03 +05:30
}, [news]);
2023-11-23 11:28:29 +05:30
2023-11-23 14:23:09 +05:30
// LOADING STATE
2023-10-15 09:27:11 +05:30
useEffect(() => {
setIsLoading(false);
}, []);
// HANDLERS
2022-12-16 21:50:46 +05:30
const resetImagePaths = () => {
logit("🔄 Resetting image paths");
2022-12-24 12:47:54 +05:30
setDimensions({
width: null,
height: null,
});
2022-12-16 21:50:46 +05:30
setProgress("");
2023-11-23 14:23:09 +05:30
setImagePath("");
2022-12-16 21:50:46 +05:30
setUpscaledImagePath("");
setBatchFolderPath("");
setUpscaledBatchFolderPath("");
};
2023-11-23 14:23:09 +05:30
// UTILS
// CHECK IF IMAGE IS VALID
const validateImagePath = (path: string) => {
if (path.length > 0) {
logit("🖼 imagePath: ", path);
const extension = path.toLocaleLowerCase().split(".").pop();
logit("🔤 Extension: ", extension);
if (!allowedFileTypes.includes(extension.toLowerCase())) {
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.INVALID_IMAGE_ERROR.TITLE"),
description: t("ERRORS.INVALID_IMAGE_ERROR.DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
2023-11-23 14:23:09 +05:30
resetImagePaths();
}
} else {
resetImagePaths();
}
};
2022-12-16 21:50:46 +05:30
// HANDLERS
2023-04-08 12:23:32 +05:30
const handleMouseMove = useCallback((e: any) => {
2022-12-21 16:02:45 +05:30
const { left, top, width, height } = e.target.getBoundingClientRect();
const x = ((e.pageX - left) / width) * 100;
const y = ((e.pageY - top) / height) * 100;
setBackgroundPosition(`${x}% ${y}%`);
2023-04-08 12:23:32 +05:30
}, []);
2022-12-21 16:02:45 +05:30
2022-11-12 02:09:28 +05:30
const selectImageHandler = async () => {
resetImagePaths();
2023-09-10 23:12:18 +05:30
var path = await window.electron.invoke(COMMAND.SELECT_FILE);
2023-11-23 14:23:09 +05:30
if (path === null) return;
logit("🖼 Selected Image Path: ", path);
setImagePath(path);
var dirname = getDirectoryFromPath(path);
2023-11-23 14:23:09 +05:30
logit("📁 Selected Image Directory: ", dirname);
if (!featureFlags.APP_STORE_BUILD) {
2024-02-14 12:02:52 +05:30
if (!rememberOutputFolder) {
setOutputPath(dirname);
}
2022-11-12 02:09:28 +05:30
}
2023-11-23 14:23:09 +05:30
validateImagePath(path);
2022-11-12 02:09:28 +05:30
};
const selectFolderHandler = async () => {
resetImagePaths();
2023-09-10 23:12:18 +05:30
var path = await window.electron.invoke(COMMAND.SELECT_FOLDER);
2023-04-09 10:46:15 +05:30
if (path !== null) {
logit("🖼 Selected Folder Path: ", path);
2022-11-12 02:09:28 +05:30
setBatchFolderPath(path);
2024-02-14 12:02:52 +05:30
if (!rememberOutputFolder) {
setOutputPath(path);
}
2023-05-01 12:04:45 +05:30
} else {
logit("🚫 Folder selection cancelled");
2023-05-01 12:04:45 +05:30
setBatchFolderPath("");
2024-02-14 12:02:52 +05:30
if (!rememberOutputFolder) {
setOutputPath("");
}
2022-11-12 02:09:28 +05:30
}
};
2023-04-08 12:23:32 +05:30
const handleModelChange = (e: any) => {
2022-11-12 02:09:28 +05:30
setModel(e.value);
logit("🔀 Model changed: ", e.value);
2022-12-24 14:15:15 +05:30
localStorage.setItem(
"model",
2024-02-07 06:53:19 +05:30
JSON.stringify({ label: e.label, value: e.value }),
2022-12-24 14:15:15 +05:30
);
2022-11-12 02:09:28 +05:30
};
2022-12-16 21:50:46 +05:30
// DRAG AND DROP HANDLERS
2022-11-12 02:09:28 +05:30
const handleDragEnter = (e) => {
e.preventDefault();
console.log("drag enter");
};
const handleDragLeave = (e) => {
e.preventDefault();
console.log("drag leave");
};
const handleDragOver = (e) => {
e.preventDefault();
console.log("drag over");
};
const openFolderHandler = (e) => {
logit("📂 OPEN_FOLDER: ", upscaledBatchFolderPath);
2023-09-10 23:12:18 +05:30
window.electron.send(COMMAND.OPEN_FOLDER, upscaledBatchFolderPath);
2022-11-12 02:09:28 +05:30
};
const handleDrop = (e) => {
e.preventDefault();
resetImagePaths();
2023-04-21 00:04:49 +05:30
if (
e.dataTransfer.items.length === 0 ||
e.dataTransfer.files.length === 0
) {
logit("👎 No valid files dropped");
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.INVALID_IMAGE_ERROR.TITLE"),
description: t("ERRORS.INVALID_IMAGE_ERROR.ADDITIONAL_DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
2023-04-21 00:04:49 +05:30
return;
}
2022-11-12 02:09:28 +05:30
const type = e.dataTransfer.items[0].type;
const filePath = e.dataTransfer.files[0].path;
const extension = e.dataTransfer.files[0].name.split(".").at(-1);
logit("⤵️ Dropped file: ", JSON.stringify({ type, filePath, extension }));
2022-11-12 02:09:28 +05:30
if (
2023-08-14 16:25:30 +05:30
!type.includes("image") ||
!allowedFileTypes.includes(extension.toLowerCase())
2022-11-12 02:09:28 +05:30
) {
logit("🚫 Invalid file dropped");
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.INVALID_IMAGE_ERROR.TITLE"),
description: t("ERRORS.INVALID_IMAGE_ERROR.ADDITIONAL_DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
2022-11-12 02:09:28 +05:30
} else {
2023-08-14 16:25:30 +05:30
logit("🖼 Setting image path: ", filePath);
2023-11-23 14:23:09 +05:30
setImagePath(filePath);
var dirname = getDirectoryFromPath(filePath);
logit("🗂 Setting output path: ", dirname);
2024-02-14 12:02:52 +05:30
if (!featureFlags.APP_STORE_BUILD) {
if (!rememberOutputFolder) {
setOutputPath(dirname);
}
}
2023-11-23 14:23:09 +05:30
validateImagePath(filePath);
2022-11-12 02:09:28 +05:30
}
};
const handlePaste = (e) => {
resetImagePaths();
e.preventDefault();
const type = e.clipboardData.items[0].type;
const filePath = e.clipboardData.files[0].path;
const extension = e.clipboardData.files[0].name.split(".").at(-1);
logit("📋 Pasted file: ", JSON.stringify({ type, filePath, extension }));
2022-11-12 02:09:28 +05:30
if (
!type.includes("image") &&
!allowedFileTypes.includes(extension.toLowerCase())
) {
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.INVALID_IMAGE_ERROR.TITLE"),
description: t("ERRORS.INVALID_IMAGE_ERROR.ADDITIONAL_DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
2022-11-12 02:09:28 +05:30
} else {
2023-11-23 14:23:09 +05:30
setImagePath(filePath);
var dirname = getDirectoryFromPath(filePath);
logit("🗂 Setting output path: ", dirname);
2024-02-14 12:02:52 +05:30
if (!rememberOutputFolder) {
setOutputPath(dirname);
2023-04-28 18:17:51 +05:30
}
2022-11-12 02:09:28 +05:30
}
};
const upscaylHandler = async () => {
2023-08-14 16:25:30 +05:30
logit("🔄 Resetting Upscaled Image Path");
setUpscaledImagePath("");
2023-09-09 18:31:31 +05:30
setUpscaledBatchFolderPath("");
2023-08-14 16:25:30 +05:30
if (imagePath !== "" || batchFolderPath !== "") {
setProgress(t("APP.PROGRESS.WAIT_TITLE"));
2023-11-22 21:24:02 +05:30
// Double Upscayl
2022-11-12 02:09:28 +05:30
if (doubleUpscayl) {
2023-11-22 21:24:02 +05:30
window.electron.send<DoubleUpscaylPayload>(COMMAND.DOUBLE_UPSCAYL, {
2022-11-12 02:09:28 +05:30
imagePath,
outputPath,
model,
2022-12-16 21:50:46 +05:30
gpuId: gpuId.length === 0 ? null : gpuId,
saveImageAs,
2023-04-14 16:01:37 +05:30
scale,
2023-11-22 21:24:02 +05:30
noImageProcessing,
2023-11-23 11:09:46 +05:30
compression: compression.toString(),
customWidth: customWidth > 0 ? customWidth.toString() : null,
2024-04-20 21:14:42 +05:30
useCustomWidth,
tileSize,
2022-11-12 02:09:28 +05:30
});
logit("🏁 DOUBLE_UPSCAYL");
2022-11-12 02:09:28 +05:30
} else if (batchMode) {
2023-11-22 21:24:02 +05:30
// Batch Upscayl
2022-11-12 02:09:28 +05:30
setDoubleUpscayl(false);
2023-11-22 21:24:02 +05:30
window.electron.send<BatchUpscaylPayload>(COMMAND.FOLDER_UPSCAYL, {
2022-11-12 02:09:28 +05:30
batchFolderPath,
outputPath,
model,
2022-12-16 21:50:46 +05:30
gpuId: gpuId.length === 0 ? null : gpuId,
saveImageAs,
2023-04-14 16:01:37 +05:30
scale,
2023-11-22 21:24:02 +05:30
noImageProcessing,
2023-11-23 11:09:46 +05:30
compression: compression.toString(),
customWidth: customWidth > 0 ? customWidth.toString() : null,
2024-04-20 21:14:42 +05:30
useCustomWidth,
tileSize,
2022-11-12 02:09:28 +05:30
});
logit("🏁 FOLDER_UPSCAYL");
2022-11-12 02:09:28 +05:30
} else {
2023-11-22 21:24:02 +05:30
// Single Image Upscayl
window.electron.send<ImageUpscaylPayload>(COMMAND.UPSCAYL, {
2022-11-12 02:09:28 +05:30
imagePath,
outputPath,
model,
2022-12-16 21:50:46 +05:30
gpuId: gpuId.length === 0 ? null : gpuId,
saveImageAs,
2023-04-14 16:01:37 +05:30
scale,
2023-08-27 21:41:01 +05:30
overwrite,
2023-11-22 21:24:02 +05:30
noImageProcessing,
2023-11-23 11:09:46 +05:30
compression: compression.toString(),
customWidth: customWidth > 0 ? customWidth.toString() : null,
2024-04-20 21:14:42 +05:30
useCustomWidth,
tileSize,
2022-11-12 02:09:28 +05:30
});
logit("🏁 UPSCAYL");
2022-11-12 02:09:28 +05:30
}
2023-11-22 21:24:02 +05:30
} else {
2024-04-21 19:34:59 +05:30
toast({
title: t("ERRORS.NO_IMAGE_ERROR.TITLE"),
description: t("ERRORS.NO_IMAGE_ERROR.DESCRIPTION"),
2024-04-21 19:34:59 +05:30
});
logit("🚫 No valid image selected");
2022-11-12 02:09:28 +05:30
}
};
2023-04-28 23:51:42 +05:30
const stopHandler = () => {
2023-09-10 23:12:18 +05:30
window.electron.send(COMMAND.STOP);
logit("🛑 Stopping Upscayl");
2023-04-28 23:51:42 +05:30
resetImagePaths();
};
2023-10-15 09:27:11 +05:30
if (isLoading) {
2023-08-30 10:24:16 +05:30
return (
2024-04-24 16:38:27 +05:30
<Logo className="absolute left-1/2 top-1/2 w-36 -translate-x-1/2 -translate-y-1/2 animate-pulse" />
2023-08-30 10:24:16 +05:30
);
}
2022-11-12 02:09:28 +05:30
return (
2022-11-12 03:02:24 +05:30
<div className="flex h-screen w-screen flex-row overflow-hidden bg-base-300">
2024-04-17 21:48:45 +05:30
{/* TOP LOGO WHEN SIDEBAR IS HIDDEN */}
{!showSidebar && (
<div className="fixed right-2 top-2 z-50 flex items-center justify-center gap-2 rounded-[7px] bg-base-300 px-2 py-1 font-medium text-base-content ">
2024-04-24 16:38:27 +05:30
<Logo className="w-5" />
{t("TITLE")}
2024-04-17 21:48:45 +05:30
</div>
)}
{/* SIDEBAR BUTTON */}
<button
className={cn(
2024-04-22 13:25:44 +05:30
"fixed left-0 top-1/2 z-[999] -translate-y-1/2 rounded-r-full bg-base-100 p-4 ",
2024-04-17 21:48:45 +05:30
showSidebar ? "hidden" : "",
)}
onClick={() => setShowSidebar((prev) => !prev)}
>
2024-04-22 13:25:44 +05:30
<ChevronRightIcon />
2024-04-17 21:48:45 +05:30
</button>
2024-04-19 00:15:12 +05:30
2024-04-17 21:48:45 +05:30
{/* LEFT PANE */}
<div
className={`relative flex h-screen min-w-[350px] max-w-[350px] flex-col bg-base-100 ${showSidebar ? "" : "hidden"}`}
>
<button
2024-04-22 13:25:44 +05:30
className="absolute -right-0 top-1/2 z-[999] -translate-y-1/2 translate-x-1/2 rounded-full bg-base-100 p-4"
2024-04-17 21:48:45 +05:30
onClick={() => setShowSidebar((prev) => !prev)}
>
2024-04-22 13:25:44 +05:30
<ChevronLeftIcon />
2024-04-17 21:48:45 +05:30
</button>
2024-04-19 00:15:12 +05:30
2024-04-17 21:48:45 +05:30
{/* UPSCAYL CLOUD MODAL */}
2023-11-02 20:33:21 +05:30
{featureFlags.SHOW_UPSCAYL_CLOUD_INFO && (
<UpscaylCloudModal
show={showCloudModal}
setShow={setShowCloudModal}
setDontShowCloudModal={setDontShowCloudModal}
/>
)}
2024-04-17 21:48:45 +05:30
{/* MACOS TITLEBAR */}
2023-08-30 10:24:16 +05:30
{window.electron.platform === "mac" && (
2024-02-07 06:53:19 +05:30
<div className="mac-titlebar pt-8"></div>
2023-08-30 10:24:16 +05:30
)}
2022-11-12 02:09:28 +05:30
{/* HEADER */}
2023-08-30 13:28:15 +05:30
<Header version={version} />
2023-11-02 20:33:21 +05:30
{!dontShowCloudModal && featureFlags.SHOW_UPSCAYL_CLOUD_INFO && (
2023-09-03 14:46:48 +05:30
<button
2024-04-19 03:28:47 +05:30
className="mx-5 mb-5 animate-pulse rounded-btn bg-success p-1 text-sm text-slate-50 shadow-lg shadow-success/40"
2023-09-03 14:46:48 +05:30
onClick={() => {
setShowCloudModal(true);
2024-02-07 06:53:19 +05:30
}}
>
{t("INTRO")}
2023-09-03 14:46:48 +05:30
</button>
)}
2023-08-30 11:05:40 +05:30
2024-04-17 21:48:45 +05:30
{/* NEWS DIALOG */}
2023-11-23 14:55:55 +05:30
<NewsModal
show={showNewsModal}
setShow={(val: boolean) => {
setShowNewsModal(val);
setNews((prev) => ({ ...prev, seen: true }));
}}
news={news}
/>
2023-03-18 22:38:50 +05:30
<Tabs selectedTab={selectedTab} setSelectedTab={setSelectedTab} />
2023-08-14 16:25:30 +05:30
2023-03-18 22:38:50 +05:30
{selectedTab === 0 && (
2022-11-23 23:54:30 +05:30
<LeftPaneImageSteps
selectImageHandler={selectImageHandler}
selectFolderHandler={selectFolderHandler}
handleModelChange={handleModelChange}
upscaylHandler={upscaylHandler}
batchMode={batchMode}
setBatchMode={setBatchMode}
imagePath={imagePath}
doubleUpscayl={doubleUpscayl}
setDoubleUpscayl={setDoubleUpscayl}
2022-12-24 12:47:54 +05:30
dimensions={dimensions}
2023-05-01 14:53:11 +05:30
setGpuId={setGpuId}
2023-09-19 20:44:15 +05:30
model={model}
2023-05-01 14:53:11 +05:30
setModel={setModel}
setSaveImageAs={setSaveImageAs}
2022-11-23 23:54:30 +05:30
/>
)}
2023-03-18 22:38:50 +05:30
{selectedTab === 1 && (
<SettingsTab
batchMode={batchMode}
setModel={setModel}
2023-09-13 19:37:45 +05:30
compression={compression}
setCompression={setCompression}
2023-03-18 22:38:50 +05:30
gpuId={gpuId}
setGpuId={setGpuId}
saveImageAs={saveImageAs}
setSaveImageAs={setSaveImageAs}
2023-04-08 12:23:32 +05:30
logData={logData}
2023-08-14 16:25:30 +05:30
os={os}
2023-09-03 14:46:48 +05:30
show={showCloudModal}
setShow={setShowCloudModal}
setDontShowCloudModal={setDontShowCloudModal}
2023-03-18 22:38:50 +05:30
/>
)}
{/* )} */}
2022-11-12 02:09:28 +05:30
<Footer />
</div>
{/* RIGHT PANE */}
<div
className="relative flex h-screen w-full flex-col items-center justify-center"
onDrop={(e) => handleDrop(e)}
onDragOver={(e) => handleDragOver(e)}
onDragEnter={(e) => handleDragEnter(e)}
onDragLeave={(e) => handleDragLeave(e)}
2023-11-10 18:53:55 +05:30
onDoubleClick={() => {
if (batchMode) {
selectFolderHandler();
} else {
selectImageHandler();
}
}}
2024-02-07 06:53:19 +05:30
onPaste={(e) => handlePaste(e)}
>
2023-08-30 10:24:16 +05:30
{window.electron.platform === "mac" && (
2024-02-07 06:53:19 +05:30
<div className="mac-titlebar absolute top-0 h-8 w-full"></div>
2023-08-30 10:24:16 +05:30
)}
2022-11-12 02:09:28 +05:30
{progress.length > 0 &&
upscaledImagePath.length === 0 &&
2023-08-14 16:25:30 +05:30
upscaledBatchFolderPath.length === 0 ? (
2022-12-02 19:51:42 +05:30
<ProgressBar
batchMode={batchMode}
2022-12-02 19:51:42 +05:30
progress={progress}
doubleUpscaylCounter={doubleUpscaylCounter}
2023-04-28 23:51:42 +05:30
stopHandler={stopHandler}
2022-12-02 19:51:42 +05:30
/>
2022-11-12 02:09:28 +05:30
) : null}
2022-11-25 13:00:03 +05:30
{/* DEFAULT PANE INFO */}
2023-08-14 16:25:30 +05:30
{((!batchMode &&
imagePath.length === 0 &&
2022-11-12 02:09:28 +05:30
upscaledImagePath.length === 0) ||
2023-08-14 16:25:30 +05:30
(batchMode &&
batchFolderPath.length === 0 &&
2023-08-14 16:25:30 +05:30
upscaledBatchFolderPath.length === 0)) && (
<RightPaneInfo version={version} batchMode={batchMode} />
2022-11-12 02:09:28 +05:30
)}
2022-11-25 13:00:03 +05:30
{/* SHOW SELECTED IMAGE */}
2022-11-12 02:09:28 +05:30
{!batchMode &&
upscaledImagePath.length === 0 &&
imagePath.length > 0 && (
2022-12-24 13:10:45 +05:30
<>
<ImageOptions
zoomAmount={zoomAmount}
setZoomAmount={setZoomAmount}
resetImagePaths={resetImagePaths}
hideZoomOptions={true}
/>
<img
src={"file:///" + sanitizePath(imagePath)}
2022-12-24 13:10:45 +05:30
onLoad={(e: any) => {
setDimensions({
width: e.target.naturalWidth,
height: e.target.naturalHeight,
});
}}
draggable="false"
alt=""
2023-10-14 06:30:49 +05:30
className="h-full w-full bg-gradient-to-br from-base-300 to-base-100 object-contain"
2022-12-24 13:10:45 +05:30
/>
</>
2022-11-12 02:09:28 +05:30
)}
2022-11-25 13:00:03 +05:30
{/* BATCH UPSCALE SHOW SELECTED FOLDER */}
2022-11-12 02:09:28 +05:30
{batchMode &&
upscaledBatchFolderPath.length === 0 &&
batchFolderPath.length > 0 && (
<p className="select-none text-base-content">
<span className="font-bold">
{t("APP.PROGRESS.BATCH.SELECTED_FOLDER_TITLE")}
</span>{" "}
2023-09-09 18:31:31 +05:30
{batchFolderPath}
2022-11-12 02:09:28 +05:30
</p>
)}
2022-11-25 13:00:03 +05:30
{/* BATCH UPSCALE DONE INFO */}
2022-11-12 02:09:28 +05:30
{batchMode && upscaledBatchFolderPath.length > 0 && (
2024-02-14 12:02:52 +05:30
<div className="z-50 flex flex-col items-center">
<p className="select-none py-4 font-bold text-base-content">
{t("APP.PROGRESS.BATCH.DONE_TITLE")}
2022-11-12 02:09:28 +05:30
</p>
<button
2024-02-07 06:53:19 +05:30
className="bg-gradient-blue btn btn-primary rounded-btn p-3 font-medium text-white/90 transition-colors"
onClick={openFolderHandler}
>
{t("APP.PROGRESS.BATCH.OPEN_UPSCAYLED_FOLDER_TITLE")}
2022-11-12 02:09:28 +05:30
</button>
2024-02-14 12:02:52 +05:30
</div>
2022-11-12 02:09:28 +05:30
)}
2024-01-23 14:14:32 +05:30
<ImageOptions
zoomAmount={zoomAmount}
setZoomAmount={setZoomAmount}
resetImagePaths={resetImagePaths}
/>
{!batchMode &&
viewType === "lens" &&
upscaledImagePath &&
imagePath && (
<div
2024-02-07 07:04:06 +05:30
className="group relative h-full w-full overflow-hidden"
2024-02-07 06:53:19 +05:30
onMouseMove={handleMouseMoveCompare}
>
2024-01-23 14:14:32 +05:30
<img
2024-02-07 06:53:19 +05:30
className={`absolute left-0 top-0 h-full w-full object-contain transition-transform group-hover:scale-[${
2024-01-23 14:14:32 +05:30
zoomAmount + "%"
}]`}
src={"file:///" + imagePath}
style={{
backgroundPosition: "0% 0%",
transformOrigin: backgroundPosition,
}}
/>
<div
2024-02-07 06:57:54 +05:30
className={`invisible absolute left-0 top-0 h-full w-full bg-white mix-blend-difference group-hover:visible group-hover:scale-[${
2024-01-23 14:14:32 +05:30
zoomAmount + "%"
}]`}
style={{
clipPath: `circle(${
2024-02-07 07:04:06 +05:30
(lensSize + 2 * (parseInt(zoomAmount) / 100)) /
(parseInt(zoomAmount) / 100)
2024-01-23 14:14:32 +05:30
}px at ${cursorPosition.x}px ${cursorPosition.y}px)`,
backgroundPosition: "0% 0%",
transformOrigin: backgroundPosition,
}}
/>
<img
2024-02-07 06:53:19 +05:30
className={`absolute h-full w-full object-contain transition-transform group-hover:scale-[${
2024-01-23 14:14:32 +05:30
zoomAmount + "%"
}]`}
src={"file:///" + upscaledImagePath}
style={{
clipPath: `circle(${
2024-02-07 07:04:06 +05:30
(lensSize + parseInt(zoomAmount) / 100) /
2024-02-07 06:53:19 +05:30
(parseInt(zoomAmount) / 100)
2024-01-23 14:14:32 +05:30
}px at ${cursorPosition.x}px ${cursorPosition.y}px)`,
2024-02-07 06:53:19 +05:30
backgroundPosition: backgroundPosition,
2024-01-23 14:14:32 +05:30
transformOrigin: backgroundPosition,
}}
/>
</div>
)}
2022-11-25 13:00:03 +05:30
{/* COMPARISON SLIDER */}
2024-01-23 14:14:32 +05:30
{!batchMode &&
viewType === "slider" &&
imagePath.length > 0 &&
upscaledImagePath.length > 0 && (
<>
<ReactCompareSlider
itemOne={
<>
<p className="absolute bottom-1 left-1 rounded-md bg-black p-1 text-sm font-medium text-white opacity-30">
{t("APP.SLIDER.ORIGINAL_TITLE")}
2024-01-23 14:14:32 +05:30
</p>
<img
/* USE REGEX TO GET THE FILENAME AND ENCODE IT INTO PROPER FORM IN ORDER TO AVOID ERRORS DUE TO SPECIAL CHARACTERS */
src={"file:///" + sanitizedImagePath}
alt={t("APP.SLIDER.ORIGINAL_TITLE")}
2024-01-23 14:14:32 +05:30
onMouseMove={handleMouseMove}
style={{
objectFit: "contain",
backgroundPosition: "0% 0%",
transformOrigin: backgroundPosition,
}}
2024-01-23 14:24:08 +05:30
className={`h-full w-full bg-gradient-to-br from-base-300 to-base-100 transition-transform group-hover:scale-[${zoomAmount}%]`}
2024-01-23 14:14:32 +05:30
/>
</>
}
itemTwo={
<>
<p className="absolute bottom-1 right-1 rounded-md bg-black p-1 text-sm font-medium text-white opacity-30">
{t("APP.SLIDER.UPSCAYLED_TITLE")}
2024-01-23 14:14:32 +05:30
</p>
<img
/* USE REGEX TO GET THE FILENAME AND ENCODE IT INTO PROPER FORM IN ORDER TO AVOID ERRORS DUE TO SPECIAL CHARACTERS */
src={"file:///" + sanitizedUpscaledImagePath}
alt={t("APP.SLIDER.UPSCAYLED_TITLE")}
2024-01-23 14:14:32 +05:30
style={{
objectFit: "contain",
backgroundPosition: "0% 0%",
transformOrigin: backgroundPosition,
}}
onMouseMove={handleMouseMove}
className={`h-full w-full bg-gradient-to-br from-base-300 to-base-100 transition-transform group-hover:scale-[${
zoomAmount || "100%"
}%]`}
/>
</>
}
className="group h-screen"
/>
</>
)}
2022-11-12 02:09:28 +05:30
</div>
</div>
);
};
export default Home;