feat: choose icon from .svg
This commit is contained in:
parent
79835b2f03
commit
183a388495
5 changed files with 61 additions and 54 deletions
|
@ -163,8 +163,8 @@ handle(IpcEvents.SET_TRAY_ICON, (_, iconURI) => setTrayIcon(iconURI));
|
||||||
handle(IpcEvents.GET_TRAY_ICON, (_, iconPath) => getTrayIconFile(iconPath));
|
handle(IpcEvents.GET_TRAY_ICON, (_, iconPath) => getTrayIconFile(iconPath));
|
||||||
handleSync(IpcEvents.GET_TRAY_ICON_SYNC, (_, iconPath) => getTrayIconFileSync(iconPath));
|
handleSync(IpcEvents.GET_TRAY_ICON_SYNC, (_, iconPath) => getTrayIconFileSync(iconPath));
|
||||||
handle(IpcEvents.GET_SYSTEM_ACCENT_COLOR, () => getAccentColor());
|
handle(IpcEvents.GET_SYSTEM_ACCENT_COLOR, () => getAccentColor());
|
||||||
handle(IpcEvents.CREATE_TRAY_ICON_RESPONSE, (_, iconName, dataURL, isCustomIcon) =>
|
handle(IpcEvents.CREATE_TRAY_ICON_RESPONSE, (_, iconName, dataURL, isCustomIcon, isSvg) =>
|
||||||
createTrayIcon(iconName, dataURL, isCustomIcon)
|
createTrayIcon(iconName, dataURL, isCustomIcon, isSvg)
|
||||||
);
|
);
|
||||||
handle(IpcEvents.GENERATE_TRAY_ICONS, () => generateTrayIcons());
|
handle(IpcEvents.GENERATE_TRAY_ICONS, () => generateTrayIcons());
|
||||||
handle(IpcEvents.SELECT_TRAY_ICON, async (_, iconName) => pickTrayIcon(iconName));
|
handle(IpcEvents.SELECT_TRAY_ICON, async (_, iconName) => pickTrayIcon(iconName));
|
||||||
|
|
|
@ -112,14 +112,19 @@ export function getTrayIconFileSync(iconName: string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function createTrayIcon(iconName: string, iconDataURL: string, isCustomIcon: boolean = false) {
|
export async function createTrayIcon(
|
||||||
|
iconName: string,
|
||||||
|
iconDataURL: string,
|
||||||
|
isCustomIcon: boolean = false,
|
||||||
|
isSvg: boolean = false
|
||||||
|
) {
|
||||||
// creates .png at config/TrayIcons/iconName.png from given iconDataURL
|
// creates .png at config/TrayIcons/iconName.png from given iconDataURL
|
||||||
// primarily called from renderer using CREATE_TRAY_ICON_RESPONSE IPC call
|
// primarily called from renderer using CREATE_TRAY_ICON_RESPONSE IPC call
|
||||||
iconDataURL = iconDataURL.replace(/^data:image\/png;base64,/, "");
|
iconDataURL = iconDataURL.replace(/^data:image\/png;base64,/, "");
|
||||||
if (isCustomIcon) {
|
if (isCustomIcon) {
|
||||||
const img = nativeImage.createFromDataURL(iconDataURL).resize({ width: 128, height: 128 });
|
const img = nativeImage.createFromDataURL(iconDataURL).resize({ width: 128, height: 128 });
|
||||||
// writeFileSync(join(ICONS_DIR, iconName + "_custom.png"), img.toDataURL(), "base64");
|
if (isSvg) writeFileSync(join(ICONS_DIR, iconName + "_custom.png"), iconDataURL, "base64");
|
||||||
writeFileSync(join(ICONS_DIR, iconName + "_custom.png"), img.toPNG());
|
else writeFileSync(join(ICONS_DIR, iconName + "_custom.png"), img.toPNG());
|
||||||
} else {
|
} else {
|
||||||
writeFileSync(join(ICONS_DIR, iconName + ".png"), iconDataURL, "base64");
|
writeFileSync(join(ICONS_DIR, iconName + ".png"), iconDataURL, "base64");
|
||||||
}
|
}
|
||||||
|
@ -144,11 +149,15 @@ export async function pickTrayIcon(iconName: string) {
|
||||||
|
|
||||||
const res = await dialog.showOpenDialog(mainWin!, {
|
const res = await dialog.showOpenDialog(mainWin!, {
|
||||||
properties: ["openFile"],
|
properties: ["openFile"],
|
||||||
filters: [{ name: "Image", extensions: ["png", "jpg"] }]
|
filters: [{ name: "Image", extensions: ["png", "jpg", "svg"] }]
|
||||||
});
|
});
|
||||||
if (!res.filePaths.length) return "cancelled";
|
if (!res.filePaths.length) return "cancelled";
|
||||||
const dir = res.filePaths[0];
|
const dir = res.filePaths[0];
|
||||||
// add .svg !!
|
// add .svg !!
|
||||||
|
if (dir.split(".").pop() === "svg") {
|
||||||
|
mainWin.webContents.send(IpcEvents.CREATE_TRAY_ICON_REQUEST, iconName, readFileSync(dir, "utf-8"));
|
||||||
|
return "svg";
|
||||||
|
}
|
||||||
const image = nativeImage.createFromPath(dir);
|
const image = nativeImage.createFromPath(dir);
|
||||||
if (image.isEmpty()) return "invalid";
|
if (image.isEmpty()) return "invalid";
|
||||||
const img = nativeImage.createFromPath(dir).resize({ width: 128, height: 128 });
|
const img = nativeImage.createFromPath(dir).resize({ width: 128, height: 128 });
|
||||||
|
|
|
@ -86,10 +86,16 @@ export const VesktopNative = {
|
||||||
setIcon: (iconURI: string) => invoke<void>(IpcEvents.SET_TRAY_ICON, iconURI),
|
setIcon: (iconURI: string) => invoke<void>(IpcEvents.SET_TRAY_ICON, iconURI),
|
||||||
getIcon: (iconName: string) => invoke<string>(IpcEvents.GET_TRAY_ICON, iconName),
|
getIcon: (iconName: string) => invoke<string>(IpcEvents.GET_TRAY_ICON, iconName),
|
||||||
getIconSync: (iconName: string) => sendSync<string>(IpcEvents.GET_TRAY_ICON_SYNC, iconName),
|
getIconSync: (iconName: string) => sendSync<string>(IpcEvents.GET_TRAY_ICON_SYNC, iconName),
|
||||||
createIconResponse: (iconName: string, iconDataURL: string, isCustomIcon: boolean = true) =>
|
createIconResponse: (
|
||||||
invoke<void>(IpcEvents.CREATE_TRAY_ICON_RESPONSE, iconName, iconDataURL, isCustomIcon),
|
iconName: string,
|
||||||
createIconRequest: (listener: (iconName: string) => void) => {
|
iconDataURL: string,
|
||||||
ipcRenderer.on(IpcEvents.CREATE_TRAY_ICON_REQUEST, (_, iconPath: string) => listener(iconPath));
|
isCustomIcon: boolean = true,
|
||||||
|
isSvg: boolean = true
|
||||||
|
) => invoke<void>(IpcEvents.CREATE_TRAY_ICON_RESPONSE, iconName, iconDataURL, isCustomIcon, isSvg),
|
||||||
|
createIconRequest: (listener: (iconName: string, svg: string) => void) => {
|
||||||
|
ipcRenderer.on(IpcEvents.CREATE_TRAY_ICON_REQUEST, (_, iconPath: string, svg: string) =>
|
||||||
|
listener(iconPath, svg)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
generateTrayIcons: () => invoke<void>(IpcEvents.GENERATE_TRAY_ICONS),
|
generateTrayIcons: () => invoke<void>(IpcEvents.GENERATE_TRAY_ICONS),
|
||||||
setCurrentVoiceIcon: (listener: (...args: any[]) => void) => {
|
setCurrentVoiceIcon: (listener: (...args: any[]) => void) => {
|
||||||
|
|
|
@ -41,6 +41,36 @@ const statusToSettingsKey = {
|
||||||
deafened: { key: "trayDeafenedOverride", label: "Deafened Icon" }
|
deafened: { key: "trayDeafenedOverride", label: "Deafened Icon" }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
async function changeIcon(iconName, settings) {
|
||||||
|
const choice = await VesktopNative.fileManager.selectTrayIcon(iconName);
|
||||||
|
switch (choice) {
|
||||||
|
case "cancelled":
|
||||||
|
return;
|
||||||
|
case "invalid":
|
||||||
|
Toasts.show({
|
||||||
|
message: "Please select a valid .png or .jpg image!",
|
||||||
|
id: Toasts.genId(),
|
||||||
|
type: Toasts.Type.FAILURE
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const updateIcon = () => {
|
||||||
|
const iconKey = statusToSettingsKey[iconName as keyof typeof statusToSettingsKey].key;
|
||||||
|
settings[iconKey] = true;
|
||||||
|
const iconDataURL = VesktopNative.tray.getIconSync(iconName);
|
||||||
|
const img = document.getElementById(iconName) as HTMLImageElement;
|
||||||
|
if (img) {
|
||||||
|
img.src = iconDataURL;
|
||||||
|
}
|
||||||
|
setCurrentTrayIcon();
|
||||||
|
};
|
||||||
|
|
||||||
|
// sometimes new icon may not be generated in time and will be used old icon :c
|
||||||
|
if (choice === "svg") setTimeout(updateIcon, 50);
|
||||||
|
else updateIcon();
|
||||||
|
}
|
||||||
|
|
||||||
function trayEditButton(iconName: string) {
|
function trayEditButton(iconName: string) {
|
||||||
const Settings = useSettings();
|
const Settings = useSettings();
|
||||||
return (
|
return (
|
||||||
|
@ -58,27 +88,7 @@ function trayEditButton(iconName: string) {
|
||||||
width="40"
|
width="40"
|
||||||
height="40"
|
height="40"
|
||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
const choice = await VesktopNative.fileManager.selectTrayIcon(iconName);
|
changeIcon(iconName, Settings);
|
||||||
switch (choice) {
|
|
||||||
case "cancelled":
|
|
||||||
return;
|
|
||||||
case "invalid":
|
|
||||||
Toasts.show({
|
|
||||||
message: "Please select a valid .png or .jpg image!",
|
|
||||||
id: Toasts.genId(),
|
|
||||||
type: Toasts.Type.FAILURE
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const iconKey = statusToSettingsKey[iconName as keyof typeof statusToSettingsKey].key;
|
|
||||||
Settings[iconKey] = true;
|
|
||||||
const iconDataURL = VesktopNative.tray.getIconSync(iconName);
|
|
||||||
const img = document.getElementById(iconName) as HTMLImageElement;
|
|
||||||
if (img) {
|
|
||||||
img.src = iconDataURL;
|
|
||||||
}
|
|
||||||
setCurrentTrayIcon();
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
@ -102,26 +112,7 @@ function TrayModalComponent({ modalProps, close }: { modalProps: any; close: ()
|
||||||
{trayEditButton(iconName)}
|
{trayEditButton(iconName)}
|
||||||
<Button
|
<Button
|
||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
const choice = await VesktopNative.fileManager.selectTrayIcon(iconName);
|
changeIcon(iconName, Settings);
|
||||||
switch (choice) {
|
|
||||||
case "cancelled":
|
|
||||||
return;
|
|
||||||
case "invalid":
|
|
||||||
Toasts.show({
|
|
||||||
message: "Please select a valid .png or .jpg image!",
|
|
||||||
id: Toasts.genId(),
|
|
||||||
type: Toasts.Type.FAILURE
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Settings[key] = true;
|
|
||||||
const iconDataURL = VesktopNative.tray.getIconSync(iconName);
|
|
||||||
const img = document.getElementById(iconName) as HTMLImageElement;
|
|
||||||
if (img) {
|
|
||||||
img.src = iconDataURL;
|
|
||||||
}
|
|
||||||
setCurrentTrayIcon();
|
|
||||||
}}
|
}}
|
||||||
look={Button.Looks.OUTLINED}
|
look={Button.Looks.OUTLINED}
|
||||||
>
|
>
|
||||||
|
|
|
@ -39,9 +39,9 @@ function changeColorsInSvg(svg: string, stockColor: string) {
|
||||||
return svg;
|
return svg;
|
||||||
}
|
}
|
||||||
|
|
||||||
VesktopNative.tray.createIconRequest(async (iconName: string) => {
|
VesktopNative.tray.createIconRequest(async (iconName: string, svgIcon: string = "") => {
|
||||||
try {
|
try {
|
||||||
var svg = await VesktopNative.tray.getIcon(iconName);
|
var svg = svgIcon || (await VesktopNative.tray.getIcon(iconName));
|
||||||
svg = changeColorsInSvg(svg, "#f6bfac");
|
svg = changeColorsInSvg(svg, "#f6bfac");
|
||||||
const canvas = document.createElement("canvas");
|
const canvas = document.createElement("canvas");
|
||||||
canvas.width = 128;
|
canvas.width = 128;
|
||||||
|
@ -54,7 +54,8 @@ VesktopNative.tray.createIconRequest(async (iconName: string) => {
|
||||||
if (ctx) {
|
if (ctx) {
|
||||||
ctx.drawImage(img, 0, 0);
|
ctx.drawImage(img, 0, 0);
|
||||||
const dataURL = canvas.toDataURL("image/png");
|
const dataURL = canvas.toDataURL("image/png");
|
||||||
VesktopNative.tray.createIconResponse(iconName, dataURL, false);
|
const isSvg = svgIcon !== "";
|
||||||
|
VesktopNative.tray.createIconResponse(iconName, dataURL, isSvg, isSvg); // custom if svgIcon is provided
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
img.src = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
|
img.src = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
|
||||||
|
|
Loading…
Reference in a new issue