mirror of
https://github.com/upscayl/upscayl.git
synced 2025-01-19 01:24:09 +01:00
Update quality
This commit is contained in:
parent
30557a8dc7
commit
b7308a146b
@ -5,7 +5,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { join, dirname, resolve } from "path";
|
import { join, dirname, resolve } from "path";
|
||||||
import { getArch, getPlatform } from "./getDeviceSpecs";
|
import { getPlatform } from "./getDeviceSpecs";
|
||||||
import isDev from "electron-is-dev";
|
import isDev from "electron-is-dev";
|
||||||
import { app } from "electron";
|
import { app } from "electron";
|
||||||
|
|
||||||
|
@ -23,6 +23,8 @@ const commands = {
|
|||||||
LOG: "Log",
|
LOG: "Log",
|
||||||
STOP: "Stop the current operation",
|
STOP: "Stop the current operation",
|
||||||
OS: "Get OS",
|
OS: "Get OS",
|
||||||
|
SCALING_AND_CONVERTING: "Adding some finishing touches",
|
||||||
|
UPSCAYL_ERROR: "Upscaling Error",
|
||||||
};
|
};
|
||||||
|
|
||||||
export default commands;
|
export default commands;
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
// Native
|
// Native
|
||||||
import { autoUpdater } from "electron-updater";
|
import { autoUpdater } from "electron-updater";
|
||||||
import { getPlatform } from "./getDeviceSpecs";
|
import { getPlatform } from "./getDeviceSpecs";
|
||||||
import { join, parse } from "path";
|
import path, { join, parse } from "path";
|
||||||
import log from "electron-log";
|
import log from "electron-log";
|
||||||
import { format } from "url";
|
import { format } from "url";
|
||||||
import fs from "fs";
|
import fs from "fs";
|
||||||
|
import sharp from "sharp";
|
||||||
|
|
||||||
import { execPath, modelsPath } from "./binaries";
|
import { execPath, modelsPath } from "./binaries";
|
||||||
// Packages
|
// Packages
|
||||||
@ -29,7 +30,6 @@ import {
|
|||||||
getSingleImageArguments,
|
getSingleImageArguments,
|
||||||
} from "./utils/getArguments";
|
} from "./utils/getArguments";
|
||||||
import { spawnUpscayl } from "./upscayl";
|
import { spawnUpscayl } from "./upscayl";
|
||||||
import Jimp from "jimp";
|
|
||||||
|
|
||||||
let childProcesses: {
|
let childProcesses: {
|
||||||
process: ChildProcessWithoutNullStreams;
|
process: ChildProcessWithoutNullStreams;
|
||||||
@ -48,7 +48,7 @@ let folderPath: string | undefined = undefined;
|
|||||||
let customModelsFolderPath: string | undefined = undefined;
|
let customModelsFolderPath: string | undefined = undefined;
|
||||||
let outputFolderPath: string | undefined = undefined;
|
let outputFolderPath: string | undefined = undefined;
|
||||||
let saveOutputFolder = false;
|
let saveOutputFolder = false;
|
||||||
let quality = 100;
|
let quality = 0;
|
||||||
let overwrite = false;
|
let overwrite = false;
|
||||||
|
|
||||||
let stopped = false;
|
let stopped = false;
|
||||||
@ -162,7 +162,11 @@ app.on("ready", async () => {
|
|||||||
.executeJavaScript('localStorage.getItem("quality");', true)
|
.executeJavaScript('localStorage.getItem("quality");', true)
|
||||||
.then((lastSavedQuality: string | null) => {
|
.then((lastSavedQuality: string | null) => {
|
||||||
if (lastSavedQuality !== null) {
|
if (lastSavedQuality !== null) {
|
||||||
quality = parseInt(lastSavedQuality);
|
if (parseInt(lastSavedQuality) === 100) {
|
||||||
|
quality = 99;
|
||||||
|
} else {
|
||||||
|
quality = parseInt(lastSavedQuality);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
mainWindow.webContents.send(commands.OS, getPlatform());
|
mainWindow.webContents.send(commands.OS, getPlatform());
|
||||||
@ -229,6 +233,46 @@ const getModels = (folderPath: string) => {
|
|||||||
return models;
|
return models;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//------------------------Save Last Paths-----------------------------//
|
||||||
|
const convertAndScale = async (
|
||||||
|
originalImagePath: string,
|
||||||
|
upscaledImagePath: string,
|
||||||
|
processedImagePath: string,
|
||||||
|
scale: string,
|
||||||
|
saveImageAs: string,
|
||||||
|
onError: (error: any) => void
|
||||||
|
) => {
|
||||||
|
const originalImage = await sharp(originalImagePath).metadata();
|
||||||
|
if (!mainWindow || !originalImage) {
|
||||||
|
throw new Error("Could not grab the original image!");
|
||||||
|
}
|
||||||
|
// Resize the image to the scale
|
||||||
|
const newImage = sharp(upscaledImagePath)
|
||||||
|
.resize(
|
||||||
|
originalImage.width && originalImage.width * parseInt(scale),
|
||||||
|
originalImage.height && originalImage.height * parseInt(scale)
|
||||||
|
)
|
||||||
|
.withMetadata(); // Keep metadata
|
||||||
|
// Change the output according to the saveImageAs
|
||||||
|
if (saveImageAs === "png") {
|
||||||
|
newImage.png({ compressionLevel: quality });
|
||||||
|
} else if (saveImageAs === "jpg") {
|
||||||
|
console.log("Quality: ", quality);
|
||||||
|
newImage.jpeg({ quality: 100 - quality });
|
||||||
|
}
|
||||||
|
// Save the image
|
||||||
|
const buffer = await newImage.toBuffer();
|
||||||
|
sharp(buffer)
|
||||||
|
.toFile(processedImagePath)
|
||||||
|
.then(() => {
|
||||||
|
logit("✅ Done converting to: ", upscaledImagePath);
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
logit("❌ Error converting to: ", saveImageAs, error);
|
||||||
|
onError(error);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
//------------------------Open Folder-----------------------------//
|
//------------------------Open Folder-----------------------------//
|
||||||
ipcMain.on(commands.OPEN_FOLDER, async (event, payload) => {
|
ipcMain.on(commands.OPEN_FOLDER, async (event, payload) => {
|
||||||
logit("📂 Opening Folder: ", payload);
|
logit("📂 Opening Folder: ", payload);
|
||||||
@ -435,7 +479,7 @@ ipcMain.on(commands.UPSCAYL, async (event, payload) => {
|
|||||||
model,
|
model,
|
||||||
scale,
|
scale,
|
||||||
gpuId,
|
gpuId,
|
||||||
saveImageAs
|
"png"
|
||||||
),
|
),
|
||||||
logit
|
logit
|
||||||
);
|
);
|
||||||
@ -473,31 +517,31 @@ ipcMain.on(commands.UPSCAYL, async (event, payload) => {
|
|||||||
if (!failed && !stopped) {
|
if (!failed && !stopped) {
|
||||||
logit("💯 Done upscaling");
|
logit("💯 Done upscaling");
|
||||||
logit("♻ Scaling and converting now...");
|
logit("♻ Scaling and converting now...");
|
||||||
const originalImage = await Jimp.read(inputDir + slash + fullfileName);
|
// Free up memory
|
||||||
|
upscayl.kill();
|
||||||
try {
|
try {
|
||||||
const newImage = await Jimp.read(
|
if (!mainWindow) return;
|
||||||
isAlpha ? outFile + ".png" : outFile
|
await convertAndScale(
|
||||||
|
inputDir + slash + fullfileName,
|
||||||
|
isAlpha ? outFile + ".png" : outFile,
|
||||||
|
isAlpha ? outFile + ".png" : outFile,
|
||||||
|
payload.scale,
|
||||||
|
saveImageAs,
|
||||||
|
onError
|
||||||
);
|
);
|
||||||
try {
|
mainWindow.setProgressBar(-1);
|
||||||
if (!mainWindow) return;
|
mainWindow.webContents.send(commands.UPSCAYL_DONE, outFile);
|
||||||
newImage
|
|
||||||
.quality(100 - quality)
|
|
||||||
.scaleToFit(
|
|
||||||
originalImage.getWidth() * parseInt(payload.scale),
|
|
||||||
originalImage.getHeight() * parseInt(payload.scale)
|
|
||||||
)
|
|
||||||
.write(isAlpha ? outFile + ".png" : outFile);
|
|
||||||
mainWindow.setProgressBar(-1);
|
|
||||||
mainWindow.webContents.send(
|
|
||||||
commands.UPSCAYL_DONE,
|
|
||||||
isAlpha ? outFile + ".png" : outFile
|
|
||||||
);
|
|
||||||
} catch (error) {
|
|
||||||
logit("❌ Error converting to PNG: ", error);
|
|
||||||
onError(error);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logit("❌ Error reading original image metadata", error);
|
logit(
|
||||||
|
"❌ Error processing (scaling and converting) the image. Please report this error on GitHub.",
|
||||||
|
error
|
||||||
|
);
|
||||||
|
upscayl.kill();
|
||||||
|
mainWindow &&
|
||||||
|
mainWindow.webContents.send(
|
||||||
|
commands.UPSCAYL_ERROR,
|
||||||
|
"Error processing (scaling and converting) the image. Please report this error on GitHub."
|
||||||
|
);
|
||||||
onError(error);
|
onError(error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -599,26 +643,38 @@ ipcMain.on(commands.FOLDER_UPSCAYL, async (event, payload) => {
|
|||||||
if (!failed && !stopped) {
|
if (!failed && !stopped) {
|
||||||
logit("💯 Done upscaling");
|
logit("💯 Done upscaling");
|
||||||
logit("♻ Scaling and converting now...");
|
logit("♻ Scaling and converting now...");
|
||||||
|
upscayl.kill();
|
||||||
// Get number of files in output folder
|
// Get number of files in output folder
|
||||||
const files = fs.readdirSync(inputDir);
|
const files = fs.readdirSync(inputDir);
|
||||||
files.forEach(async (file) => {
|
try {
|
||||||
console.log("Filename: ", file.slice(0, -3));
|
files.forEach(async (file) => {
|
||||||
// Resize the image to the original size
|
console.log("Filename: ", file.slice(0, -3));
|
||||||
const originalImage = await Jimp.read(inputDir + slash + file);
|
await convertAndScale(
|
||||||
const newImage = await Jimp.read(
|
inputDir + slash + file,
|
||||||
outputDir + slash + file.slice(0, -3) + "png"
|
outputDir + slash + file.slice(0, -3) + "png",
|
||||||
|
outputDir + slash + file,
|
||||||
|
payload.scale,
|
||||||
|
saveImageAs,
|
||||||
|
onError
|
||||||
|
);
|
||||||
|
// Remove the png file (default) if the saveImageAs is not png
|
||||||
|
if (saveImageAs !== "png") {
|
||||||
|
fs.unlinkSync(outputDir + slash + file.slice(0, -3) + "png");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logit(
|
||||||
|
"❌ Error processing (scaling and converting) the image. Please report this error on GitHub.",
|
||||||
|
error
|
||||||
);
|
);
|
||||||
newImage
|
upscayl.kill();
|
||||||
.quality(100 - quality)
|
mainWindow &&
|
||||||
.scaleToFit(
|
mainWindow.webContents.send(
|
||||||
originalImage.getWidth() * parseInt(payload.scale),
|
commands.UPSCAYL_ERROR,
|
||||||
originalImage.getHeight() * parseInt(payload.scale)
|
"Error processing (scaling and converting) the image. Please report this error on GitHub."
|
||||||
)
|
);
|
||||||
.write(outputDir + slash + file);
|
onError(error);
|
||||||
if (saveImageAs !== "png") {
|
}
|
||||||
fs.unlinkSync(outputDir + slash + file.slice(0, -3) + "png");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
mainWindow.webContents.send(commands.FOLDER_UPSCAYL_DONE, outputDir);
|
mainWindow.webContents.send(commands.FOLDER_UPSCAYL_DONE, outputDir);
|
||||||
} else {
|
} else {
|
||||||
@ -636,8 +692,9 @@ ipcMain.on(commands.DOUBLE_UPSCAYL, async (event, payload) => {
|
|||||||
if (!mainWindow) return;
|
if (!mainWindow) return;
|
||||||
|
|
||||||
const model = payload.model as string;
|
const model = payload.model as string;
|
||||||
let inputDir = (payload.imagePath.match(/(.*)[\/\\]/)[1] || "") as string;
|
const imagePath = payload.imagePath;
|
||||||
let outputDir = payload.outputPath as string;
|
let inputDir = (imagePath.match(/(.*)[\/\\]/) || [""])[1];
|
||||||
|
let outputDir = path.normalize(payload.outputPath);
|
||||||
|
|
||||||
if (saveOutputFolder === true && outputFolderPath) {
|
if (saveOutputFolder === true && outputFolderPath) {
|
||||||
outputDir = outputFolderPath;
|
outputDir = outputFolderPath;
|
||||||
@ -649,7 +706,7 @@ ipcMain.on(commands.DOUBLE_UPSCAYL, async (event, payload) => {
|
|||||||
|
|
||||||
// COPY IMAGE TO TMP FOLDER
|
// COPY IMAGE TO TMP FOLDER
|
||||||
|
|
||||||
const fullfileName = payload.imagePath.split(slash).slice(-1)[0] as string;
|
const fullfileName = imagePath.split(slash).slice(-1)[0] as string;
|
||||||
const fileName = parse(fullfileName).name;
|
const fileName = parse(fullfileName).name;
|
||||||
const outFile =
|
const outFile =
|
||||||
outputDir + slash + fileName + "_upscayl_16x_" + model + "." + saveImageAs;
|
outputDir + slash + fileName + "_upscayl_16x_" + model + "." + saveImageAs;
|
||||||
@ -718,28 +775,51 @@ ipcMain.on(commands.DOUBLE_UPSCAYL, async (event, payload) => {
|
|||||||
if (!failed2 && !stopped) {
|
if (!failed2 && !stopped) {
|
||||||
logit("💯 Done upscaling");
|
logit("💯 Done upscaling");
|
||||||
logit("♻ Scaling and converting now...");
|
logit("♻ Scaling and converting now...");
|
||||||
const originalImage = await Jimp.read(inputDir + slash + fullfileName);
|
mainWindow.webContents.send(commands.SCALING_AND_CONVERTING);
|
||||||
try {
|
try {
|
||||||
const newImage = await Jimp.read(isAlpha ? outFile + ".png" : outFile);
|
const originalImage = await sharp(
|
||||||
try {
|
inputDir + slash + fullfileName
|
||||||
newImage
|
).metadata();
|
||||||
.quality(100 - quality)
|
if (!mainWindow || !originalImage) {
|
||||||
.scaleToFit(
|
throw new Error("Could not grab the original image!");
|
||||||
originalImage.getWidth() * parseInt(payload.scale),
|
|
||||||
originalImage.getHeight() * parseInt(payload.scale)
|
|
||||||
)
|
|
||||||
.write(isAlpha ? outFile + ".png" : outFile);
|
|
||||||
mainWindow.setProgressBar(-1);
|
|
||||||
mainWindow.webContents.send(
|
|
||||||
commands.DOUBLE_UPSCAYL_DONE,
|
|
||||||
isAlpha ? outFile + ".png" : outFile
|
|
||||||
);
|
|
||||||
} catch (error) {
|
|
||||||
logit("❌ Error converting to PNG: ", error);
|
|
||||||
onError(error);
|
|
||||||
}
|
}
|
||||||
|
// Resize the image to the scale
|
||||||
|
const newImage = sharp(isAlpha ? outFile + ".png" : outFile)
|
||||||
|
.resize(
|
||||||
|
originalImage.width &&
|
||||||
|
originalImage.width * parseInt(payload.scale),
|
||||||
|
originalImage.height &&
|
||||||
|
originalImage.height * parseInt(payload.scale)
|
||||||
|
)
|
||||||
|
.withMetadata(); // Keep metadata
|
||||||
|
// Change the output according to the saveImageAs
|
||||||
|
if (saveImageAs === "png") {
|
||||||
|
newImage.png({ quality: 100 - quality });
|
||||||
|
} else if (saveImageAs === "jpg") {
|
||||||
|
newImage.jpeg({ quality: 100 - quality });
|
||||||
|
}
|
||||||
|
// Save the image
|
||||||
|
await newImage
|
||||||
|
.toFile(isAlpha ? outFile + ".png" : outFile)
|
||||||
|
.then(() => {
|
||||||
|
logit(
|
||||||
|
"✅ Done converting to: ",
|
||||||
|
isAlpha ? outFile + ".png" : outFile
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
logit("❌ Error converting to: ", saveImageAs, error);
|
||||||
|
upscayl.kill();
|
||||||
|
onError(error);
|
||||||
|
});
|
||||||
|
mainWindow.setProgressBar(-1);
|
||||||
|
mainWindow.webContents.send(
|
||||||
|
commands.DOUBLE_UPSCAYL_DONE,
|
||||||
|
isAlpha ? outFile + ".png" : outFile
|
||||||
|
);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logit("❌ Error reading original image metadata", error);
|
logit("❌ Error reading original image metadata", error);
|
||||||
|
upscayl.kill();
|
||||||
onError(error);
|
onError(error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
1884
package-lock.json
generated
1884
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -190,11 +190,11 @@
|
|||||||
"electron-next": "^3.1.5",
|
"electron-next": "^3.1.5",
|
||||||
"electron-updater": "^6.1.1",
|
"electron-updater": "^6.1.1",
|
||||||
"firebase": "^10.3.0",
|
"firebase": "^10.3.0",
|
||||||
"jimp": "^0.22.8",
|
|
||||||
"jotai": "^2.2.2",
|
"jotai": "^2.2.2",
|
||||||
"react-compare-slider": "^2.2.0",
|
"react-compare-slider": "^2.2.0",
|
||||||
"react-select": "^5.7.4",
|
"react-select": "^5.7.4",
|
||||||
"react-tooltip": "^5.18.1",
|
"react-tooltip": "^5.18.1",
|
||||||
|
"sharp": "^0.32.5",
|
||||||
"tailwind-scrollbar": "^3.0.4",
|
"tailwind-scrollbar": "^3.0.4",
|
||||||
"theme-change": "^2.5.0"
|
"theme-change": "^2.5.0"
|
||||||
},
|
},
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
|
|
||||||
type QualityInputProps = {
|
type QualityInputProps = {
|
||||||
quality: number;
|
quality: number;
|
||||||
|
@ -127,6 +127,19 @@ function SettingsTab({
|
|||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!localStorage.getItem("quality")) {
|
||||||
|
logit("⚙️ Setting quality to 100%");
|
||||||
|
localStorage.setItem("quality", JSON.stringify(quality));
|
||||||
|
} else {
|
||||||
|
const currentlySavedQuality = localStorage.getItem("quality");
|
||||||
|
logit("⚙️ Getting quality from localStorage", quality);
|
||||||
|
if (currentlySavedQuality) {
|
||||||
|
setQuality(JSON.parse(currentlySavedQuality));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
// HANDLERS
|
// HANDLERS
|
||||||
const setExportType = (format: string) => {
|
const setExportType = (format: string) => {
|
||||||
setSaveImageAs(format);
|
setSaveImageAs(format);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user