feat: init
30
.gitignore
vendored
Normal file
@ -0,0 +1,30 @@
|
||||
# dependencies
|
||||
/node_modules
|
||||
/.pnp
|
||||
.pnp.js
|
||||
|
||||
# testing
|
||||
/coverage
|
||||
|
||||
# production
|
||||
/build-web
|
||||
/build-electron
|
||||
/build
|
||||
/dist
|
||||
/packs
|
||||
|
||||
# misc
|
||||
.DS_Store
|
||||
.env.local
|
||||
.env.development.local
|
||||
.env.test.local
|
||||
.env.production.local
|
||||
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
|
||||
yarn.lock
|
||||
package-lock.json
|
||||
|
||||
.parcel-cache
|
0
.prettierignore
Normal file
15
.prettierrc
Normal file
@ -0,0 +1,15 @@
|
||||
{
|
||||
"singleQuote": true,
|
||||
"trailingComma": "all",
|
||||
"printWidth": 100,
|
||||
"proseWrap": "never",
|
||||
"arrowParens": "avoid",
|
||||
"overrides": [
|
||||
{
|
||||
"files": ".prettierrc",
|
||||
"options": {
|
||||
"parser": "json"
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
17
config-overrides.js
Normal file
@ -0,0 +1,17 @@
|
||||
const { override, addLessLoader, adjustStyleLoaders } = require('customize-cra');
|
||||
|
||||
module.exports = override(
|
||||
addLessLoader({
|
||||
lessOptions: {
|
||||
javascriptEnabled: true,
|
||||
},
|
||||
}),
|
||||
adjustStyleLoaders(({ use: [, , postcss] }) => {
|
||||
const postcssOptions = postcss.options;
|
||||
postcss.options = { postcssOptions };
|
||||
}),
|
||||
function (config) {
|
||||
config.target = 'electron-renderer';
|
||||
return config;
|
||||
},
|
||||
);
|
24
electron/const.js
Normal file
@ -0,0 +1,24 @@
|
||||
import path from "path";
|
||||
import isDev from "electron-is-dev";
|
||||
import url from "url";
|
||||
import { app } from "electron";
|
||||
|
||||
const APP_PATH = app.getAppPath();
|
||||
// 对于一些 shell 去执行的文件,asar 目录下无法使用。配合 extraResources
|
||||
const EXECUTABLE_PATH = path.join(
|
||||
APP_PATH.indexOf("app.asar") > -1
|
||||
? APP_PATH.substring(0, APP_PATH.indexOf("app.asar"))
|
||||
: APP_PATH,
|
||||
"public"
|
||||
);
|
||||
|
||||
export default {
|
||||
APP_START_URL: isDev
|
||||
? "http://localhost:3000"
|
||||
: url.format({
|
||||
pathname: path.join(APP_PATH, "./build/index.html"),
|
||||
protocol: "file:",
|
||||
slashes: true,
|
||||
}),
|
||||
IS_DEV: isDev,
|
||||
};
|
49
electron/index.js
Normal file
@ -0,0 +1,49 @@
|
||||
import { app, BrowserWindow } from "electron";
|
||||
import log from "electron-log";
|
||||
import CONFIG from "./const";
|
||||
import { checkUpdate } from "./utils";
|
||||
|
||||
app.commandLine.appendSwitch("--no-proxy-server");
|
||||
|
||||
function createWindow() {
|
||||
// electron.Menu.setApplicationMenu(null);
|
||||
checkUpdate(
|
||||
"https://cdn.jsdelivr.net/gh/lecepin/electron-react-tpl/package.json",
|
||||
"https://github.com/lecepin/electron-react-tpl/releases"
|
||||
);
|
||||
|
||||
mainWindow = new BrowserWindow({
|
||||
width: 800,
|
||||
height: 600,
|
||||
// resizable: false,
|
||||
// maximizable: false,
|
||||
webPreferences: {
|
||||
webSecurity: false,
|
||||
nodeIntegration: true,
|
||||
contextIsolation: false,
|
||||
},
|
||||
});
|
||||
|
||||
mainWindow.loadURL(CONFIG.APP_START_URL);
|
||||
CONFIG.IS_DEV && mainWindow.webContents.openDevTools();
|
||||
|
||||
mainWindow.on("closed", function () {
|
||||
mainWindow = null;
|
||||
});
|
||||
}
|
||||
|
||||
app.whenReady().then(() => {
|
||||
createWindow();
|
||||
|
||||
app.on("activate", () => {
|
||||
if (BrowserWindow.getAllWindows().length === 0) {
|
||||
createWindow();
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
app.on("window-all-closed", () => {
|
||||
if (process.platform !== "darwin") {
|
||||
app.quit();
|
||||
}
|
||||
});
|
30
electron/utils.js
Normal file
@ -0,0 +1,30 @@
|
||||
import { get } from "axios";
|
||||
const { app, dialog, shell } = require("electron");
|
||||
import semver from "semver";
|
||||
|
||||
// packageUrl 需要包含 { "version": "1.0.0" } 结构
|
||||
function checkUpdate(
|
||||
// 可以使用加速地址 https://cdn.jsdelivr.net/gh/lecepin/electron-react-tpl/package.json
|
||||
packageUrl = "https://raw.githubusercontent.com/lecepin/electron-react-tpl/master/package.json",
|
||||
downloadUrl = "https://github.com/lecepin/electron-react-tpl/releases"
|
||||
) {
|
||||
get(packageUrl)
|
||||
.then(({ data }) => {
|
||||
if (semver.gt(data?.version, app.getVersion())) {
|
||||
const result = dialog.showMessageBoxSync({
|
||||
message: "发现新版本,是否更新?",
|
||||
type: "question",
|
||||
cancelId: 1,
|
||||
defaultId: 0,
|
||||
buttons: ["进入新版本下载页面", "取消"],
|
||||
});
|
||||
|
||||
if (result === 0 && downloadUrl) {
|
||||
shell.openExternal(downloadUrl);
|
||||
}
|
||||
}
|
||||
})
|
||||
.catch((err) => {});
|
||||
}
|
||||
|
||||
export { checkUpdate };
|
110
package.json
Normal file
@ -0,0 +1,110 @@
|
||||
{
|
||||
"name": "electron-react-tpl",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"main": "./build-electron/index.js",
|
||||
"homepage": "./",
|
||||
"scripts": {
|
||||
"postinstall": "husky install",
|
||||
"start": "concurrently \"cross-env BROWSER=none npm run start-web\" \"wait-on http://localhost:3000 && npm run start-electron\" ",
|
||||
"start-web": "react-app-rewired start",
|
||||
"start-electron": "parcel build --target electron --no-cache && electron .",
|
||||
"build-web": "react-app-rewired build",
|
||||
"build-electron": "parcel build --target electron --no-cache",
|
||||
"build-all": "rm -rf ./build && rm -rf ./build-electron && npm run build-electron && npm run build-web",
|
||||
"pack": "npm run build-all && electron-builder",
|
||||
"gen-icon": "electron-icon-builder --input=./public/icon/icon.png --output=./public/icon"
|
||||
},
|
||||
"targets": {
|
||||
"electron": {
|
||||
"source": "electron/index.js",
|
||||
"context": "electron-main",
|
||||
"distDir": "build-electron"
|
||||
}
|
||||
},
|
||||
"husky": {
|
||||
"hooks": {
|
||||
"pre-commit": "prettier -c --write \"(src/**/*|electron/**/*)\" && git add -A ."
|
||||
}
|
||||
},
|
||||
"devDependencies": {
|
||||
"concurrently": "^7.1.0",
|
||||
"cross-env": "^7.0.3",
|
||||
"customize-cra": "^1.0.0",
|
||||
"electron": "^18.2.3",
|
||||
"electron-builder": "^23.0.3",
|
||||
"electron-icon-builder": "^2.0.1",
|
||||
"husky": "^8.0.1",
|
||||
"less": "^4.1.2",
|
||||
"less-loader": "^11.0.0",
|
||||
"parcel": "^2.5.0",
|
||||
"prettier": "^2.6.2",
|
||||
"react-app-rewired": "^2.2.1",
|
||||
"react-scripts": "5.0.1",
|
||||
"wait-on": "^6.0.1"
|
||||
},
|
||||
"dependencies": {
|
||||
"axios": "^0.27.2",
|
||||
"electron-is-dev": "^2.0.0",
|
||||
"electron-log": "^4.4.7",
|
||||
"react": "^18.1.0",
|
||||
"react-dom": "^18.1.0",
|
||||
"semver": "^7.3.7"
|
||||
},
|
||||
"author": "lecepin",
|
||||
"license": "ISC",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/lecepin/electron-react-tpl.git"
|
||||
},
|
||||
"build": {
|
||||
"extends": null,
|
||||
"productName": "Lecepin Tpl",
|
||||
"appId": "com.lecepin.tpl",
|
||||
"directories": {
|
||||
"output": "packs"
|
||||
},
|
||||
"npmRebuild": false,
|
||||
"files": [
|
||||
"build/**/*",
|
||||
"build-electron/**/*",
|
||||
"public/**/*"
|
||||
],
|
||||
"mac": {
|
||||
"icon": "public/icon/icons/mac/icon.icns"
|
||||
},
|
||||
"win": {
|
||||
"target": [
|
||||
{
|
||||
"target": "nsis",
|
||||
"arch": [
|
||||
"x64",
|
||||
"ia32"
|
||||
]
|
||||
}
|
||||
],
|
||||
"icon": "public/icon/icons/win/icon.ico"
|
||||
},
|
||||
"nsis": {
|
||||
"oneClick": false,
|
||||
"perMachine": false,
|
||||
"allowElevation": true,
|
||||
"allowToChangeInstallationDirectory": true
|
||||
},
|
||||
"extraResources": [
|
||||
"public"
|
||||
]
|
||||
},
|
||||
"browserslist": {
|
||||
"production": [
|
||||
">0.2%",
|
||||
"not dead",
|
||||
"not op_mini all"
|
||||
],
|
||||
"development": [
|
||||
"last 1 chrome version",
|
||||
"last 1 firefox version",
|
||||
"last 1 safari version"
|
||||
]
|
||||
}
|
||||
}
|
BIN
public/icon/icon.png
Normal file
After Width: | Height: | Size: 194 KiB |
BIN
public/icon/icons/mac/icon.icns
Normal file
BIN
public/icon/icons/png/1024x1024.png
Normal file
After Width: | Height: | Size: 199 KiB |
BIN
public/icon/icons/png/128x128.png
Normal file
After Width: | Height: | Size: 12 KiB |
BIN
public/icon/icons/png/16x16.png
Normal file
After Width: | Height: | Size: 716 B |
BIN
public/icon/icons/png/24x24.png
Normal file
After Width: | Height: | Size: 1.3 KiB |
BIN
public/icon/icons/png/256x256.png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
public/icon/icons/png/32x32.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
public/icon/icons/png/48x48.png
Normal file
After Width: | Height: | Size: 3.4 KiB |
BIN
public/icon/icons/png/512x512.png
Normal file
After Width: | Height: | Size: 68 KiB |
BIN
public/icon/icons/png/64x64.png
Normal file
After Width: | Height: | Size: 5.0 KiB |
BIN
public/icon/icons/win/icon.ico
Normal file
After Width: | Height: | Size: 353 KiB |
12
public/index.html
Normal file
@ -0,0 +1,12 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<title></title>
|
||||
</head>
|
||||
<body>
|
||||
<noscript>You need to enable JavaScript to run this app.</noscript>
|
||||
<div id="root"></div>
|
||||
</body>
|
||||
</html>
|
38
src/App.css
Normal file
@ -0,0 +1,38 @@
|
||||
.App {
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.App-logo {
|
||||
height: 40vmin;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
.App-logo {
|
||||
animation: App-logo-spin infinite 20s linear;
|
||||
}
|
||||
}
|
||||
|
||||
.App-header {
|
||||
background-color: #282c34;
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
font-size: calc(10px + 2vmin);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.App-link {
|
||||
color: #61dafb;
|
||||
}
|
||||
|
||||
@keyframes App-logo-spin {
|
||||
from {
|
||||
transform: rotate(0deg);
|
||||
}
|
||||
to {
|
||||
transform: rotate(360deg);
|
||||
}
|
||||
}
|
30
src/App.jsx
Normal file
@ -0,0 +1,30 @@
|
||||
import logo from "./logo.png";
|
||||
import { shell } from "electron";
|
||||
import "./App.css";
|
||||
|
||||
function App() {
|
||||
return (
|
||||
<div className="App">
|
||||
<header className="App-header">
|
||||
<img src={logo} className="App-logo" alt="logo" />
|
||||
<p>
|
||||
Edit <code>src/App.js</code> and save to reload.
|
||||
</p>
|
||||
<a
|
||||
onClick={(e) => {
|
||||
e.preventDefault();
|
||||
shell.openExternal("https://github.com/lecepin/electron-react-tpl");
|
||||
}}
|
||||
className="App-link"
|
||||
href="#"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
Open Github
|
||||
</a>
|
||||
</header>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default App;
|
13
src/index.css
Normal file
@ -0,0 +1,13 @@
|
||||
body {
|
||||
margin: 0;
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
|
||||
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
|
||||
sans-serif;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
}
|
||||
|
||||
code {
|
||||
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
|
||||
monospace;
|
||||
}
|
11
src/index.js
Normal file
@ -0,0 +1,11 @@
|
||||
import React from 'react';
|
||||
import ReactDOM from 'react-dom/client';
|
||||
import './index.css';
|
||||
import App from './App';
|
||||
|
||||
const root = ReactDOM.createRoot(document.getElementById('root'));
|
||||
root.render(
|
||||
<React.StrictMode>
|
||||
<App />
|
||||
</React.StrictMode>
|
||||
);
|
BIN
src/logo.png
Normal file
After Width: | Height: | Size: 194 KiB |