mirror of
https://github.com/LifeArchiveProject/WeChatDataAnalysis.git
synced 2026-02-02 22:10:50 +08:00
ci: release on tag
This commit is contained in:
78
.github/workflows/release.yml
vendored
Normal file
78
.github/workflows/release.yml
vendored
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
name: Release (Windows)
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- "v*"
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-windows-installer:
|
||||||
|
runs-on: windows-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Derive version from tag
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
$tag = "${{ github.ref_name }}"
|
||||||
|
$version = $tag -replace '^v', ''
|
||||||
|
if ([string]::IsNullOrWhiteSpace($version)) {
|
||||||
|
throw "Invalid tag: '$tag' (expected like v1.0.0)"
|
||||||
|
}
|
||||||
|
"TAG_NAME=$tag" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
"VERSION=$version" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
|
||||||
|
- name: Setup Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: "20"
|
||||||
|
cache: "npm"
|
||||||
|
cache-dependency-path: |
|
||||||
|
frontend/package-lock.json
|
||||||
|
desktop/package-lock.json
|
||||||
|
|
||||||
|
- name: Setup Python
|
||||||
|
uses: actions/setup-python@v5
|
||||||
|
with:
|
||||||
|
python-version-file: ".python-version"
|
||||||
|
|
||||||
|
- name: Install uv
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
python -m pip install uv
|
||||||
|
|
||||||
|
- name: Install frontend deps
|
||||||
|
working-directory: frontend
|
||||||
|
run: npm ci
|
||||||
|
|
||||||
|
- name: Install desktop deps
|
||||||
|
working-directory: desktop
|
||||||
|
run: npm ci
|
||||||
|
|
||||||
|
- name: Set desktop app version (electron-builder)
|
||||||
|
working-directory: desktop
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
npm version $env:VERSION --no-git-tag-version
|
||||||
|
|
||||||
|
- name: Build Windows installer
|
||||||
|
working-directory: desktop
|
||||||
|
run: npm run dist
|
||||||
|
|
||||||
|
- name: Create GitHub Release and upload installer
|
||||||
|
uses: softprops/action-gh-release@v2
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
tag_name: ${{ env.TAG_NAME }}
|
||||||
|
name: ${{ env.TAG_NAME }}
|
||||||
|
generate_release_notes: true
|
||||||
|
files: |
|
||||||
|
desktop/dist/*Setup*.exe
|
||||||
|
desktop/dist/*Setup*.exe.blockmap
|
||||||
|
|
||||||
@@ -14,7 +14,7 @@
|
|||||||
"build": {
|
"build": {
|
||||||
"appId": "com.lifearchive.wechatdataanalysis",
|
"appId": "com.lifearchive.wechatdataanalysis",
|
||||||
"productName": "WeChatDataAnalysis",
|
"productName": "WeChatDataAnalysis",
|
||||||
"icon": "resources/icon.ico",
|
"icon": "build/icon.ico",
|
||||||
"asar": true,
|
"asar": true,
|
||||||
"directories": {
|
"directories": {
|
||||||
"output": "dist"
|
"output": "dist"
|
||||||
@@ -34,15 +34,18 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"win": {
|
"win": {
|
||||||
"icon": "resources/icon.ico",
|
"icon": "build/icon.ico",
|
||||||
"target": [
|
"target": [
|
||||||
"nsis"
|
"nsis"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"nsis": {
|
"nsis": {
|
||||||
"installerIcon": "resources/icon.ico",
|
"oneClick": false,
|
||||||
"uninstallerIcon": "resources/icon.ico",
|
"allowToChangeInstallationDirectory": true,
|
||||||
"installerHeaderIcon": "resources/icon.ico"
|
"include": "scripts/installer-custom.nsh",
|
||||||
|
"installerIcon": "build/installerIcon.ico",
|
||||||
|
"uninstallerIcon": "build/uninstallerIcon.ico",
|
||||||
|
"installerHeaderIcon": "build/installerHeaderIcon.ico"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|||||||
@@ -6,7 +6,15 @@ const { PNG } = require("pngjs");
|
|||||||
|
|
||||||
const repoRoot = path.resolve(__dirname, "..", "..");
|
const repoRoot = path.resolve(__dirname, "..", "..");
|
||||||
const srcPng = path.join(repoRoot, "frontend", "public", "logo.png");
|
const srcPng = path.join(repoRoot, "frontend", "public", "logo.png");
|
||||||
const dstIco = path.join(repoRoot, "desktop", "resources", "icon.ico");
|
// Write the generated ICO to the locations electron-builder expects for app+installer icons.
|
||||||
|
// Also keep a copy in desktop/resources for convenience.
|
||||||
|
const dstIcos = [
|
||||||
|
path.join(repoRoot, "desktop", "resources", "icon.ico"),
|
||||||
|
path.join(repoRoot, "desktop", "build", "icon.ico"),
|
||||||
|
path.join(repoRoot, "desktop", "build", "installerIcon.ico"),
|
||||||
|
path.join(repoRoot, "desktop", "build", "uninstallerIcon.ico"),
|
||||||
|
path.join(repoRoot, "desktop", "build", "installerHeaderIcon.ico"),
|
||||||
|
];
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
if (!fs.existsSync(srcPng)) {
|
if (!fs.existsSync(srcPng)) {
|
||||||
@@ -35,11 +43,13 @@ async function main() {
|
|||||||
fs.writeFileSync(tmpPng, PNG.sync.write(square));
|
fs.writeFileSync(tmpPng, PNG.sync.write(square));
|
||||||
|
|
||||||
const buf = await pngToIco(tmpPng);
|
const buf = await pngToIco(tmpPng);
|
||||||
fs.mkdirSync(path.dirname(dstIco), { recursive: true });
|
for (const dstIco of dstIcos) {
|
||||||
fs.writeFileSync(dstIco, buf);
|
fs.mkdirSync(path.dirname(dstIco), { recursive: true });
|
||||||
|
fs.writeFileSync(dstIco, buf);
|
||||||
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(`Generated icon: ${dstIco}`);
|
console.log(`Generated icon(s):\n${dstIcos.map((p) => `- ${p}`).join("\n")}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
main().catch((err) => {
|
main().catch((err) => {
|
||||||
|
|||||||
80
desktop/scripts/installer-custom.nsh
Normal file
80
desktop/scripts/installer-custom.nsh
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
; This file is included for both installer and uninstaller builds.
|
||||||
|
; Guard installer-only pages/functions to avoid "function not referenced" warnings
|
||||||
|
; when electron-builder compiles the standalone uninstaller.
|
||||||
|
!ifndef BUILD_UNINSTALLER
|
||||||
|
!include nsDialogs.nsh
|
||||||
|
!include LogicLib.nsh
|
||||||
|
|
||||||
|
; Directory page is a "parent folder" picker. When users browse to a new folder,
|
||||||
|
; NSIS will set $INSTDIR to exactly what they pick (without app sub-folder),
|
||||||
|
; and electron-builder later appends "\${APP_FILENAME}" before installation.
|
||||||
|
; Make this explicit on the directory page to reduce confusion.
|
||||||
|
!define /ifndef MUI_DIRECTORYPAGE_TEXT_TOP "请选择安装位置(将自动创建并使用“${APP_FILENAME}”子文件夹)。"
|
||||||
|
!define /ifndef MUI_DIRECTORYPAGE_TEXT_DESTINATION "安装位置:"
|
||||||
|
|
||||||
|
Var WDA_InstallDirPage
|
||||||
|
|
||||||
|
!macro customPageAfterChangeDir
|
||||||
|
; Add a confirmation page after the directory picker so users clearly see
|
||||||
|
; the final install location (includes the app sub-folder).
|
||||||
|
!ifdef allowToChangeInstallationDirectory
|
||||||
|
Page custom WDA_InstallDirPageCreate WDA_InstallDirPageLeave
|
||||||
|
!endif
|
||||||
|
!macroend
|
||||||
|
|
||||||
|
Function WDA_EnsureAppSubDir
|
||||||
|
; Normalize $INSTDIR to always end with "\${APP_FILENAME}" (avoid cluttering a parent folder).
|
||||||
|
StrCpy $0 "$INSTDIR"
|
||||||
|
|
||||||
|
; Trim trailing "\" (except for drive root like "C:\").
|
||||||
|
StrLen $1 "$0"
|
||||||
|
${If} $1 > 3
|
||||||
|
StrCpy $2 "$0" 1 -1
|
||||||
|
${If} $2 == "\"
|
||||||
|
IntOp $1 $1 - 1
|
||||||
|
StrCpy $0 "$0" $1
|
||||||
|
${EndIf}
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
|
; If already ends with APP_FILENAME, keep it.
|
||||||
|
StrLen $3 "$0"
|
||||||
|
StrLen $4 "${APP_FILENAME}"
|
||||||
|
${If} $3 >= $4
|
||||||
|
IntOp $5 $3 - $4
|
||||||
|
StrCpy $6 "$0" $4 $5
|
||||||
|
${If} $6 == "${APP_FILENAME}"
|
||||||
|
StrCpy $INSTDIR "$0"
|
||||||
|
Return
|
||||||
|
${EndIf}
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
|
; Otherwise append the app folder name.
|
||||||
|
StrCpy $INSTDIR "$0\${APP_FILENAME}"
|
||||||
|
FunctionEnd
|
||||||
|
|
||||||
|
Function WDA_InstallDirPageCreate
|
||||||
|
Call WDA_EnsureAppSubDir
|
||||||
|
|
||||||
|
nsDialogs::Create 1018
|
||||||
|
Pop $WDA_InstallDirPage
|
||||||
|
|
||||||
|
${If} $WDA_InstallDirPage == error
|
||||||
|
Abort
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
|
${NSD_CreateLabel} 0u 0u 100% 24u "程序将安装到:"
|
||||||
|
Pop $0
|
||||||
|
|
||||||
|
${NSD_CreateLabel} 0u 22u 100% 24u "$INSTDIR"
|
||||||
|
Pop $0
|
||||||
|
|
||||||
|
${NSD_CreateLabel} 0u 50u 100% 36u "为避免把文件直接安装到父目录,安装程序会自动创建“${APP_FILENAME}”子文件夹。"
|
||||||
|
Pop $0
|
||||||
|
|
||||||
|
nsDialogs::Show
|
||||||
|
FunctionEnd
|
||||||
|
|
||||||
|
Function WDA_InstallDirPageLeave
|
||||||
|
FunctionEnd
|
||||||
|
|
||||||
|
!endif
|
||||||
Reference in New Issue
Block a user