From 57a7f3745de5a816dca875b509b00d91d4e966a6 Mon Sep 17 00:00:00 2001 From: lecepin <383810086@qq.com> Date: Sat, 21 May 2022 22:47:51 +0800 Subject: [PATCH] =?UTF-8?q?feat:=20=E5=85=A8=E9=83=A8=E5=AE=8C=E6=88=90?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- electron/index.js | 5 +- electron/ipc.js | 42 ++++++++++-- electron/proxyServer.js | 56 ++++++++++------ electron/utils.js | 36 ++++++++++- package.json | 4 ++ src/App.jsx | 134 ++++++++++++++++++++++++++++++++++++-- src/App.less | 50 +++++++++++++++ src/fsm.js | 139 ++++++++++++++++++++++++++++++++++++---- src/index.js | 2 + 9 files changed, 422 insertions(+), 46 deletions(-) diff --git a/electron/index.js b/electron/index.js index 6d0c282..7f2bbb8 100644 --- a/electron/index.js +++ b/electron/index.js @@ -1,9 +1,11 @@ import { app, BrowserWindow, Menu } from 'electron'; import CONFIG from './const'; import { checkUpdate } from './utils'; -import initIPC from './ipc'; +import initIPC, { setWin } from './ipc'; app.commandLine.appendSwitch('--no-proxy-server'); +process.on('uncaughtException', () => {}); +process.on('unhandledRejection', () => {}); function createWindow() { Menu.setApplicationMenu(null); @@ -24,6 +26,7 @@ function createWindow() { }, }); + setWin(mainWindow); mainWindow.loadURL(CONFIG.APP_START_URL); CONFIG.IS_DEV && mainWindow.webContents.openDevTools(); } diff --git a/electron/ipc.js b/electron/ipc.js index 5312c51..99a7616 100644 --- a/electron/ipc.js +++ b/electron/ipc.js @@ -1,7 +1,11 @@ -import { ipcMain } from 'electron'; +import { ipcMain, dialog } from 'electron'; import log from 'electron-log'; +import { throttle } from 'lodash'; import { startServer } from './proxyServer'; import { installCert, checkCertInstalled } from './cert'; +import { downloadFile } from './utils'; + +let win; export default function initIPC() { ipcMain.handle('invoke_初始化信息', async (event, arg) => { @@ -13,14 +17,42 @@ export default function initIPC() { }); ipcMain.handle('invoke_启动服务', async (event, arg) => { - console.log('invoke_启动服务'); return startServer({ - interceptCallback: async req => { - console.log('=========> intercept', req.url); + interceptCallback: phase => async (req, res) => { + if (phase === 'response' && res?._data?.headers?.['content-type'] == 'video/mp4') { + win?.webContents?.send?.('VIDEO_CAPTURE', { + url: req.fullUrl(), + size: res?._data?.headers?.['content-length'] ?? 0, + }); + } }, setProxyErrorCallback: err => { - console.log({ err }); + console.log('开启代理失败', err); }, }); }); + + ipcMain.handle('invoke_选择下载位置', async (event, arg) => { + const result = dialog.showOpenDialogSync({ title: '保存', properties: ['openDirectory'] }); + + if (!result?.[0]) { + throw '取消'; + } + + return result?.[0]; + }); + + ipcMain.handle('invoke_下载视频', async (event, { url, savePath }) => { + return downloadFile( + url, + `${savePath}/${Date.now()}.mp4`, + throttle(value => win?.webContents?.send?.('e_进度变化', value), 1000), + ).catch(err => { + console; + }); + }); +} + +export function setWin(w) { + win = w; } diff --git a/electron/proxyServer.js b/electron/proxyServer.js index 24ab503..4ff39dd 100644 --- a/electron/proxyServer.js +++ b/electron/proxyServer.js @@ -5,34 +5,50 @@ import { app } from 'electron'; import CONFIG from './const'; import { setProxy, closeProxy } from './setProxy'; -export async function startServer({ interceptCallback = f => f, setProxyErrorCallback = f => f }) { +export async function startServer({ + interceptCallback = f => f => f, + setProxyErrorCallback = f => f, +}) { const port = await getPort(); - const proxy = hoxy - .createServer({ - certAuthority: { - key: fs.readFileSync(CONFIG.CERT_PRIVATE_PATH), - cert: fs.readFileSync(CONFIG.CERT_PUBLIC_PATH), - }, - }) - .listen(port, () => { - setProxy('127.0.0.1', port).catch(setProxyErrorCallback); - }) - .on('error', e => { - console.log('proxy lib error', e); - }); - proxy.intercept( - { - phase: 'request', - }, - interceptCallback, - ); + return new Promise(async (resolve, reject) => { + const proxy = hoxy + .createServer({ + certAuthority: { + key: fs.readFileSync(CONFIG.CERT_PRIVATE_PATH), + cert: fs.readFileSync(CONFIG.CERT_PUBLIC_PATH), + }, + }) + .listen(port, () => { + setProxy('127.0.0.1', port) + .then(() => resolve()) + .catch(() => { + setProxyErrorCallback(data); + reject('设置代理失败'); + }); + }); + + proxy.intercept( + { + phase: 'request', + }, + interceptCallback('request'), + ); + + proxy.intercept( + { + phase: 'response', + }, + interceptCallback('response'), + ); + }); } app.on('before-quit', async e => { e.preventDefault(); try { await closeProxy(); + console.log('close proxy success'); } catch (error) {} app.exit(); diff --git a/electron/utils.js b/electron/utils.js index 61e9648..d900c77 100644 --- a/electron/utils.js +++ b/electron/utils.js @@ -1,6 +1,7 @@ import { get } from 'axios'; -const { app, dialog, shell } = require('electron'); +import { app, dialog, shell } from 'electron'; import semver from 'semver'; +import fs from 'fs'; // packageUrl 需要包含 { "version": "1.0.0" } 结构 function checkUpdate( @@ -27,4 +28,35 @@ function checkUpdate( .catch(err => {}); } -export { checkUpdate }; +function downloadFile(url, fullFileName, progressCallback) { + return get(url, { + responseType: 'stream', + headers: { + 'User-Agent': + 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.127 Safari/537.36', + }, + }).then(({ data, headers }) => { + let currentLen = 0; + const totalLen = headers['content-length']; + + return new Promise((resolve, reject) => { + data.on('data', ({ length }) => { + currentLen += length; + progressCallback?.(currentLen / totalLen); + }); + + data.on('error', err => reject(err)); + + data.pipe( + fs.createWriteStream(fullFileName).on('finish', () => { + resolve({ + fullFileName, + totalLen, + }); + }), + ); + }); + }); +} + +export { checkUpdate, downloadFile }; diff --git a/package.json b/package.json index 18de558..e832325 100644 --- a/package.json +++ b/package.json @@ -48,13 +48,17 @@ "webpack-cli": "^4.9.2" }, "dependencies": { + "@ant-design/icons": "^4.7.0", "@xstate/react": "^3.0.0", + "antd": "^4.20.5", "axios": "^0.27.2", "electron-is-dev": "^2.0.0", "electron-log": "^4.4.7", "get-port": "^6.1.2", "hoxy": "^3.3.1", + "lodash": "^4.17.21", "mkdirp": "^1.0.4", + "pretty-bytes": "^6.0.0", "react": "^18.1.0", "react-dom": "^18.1.0", "regedit": "5.0.0", diff --git a/src/App.jsx b/src/App.jsx index 3c1ece6..2e77ffc 100644 --- a/src/App.jsx +++ b/src/App.jsx @@ -1,20 +1,142 @@ import { useMachine } from '@xstate/react'; +import { Table, Button, Progress, Alert } from 'antd'; +import { shell } from 'electron'; +import { + DownloadOutlined, + PlaySquareOutlined, + ClearOutlined, + GithubOutlined, + EyeOutlined, + FormatPainterOutlined, + RedoOutlined, +} from '@ant-design/icons'; import fsm from './fsm'; import './App.less'; function App() { const [state, send] = useMachine(fsm); - const {} = state.context; + const { captureList, currentUrl, downloadProgress } = state.context; return (
未初始化
- - +