Files
electron-4/src/main/index.ts

439 lines
13 KiB
TypeScript
Raw Normal View History

2025-09-11 18:27:56 +08:00
import { app, shell, BrowserWindow, ipcMain, globalShortcut, dialog } from 'electron'
2025-08-29 09:35:52 +08:00
import path, { join } from 'path'
import { electronApp, optimizer, is } from '@electron-toolkit/utils'
import icon from '../../resources/earth.png?asset'
2025-09-11 18:06:03 +08:00
import { Recorder } from "../preload/recorder";
2025-08-29 09:35:52 +08:00
import fs from 'fs'
2025-09-11 18:27:56 +08:00
import { exec } from 'child_process'
2025-09-11 18:06:03 +08:00
import dayjs from 'dayjs'
2025-09-23 11:17:50 +08:00
import { GetHomeDir } from './config'
2025-09-25 16:21:00 +08:00
import { start, getServer } from "./app";
2025-08-29 09:35:52 +08:00
2025-09-09 10:06:18 +08:00
2025-08-29 09:35:52 +08:00
// 开发环境路径处理 - 确保添加正确的file协议
const devSplashPath = path.resolve(
app.getAppPath(),
'src',
'renderer',
'public',
'startUp',
'startUp.html'
)
// 开发环境必须添加file协议使用URL构造器确保格式正确
const devSplashURL = new URL(`file:///${devSplashPath}`).href
// 生产环境路径处理
let prodSplashURL = ''
const prodPossiblePaths = [
path.join(process.resourcesPath, 'app.asar', 'out', 'renderer', 'startUp', 'startUp.html'),
path.join(process.resourcesPath, 'app.asar', 'renderer', 'startUp', 'startUp.html'),
path.join(process.resourcesPath, 'out', 'renderer', 'startUp', 'startUp.html'),
// 增加asar包外可能的路径
path.join(process.resourcesPath, 'app.asar.unpacked', 'resources', 'startUp', 'startUp.html')
]
// 寻找存在的生产环境路径
const prodSplashPath = prodPossiblePaths.find((p) => {
try {
// 检查路径是否存在处理asar内部文件检查
return fs.existsSync(p)
} catch (e: any) {
console.error('检查路径时出错:', p, e.message)
return false
}
})
if (prodSplashPath) {
// 使用URL构造器处理路径自动编码特殊字符
// 对于Windows系统需要处理盘符前的斜杠问题
const normalizedPath =
process.platform === 'win32' ? prodSplashPath.replace(/^(\w:)/, '/$1') : prodSplashPath
prodSplashURL = new URL(`file://${normalizedPath}`).href
} else {
console.error('未找到有效的启动页路径,检查打包配置和文件是否存在')
// 可以在这里添加默认路径或错误处理
}
// 最终的启动页URL
const splashURL = process.env.NODE_ENV === 'development' ? devSplashURL : prodSplashURL
2025-09-25 16:21:00 +08:00
let startBatPath = process.env.NODE_ENV === 'development' ? path.resolve(app.getAppPath(), 'resources', 'java', 'start.bat') : path.join(process.resourcesPath, 'app.asar.unpacked', 'resources', 'java', 'start.bat')
2025-09-11 18:27:56 +08:00
startBatPath = process.platform === 'win32' ? startBatPath.replace(/^(\w:)/, '/$1') : startBatPath
2025-09-25 16:21:00 +08:00
let stopBatPath = process.env.NODE_ENV === 'development' ? path.resolve(app.getAppPath(), 'resources', 'java', 'stop.bat') : path.join(process.resourcesPath, 'app.asar.unpacked', 'resources', 'java', 'stop.bat')
2025-09-11 18:27:56 +08:00
stopBatPath = process.platform === 'win32' ? stopBatPath.replace(/^(\w:)/, '/$1') : stopBatPath
2025-08-29 09:35:52 +08:00
// const splashURL =
// process.env.NODE_ENV === 'development'
// ? `${join(app.getAppPath(), 'src/renderer/public/startUp/startUp.html')}`
// : `file://${join(app.getAppPath(), 'resources/app.asar/out/renderer/startUp/startUp.html')}`
function createWindow(): void {
// Create the browser window.
2025-09-23 11:17:50 +08:00
start();
2025-08-29 09:35:52 +08:00
// 创建启动窗口
const splashWindow = new BrowserWindow({
width: 896,
height: 510,
frame: false,
alwaysOnTop: true,
show: false,
webPreferences: {
nodeIntegration: true, // 开启 Node 集成
contextIsolation: false, // 关闭上下文隔离
devTools: true,
webSecurity: false,
allowRunningInsecureContent: true
}
})
// 显示启动页
splashWindow.loadURL(splashURL)
splashWindow.show()
// 创建主窗口(保持原有配置,但先不显示)
const mainWindow = new BrowserWindow({
minWidth: 1780, // 添加最小宽度限制
minHeight: 920, // 添加最小高度限制
// fullscreen: true,
show: false,
frame: true,
autoHideMenuBar: true,
useContentSize: true, // 窗口尺寸包含内容区域而非边框
simpleFullscreen: true, // 使用简单全屏模式(仅macOS有效)
backgroundColor: '#00000000', // 添加这行设置透明背景
...(process.platform === 'linux' ? { icon } : {}),
webPreferences: {
preload: join(__dirname, '../preload/index.js'),
sandbox: false,
nodeIntegration: true, // 开启 Node 集成
contextIsolation: false, // 关闭上下文隔离
devTools: true,
webSecurity: false,
allowRunningInsecureContent: true
}
})
2025-09-26 11:37:44 +08:00
ipcMain.on("restart", () => {
app.relaunch();
forceQuit();
});
2025-08-29 09:35:52 +08:00
// 监听启动页完成的消息
ipcMain.on('splash-completed', () => {
// 启动页进度条已完成,可以关闭启动页并显示主窗口
setTimeout(() => {
splashWindow.destroy()
mainWindow.maximize() // 先最大化
mainWindow.show()
setTimeout(() => {
mainWindow.webContents.send('start-login-video')
}, 200) // 给一点时间让窗口完全显示
}, 1000)
})
ipcMain.on('quit-app', () => {
2025-09-25 16:21:00 +08:00
windowAllClosed()
2025-08-29 09:35:52 +08:00
})
ipcMain.on('renderNode', () => {
// 获取所有窗口并转发消息
BrowserWindow.getAllWindows().forEach((win) => {
// 移除条件判断,确保所有窗口都能收到消息
setTimeout(() => {
win.webContents.send('renderNode-reply')
}, 200)
})
})
2025-09-11 18:27:56 +08:00
ipcMain.on("open-directory-dialog", (event, option) => {
// @ts-ignore
dialog.showOpenDialog(BrowserWindow.getFocusedWindow(), {
2025-09-18 20:48:30 +08:00
properties: option.properties,
filters: option.filters,
})
2025-09-11 18:27:56 +08:00
.then((files) => {
let arr = [];
if (!files.canceled) {
files.filePaths.forEach((url) => {
// @ts-ignore
arr.push(url.replace(/\\/g, "/"));
});
}
event.sender.send("selectedItem", arr);
});
});
2025-09-11 18:06:03 +08:00
ipcMain.on("saveFile", (event, { title, filename, filters }) => {
dialog
.showSaveDialog({
title,
defaultPath: filename,
filters,
})
.then((files) => {
let path = "";
if (!files.canceled) {
path = files.filePath.replace(/\\/g, "/");
}
event.sender.send("selectedFileItem", path);
});
});
let recorder;
2025-09-11 19:17:52 +08:00
ipcMain.on("startRecoder", () => {
2025-09-11 18:06:03 +08:00
console.log("开始录制");
recorder = new Recorder();
recorder.start();
});
2025-09-11 19:17:52 +08:00
ipcMain.on("endRecoder", () => {
2025-09-11 18:06:03 +08:00
console.log("结束录制");
// 判断是否存在recorder是否有recorder.end方法
if (!recorder) {
console.log("recorder不存在");
return;
}
recorder.end(() => {
let path = dialog.showSaveDialogSync({
title: "保存视频文件",
defaultPath: dayjs().format("YYYYMMDDHHmmss") + "视频录制.mp4",
filters: [{ name: "文件类型", extensions: ["mp4"] }],
});
if (path != undefined) {
recorder.move(path, () => {
recorder = null;
});
} else {
recorder = null;
}
});
});
2025-09-23 11:17:50 +08:00
ipcMain.on("requireGEMarkerName", (event, obj) => {
/*
* obj= {
dirName: "GEMarker",
dirName1s: "GEMarker1s"
}
* */
2025-09-26 19:14:47 +08:00
// console.log('GetHomeDir()', GetHomeDir())
let prefix =
process.env.NODE_ENV === "development"
? "src/renderer/public"
: "resources/app.asar/out/renderer";
2025-09-23 11:17:50 +08:00
let data = {};
for (const objKey in obj) {
let files = fs.readdirSync(
path.join(
2025-09-26 19:14:47 +08:00
GetHomeDir(),
prefix,
2025-09-23 11:17:50 +08:00
obj[objKey]
)
);
2025-09-26 19:14:47 +08:00
// console.log(files);
2025-09-23 11:17:50 +08:00
for (let i = 0; i < files.length; i++) {
files[i] = obj[objKey] + "/" + files[i];
}
data[obj[objKey]] = files;
}
// let files = fs.readdirSync(path.join(global.__static ? global.__static : GetHomeDir() + "/static", dirName))
// data[dirName] = files
event.sender.send("dirFiles", data);
});
2025-09-28 16:03:15 +08:00
ipcMain.on("restart", (e) => {
closeChild();
app.relaunch();
app.exit();
});
2025-09-23 11:17:50 +08:00
let _winMap = new Map();
2025-09-18 20:48:30 +08:00
// 监听渲染进程创建新窗口的请求
// @ts-ignore
2025-09-23 11:17:50 +08:00
ipcMain.handle('create-new-window', async (event, params, url, option, id) => {
2025-09-18 20:48:30 +08:00
try {
const newWindow = await new BrowserWindow(params)
if (url) {
await newWindow.loadURL(url)
await newWindow.webContents.send("data", option)
2025-09-23 11:17:50 +08:00
newWindow.on('closed', () => {
_winMap.delete(id);
// a.delete(newWindow.id)
// closeCB(newWindow.id)
});
2025-09-18 20:48:30 +08:00
}
2025-09-23 11:17:50 +08:00
_winMap.set(id, newWindow.id);
return _winMap
2025-09-18 20:48:30 +08:00
} catch (error) {
console.error('创建窗口失败:', error);
throw error; // 抛出错误以便渲染进程捕获
}
})
2025-09-23 11:17:50 +08:00
//@ts-ignore
ipcMain.handle('show-window-by-id', async (event, id) => {
BrowserWindow.fromId(id)?.show()
})
2025-09-25 16:21:00 +08:00
ipcMain.handle('get-_winMap', () => {
2025-09-23 11:17:50 +08:00
return _winMap
})
2025-08-29 09:35:52 +08:00
// 设置窗口标题和图标
mainWindow.webContents.setWindowOpenHandler((details) => {
shell.openExternal(details.url)
return { action: 'deny' }
})
// 注册 F5 快捷键刷新
globalShortcut.register('CommandOrControl+F5', () => {
if (mainWindow) {
mainWindow.reload()
setTimeout(() => {
mainWindow.webContents.send('start-login-video')
}, 200) // 给一点时间让窗口完成刷新
}
})
globalShortcut.register('CommandOrControl+F12', () => {
mainWindow.webContents.openDevTools()
})
// HMR for renderer base on electron-vite cli.
// Load the remote URL for development or the local html file for production.
if (is.dev && process.env['ELECTRON_RENDERER_URL']) {
mainWindow.loadURL(process.env['ELECTRON_RENDERER_URL'])
} else {
mainWindow.loadFile(join(__dirname, '../renderer/index.html'))
}
}
// This method will be called when Electron has finished
// initialization and is ready to create browser windows.
// Some APIs can only be used after this event occurs.
app.whenReady().then(() => {
2025-09-11 18:27:56 +08:00
// alert(devSplashURL)
// alert(prodSplashURL)
// dialog.showMessageBox({
// type: 'info',
// title: '信息1',
// message: devSplashURL,
// buttons: ['确定']
// })
// 执行批处理文件
exec(startBatPath.substring(1, 200), (error, stdout, stderr) => {
if (error) {
console.error(`执行错误: ${error.message}`);
return;
}
if (stderr) {
console.error(`错误输出: ${stderr}`);
return;
}
console.log(`批处理输出: ${stdout}`);
});
2025-08-29 09:35:52 +08:00
// Set app user model id for windows
electronApp.setAppUserModelId('com.electron')
// Default open or close DevTools by F12 in development
// and ignore CommandOrControl + R in production.
// see https://github.com/alex8088/electron-toolkit/tree/master/packages/utils
app.on('browser-window-created', (_, window) => {
optimizer.watchWindowShortcuts(window)
})
// IPC test
ipcMain.on('ping', () => console.log('pong'))
createWindow()
app.on('activate', function () {
// On macOS it's common to re-create a window in the app when the
// dock icon is clicked and there are no other windows open.
if (BrowserWindow.getAllWindows().length === 0) createWindow()
})
})
// Quit when all windows are closed, except on macOS. There, it's common
// for applications and their menu bar to stay active until the user quits
// explicitly with Cmd + Q.
2025-09-25 16:21:00 +08:00
// app.on('window-all-closed', () => {
// if (process.platform !== 'darwin') {
// getServer().close(() => {
// // 关闭后台服务
// exec(stopBatPath.substring(1, 200), () => {
// app.quit()
// });
// })
// }
// })
2025-08-29 09:35:52 +08:00
// 退出时注销所有快捷键
app.on('will-quit', () => {
2025-09-23 11:17:50 +08:00
globalShortcut.unregisterAll()
2025-08-29 09:35:52 +08:00
})
2025-09-25 16:21:00 +08:00
// 用于跟踪是否正在执行退出流程
let isQuitting = false;
app.on('window-all-closed', () => {
windowAllClosed()
});
function windowAllClosed() {
// 防止重复触发退出流程
if (isQuitting) return;
isQuitting = true;
// macOS特殊处理保持应用活跃
if (process.platform === 'darwin') {
return;
}
console.log('所有窗口已关闭,执行清理脚本...');
getServer().close(() => {
// 执行批处理文件
const cleanupProcess = exec(stopBatPath.substring(1, 200), (error, stdout, stderr) => {
if (error) {
console.error(`清理脚本执行失败: ${error.message}`);
}
if (stderr) {
console.error(`清理脚本错误输出: ${stderr}`);
}
if (stdout) {
console.log(`清理脚本输出: ${stdout}`);
}
// 脚本执行完成后强制退出
forceQuit();
});
// 监听子进程退出事件(确保即使脚本出错也能退出)
cleanupProcess.on('exit', (code) => {
console.log(`清理脚本退出,代码: ${code}`);
if (isQuitting) {
forceQuit();
}
});
// 超时保护:防止脚本卡住导致应用无法退出
setTimeout(() => {
if (isQuitting) {
console.log('清理脚本执行超时,强制退出');
cleanupProcess.kill(); // 终止卡住的脚本
forceQuit();
}
}, 2000); // 2秒超时
})
}
// 强制退出应用的辅助函数
function forceQuit() {
// 终止所有剩余的子进程
if (process.platform === 'win32') {
// Windows系统特殊处理
exec('taskkill /F /T /PID ' + process.pid, (error) => {
if (error) console.error('强制终止失败:', error);
});
} else {
// 其他系统
process.exit(0);
}
}
2025-09-23 11:17:50 +08:00
console.log('=================================================')
global.sharedObject = {
hasService: false,
};
2025-08-29 09:35:52 +08:00
// In this file you can include the rest of your app's specific main process
// code. You can also put them in separate files and require them here.