* add login code api * use container for webview * improve browserview * may login via xuesheng account * locale * fix ui
408 lines
9.5 KiB
TypeScript
408 lines
9.5 KiB
TypeScript
import {
|
|
app,
|
|
BrowserWindow,
|
|
BrowserView,
|
|
Menu,
|
|
ipcMain,
|
|
shell,
|
|
dialog,
|
|
systemPreferences,
|
|
} from "electron";
|
|
import path from "path";
|
|
import db from "@main/db";
|
|
import settings from "@main/settings";
|
|
import downloader from "@main/downloader";
|
|
import whisper from "@main/whisper";
|
|
import fs from "fs-extra";
|
|
import "@main/i18n";
|
|
import log from "electron-log/main";
|
|
import { WEB_API_URL, REPO_URL } from "@/constants";
|
|
import { AudibleProvider, TedProvider } from "@main/providers";
|
|
import { FfmpegDownloader } from "@main/ffmpeg";
|
|
import { Waveform } from "./waveform";
|
|
|
|
log.initialize({ preload: true });
|
|
const logger = log.scope("window");
|
|
|
|
const audibleProvider = new AudibleProvider();
|
|
const tedProvider = new TedProvider();
|
|
const ffmpegDownloader = new FfmpegDownloader();
|
|
const waveform = new Waveform();
|
|
|
|
const main = {
|
|
win: null as BrowserWindow | null,
|
|
init: () => {},
|
|
};
|
|
|
|
main.init = () => {
|
|
if (main.win) {
|
|
main.win.show();
|
|
return;
|
|
}
|
|
|
|
// Prepare local database
|
|
db.registerIpcHandlers();
|
|
|
|
// Prepare Settings
|
|
settings.registerIpcHandlers();
|
|
|
|
// Whisper
|
|
whisper.registerIpcHandlers();
|
|
|
|
// Waveform
|
|
waveform.registerIpcHandlers();
|
|
|
|
// Downloader
|
|
downloader.registerIpcHandlers();
|
|
|
|
// FfmpegDownloader
|
|
ffmpegDownloader.registerIpcHandlers();
|
|
|
|
// AudibleProvider
|
|
audibleProvider.registerIpcHandlers();
|
|
|
|
// TedProvider
|
|
tedProvider.registerIpcHandlers();
|
|
|
|
// BrowserView
|
|
ipcMain.handle(
|
|
"view-load",
|
|
(
|
|
event,
|
|
url,
|
|
bounds: { x: number; y: number; width: number; height: number },
|
|
options?: {
|
|
navigatable?: boolean;
|
|
}
|
|
) => {
|
|
const {
|
|
x = 0,
|
|
y = 0,
|
|
width = mainWindow.getBounds().width,
|
|
height = mainWindow.getBounds().height,
|
|
} = bounds;
|
|
const { navigatable = false } = options || {};
|
|
|
|
logger.debug("view-load", url);
|
|
const view = new BrowserView();
|
|
view.setBackgroundColor("#fff");
|
|
mainWindow.setBrowserView(view);
|
|
|
|
view.setBounds({
|
|
x,
|
|
y,
|
|
width,
|
|
height,
|
|
});
|
|
view.setAutoResize({
|
|
width: true,
|
|
height: true,
|
|
horizontal: true,
|
|
vertical: true,
|
|
});
|
|
view.webContents.on("did-navigate", (_event, url) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-navigate",
|
|
url,
|
|
});
|
|
});
|
|
view.webContents.on(
|
|
"did-fail-load",
|
|
(_event, _errorCode, errrorDescription, validatedURL) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-fail-load",
|
|
error: errrorDescription,
|
|
url: validatedURL,
|
|
});
|
|
(view.webContents as any).destroy();
|
|
mainWindow.removeBrowserView(view);
|
|
}
|
|
);
|
|
view.webContents.on("did-finish-load", () => {
|
|
view.webContents
|
|
.executeJavaScript(`document.documentElement.innerHTML`)
|
|
.then((html) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-finish-load",
|
|
html,
|
|
});
|
|
});
|
|
});
|
|
|
|
view.webContents.on("will-redirect", (detail) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "will-redirect",
|
|
url: detail.url,
|
|
});
|
|
|
|
logger.debug("will-redirect", detail.url);
|
|
});
|
|
|
|
view.webContents.on("will-navigate", (detail) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "will-navigate",
|
|
url: detail.url,
|
|
});
|
|
|
|
logger.debug("will-navigate", detail.url);
|
|
if (!navigatable) {
|
|
logger.debug("prevent navigation", detail.url);
|
|
detail.preventDefault();
|
|
}
|
|
});
|
|
view.webContents.loadURL(url);
|
|
}
|
|
);
|
|
|
|
ipcMain.handle("view-remove", () => {
|
|
logger.debug("view-remove");
|
|
mainWindow.getBrowserViews().forEach((view) => {
|
|
(view.webContents as any).destroy();
|
|
mainWindow.removeBrowserView(view);
|
|
});
|
|
});
|
|
|
|
ipcMain.handle("view-hide", () => {
|
|
logger.debug("view-hide");
|
|
const view = mainWindow.getBrowserView();
|
|
if (!view) return;
|
|
|
|
const bounds = view.getBounds();
|
|
logger.debug("current view bounds", bounds);
|
|
|
|
view.setBounds({
|
|
x: -bounds.width,
|
|
y: -bounds.height,
|
|
width: 0,
|
|
height: 0,
|
|
});
|
|
});
|
|
|
|
ipcMain.handle(
|
|
"view-show",
|
|
(
|
|
_event,
|
|
bounds: {
|
|
x: number;
|
|
y: number;
|
|
width: number;
|
|
height: number;
|
|
}
|
|
) => {
|
|
const view = mainWindow.getBrowserView();
|
|
if (!view) return;
|
|
|
|
logger.debug("view-show", bounds);
|
|
view.setBounds(bounds);
|
|
}
|
|
);
|
|
|
|
ipcMain.handle("view-scrape", (event, url) => {
|
|
logger.debug("view-scrape", url);
|
|
const view = new BrowserView();
|
|
mainWindow.setBrowserView(view);
|
|
|
|
view.webContents.on("did-navigate", (_event, url) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-navigate",
|
|
url,
|
|
});
|
|
});
|
|
view.webContents.on(
|
|
"did-fail-load",
|
|
(_event, _errorCode, errrorDescription, validatedURL) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-fail-load",
|
|
error: errrorDescription,
|
|
url: validatedURL,
|
|
});
|
|
(view.webContents as any).destroy();
|
|
mainWindow.removeBrowserView(view);
|
|
}
|
|
);
|
|
view.webContents.on("did-finish-load", () => {
|
|
view.webContents
|
|
.executeJavaScript(`document.documentElement.innerHTML`)
|
|
.then((html) => {
|
|
event.sender.send("view-on-state", {
|
|
state: "did-finish-load",
|
|
html,
|
|
});
|
|
(view.webContents as any).destroy();
|
|
mainWindow.removeBrowserView(view);
|
|
});
|
|
});
|
|
|
|
view.webContents.loadURL(url);
|
|
});
|
|
|
|
// App options
|
|
ipcMain.handle("app-reset", () => {
|
|
fs.removeSync(settings.userDataPath());
|
|
fs.removeSync(settings.file());
|
|
|
|
app.relaunch();
|
|
app.exit();
|
|
});
|
|
|
|
ipcMain.handle("app-reset-settings", () => {
|
|
fs.removeSync(settings.file());
|
|
|
|
app.relaunch();
|
|
app.exit();
|
|
});
|
|
|
|
ipcMain.handle("app-relaunch", () => {
|
|
app.relaunch();
|
|
app.exit();
|
|
});
|
|
|
|
ipcMain.handle("app-reload", () => {
|
|
if (MAIN_WINDOW_VITE_DEV_SERVER_URL) {
|
|
mainWindow.loadURL(MAIN_WINDOW_VITE_DEV_SERVER_URL);
|
|
} else {
|
|
mainWindow.loadFile(
|
|
path.join(__dirname, `../renderer/${MAIN_WINDOW_VITE_NAME}/index.html`)
|
|
);
|
|
}
|
|
});
|
|
|
|
ipcMain.handle("app-is-packaged", () => {
|
|
return app.isPackaged;
|
|
});
|
|
|
|
ipcMain.handle("app-api-url", () => {
|
|
return process.env.WEB_API_URL || WEB_API_URL;
|
|
});
|
|
|
|
ipcMain.handle("app-quit", () => {
|
|
app.quit();
|
|
});
|
|
|
|
ipcMain.handle("app-open-dev-tools", () => {
|
|
mainWindow.webContents.openDevTools();
|
|
});
|
|
|
|
ipcMain.handle("app-create-issue", (_event, title, log) => {
|
|
const body = `**Version**
|
|
|
|
${app.getVersion()}
|
|
|
|
**Platform**
|
|
|
|
${process.platform} ${process.arch} ${process.getSystemVersion()}
|
|
|
|
**Log**
|
|
\`\`\`
|
|
${log}
|
|
\`\`\`
|
|
`;
|
|
|
|
const params = {
|
|
title,
|
|
body,
|
|
};
|
|
|
|
shell.openExternal(
|
|
`${REPO_URL}/issues/new?${new URLSearchParams(params).toString()}`
|
|
);
|
|
});
|
|
|
|
ipcMain.handle(
|
|
"system-preferences-media-access",
|
|
async (_event, mediaType: "microphone" | "camera") => {
|
|
if (process.platform === "linux") return true;
|
|
if (process.platform === "win32")
|
|
return systemPreferences.getMediaAccessStatus(mediaType) === "granted";
|
|
|
|
if (process.platform === "darwin") {
|
|
const status = systemPreferences.getMediaAccessStatus(mediaType);
|
|
if (status !== "granted") {
|
|
const result = await systemPreferences.askForMediaAccess(mediaType);
|
|
return result;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
);
|
|
|
|
// Shell
|
|
ipcMain.handle("shell-open-external", (_event, url) => {
|
|
shell.openExternal(url);
|
|
});
|
|
|
|
ipcMain.handle("shell-open-path", (_event, path) => {
|
|
shell.openPath(path);
|
|
});
|
|
|
|
// Dialog
|
|
ipcMain.handle("dialog-show-open-dialog", (event, options) => {
|
|
return dialog.showOpenDialogSync(
|
|
BrowserWindow.fromWebContents(event.sender),
|
|
options
|
|
);
|
|
});
|
|
|
|
ipcMain.handle("dialog-show-save-dialog", (event, options) => {
|
|
return dialog.showSaveDialogSync(
|
|
BrowserWindow.fromWebContents(event.sender),
|
|
options
|
|
);
|
|
});
|
|
|
|
ipcMain.handle("dialog-show-message-box", (event, options) => {
|
|
return dialog.showMessageBoxSync(
|
|
BrowserWindow.fromWebContents(event.sender),
|
|
options
|
|
);
|
|
});
|
|
|
|
ipcMain.handle("dialog-show-error-box", (_event, title, content) => {
|
|
return dialog.showErrorBox(title, content);
|
|
});
|
|
|
|
// Create the browser window.
|
|
const mainWindow = new BrowserWindow({
|
|
icon: "./assets/icon.png",
|
|
width: 1600,
|
|
height: 1200,
|
|
minWidth: 1024,
|
|
minHeight: 768,
|
|
webPreferences: {
|
|
preload: path.join(__dirname, "preload.js"),
|
|
},
|
|
});
|
|
|
|
mainWindow.webContents.setWindowOpenHandler(({ url }) => {
|
|
if (url.startsWith("http")) {
|
|
logger.info(`Opening ${url}`);
|
|
shell.openExternal(url);
|
|
return { action: "deny" };
|
|
} else {
|
|
return { action: "allow" };
|
|
}
|
|
});
|
|
|
|
// and load the index.html of the app.
|
|
if (MAIN_WINDOW_VITE_DEV_SERVER_URL) {
|
|
mainWindow.loadURL(MAIN_WINDOW_VITE_DEV_SERVER_URL);
|
|
|
|
// Open the DevTools.
|
|
setTimeout(() => {
|
|
mainWindow.webContents.openDevTools();
|
|
}, 100);
|
|
} else {
|
|
mainWindow.loadFile(
|
|
path.join(__dirname, `../renderer/${MAIN_WINDOW_VITE_NAME}/index.html`)
|
|
);
|
|
// mainWindow.webContents.openDevTools();
|
|
}
|
|
|
|
Menu.setApplicationMenu(null);
|
|
|
|
main.win = mainWindow;
|
|
};
|
|
|
|
export default main;
|