223 lines
6.6 KiB
TypeScript
223 lines
6.6 KiB
TypeScript
/* eslint-disable prettier/prettier */
|
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
import { electronApp, is, optimizer } from '@electron-toolkit/utils'
|
|
import { app, BrowserWindow, globalShortcut, ipcMain, Menu, screen, shell, Tray } from 'electron'
|
|
import fs from 'fs'
|
|
import path, { join } from 'path'
|
|
import icon from '../../resources/icon.png?asset'
|
|
|
|
let mainWindow: null | BrowserWindow = null
|
|
let isQuiting = false
|
|
const startupFlagFile = path.join(app.getPath('userData'), 'startup-set.flag')
|
|
|
|
function createWindow(): void {
|
|
// Get Screen width, height
|
|
const { width, height } = screen.getPrimaryDisplay().workAreaSize
|
|
|
|
mainWindow = new BrowserWindow({
|
|
width: width / 2,
|
|
height: height / 2,
|
|
show: false,
|
|
autoHideMenuBar: true,
|
|
...(process.platform === 'linux' ? { icon } : {}),
|
|
webPreferences: {
|
|
preload: join(__dirname, '../preload/index.js'),
|
|
sandbox: false
|
|
}
|
|
})
|
|
|
|
// Set App Show Position
|
|
mainWindow.setPosition(width / 2, height / 2)
|
|
|
|
mainWindow.on('ready-to-show', () => {
|
|
mainWindow?.show()
|
|
})
|
|
|
|
mainWindow.webContents.setWindowOpenHandler((details) => {
|
|
shell.openExternal(details.url)
|
|
return { action: 'deny' }
|
|
})
|
|
|
|
// Make the window always on top
|
|
mainWindow.setAlwaysOnTop(true, 'normal')
|
|
|
|
// mainWindow.webContents.openDevTools()
|
|
|
|
// Inspect element with shortcut
|
|
mainWindow.webContents.once('did-finish-load', () => {
|
|
const shortcut = 'Control+Shift+C'
|
|
|
|
mainWindow?.on('focus', () => {
|
|
globalShortcut.register(shortcut, () => {
|
|
const pos = screen.getCursorScreenPoint()
|
|
mainWindow?.webContents.inspectElement(pos.x, pos.y)
|
|
mainWindow?.webContents.focus() // optional: refocus back
|
|
})
|
|
})
|
|
|
|
mainWindow?.on('blur', () => {
|
|
globalShortcut.unregister(shortcut)
|
|
})
|
|
|
|
mainWindow?.on('closed', () => {
|
|
globalShortcut.unregister(shortcut)
|
|
})
|
|
})
|
|
|
|
// 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'))
|
|
}
|
|
|
|
// Khi bấm dấu X
|
|
mainWindow.on('close', (event) => {
|
|
if (!isQuiting) {
|
|
event.preventDefault()
|
|
mainWindow?.hide()
|
|
}
|
|
})
|
|
}
|
|
|
|
function createTray() {
|
|
const tray = new Tray(icon)
|
|
|
|
const contextMenu = Menu.buildFromTemplate([
|
|
{
|
|
label: 'Show',
|
|
click: () => {
|
|
mainWindow?.show()
|
|
}
|
|
},
|
|
{
|
|
label: 'Quit',
|
|
click: () => {
|
|
isQuiting = true
|
|
app.quit()
|
|
}
|
|
}
|
|
])
|
|
|
|
tray.setToolTip('Zulip notes')
|
|
tray.setContextMenu(contextMenu)
|
|
|
|
tray.on('double-click', () => {
|
|
mainWindow?.show()
|
|
})
|
|
}
|
|
|
|
// 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(() => {
|
|
if (!fs.existsSync(startupFlagFile)) {
|
|
app.setLoginItemSettings({
|
|
openAtLogin: true,
|
|
path: process.execPath,
|
|
args: []
|
|
})
|
|
|
|
fs.writeFileSync(startupFlagFile, 'ok')
|
|
}
|
|
|
|
// 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'))
|
|
|
|
// Create main window
|
|
createWindow()
|
|
|
|
// Create tray icon
|
|
createTray()
|
|
|
|
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.
|
|
app.on('window-all-closed', () => {
|
|
if (process.platform !== 'darwin') {
|
|
app.quit()
|
|
}
|
|
})
|
|
|
|
// 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.
|
|
// Custom events
|
|
|
|
ipcMain.handle('save-config-file', async (_event, jsonContent: any) => {
|
|
// const filePath = path.join(__dirname, 'config-data.json')
|
|
const userDataPath = app.getPath('userData')
|
|
const filePath = path.join(userDataPath, 'config-data.json')
|
|
|
|
try {
|
|
let fileData: any = {}
|
|
|
|
// 👀 Kiểm tra nếu file đã tồn tại
|
|
if (fs.existsSync(filePath)) {
|
|
const rawData = fs.readFileSync(filePath, 'utf-8')
|
|
fileData = JSON.parse(rawData)
|
|
}
|
|
|
|
const now = new Date().getTime()
|
|
|
|
// 📦 Ghi dữ liệu mới
|
|
const newData = {
|
|
...jsonContent, // giữ dữ liệu mới (nếu bạn muốn merge thì sửa dòng này)
|
|
created_at: fileData.created_at || now, // nếu có created_at thì giữ nguyên, không có thì lấy now
|
|
updated_at: now // luôn cập nhật updated_at
|
|
}
|
|
|
|
fs.writeFileSync(filePath, JSON.stringify(newData, null, 2), 'utf-8')
|
|
|
|
return { success: true, path: filePath }
|
|
} catch (error: any) {
|
|
console.error('Error saving file:', error)
|
|
return { success: false, error: error.message }
|
|
}
|
|
})
|
|
|
|
ipcMain.handle('get-config-file', async () => {
|
|
const userDataPath = app.getPath('userData')
|
|
const filePath = path.join(userDataPath, 'config-data.json')
|
|
|
|
try {
|
|
// 📂 Kiểm tra file có tồn tại không
|
|
if (!fs.existsSync(filePath)) {
|
|
return { success: false, error: 'Config file does not exist' }
|
|
}
|
|
|
|
const rawData = fs.readFileSync(filePath, 'utf-8')
|
|
const fileData = JSON.parse(rawData)
|
|
|
|
return fileData
|
|
} catch (error: any) {
|
|
console.error('Error reading config file:', error)
|
|
return null
|
|
}
|
|
})
|
|
|
|
ipcMain.handle('show-window', async () => {
|
|
mainWindow?.show()
|
|
})
|
|
|
|
ipcMain.handle('hide-window', async () => {
|
|
mainWindow?.hide()
|
|
})
|