Compare commits
229 Commits
v4.2.5-bet
...
v4.3.11
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e746756e56 | ||
|
|
1829d1cd0b | ||
|
|
fb979e5639 | ||
|
|
e7d0a85ad5 | ||
|
|
a384711327 | ||
|
|
3fd4778a48 | ||
|
|
4841dc09b3 | ||
|
|
b3aa4fc776 | ||
|
|
a9b3b8b6f4 | ||
|
|
56ef196695 | ||
| 242238d341 | |||
| f66f6d38fe | |||
| d58077f58b | |||
| 4d4d6dbedf | |||
| f60b276916 | |||
| 87857fd499 | |||
| 3c371cd079 | |||
|
|
428b849bcc | ||
|
|
85f3b4f607 | ||
|
|
916396f855 | ||
|
|
211c8d2b04 | ||
|
|
92e274d3fd | ||
|
|
d511ea48d5 | ||
|
|
1aa4da1adf | ||
|
|
0e8b6b0b6b | ||
|
|
1a2c1b976f | ||
|
|
1cc242fa51 | ||
|
|
18dfdba15d | ||
|
|
b04ac4eec6 | ||
|
|
c009f0c891 | ||
|
|
d2dc0bd295 | ||
|
|
ddbb5b7f19 | ||
|
|
954c25090b | ||
|
|
0b6cc59de1 | ||
|
|
2271b5741d | ||
|
|
8a438b041f | ||
|
|
dd92fcc4d8 | ||
|
|
8f66ca0e16 | ||
|
|
895ba1d24a | ||
| e49b807bef | |||
|
|
73c15b5e93 | ||
|
|
e505ea8c51 | ||
|
|
21e7df7c3e | ||
|
|
2d72ca66a4 | ||
|
|
4725a30165 | ||
|
|
f3c977f1b3 | ||
|
|
9a0e7265c6 | ||
|
|
3f8e2fbe6b | ||
|
|
590b13e916 | ||
|
|
0f6aee56e5 | ||
|
|
daf18e7295 | ||
|
|
9bcc87f663 | ||
|
|
e7205ce0aa | ||
|
|
e3c4b2edc8 | ||
|
|
222a3b35a2 | ||
|
|
cd5dfd56b2 | ||
|
|
7d5c6b8222 | ||
|
|
4dbf4736e4 | ||
|
|
d50504181e | ||
|
|
c7e94dfcd1 | ||
|
|
a752b67ca1 | ||
|
|
078736337d | ||
|
|
de1058a28c | ||
|
|
740797a689 | ||
|
|
26328920a2 | ||
|
|
9c447bbdf9 | ||
|
|
fac85a889f | ||
|
|
f5d898c89e | ||
| 974a4b634a | |||
| 3127c83603 | |||
|
|
8d69e43f72 | ||
|
|
86df9e7a50 | ||
| 59ff9bf818 | |||
|
|
1641e32e3d | ||
|
|
a482087abd | ||
| bc5b15cec2 | |||
| 3787c25a77 | |||
|
|
0b06b499e4 | ||
|
|
04079dd57b | ||
|
|
34ac0c5ab3 | ||
| 0d904b229e | |||
| c0f887ff9d | |||
|
|
cf95075d01 | ||
|
|
d78a764d87 | ||
|
|
a368f4b722 | ||
|
|
803fe4568f | ||
|
|
1162d5dcc1 | ||
|
|
aa83058e39 | ||
|
|
061f205224 | ||
|
|
5d966f98df | ||
|
|
0037914db8 | ||
|
|
13d0115475 | ||
|
|
5bdb5ad2bf | ||
| a5d733c8df | |||
|
|
0b038e2997 | ||
|
|
5e46040db6 | ||
|
|
f2b04dd0f6 | ||
|
|
2177c1b40e | ||
|
|
d1f4cffe8f | ||
|
|
74ce441b90 | ||
|
|
5893aa2426 | ||
|
|
cb7e7bf9d4 | ||
|
|
fbfdc6aa12 | ||
|
|
e7b6743e10 | ||
|
|
ff4283e917 | ||
|
|
890886d62d | ||
|
|
fd75dda2b1 | ||
|
|
f22c1aeae3 | ||
|
|
d68d49a469 | ||
|
|
1900d4eaf5 | ||
|
|
02833209d5 | ||
| 2058c0218c | |||
|
|
8896e723eb | ||
|
|
edcc614833 | ||
|
|
23fe1ff0be | ||
|
|
19d1dc9f28 | ||
|
|
24b93cfcad | ||
|
|
d3298fac8a | ||
|
|
fba5395bf0 | ||
|
|
2c4508ee16 | ||
| d239443555 | |||
| e45ad08fab | |||
| ddf5d26c4b | |||
|
|
ce74dcf912 | ||
|
|
41412e1ef4 | ||
|
|
1395d48cd0 | ||
|
|
418c3d4742 | ||
|
|
17ec962a22 | ||
|
|
989ee73549 | ||
|
|
7e452e1253 | ||
|
|
5bdb5c8025 | ||
|
|
924a5fea0b | ||
|
|
b51a57a6ee | ||
|
|
4079188881 | ||
|
|
174163e305 | ||
|
|
0886439685 | ||
|
|
34bf5a4fe8 | ||
|
|
e6a97f2b17 | ||
|
|
fecff625a3 | ||
|
|
6f540036a0 | ||
|
|
86d72aec39 | ||
|
|
39876832f3 | ||
|
|
f3af6ddbbc | ||
|
|
ba7299e20c | ||
|
|
5db9d934b2 | ||
|
|
5c8eebf12c | ||
|
|
e725f6d2b2 | ||
|
|
494b655156 | ||
|
|
2940f2557c | ||
|
|
5e4660670f | ||
| e8d592ae76 | |||
|
|
97ea51df59 | ||
|
|
986061dc97 | ||
|
|
fe1910d16f | ||
|
|
63cb1aaa74 | ||
| 49ebd50077 | |||
|
|
4a6f874210 | ||
|
|
9394c7a9c5 | ||
|
|
7e502420fa | ||
|
|
12f4b764de | ||
|
|
4da4b7d552 | ||
|
|
d38abbbaa0 | ||
|
|
67bf7f649e | ||
|
|
acb35403b0 | ||
|
|
7d5dccc649 | ||
|
|
a7e0e7b217 | ||
|
|
9ce75b2dda | ||
|
|
d2022819f6 | ||
|
|
c8b342ba01 | ||
|
|
63823d5c89 | ||
|
|
cb17cc32da | ||
|
|
14d0e6d438 | ||
|
|
878fbad06a | ||
|
|
deb0506163 | ||
|
|
c4aeb673fd | ||
|
|
915ee59643 | ||
|
|
1568e120be | ||
|
|
d19dd3496d | ||
|
|
62c86ce477 | ||
|
|
c727eddc54 | ||
|
|
9c946ef6dc | ||
|
|
38a04fc4b2 | ||
| bded794647 | |||
|
|
539cb1de99 | ||
| 2e9ff47dbb | |||
|
|
c01079af1b | ||
|
|
cca1acb6f6 | ||
|
|
d7e502e22f | ||
|
|
bbeab360bc | ||
|
|
a78b7fdb29 | ||
| 273fbe2261 | |||
| ba9855c616 | |||
| c54f894f4f | |||
|
|
9f88f92ec0 | ||
|
|
a80e96c2cd | ||
|
|
7774612810 | ||
| 088ea1817c | |||
|
|
f362c8f7ef | ||
|
|
648f42b7e0 | ||
|
|
9a56cc350d | ||
|
|
50cd49217f | ||
|
|
7ed4b7db57 | ||
| b359cd623b | |||
|
|
a363e8dc34 | ||
|
|
52affc0d76 | ||
|
|
fe26f29f93 | ||
|
|
67b8725156 | ||
|
|
2a235b2bc9 | ||
|
|
dd022cf356 | ||
|
|
62e5bb30e2 | ||
|
|
675e11960a | ||
|
|
0c274ecbe0 | ||
|
|
2dfcd3f131 | ||
|
|
053acd138f | ||
|
|
3f20ae62be | ||
|
|
d342c7c827 | ||
|
|
3da0cfd0d0 | ||
|
|
acc4045580 | ||
|
|
6ee577302f | ||
|
|
d52856180a | ||
|
|
d4d479ca20 | ||
|
|
364af4b9c5 | ||
|
|
9e0d81fb1d | ||
|
|
2ee2c37479 | ||
|
|
528925b969 | ||
| 4851b40777 | |||
|
|
6372ad4e0a | ||
|
|
465bc9137e | ||
|
|
e8b6f5d893 |
247
.github/workflows/build-app.yml
vendored
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,24 +16,23 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
name: Build AUTO_MAA
|
name: Build AUTO_MAA
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
workflow_dispatch:
|
||||||
branches: [ "main" ]
|
|
||||||
paths-ignore:
|
|
||||||
- '**.md'
|
|
||||||
- 'LICENSE'
|
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: write
|
||||||
|
actions: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
pre_check:
|
pre_check:
|
||||||
name: Pre Checks
|
name: Pre Checks
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Repo Check
|
- name: Repo Check
|
||||||
id: repo_check
|
id: repo_check
|
||||||
@@ -43,120 +42,236 @@ jobs:
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
exit 0
|
exit 0
|
||||||
|
|
||||||
build_AUTO_MAA:
|
build_AUTO_MAA:
|
||||||
runs-on: windows-latest
|
runs-on: windows-latest
|
||||||
needs: pre_check
|
needs: pre_check
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
- name: Set up Python 3.12
|
|
||||||
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: "3.12"
|
python-version: '3.12'
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
python -m pip install --upgrade pip
|
python -m pip install --upgrade pip
|
||||||
pip install flake8 pytest
|
pip install flake8 pytest
|
||||||
pip install -r requirements.txt
|
pip install -r requirements.txt
|
||||||
|
|
||||||
- name: Lint with flake8
|
- name: Lint with flake8
|
||||||
run: |
|
run: |
|
||||||
# stop the build if there are Python syntax errors or undefined names
|
# stop the build if there are Python syntax errors or undefined names
|
||||||
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
||||||
# exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide
|
# exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide
|
||||||
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics
|
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics
|
||||||
- name: Package
|
|
||||||
id: package
|
- name: Get version
|
||||||
|
id: get_version
|
||||||
run: |
|
run: |
|
||||||
copy app\utils\package.py .\
|
$version = (Get-Content resources/version.json | ConvertFrom-Json).main_version
|
||||||
python package.py
|
echo "main_version=$version" >> $env:GITHUB_OUTPUT
|
||||||
- name: Read version
|
|
||||||
id: read_version
|
- name: Nuitka build main program
|
||||||
|
uses: Nuitka/Nuitka-Action@main
|
||||||
|
with:
|
||||||
|
nuitka-version: 2.7.6
|
||||||
|
script-name: main.py
|
||||||
|
mode: app
|
||||||
|
enable-plugins: pyside6
|
||||||
|
onefile-tempdir-spec: "{TEMP}/AUTO_MAA"
|
||||||
|
windows-console-mode: attach
|
||||||
|
windows-icon-from-ico: resources/icons/AUTO_MAA.ico
|
||||||
|
company-name: AUTO_MAA Team
|
||||||
|
product-name: AUTO_MAA
|
||||||
|
file-version: ${{ steps.get_version.outputs.main_version }}
|
||||||
|
product-version: ${{ steps.get_version.outputs.main_version }}
|
||||||
|
file-description: AUTO_MAA Component
|
||||||
|
copyright: Copyright © 2024-2025 DLmaster361
|
||||||
|
assume-yes-for-downloads: true
|
||||||
|
output-file: AUTO_MAA
|
||||||
|
output-dir: AUTO_MAA
|
||||||
|
|
||||||
|
- name: Upload unsigned main program
|
||||||
|
id: upload-unsigned-main-program
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: AUTO_MAA
|
||||||
|
path: AUTO_MAA/AUTO_MAA.exe
|
||||||
|
|
||||||
|
- name: Sign main program
|
||||||
|
id: sign_main_program
|
||||||
|
uses: signpath/github-action-submit-signing-request@v1.2
|
||||||
|
with:
|
||||||
|
api-token: '${{ secrets.SIGNPATH_API_TOKEN }}'
|
||||||
|
organization-id: '787a1d5f-6177-4f30-9559-d2646473584a'
|
||||||
|
project-slug: 'AUTO_MAA'
|
||||||
|
signing-policy-slug: 'release-signing'
|
||||||
|
artifact-configuration-slug: "AUTO_MAA"
|
||||||
|
github-artifact-id: '${{ steps.upload-unsigned-main-program.outputs.artifact-id }}'
|
||||||
|
wait-for-completion: true
|
||||||
|
output-artifact-directory: 'AUTO_MAA'
|
||||||
|
|
||||||
|
- name: Add other resources
|
||||||
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$MAIN_VERSION=(Get-Content -Path "version_info.txt" -TotalCount 1).Trim()
|
$root = "${{ github.workspace }}"
|
||||||
"AUTO_MAA_version=$MAIN_VERSION" | Out-File -FilePath $env:GITHUB_ENV -Append
|
$ver = "${{ steps.get_version.outputs.main_version }}"
|
||||||
$UPDATER_VERSION=(Get-Content -Path "version_info.txt" -TotalCount 2 | Select-Object -Index 1).Trim()
|
Copy-Item "$root/app" "$root/AUTO_MAA/app" -Recurse
|
||||||
"updater_version=$UPDATER_VERSION" | Out-File -FilePath $env:GITHUB_ENV -Append
|
Copy-Item "$root/resources" "$root/AUTO_MAA/resources" -Recurse
|
||||||
- name: Create Zip
|
Copy-Item "$root/main.py" "$root/AUTO_MAA/"
|
||||||
id: create_zip
|
Copy-Item "$root/requirements.txt" "$root/AUTO_MAA/"
|
||||||
|
Copy-Item "$root/README.md" "$root/AUTO_MAA/"
|
||||||
|
Copy-Item "$root/LICENSE" "$root/AUTO_MAA/"
|
||||||
|
|
||||||
|
- name: Create Inno Setup script
|
||||||
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
Compress-Archive -Path app,resources,main.py,AUTO_MAA.exe,requirements.txt,README.md,LICENSE -DestinationPath AUTO_MAA_${{ env.AUTO_MAA_version }}.zip
|
$root = "${{ github.workspace }}"
|
||||||
Compress-Archive -Path Updater.exe -DestinationPath Updater_${{ env.updater_version }}.zip
|
$ver = "${{ steps.get_version.outputs.main_version }}"
|
||||||
|
$iss = Get-Content "$root/app/utils/AUTO_MAA.iss" -Raw
|
||||||
|
$iss = $iss -replace '#define MyAppVersion ""', "#define MyAppVersion `"$ver`""
|
||||||
|
$iss = $iss -replace '#define MyAppPath ""', "#define MyAppPath `"$root/AUTO_MAA`""
|
||||||
|
$iss = $iss -replace '#define OutputDir ""', "#define OutputDir `"$root`""
|
||||||
|
Set-Content -Path "$root/AUTO_MAA.iss" -Value $iss
|
||||||
|
|
||||||
|
- name: Build setup program
|
||||||
|
uses: Minionguyjpro/Inno-Setup-Action@v1.2.5
|
||||||
|
with:
|
||||||
|
path: AUTO_MAA.iss
|
||||||
|
|
||||||
|
- name: Upload unsigned setup program
|
||||||
|
id: upload-unsigned-setup-program
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: AUTO_MAA-Setup
|
||||||
|
path: AUTO_MAA-Setup.exe
|
||||||
|
|
||||||
|
- name: Sign setup program
|
||||||
|
id: sign_setup_program
|
||||||
|
uses: signpath/github-action-submit-signing-request@v1.2
|
||||||
|
with:
|
||||||
|
api-token: '${{ secrets.SIGNPATH_API_TOKEN }}'
|
||||||
|
organization-id: '787a1d5f-6177-4f30-9559-d2646473584a'
|
||||||
|
project-slug: 'AUTO_MAA'
|
||||||
|
signing-policy-slug: 'release-signing'
|
||||||
|
artifact-configuration-slug: "AUTO_MAA-Setup"
|
||||||
|
github-artifact-id: '${{ steps.upload-unsigned-setup-program.outputs.artifact-id }}'
|
||||||
|
wait-for-completion: true
|
||||||
|
output-artifact-directory: 'AUTO_MAA_Setup'
|
||||||
|
|
||||||
|
- name: Compress setup exe
|
||||||
|
shell: pwsh
|
||||||
|
run: Compress-Archive -Path AUTO_MAA_Setup/* -DestinationPath AUTO_MAA_${{ steps.get_version.outputs.main_version }}.zip
|
||||||
|
|
||||||
|
- name: Generate version info
|
||||||
|
shell: python
|
||||||
|
run: |
|
||||||
|
import json
|
||||||
|
from pathlib import Path
|
||||||
|
def version_text(version_numb):
|
||||||
|
while len(version_numb) < 4:
|
||||||
|
version_numb.append(0)
|
||||||
|
if version_numb[3] == 0:
|
||||||
|
return f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
||||||
|
else:
|
||||||
|
return f"v{'.'.join(str(_) for _ in version_numb[0:3])}-beta.{version_numb[3]}"
|
||||||
|
def version_info_markdown(info):
|
||||||
|
version_info = ""
|
||||||
|
for key, value in info.items():
|
||||||
|
version_info += f"## {key}\n"
|
||||||
|
for v in value:
|
||||||
|
version_info += f"- {v}\n"
|
||||||
|
return version_info
|
||||||
|
root_path = Path(".")
|
||||||
|
version = json.loads((root_path / "resources/version.json").read_text(encoding="utf-8"))
|
||||||
|
main_version_numb = list(map(int, version["main_version"].split(".")))
|
||||||
|
all_version_info = {}
|
||||||
|
for v_i in version["version_info"].values():
|
||||||
|
for key, value in v_i.items():
|
||||||
|
if key in all_version_info:
|
||||||
|
all_version_info[key] += value.copy()
|
||||||
|
else:
|
||||||
|
all_version_info[key] = value.copy()
|
||||||
|
(root_path / "version_info.txt").write_text(
|
||||||
|
f"{version_text(main_version_numb)}\n\n<!--{json.dumps(version['version_info'], ensure_ascii=False)}-->\n{version_info_markdown(all_version_info)}",
|
||||||
|
encoding="utf-8",
|
||||||
|
)
|
||||||
|
|
||||||
- name: Upload Artifact
|
- name: Upload Artifact
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: AUTO_MAA_${{ env.AUTO_MAA_version }}
|
name: AUTO_MAA_${{ steps.get_version.outputs.main_version }}
|
||||||
path: |
|
path: AUTO_MAA_${{ steps.get_version.outputs.main_version }}.zip
|
||||||
AUTO_MAA_${{ env.AUTO_MAA_version }}.zip
|
|
||||||
Updater_${{ env.updater_version }}.zip
|
|
||||||
- name: Upload Version_Info Artifact
|
- name: Upload Version_Info Artifact
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: version_info
|
name: version_info
|
||||||
path: version_info.txt
|
path: version_info.txt
|
||||||
|
|
||||||
publish_release:
|
publish_release:
|
||||||
name: Publish release
|
name: Publish release
|
||||||
needs: build_AUTO_MAA
|
needs: build_AUTO_MAA
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Download artifacts
|
- name: Download artifacts
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v4
|
||||||
with:
|
with:
|
||||||
pattern: AUTO_MAA_*
|
pattern: AUTO_MAA_*
|
||||||
merge-multiple: true
|
merge-multiple: true
|
||||||
path: artifacts
|
path: artifacts
|
||||||
|
|
||||||
- name: Download Version_Info
|
- name: Download Version_Info
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: version_info
|
name: version_info
|
||||||
path: ./
|
path: ./
|
||||||
- name: Check if release exists
|
|
||||||
id: check_if_release_exists
|
|
||||||
run: |
|
|
||||||
release_id=$(gh release view $(sed 's/\r$//g' <(head -n 1 version_info.txt)) --json id --jq .id || true)
|
|
||||||
if [[ -z $release_id ]]; then
|
|
||||||
echo "release_exists=false" >> $GITHUB_OUTPUT
|
|
||||||
else
|
|
||||||
echo "release_exists=true" >> $GITHUB_OUTPUT
|
|
||||||
fi
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
|
||||||
- name: Create release
|
- name: Create release
|
||||||
id: create_release
|
id: create_release
|
||||||
if: steps.check_if_release_exists.outputs.release_exists == 'false'
|
|
||||||
run: |
|
run: |
|
||||||
set -xe
|
set -xe
|
||||||
shopt -s nullglob
|
shopt -s nullglob
|
||||||
NAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
NAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
||||||
TAGNAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
TAGNAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
||||||
NOTES_MAIN="$(sed 's/\r$//g' <(tail -n +3 version_info.txt))"
|
NOTES_MAIN="$(sed 's/\r$//g' <(tail -n +3 version_info.txt))"
|
||||||
NOTES_TAIL="\`\`\`本release通过GitHub Actions自动构建\`\`\`"
|
NOTES="$NOTES_MAIN
|
||||||
NOTES="$NOTES_MAIN<br><br>$NOTES_TAIL"
|
|
||||||
gh release create "$TAGNAME" --target "main" --title "$NAME" --notes "$NOTES" artifacts/*
|
## 代码签名策略(Code signing policy)
|
||||||
|
|
||||||
|
Free code signing provided by [SignPath.io](https://signpath.io/), certificate by [SignPath Foundation](https://signpath.org/)
|
||||||
|
|
||||||
|
- 审批人(Approvers): [DLmaster (@DLmaster361)](https://github.com/DLmaster361)
|
||||||
|
|
||||||
|
## 隐私政策(Privacy policy)
|
||||||
|
|
||||||
|
除非用户、安装者或使用者特别要求,否则本程序不会将任何信息传输到其他网络系统。
|
||||||
|
|
||||||
|
This program will not transfer any information to other networked systems unless specifically requested by the user or the person installing or operating it.
|
||||||
|
|
||||||
|
[已有 Mirror酱 CDK ?前往 Mirror酱 高速下载](https://mirrorchyan.com/zh/projects?rid=AUTO_MAA&source=auto_maa-release)
|
||||||
|
|
||||||
|
\`\`\`本release通过GitHub Actions自动构建\`\`\`"
|
||||||
|
if [ "${{ github.ref_name }}" == "main" ]; then
|
||||||
|
PRERELEASE_FLAG=""
|
||||||
|
else
|
||||||
|
PRERELEASE_FLAG="--prerelease"
|
||||||
|
fi
|
||||||
|
gh release create "$TAGNAME" --target "main" --title "$NAME" --notes "$NOTES" $PRERELEASE_FLAG artifacts/*
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
||||||
- name: Update release
|
|
||||||
id: update_release
|
- name: Trigger MirrorChyanUploading
|
||||||
if: steps.check_if_release_exists.outputs.release_exists == 'true'
|
|
||||||
run: |
|
run: |
|
||||||
set -xe
|
gh workflow run --repo $GITHUB_REPOSITORY mirrorchyan
|
||||||
shopt -s nullglob
|
gh workflow run --repo $GITHUB_REPOSITORY mirrorchyan_release_note
|
||||||
NAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
TAGNAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
NOTES_MAIN="$(sed 's/\r$//g' <(tail -n +3 version_info.txt))"
|
|
||||||
NOTES_TAIL="\`\`\`本release通过GitHub Actions自动构建\`\`\`"
|
|
||||||
NOTES="$NOTES_MAIN<br><br>$NOTES_TAIL"
|
|
||||||
gh release delete "$TAGNAME" --yes
|
|
||||||
gh release create "$TAGNAME" --target "main" --title "$NAME" --notes "$NOTES" artifacts/*
|
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
- name: Setup SSH Key
|
|
||||||
run: |
|
|
||||||
mkdir -p ~/.ssh
|
|
||||||
echo "${{ secrets.SSH_PRIVATE_KEY }}" > ~/.ssh/id_rsa
|
|
||||||
chmod 600 ~/.ssh/id_rsa
|
|
||||||
ssh-keyscan -H ${{ secrets.SERVER_IP }} >> ~/.ssh/known_hosts
|
|
||||||
- name: Upload Release to Server
|
|
||||||
run: |
|
|
||||||
scp -r artifacts/* ${{ secrets.SERVER_USER }}@${{ secrets.SERVER_IP }}:/home/user/files/AUTO_MAA/
|
|
||||||
|
|||||||
162
.github/workflows/build-pre.yml
vendored
@@ -1,162 +0,0 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
|
||||||
# Copyright © <2024> <DLmaster361>
|
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
|
||||||
|
|
||||||
# AUTO_MAA is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published
|
|
||||||
# by the Free Software Foundation, either version 3 of the License,
|
|
||||||
# or (at your option) any later version.
|
|
||||||
|
|
||||||
# AUTO_MAA is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
|
||||||
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
||||||
# the GNU General Public License for more details.
|
|
||||||
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
|
||||||
|
|
||||||
name: Build AUTO_MAA_Pre
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [ "dev" ]
|
|
||||||
paths-ignore:
|
|
||||||
- '**.md'
|
|
||||||
- 'LICENSE'
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
pre_check:
|
|
||||||
name: Pre Checks
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Repo Check
|
|
||||||
id: repo_check
|
|
||||||
run: |
|
|
||||||
if [[ "$GITHUB_REPOSITORY" != "DLmaster361/AUTO_MAA" ]]; then
|
|
||||||
echo "When forking this repository to make your own builds, you have to adjust this check."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
exit 0
|
|
||||||
build_AUTO_MAA:
|
|
||||||
runs-on: windows-latest
|
|
||||||
needs: pre_check
|
|
||||||
steps:
|
|
||||||
- name: Checkout code
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
- name: Set up Python 3.12
|
|
||||||
uses: actions/setup-python@v5
|
|
||||||
with:
|
|
||||||
python-version: "3.12"
|
|
||||||
- name: Install dependencies
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
pip install flake8 pytest
|
|
||||||
pip install -r requirements.txt
|
|
||||||
- name: Lint with flake8
|
|
||||||
run: |
|
|
||||||
# stop the build if there are Python syntax errors or undefined names
|
|
||||||
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
|
||||||
# exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide
|
|
||||||
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics
|
|
||||||
- name: Package
|
|
||||||
id: package
|
|
||||||
run: |
|
|
||||||
copy app\utils\package.py .\
|
|
||||||
python package.py
|
|
||||||
- name: Read version
|
|
||||||
id: read_version
|
|
||||||
run: |
|
|
||||||
$MAIN_VERSION=(Get-Content -Path "version_info.txt" -TotalCount 1).Trim()
|
|
||||||
"AUTO_MAA_version=$MAIN_VERSION" | Out-File -FilePath $env:GITHUB_ENV -Append
|
|
||||||
$UPDATER_VERSION=(Get-Content -Path "version_info.txt" -TotalCount 2 | Select-Object -Index 1).Trim()
|
|
||||||
"updater_version=$UPDATER_VERSION" | Out-File -FilePath $env:GITHUB_ENV -Append
|
|
||||||
- name: Create Zip
|
|
||||||
id: create_zip
|
|
||||||
run: |
|
|
||||||
Compress-Archive -Path app,resources,main.py,AUTO_MAA.exe,requirements.txt,README.md,LICENSE -DestinationPath AUTO_MAA_${{ env.AUTO_MAA_version }}.zip
|
|
||||||
Compress-Archive -Path Updater.exe -DestinationPath Updater_${{ env.updater_version }}.zip
|
|
||||||
- name: Upload Artifact
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: AUTO_MAA_${{ env.AUTO_MAA_version }}
|
|
||||||
path: |
|
|
||||||
AUTO_MAA_${{ env.AUTO_MAA_version }}.zip
|
|
||||||
Updater_${{ env.updater_version }}.zip
|
|
||||||
- name: Upload Version_Info Artifact
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: version_info
|
|
||||||
path: version_info.txt
|
|
||||||
publish_prerelease:
|
|
||||||
name: Publish prerelease
|
|
||||||
needs: build_AUTO_MAA
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
- name: Download artifacts
|
|
||||||
uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
pattern: AUTO_MAA_*
|
|
||||||
merge-multiple: true
|
|
||||||
path: artifacts
|
|
||||||
- name: Download Version_Info
|
|
||||||
uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
name: version_info
|
|
||||||
path: ./
|
|
||||||
- name: Check if release exists
|
|
||||||
id: check_if_release_exists
|
|
||||||
run: |
|
|
||||||
release_id=$(gh release view $(sed 's/\r$//g' <(head -n 1 version_info.txt)) --json id --jq .id || true)
|
|
||||||
if [[ -z $release_id ]]; then
|
|
||||||
echo "release_exists=false" >> $GITHUB_OUTPUT
|
|
||||||
else
|
|
||||||
echo "release_exists=true" >> $GITHUB_OUTPUT
|
|
||||||
fi
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
|
||||||
- name: Create prerelease
|
|
||||||
id: create_prerelease
|
|
||||||
if: steps.check_if_release_exists.outputs.release_exists == 'false'
|
|
||||||
run: |
|
|
||||||
set -xe
|
|
||||||
shopt -s nullglob
|
|
||||||
NAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
TAGNAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
NOTES_MAIN="$(sed 's/\r$//g' <(tail -n +3 version_info.txt))"
|
|
||||||
NOTES_TAIL="\`\`\`本release通过GitHub Actions自动构建\`\`\`"
|
|
||||||
NOTES="$NOTES_MAIN<br><br>$NOTES_TAIL"
|
|
||||||
gh release create "$TAGNAME" --target "main" --title "$NAME" --notes "$NOTES" --prerelease artifacts/*
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
|
||||||
- name: Update prerelease
|
|
||||||
id: update_prerelease
|
|
||||||
if: steps.check_if_release_exists.outputs.release_exists == 'true'
|
|
||||||
run: |
|
|
||||||
set -xe
|
|
||||||
shopt -s nullglob
|
|
||||||
NAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
TAGNAME="$(sed 's/\r$//g' <(head -n 1 version_info.txt))"
|
|
||||||
NOTES_MAIN="$(sed 's/\r$//g' <(tail -n +3 version_info.txt))"
|
|
||||||
NOTES_TAIL="\`\`\`本release通过GitHub Actions自动构建\`\`\`"
|
|
||||||
NOTES="$NOTES_MAIN<br><br>$NOTES_TAIL"
|
|
||||||
gh release delete "$TAGNAME" --yes
|
|
||||||
gh release create "$TAGNAME" --target "main" --title "$NAME" --notes "$NOTES" --prerelease artifacts/*
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.WORKFLOW_TOKEN }}
|
|
||||||
- name: Setup SSH Key
|
|
||||||
run: |
|
|
||||||
mkdir -p ~/.ssh
|
|
||||||
echo "${{ secrets.SSH_PRIVATE_KEY }}" > ~/.ssh/id_rsa
|
|
||||||
chmod 600 ~/.ssh/id_rsa
|
|
||||||
ssh-keyscan -H ${{ secrets.SERVER_IP }} >> ~/.ssh/known_hosts
|
|
||||||
- name: Upload Release to Server
|
|
||||||
run: |
|
|
||||||
scp -r artifacts/* ${{ secrets.SERVER_USER }}@${{ secrets.SERVER_IP }}:/home/user/files/AUTO_MAA/
|
|
||||||
21
.github/workflows/mirrorchyan.yml
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
name: mirrorchyan
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
mirrorchyan:
|
||||||
|
runs-on: macos-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- id: uploading
|
||||||
|
uses: MirrorChyan/uploading-action@v1
|
||||||
|
with:
|
||||||
|
filetype: latest-release
|
||||||
|
filename: "AUTO_MAA*.zip"
|
||||||
|
mirrorchyan_rid: AUTO_MAA
|
||||||
|
|
||||||
|
owner: DLmaster361
|
||||||
|
repo: AUTO_MAA
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
upload_token: ${{ secrets.MirrorChyanUploadToken }}
|
||||||
19
.github/workflows/mirrorchyan_release_note.yml
vendored
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
name: mirrorchyan_release_note
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
release:
|
||||||
|
types: [edited]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
mirrorchyan:
|
||||||
|
runs-on: macos-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- id: uploading
|
||||||
|
uses: MirrorChyan/release-note-action@v1
|
||||||
|
with:
|
||||||
|
mirrorchyan_rid: AUTO_MAA
|
||||||
|
|
||||||
|
upload_token: ${{ secrets.MirrorChyanUploadToken }}
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
8
.gitignore
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
__pycache__/
|
||||||
|
config/
|
||||||
|
data/
|
||||||
|
debug/
|
||||||
|
history/
|
||||||
|
resources/notice.json
|
||||||
|
resources/theme_image.json
|
||||||
|
resources/images/Home/BannerTheme.jpg
|
||||||
68
README.md
@@ -1,19 +1,21 @@
|
|||||||
# AUTO_MAA
|
<h1 align="center">AUTO_MAA</h1>
|
||||||
|
<p align="center">
|
||||||
MAA多账号管理与自动化软件
|
MAA多账号管理与自动化软件<br><br>
|
||||||
|
<img alt="软件图标" src="https://github.com/DLmaster361/AUTO_MAA/blob/main/resources/images/AUTO_MAA.png">
|
||||||

|
</p>
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
</h1>
|
<p align="center">
|
||||||
|
<a href="https://github.com/DLmaster361/AUTO_MAA/stargazers"><img alt="GitHub Stars" src="https://img.shields.io/github/stars/DLmaster361/AUTO_MAA?style=flat-square"></a>
|
||||||
[](https://github.com/DLmaster361/AUTO_MAA/stargazers)
|
<a href="https://github.com/DLmaster361/AUTO_MAA/network"><img alt="GitHub Forks" src="https://img.shields.io/github/forks/DLmaster361/AUTO_MAA?style=flat-square"></a>
|
||||||
[](https://github.com/DLmaster361/AUTO_MAA/network)
|
<a href="https://github.com/DLmaster361/AUTO_MAA/releases/latest"><img alt="GitHub Downloads" src="https://img.shields.io/github/downloads/DLmaster361/AUTO_MAA/total?style=flat-square"></a>
|
||||||
[](https://github.com/DLmaster361/AUTO_MAA/issues)
|
<a href="https://github.com/DLmaster361/AUTO_MAA/issues"><img alt="GitHub Issues" src="https://img.shields.io/github/issues/DLmaster361/AUTO_MAA?style=flat-square"></a>
|
||||||
[](https://github.com/DLmaster361/AUTO_MAA/graphs/contributors)
|
<a href="https://github.com/DLmaster361/AUTO_MAA/graphs/contributors"><img alt="GitHub Contributors" src="https://img.shields.io/github/contributors/DLmaster361/AUTO_MAA?style=flat-square"></a>
|
||||||
[](https://github.com/DLmaster361/AUTO_MAA/blob/main/LICENSE)
|
<a href="https://github.com/DLmaster361/AUTO_MAA/blob/main/LICENSE"><img alt="GitHub License" src="https://img.shields.io/github/license/DLmaster361/AUTO_MAA?style=flat-square"></a>
|
||||||
</div>
|
<a href="https://deepwiki.com/DLmaster361/AUTO_MAA"><img alt="DeepWiki" src="https://deepwiki.com/badge.svg"></a>
|
||||||
|
<a href="https://mirrorchyan.com/zh/projects?rid=AUTO_MAA&source=auto_maa-readme"><img alt="mirrorc" src="https://img.shields.io/badge/Mirror%E9%85%B1-%239af3f6?logo=countingworkspro&logoColor=4f46e5"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
## 软件介绍
|
## 软件介绍
|
||||||
|
|
||||||
@@ -21,6 +23,11 @@ MAA多账号管理与自动化软件
|
|||||||
|
|
||||||
本软件是明日方舟第三方软件`MAA`的第三方工具,即第3<sup>3</sup>方软件。旨在优化MAA多账号功能体验,并通过一些方法解决MAA项目未能解决的部分问题,提高代理的稳定性。
|
本软件是明日方舟第三方软件`MAA`的第三方工具,即第3<sup>3</sup>方软件。旨在优化MAA多账号功能体验,并通过一些方法解决MAA项目未能解决的部分问题,提高代理的稳定性。
|
||||||
|
|
||||||
|
- **集中管理**:一站式管理多个MAA脚本与多个用户配置,和凌乱的散装脚本窗口说再见!
|
||||||
|
- **无人值守**:自动处理MAA相关报错,再也不用为代理任务卡死时自己不在电脑旁烦恼啦!
|
||||||
|
- **配置灵活**:通过调度队列与脚本的组合,自由实现您能想到的所有调度需求!
|
||||||
|
- **信息统计**:自动统计用户的公招与关卡掉落物,看看这个月您的收益是多少!
|
||||||
|
|
||||||
### 原理
|
### 原理
|
||||||
|
|
||||||
本软件可以存储多个明日方舟账号数据,并通过以下流程实现代理功能:
|
本软件可以存储多个明日方舟账号数据,并通过以下流程实现代理功能:
|
||||||
@@ -31,11 +38,9 @@ MAA多账号管理与自动化软件
|
|||||||
|
|
||||||
### 优势
|
### 优势
|
||||||
|
|
||||||
- **节省运行开销:** 只需要一份MAA软件与一个模拟器,无需多开就能完成多账号代理,羸弱的电脑也能代理日常。
|
- **高效稳定**:通过日志监测、异常处理等机制,保障代理任务顺利完成。
|
||||||
- **自定义空间大:** 依靠高级用户配置模式,支持MAA几乎所有设置选项自定义,支持模拟器多开。
|
- **简洁易用**:无需手动修改配置文件,实现自动化调度与多开管理。
|
||||||
- **调度方法自由:** 通过调度队列功能,自由实现MAA多开等多种调度方式。
|
- **兼容扩展**:支持 MAA 几乎所有的配置选项,满足不同用户需求。
|
||||||
- **一键代理无忧:** 无须中途手动修改MAA配置,将繁琐交给AUTO_MAA,把游戏留给自己。
|
|
||||||
- **代理结果复核:** 通过人工排查功能核实各用户代理情况,堵住自动代理的最后一丝风险。
|
|
||||||
|
|
||||||
## 重要声明
|
## 重要声明
|
||||||
|
|
||||||
@@ -47,13 +52,10 @@ MAA多账号管理与自动化软件
|
|||||||
- **传播:** AUTO_MAA原则上允许传播者自由传播本软件,但无论在何种传播过程中,不得删除项目作者与开发者所留版权声明,不得隐瞒项目作者与相关开发者的存在。由于软件性质,项目组不希望发现任何人在明日方舟官方媒体(包括官方媒体账号与森空岛社区等)或明日方舟游戏相关内容(包括同好群、线下活动与游戏内容讨论等)下提及AUTO_MAA或MAA,希望各位理解。
|
- **传播:** AUTO_MAA原则上允许传播者自由传播本软件,但无论在何种传播过程中,不得删除项目作者与开发者所留版权声明,不得隐瞒项目作者与相关开发者的存在。由于软件性质,项目组不希望发现任何人在明日方舟官方媒体(包括官方媒体账号与森空岛社区等)或明日方舟游戏相关内容(包括同好群、线下活动与游戏内容讨论等)下提及AUTO_MAA或MAA,希望各位理解。
|
||||||
- **衍生:** AUTO_MAA允许任何人对软件本体或软件部分代码进行二次开发或利用。但依据GPL,相关成果再次分发时也必须使用GPL或兼容的协议开源。
|
- **衍生:** AUTO_MAA允许任何人对软件本体或软件部分代码进行二次开发或利用。但依据GPL,相关成果再次分发时也必须使用GPL或兼容的协议开源。
|
||||||
- **贡献:** 不论是直接参与软件的维护编写,或是撰写文档、测试、反馈BUG、给出建议、参与讨论,都为AUTO_MAA项目的发展完善做出了不可忽视的贡献。项目组提倡各位贡献者遵照GitHub开源社区惯例,发布Issues参与项目。避免私信或私发邮件(安全性漏洞或敏感问题除外),以帮助更多用户。
|
- **贡献:** 不论是直接参与软件的维护编写,或是撰写文档、测试、反馈BUG、给出建议、参与讨论,都为AUTO_MAA项目的发展完善做出了不可忽视的贡献。项目组提倡各位贡献者遵照GitHub开源社区惯例,发布Issues参与项目。避免私信或私发邮件(安全性漏洞或敏感问题除外),以帮助更多用户。
|
||||||
|
- **图像:** `AUTO_MAA主页默认图像` 并不适用开源协议,著作权归 [NARINpopo](https://space.bilibili.com/1877154) 画师所有,商业使用权归 [DLmaster (@DLmaster361)](https://github.com/DLmaster361) 所有,软件用户仅拥有非商业使用权。不得以开源协议已授权为由在未经授权的情况下使用 `AUTO_MAA主页默认图像`,不得在未经授权的情况下将 `AUTO_MAA主页默认图像` 用于任何商业用途。
|
||||||
|
|
||||||
以上细则是本项目对GPL的相关补充与强调。未提及的以GPL为准,发生冲突的以本细则为准。如有不清楚的部分,请发Issues询问。若发生纠纷,相关内容也没有在Issues上提及的,项目组拥有最终解释权。
|
以上细则是本项目对GPL的相关补充与强调。未提及的以GPL为准,发生冲突的以本细则为准。如有不清楚的部分,请发Issues询问。若发生纠纷,相关内容也没有在Issues上提及的,项目组拥有最终解释权。
|
||||||
|
|
||||||
**注意**
|
|
||||||
|
|
||||||
- 由于本软件有修改其它目录JSON文件等行为,使用前请将AUTO_MAA添加入Windows Defender信任区以及防病毒软件的信任区或开发者目录,避免被误杀。
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
# 使用方法
|
# 使用方法
|
||||||
@@ -70,6 +72,24 @@ MAA多账号管理与自动化软件
|
|||||||
|
|
||||||
可在[《AUTO_MAA开发者协作文档》](https://docs.qq.com/aio/DQ3Z5eHNxdmxFQmZX)的`开发任务`页面中查看开发进度。
|
可在[《AUTO_MAA开发者协作文档》](https://docs.qq.com/aio/DQ3Z5eHNxdmxFQmZX)的`开发任务`页面中查看开发进度。
|
||||||
|
|
||||||
|
## 代码签名策略(Code signing policy)
|
||||||
|
|
||||||
|
Free code signing provided by [SignPath.io](https://signpath.io/), certificate by [SignPath Foundation](https://signpath.org/)
|
||||||
|
|
||||||
|
- 审批人(Approvers): [DLmaster (@DLmaster361)](https://github.com/DLmaster361)
|
||||||
|
|
||||||
|
## 隐私政策(Privacy policy)
|
||||||
|
|
||||||
|
除非用户、安装者或使用者特别要求,否则本程序不会将任何信息传输到其他网络系统。
|
||||||
|
|
||||||
|
This program will not transfer any information to other networked systems unless specifically requested by the user or the person installing or operating it.
|
||||||
|
|
||||||
|
## 特别鸣谢
|
||||||
|
|
||||||
|
- 下载服务器:由[AoXuan (@ClozyA)](https://github.com/ClozyA) 个人为项目赞助。
|
||||||
|
|
||||||
|
- EXE签名: 由 [SignPath.io](https://signpath.io/)提供免费代码签名,签名来自[SignPath Foundation](https://signpath.org/)。
|
||||||
|
|
||||||
## 贡献者
|
## 贡献者
|
||||||
|
|
||||||
感谢以下贡献者对本项目做出的贡献
|
感谢以下贡献者对本项目做出的贡献
|
||||||
@@ -82,8 +102,6 @@ MAA多账号管理与自动化软件
|
|||||||
|
|
||||||

|

|
||||||
|
|
||||||
感谢 [AoXuan (@ClozyA)](https://github.com/ClozyA) 为本项目提供的下载服务器
|
|
||||||
|
|
||||||
## Star History
|
## Star History
|
||||||
|
|
||||||
[](https://star-history.com/#DLmaster361/AUTO_MAA&Date)
|
[](https://star-history.com/#DLmaster361/AUTO_MAA&Date)
|
||||||
@@ -98,4 +116,4 @@ MAA多账号管理与自动化软件
|
|||||||
|
|
||||||
如果喜欢这个项目的话,给作者来杯咖啡吧!
|
如果喜欢这个项目的话,给作者来杯咖啡吧!
|
||||||
|
|
||||||

|

|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA主程序包
|
AUTO_MAA主程序包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -29,16 +29,15 @@ __version__ = "4.2.0"
|
|||||||
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
||||||
__license__ = "GPL-3.0 license"
|
__license__ = "GPL-3.0 license"
|
||||||
|
|
||||||
from .core import AppConfig, QueueConfig, MaaConfig, Task, TaskManager, MainTimer
|
from .core import QueueConfig, MaaConfig, MaaUserConfig, Task, TaskManager, MainTimer
|
||||||
from .models import MaaManager
|
from .models import MaaManager
|
||||||
from .services import Notify, Crypto, System
|
from .services import Notify, Crypto, System
|
||||||
from .ui import AUTO_MAA
|
from .ui import AUTO_MAA
|
||||||
from .utils import Updater
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
"AppConfig",
|
|
||||||
"QueueConfig",
|
"QueueConfig",
|
||||||
"MaaConfig",
|
"MaaConfig",
|
||||||
|
"MaaUserConfig",
|
||||||
"Task",
|
"Task",
|
||||||
"TaskManager",
|
"TaskManager",
|
||||||
"MainTimer",
|
"MainTimer",
|
||||||
@@ -47,5 +46,4 @@ __all__ = [
|
|||||||
"Crypto",
|
"Crypto",
|
||||||
"System",
|
"System",
|
||||||
"AUTO_MAA",
|
"AUTO_MAA",
|
||||||
"Updater",
|
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA核心组件包
|
AUTO_MAA核心组件包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -29,17 +29,22 @@ __version__ = "4.2.0"
|
|||||||
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
||||||
__license__ = "GPL-3.0 license"
|
__license__ = "GPL-3.0 license"
|
||||||
|
|
||||||
from .config import AppConfig, QueueConfig, MaaConfig, Config
|
from .config import QueueConfig, MaaConfig, MaaUserConfig, MaaPlanConfig, Config
|
||||||
from .main_info_bar import MainInfoBar
|
from .main_info_bar import MainInfoBar
|
||||||
|
from .network import Network
|
||||||
|
from .sound_player import SoundPlayer
|
||||||
from .task_manager import Task, TaskManager
|
from .task_manager import Task, TaskManager
|
||||||
from .timer import MainTimer
|
from .timer import MainTimer
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
"AppConfig",
|
|
||||||
"Config",
|
"Config",
|
||||||
"QueueConfig",
|
"QueueConfig",
|
||||||
"MaaConfig",
|
"MaaConfig",
|
||||||
|
"MaaUserConfig",
|
||||||
|
"MaaPlanConfig",
|
||||||
"MainInfoBar",
|
"MainInfoBar",
|
||||||
|
"Network",
|
||||||
|
"SoundPlayer",
|
||||||
"Task",
|
"Task",
|
||||||
"TaskManager",
|
"TaskManager",
|
||||||
"MainTimer",
|
"MainTimer",
|
||||||
|
|||||||
1476
app/core/config.py
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,77 +16,74 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA信息通知栏
|
AUTO_MAA信息通知栏
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtCore import Qt
|
from PySide6.QtCore import Qt
|
||||||
from qfluentwidgets import (
|
from qfluentwidgets import InfoBar, InfoBarPosition
|
||||||
InfoBar,
|
|
||||||
InfoBarPosition,
|
from .config import Config
|
||||||
)
|
from .sound_player import SoundPlayer
|
||||||
|
|
||||||
|
|
||||||
class _MainInfoBar:
|
class _MainInfoBar:
|
||||||
"""信息通知栏"""
|
"""信息通知栏"""
|
||||||
|
|
||||||
def __init__(self, main_window=None):
|
# 模式到 InfoBar 方法的映射
|
||||||
|
mode_mapping = {
|
||||||
|
"success": InfoBar.success,
|
||||||
|
"warning": InfoBar.warning,
|
||||||
|
"error": InfoBar.error,
|
||||||
|
"info": InfoBar.info,
|
||||||
|
}
|
||||||
|
|
||||||
self.main_window = main_window
|
def push_info_bar(
|
||||||
|
self, mode: str, title: str, content: str, time: int, if_force: bool = False
|
||||||
def push_info_bar(self, mode: str, title: str, content: str, time: int):
|
):
|
||||||
"""推送到信息通知栏"""
|
"""推送到信息通知栏"""
|
||||||
|
if Config.main_window is None:
|
||||||
if self.main_window is None:
|
|
||||||
logger.error("信息通知栏未设置父窗口")
|
logger.error("信息通知栏未设置父窗口")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
if mode == "success":
|
# 根据 mode 获取对应的 InfoBar 方法
|
||||||
InfoBar.success(
|
info_bar_method = self.mode_mapping.get(mode)
|
||||||
|
|
||||||
|
if not info_bar_method:
|
||||||
|
logger.error(f"未知的通知栏模式: {mode}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
if Config.main_window.isVisible():
|
||||||
|
info_bar_method(
|
||||||
title=title,
|
title=title,
|
||||||
content=content,
|
content=content,
|
||||||
orient=Qt.Horizontal,
|
orient=Qt.Horizontal,
|
||||||
isClosable=True,
|
isClosable=True,
|
||||||
position=InfoBarPosition.TOP_RIGHT,
|
position=InfoBarPosition.TOP_RIGHT,
|
||||||
duration=time,
|
duration=time,
|
||||||
parent=self.main_window,
|
parent=Config.main_window,
|
||||||
)
|
|
||||||
elif mode == "warning":
|
|
||||||
InfoBar.warning(
|
|
||||||
title=title,
|
|
||||||
content=content,
|
|
||||||
orient=Qt.Horizontal,
|
|
||||||
isClosable=True,
|
|
||||||
position=InfoBarPosition.TOP_RIGHT,
|
|
||||||
duration=time,
|
|
||||||
parent=self.main_window,
|
|
||||||
)
|
|
||||||
elif mode == "error":
|
|
||||||
InfoBar.error(
|
|
||||||
title=title,
|
|
||||||
content=content,
|
|
||||||
orient=Qt.Horizontal,
|
|
||||||
isClosable=True,
|
|
||||||
position=InfoBarPosition.TOP_RIGHT,
|
|
||||||
duration=time,
|
|
||||||
parent=self.main_window,
|
|
||||||
)
|
|
||||||
elif mode == "info":
|
|
||||||
InfoBar.info(
|
|
||||||
title=title,
|
|
||||||
content=content,
|
|
||||||
orient=Qt.Horizontal,
|
|
||||||
isClosable=True,
|
|
||||||
position=InfoBarPosition.TOP_RIGHT,
|
|
||||||
duration=time,
|
|
||||||
parent=self.main_window,
|
|
||||||
)
|
)
|
||||||
|
elif if_force:
|
||||||
|
# 如果主窗口不可见且强制推送,则录入消息队列等待窗口显示后推送
|
||||||
|
info_bar_item = {
|
||||||
|
"mode": mode,
|
||||||
|
"title": title,
|
||||||
|
"content": content,
|
||||||
|
"time": time,
|
||||||
|
}
|
||||||
|
if info_bar_item not in Config.info_bar_list:
|
||||||
|
Config.info_bar_list.append(info_bar_item)
|
||||||
|
|
||||||
|
if mode == "warning":
|
||||||
|
SoundPlayer.play("发生异常")
|
||||||
|
if mode == "error":
|
||||||
|
SoundPlayer.play("发生错误")
|
||||||
|
|
||||||
|
|
||||||
MainInfoBar = _MainInfoBar()
|
MainInfoBar = _MainInfoBar()
|
||||||
|
|||||||
151
app/core/network.py
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
|
# AUTO_MAA is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published
|
||||||
|
# by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
|
||||||
|
# AUTO_MAA is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
|
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||||||
|
# the GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
|
"""
|
||||||
|
AUTO_MAA
|
||||||
|
AUTO_MAA网络请求线程
|
||||||
|
v4.3
|
||||||
|
作者:DLmaster_361
|
||||||
|
"""
|
||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
from PySide6.QtCore import QObject, QThread, QEventLoop
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
import requests
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
class NetworkThread(QThread):
|
||||||
|
"""网络请求线程类"""
|
||||||
|
|
||||||
|
max_retries = 3
|
||||||
|
timeout = 10
|
||||||
|
backoff_factor = 0.1
|
||||||
|
|
||||||
|
def __init__(self, mode: str, url: str, path: Path = None) -> None:
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
|
self.setObjectName(
|
||||||
|
f"NetworkThread-{mode}-{re.sub(r'(&cdk=)[^&]+(&)', r'\1******\2', url)}"
|
||||||
|
)
|
||||||
|
|
||||||
|
self.mode = mode
|
||||||
|
self.url = url
|
||||||
|
self.path = path
|
||||||
|
|
||||||
|
self.status_code = None
|
||||||
|
self.response_json = None
|
||||||
|
self.error_message = None
|
||||||
|
|
||||||
|
self.loop = QEventLoop()
|
||||||
|
|
||||||
|
@logger.catch
|
||||||
|
def run(self) -> None:
|
||||||
|
"""运行网络请求线程"""
|
||||||
|
|
||||||
|
if self.mode == "get":
|
||||||
|
self.get_json(self.url)
|
||||||
|
elif self.mode == "get_file":
|
||||||
|
self.get_file(self.url, self.path)
|
||||||
|
|
||||||
|
def get_json(self, url: str) -> None:
|
||||||
|
"""通过get方法获取json数据"""
|
||||||
|
|
||||||
|
response = None
|
||||||
|
|
||||||
|
for _ in range(self.max_retries):
|
||||||
|
try:
|
||||||
|
response = requests.get(url, timeout=self.timeout)
|
||||||
|
self.status_code = response.status_code
|
||||||
|
self.response_json = response.json()
|
||||||
|
self.error_message = None
|
||||||
|
break
|
||||||
|
except Exception as e:
|
||||||
|
self.status_code = response.status_code if response else None
|
||||||
|
self.response_json = None
|
||||||
|
self.error_message = str(e)
|
||||||
|
time.sleep(self.backoff_factor)
|
||||||
|
|
||||||
|
self.loop.quit()
|
||||||
|
|
||||||
|
def get_file(self, url: str, path: Path) -> None:
|
||||||
|
"""通过get方法下载文件"""
|
||||||
|
|
||||||
|
response = None
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = requests.get(url, timeout=10)
|
||||||
|
if response.status_code == 200:
|
||||||
|
with open(path, "wb") as file:
|
||||||
|
file.write(response.content)
|
||||||
|
self.status_code = response.status_code
|
||||||
|
else:
|
||||||
|
self.status_code = response.status_code
|
||||||
|
self.error_message = "下载失败"
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.status_code = response.status_code if response else None
|
||||||
|
self.error_message = str(e)
|
||||||
|
|
||||||
|
self.loop.quit()
|
||||||
|
|
||||||
|
|
||||||
|
class _Network(QObject):
|
||||||
|
"""网络请求线程类"""
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
|
self.task_queue = []
|
||||||
|
|
||||||
|
def add_task(self, mode: str, url: str, path: Path = None) -> NetworkThread:
|
||||||
|
"""添加网络请求任务"""
|
||||||
|
|
||||||
|
network_thread = NetworkThread(mode, url, path)
|
||||||
|
|
||||||
|
self.task_queue.append(network_thread)
|
||||||
|
|
||||||
|
network_thread.start()
|
||||||
|
|
||||||
|
return network_thread
|
||||||
|
|
||||||
|
def get_result(self, network_thread: NetworkThread) -> dict:
|
||||||
|
"""获取网络请求结果"""
|
||||||
|
|
||||||
|
result = {
|
||||||
|
"status_code": network_thread.status_code,
|
||||||
|
"response_json": network_thread.response_json,
|
||||||
|
"error_message": (
|
||||||
|
re.sub(r"(&cdk=)[^&]+(&)", r"\1******\2", network_thread.error_message)
|
||||||
|
if network_thread.error_message
|
||||||
|
else None
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
network_thread.quit()
|
||||||
|
network_thread.wait()
|
||||||
|
self.task_queue.remove(network_thread)
|
||||||
|
network_thread.deleteLater()
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
Network = _Network()
|
||||||
69
app/core/sound_player.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
|
# AUTO_MAA is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published
|
||||||
|
# by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
|
||||||
|
# AUTO_MAA is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
|
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||||||
|
# the GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
|
"""
|
||||||
|
AUTO_MAA
|
||||||
|
AUTO_MAA音效播放器
|
||||||
|
v4.3
|
||||||
|
作者:DLmaster_361
|
||||||
|
"""
|
||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
from PySide6.QtCore import QObject, QUrl
|
||||||
|
from PySide6.QtMultimedia import QSoundEffect
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
from .config import Config
|
||||||
|
|
||||||
|
|
||||||
|
class _SoundPlayer(QObject):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
|
self.sounds_path = Config.app_path / "resources/sounds"
|
||||||
|
|
||||||
|
def play(self, sound_name: str):
|
||||||
|
|
||||||
|
if not Config.get(Config.voice_Enabled):
|
||||||
|
return
|
||||||
|
|
||||||
|
if (self.sounds_path / f"both/{sound_name}.wav").exists():
|
||||||
|
|
||||||
|
self.play_voice(self.sounds_path / f"both/{sound_name}.wav")
|
||||||
|
|
||||||
|
elif (
|
||||||
|
self.sounds_path / Config.get(Config.voice_Type) / f"{sound_name}.wav"
|
||||||
|
).exists():
|
||||||
|
|
||||||
|
self.play_voice(
|
||||||
|
self.sounds_path / Config.get(Config.voice_Type) / f"{sound_name}.wav"
|
||||||
|
)
|
||||||
|
|
||||||
|
def play_voice(self, sound_path: Path):
|
||||||
|
|
||||||
|
effect = QSoundEffect(self)
|
||||||
|
effect.setVolume(1)
|
||||||
|
effect.setSource(QUrl.fromLocalFile(sound_path))
|
||||||
|
effect.play()
|
||||||
|
|
||||||
|
|
||||||
|
SoundPlayer = _SoundPlayer()
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,25 +16,26 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA业务调度器
|
AUTO_MAA业务调度器
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtCore import QThread, QObject, Signal
|
from PySide6.QtCore import QThread, QObject, Signal
|
||||||
from qfluentwidgets import Dialog
|
from qfluentwidgets import MessageBox
|
||||||
import json
|
|
||||||
from pathlib import Path
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
from packaging import version
|
||||||
from typing import Dict, Union
|
from typing import Dict, Union
|
||||||
|
|
||||||
from .config import Config
|
from .config import Config
|
||||||
from .main_info_bar import MainInfoBar
|
from .main_info_bar import MainInfoBar
|
||||||
|
from .network import Network
|
||||||
|
from .sound_player import SoundPlayer
|
||||||
from app.models import MaaManager
|
from app.models import MaaManager
|
||||||
from app.services import System
|
from app.services import System
|
||||||
|
|
||||||
@@ -42,10 +43,12 @@ from app.services import System
|
|||||||
class Task(QThread):
|
class Task(QThread):
|
||||||
"""业务线程"""
|
"""业务线程"""
|
||||||
|
|
||||||
|
check_maa_version = Signal(str)
|
||||||
push_info_bar = Signal(str, str, str, int)
|
push_info_bar = Signal(str, str, str, int)
|
||||||
|
play_sound = Signal(str)
|
||||||
question = Signal(str, str)
|
question = Signal(str, str)
|
||||||
question_response = Signal(bool)
|
question_response = Signal(bool)
|
||||||
update_user_info = Signal(Path, list, list, list, list, list, list)
|
update_user_info = Signal(str, dict)
|
||||||
create_task_list = Signal(list)
|
create_task_list = Signal(list)
|
||||||
create_user_list = Signal(list)
|
create_user_list = Signal(list)
|
||||||
update_task_list = Signal(list)
|
update_task_list = Signal(list)
|
||||||
@@ -58,6 +61,8 @@ class Task(QThread):
|
|||||||
):
|
):
|
||||||
super(Task, self).__init__()
|
super(Task, self).__init__()
|
||||||
|
|
||||||
|
self.setObjectName(f"Task-{mode}-{name}")
|
||||||
|
|
||||||
self.mode = mode
|
self.mode = mode
|
||||||
self.name = name
|
self.name = name
|
||||||
self.info = info
|
self.info = info
|
||||||
@@ -66,6 +71,7 @@ class Task(QThread):
|
|||||||
|
|
||||||
self.question_response.connect(lambda: print("response"))
|
self.question_response.connect(lambda: print("response"))
|
||||||
|
|
||||||
|
@logger.catch
|
||||||
def run(self):
|
def run(self):
|
||||||
|
|
||||||
if "设置MAA" in self.mode:
|
if "设置MAA" in self.mode:
|
||||||
@@ -75,104 +81,91 @@ class Task(QThread):
|
|||||||
|
|
||||||
self.task = MaaManager(
|
self.task = MaaManager(
|
||||||
self.mode,
|
self.mode,
|
||||||
Config.app_path / f"config/MaaConfig/{self.name}",
|
Config.member_dict[self.name],
|
||||||
(
|
(None if "全局" in self.mode else self.info["SetMaaInfo"]["Path"]),
|
||||||
None
|
|
||||||
if "全局" in self.mode
|
|
||||||
else Config.app_path
|
|
||||||
/ f"config/MaaConfig/{self.name}/beta/{self.info["SetMaaInfo"]["UserId"]}/{self.info["SetMaaInfo"]["SetType"]}"
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
|
self.task.check_maa_version.connect(self.check_maa_version.emit)
|
||||||
self.task.push_info_bar.connect(self.push_info_bar.emit)
|
self.task.push_info_bar.connect(self.push_info_bar.emit)
|
||||||
|
self.task.play_sound.connect(self.play_sound.emit)
|
||||||
self.task.accomplish.connect(lambda: self.accomplish.emit([]))
|
self.task.accomplish.connect(lambda: self.accomplish.emit([]))
|
||||||
|
|
||||||
self.task.run()
|
self.task.run()
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
||||||
self.member_dict = self.search_member()
|
self.task_list = [
|
||||||
self.task_dict = [
|
[
|
||||||
[value, "等待"]
|
(
|
||||||
for _, value in self.info["Queue"].items()
|
value
|
||||||
|
if Config.member_dict[value]["Config"].get(
|
||||||
|
Config.member_dict[value]["Config"].MaaSet_Name
|
||||||
|
)
|
||||||
|
== ""
|
||||||
|
else f"{value} - {Config.member_dict[value]["Config"].get(Config.member_dict[value]["Config"].MaaSet_Name)}"
|
||||||
|
),
|
||||||
|
"等待",
|
||||||
|
value,
|
||||||
|
]
|
||||||
|
for _, value in sorted(
|
||||||
|
self.info["Queue"].items(), key=lambda x: int(x[0][7:])
|
||||||
|
)
|
||||||
if value != "禁用"
|
if value != "禁用"
|
||||||
]
|
]
|
||||||
|
|
||||||
self.create_task_list.emit(self.task_dict)
|
self.create_task_list.emit(self.task_list)
|
||||||
|
|
||||||
for i in range(len(self.task_dict)):
|
for task in self.task_list:
|
||||||
|
|
||||||
if self.isInterruptionRequested():
|
if self.isInterruptionRequested():
|
||||||
break
|
break
|
||||||
|
|
||||||
self.task_dict[i][1] = "运行"
|
task[1] = "运行"
|
||||||
self.update_task_list.emit(self.task_dict)
|
self.update_task_list.emit(self.task_list)
|
||||||
|
|
||||||
if self.task_dict[i][0] in Config.running_list:
|
if task[2] in Config.running_list:
|
||||||
|
|
||||||
self.task_dict[i][1] = "跳过"
|
task[1] = "跳过"
|
||||||
self.update_task_list.emit(self.task_dict)
|
self.update_task_list.emit(self.task_list)
|
||||||
logger.info(f"跳过任务:{self.task_dict[i][0]}")
|
logger.info(f"跳过任务:{task[0]}")
|
||||||
self.push_info_bar.emit(
|
self.push_info_bar.emit("info", "跳过任务", task[0], 3000)
|
||||||
"info", "跳过任务", self.task_dict[i][0], 3000
|
|
||||||
)
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
Config.running_list.append(self.task_dict[i][0])
|
Config.running_list.append(task[2])
|
||||||
logger.info(f"任务开始:{self.task_dict[i][0]}")
|
logger.info(f"任务开始:{task[0]}")
|
||||||
self.push_info_bar.emit("info", "任务开始", self.task_dict[i][0], 3000)
|
self.push_info_bar.emit("info", "任务开始", task[0], 3000)
|
||||||
|
|
||||||
if self.member_dict[self.task_dict[i][0]][0] == "Maa":
|
if Config.member_dict[task[2]]["Type"] == "Maa":
|
||||||
|
|
||||||
self.task = MaaManager(
|
self.task = MaaManager(
|
||||||
self.mode[0:4],
|
self.mode[0:4],
|
||||||
self.member_dict[self.task_dict[i][0]][1],
|
Config.member_dict[task[2]],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
self.task.check_maa_version.connect(self.check_maa_version.emit)
|
||||||
self.task.question.connect(self.question.emit)
|
self.task.question.connect(self.question.emit)
|
||||||
self.question_response.disconnect()
|
self.question_response.disconnect()
|
||||||
self.question_response.connect(self.task.question_response.emit)
|
self.question_response.connect(self.task.question_response.emit)
|
||||||
self.task.push_info_bar.connect(self.push_info_bar.emit)
|
self.task.push_info_bar.connect(self.push_info_bar.emit)
|
||||||
|
self.task.play_sound.connect(self.play_sound.emit)
|
||||||
self.task.create_user_list.connect(self.create_user_list.emit)
|
self.task.create_user_list.connect(self.create_user_list.emit)
|
||||||
self.task.update_user_list.connect(self.update_user_list.emit)
|
self.task.update_user_list.connect(self.update_user_list.emit)
|
||||||
self.task.update_log_text.connect(self.update_log_text.emit)
|
self.task.update_log_text.connect(self.update_log_text.emit)
|
||||||
self.task.update_user_info.connect(
|
self.task.update_user_info.connect(self.update_user_info.emit)
|
||||||
lambda modes, uids, days, lasts, notes, numbs: self.update_user_info.emit(
|
|
||||||
self.member_dict[self.task_dict[i][0]][1],
|
|
||||||
modes,
|
|
||||||
uids,
|
|
||||||
days,
|
|
||||||
lasts,
|
|
||||||
notes,
|
|
||||||
numbs,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.task.accomplish.connect(
|
self.task.accomplish.connect(
|
||||||
lambda log: self.task_accomplish(self.task_dict[i][0], log)
|
lambda log: self.task_accomplish(task[2], log)
|
||||||
)
|
)
|
||||||
|
|
||||||
self.task.run()
|
self.task.run()
|
||||||
|
|
||||||
Config.running_list.remove(self.task_dict[i][0])
|
Config.running_list.remove(task[2])
|
||||||
|
|
||||||
self.task_dict[i][1] = "完成"
|
task[1] = "完成"
|
||||||
logger.info(f"任务完成:{self.task_dict[i][0]}")
|
self.update_task_list.emit(self.task_list)
|
||||||
self.push_info_bar.emit("info", "任务完成", self.task_dict[i][0], 3000)
|
logger.info(f"任务完成:{task[0]}")
|
||||||
|
self.push_info_bar.emit("info", "任务完成", task[0], 3000)
|
||||||
|
|
||||||
self.accomplish.emit(self.logs)
|
self.accomplish.emit(self.logs)
|
||||||
|
|
||||||
def search_member(self) -> dict:
|
|
||||||
"""搜索所有脚本实例及其路径"""
|
|
||||||
|
|
||||||
member_dict = {}
|
|
||||||
|
|
||||||
if (Config.app_path / "config/MaaConfig").exists():
|
|
||||||
for subdir in (Config.app_path / "config/MaaConfig").iterdir():
|
|
||||||
if subdir.is_dir():
|
|
||||||
|
|
||||||
member_dict[subdir.name] = ["Maa", subdir]
|
|
||||||
|
|
||||||
return member_dict
|
|
||||||
|
|
||||||
def task_accomplish(self, name: str, log: dict):
|
def task_accomplish(self, name: str, log: dict):
|
||||||
"""保存保存任务结果"""
|
"""保存保存任务结果"""
|
||||||
|
|
||||||
@@ -185,7 +178,6 @@ class _TaskManager(QObject):
|
|||||||
|
|
||||||
create_gui = Signal(Task)
|
create_gui = Signal(Task)
|
||||||
connect_gui = Signal(Task)
|
connect_gui = Signal(Task)
|
||||||
push_info_bar = Signal(str, str, str, int)
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(_TaskManager, self).__init__()
|
super(_TaskManager, self).__init__()
|
||||||
@@ -205,13 +197,16 @@ class _TaskManager(QObject):
|
|||||||
|
|
||||||
logger.info(f"任务开始:{name}")
|
logger.info(f"任务开始:{name}")
|
||||||
MainInfoBar.push_info_bar("info", "任务开始", name, 3000)
|
MainInfoBar.push_info_bar("info", "任务开始", name, 3000)
|
||||||
|
SoundPlayer.play("任务开始")
|
||||||
|
|
||||||
Config.running_list.append(name)
|
Config.running_list.append(name)
|
||||||
self.task_dict[name] = Task(mode, name, info)
|
self.task_dict[name] = Task(mode, name, info)
|
||||||
|
self.task_dict[name].check_maa_version.connect(self.check_maa_version)
|
||||||
self.task_dict[name].question.connect(
|
self.task_dict[name].question.connect(
|
||||||
lambda title, content: self.push_dialog(name, title, content)
|
lambda title, content: self.push_dialog(name, title, content)
|
||||||
)
|
)
|
||||||
self.task_dict[name].push_info_bar.connect(MainInfoBar.push_info_bar)
|
self.task_dict[name].push_info_bar.connect(MainInfoBar.push_info_bar)
|
||||||
|
self.task_dict[name].play_sound.connect(SoundPlayer.play)
|
||||||
self.task_dict[name].update_user_info.connect(Config.change_user_info)
|
self.task_dict[name].update_user_info.connect(Config.change_user_info)
|
||||||
self.task_dict[name].accomplish.connect(
|
self.task_dict[name].accomplish.connect(
|
||||||
lambda logs: self.remove_task(mode, name, logs)
|
lambda logs: self.remove_task(mode, name, logs)
|
||||||
@@ -247,50 +242,91 @@ class _TaskManager(QObject):
|
|||||||
self.task_dict[name].quit()
|
self.task_dict[name].quit()
|
||||||
self.task_dict[name].wait()
|
self.task_dict[name].wait()
|
||||||
|
|
||||||
def remove_task(self, mode: str, name: str, logs: str):
|
def remove_task(self, mode: str, name: str, logs: list):
|
||||||
"""任务结束后的处理"""
|
"""任务结束后的处理"""
|
||||||
|
|
||||||
logger.info(f"任务结束:{name}")
|
logger.info(f"任务结束:{name}")
|
||||||
MainInfoBar.push_info_bar("info", "任务结束", name, 3000)
|
MainInfoBar.push_info_bar("info", "任务结束", name, 3000)
|
||||||
|
SoundPlayer.play("任务结束")
|
||||||
|
|
||||||
self.task_dict[name].deleteLater()
|
self.task_dict[name].deleteLater()
|
||||||
|
|
||||||
if len(logs) > 0:
|
|
||||||
time = logs[0][1]["Time"]
|
|
||||||
history = ""
|
|
||||||
for log in logs:
|
|
||||||
Config.save_history(log[0], log[1])
|
|
||||||
history += (
|
|
||||||
f"任务名称:{log[0]},{log[1]["History"].replace("\n","\n ")}\n"
|
|
||||||
)
|
|
||||||
Config.save_history(name, {"Time": time, "History": history})
|
|
||||||
else:
|
|
||||||
Config.save_history(
|
|
||||||
name,
|
|
||||||
{
|
|
||||||
"Time": datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
|
|
||||||
"History": "没有任务被执行",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
self.task_dict.pop(name)
|
self.task_dict.pop(name)
|
||||||
Config.running_list.remove(name)
|
Config.running_list.remove(name)
|
||||||
|
|
||||||
if "调度队列" in name and "人工排查" not in mode:
|
if "调度队列" in name and "人工排查" not in mode:
|
||||||
with (Config.app_path / f"config/QueueConfig/{name}.json").open(
|
|
||||||
"r", encoding="utf-8"
|
if len(logs) > 0:
|
||||||
) as f:
|
time = logs[0][1]["Time"]
|
||||||
info = json.load(f)
|
history = ""
|
||||||
System.set_power(info["QueueSet"]["AfterAccomplish"])
|
for log in logs:
|
||||||
|
history += f"任务名称:{log[0]},{log[1]["History"].replace("\n","\n ")}\n"
|
||||||
|
Config.save_history(name, {"Time": time, "History": history})
|
||||||
|
else:
|
||||||
|
Config.save_history(
|
||||||
|
name,
|
||||||
|
{
|
||||||
|
"Time": datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
|
||||||
|
"History": "没有任务被执行",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
Config.queue_dict[name]["Config"].get(
|
||||||
|
Config.queue_dict[name]["Config"].queueSet_AfterAccomplish
|
||||||
|
)
|
||||||
|
!= "NoAction"
|
||||||
|
and Config.power_sign == "NoAction"
|
||||||
|
):
|
||||||
|
Config.set_power_sign(
|
||||||
|
Config.queue_dict[name]["Config"].get(
|
||||||
|
Config.queue_dict[name]["Config"].queueSet_AfterAccomplish
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
if Config.args.mode == "cli" and Config.power_sign == "NoAction":
|
||||||
|
Config.set_power_sign("KillSelf")
|
||||||
|
|
||||||
|
def check_maa_version(self, v: str):
|
||||||
|
"""检查MAA版本"""
|
||||||
|
|
||||||
|
network = Network.add_task(
|
||||||
|
mode="get",
|
||||||
|
url="https://mirrorchyan.com/api/resources/MAA/latest?user_agent=AutoMaaGui&os=win&arch=x64&channel=stable",
|
||||||
|
)
|
||||||
|
network.loop.exec()
|
||||||
|
network_result = Network.get_result(network)
|
||||||
|
if network_result["status_code"] == 200:
|
||||||
|
maa_info = network_result["response_json"]
|
||||||
|
else:
|
||||||
|
logger.warning(f"获取MAA版本信息时出错:{network_result['error_message']}")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning",
|
||||||
|
"获取MAA版本信息时出错",
|
||||||
|
f"网络错误:{network_result['status_code']}",
|
||||||
|
5000,
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if version.parse(maa_info["data"]["version_name"]) > version.parse(v):
|
||||||
|
|
||||||
|
logger.info(
|
||||||
|
f"检测到MAA版本过低:{v},最新版本:{maa_info['data']['version_name']}"
|
||||||
|
)
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"info",
|
||||||
|
"MAA版本过低",
|
||||||
|
f"当前版本:{v},最新稳定版:{maa_info['data']['version_name']}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
|
||||||
def push_dialog(self, name: str, title: str, content: str):
|
def push_dialog(self, name: str, title: str, content: str):
|
||||||
"""推送对话框"""
|
"""推送对话框"""
|
||||||
|
|
||||||
choice = Dialog(title, content, None)
|
choice = MessageBox(title, content, Config.main_window)
|
||||||
choice.yesButton.setText("是")
|
choice.yesButton.setText("是")
|
||||||
choice.cancelButton.setText("否")
|
choice.cancelButton.setText("否")
|
||||||
|
|
||||||
self.task_dict[name].question_response.emit(bool(choice.exec_()))
|
self.task_dict[name].question_response.emit(bool(choice.exec()))
|
||||||
|
|
||||||
|
|
||||||
TaskManager = _TaskManager()
|
TaskManager = _TaskManager()
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,110 +16,123 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA主业务定时器
|
AUTO_MAA主业务定时器
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtWidgets import QWidget
|
from PySide6.QtCore import QObject, QTimer
|
||||||
from PySide6.QtCore import QTimer
|
|
||||||
import json
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
import pyautogui
|
from pathlib import Path
|
||||||
|
import keyboard
|
||||||
|
|
||||||
from .config import Config
|
from .config import Config
|
||||||
from .task_manager import TaskManager
|
from .task_manager import TaskManager
|
||||||
from app.services import System
|
from app.services import System
|
||||||
|
|
||||||
|
|
||||||
class _MainTimer(QWidget):
|
class _MainTimer(QObject):
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super().__init__(parent)
|
super().__init__(parent)
|
||||||
|
|
||||||
self.if_FailSafeException = False
|
|
||||||
|
|
||||||
self.Timer = QTimer()
|
self.Timer = QTimer()
|
||||||
self.Timer.timeout.connect(self.timed_start)
|
self.Timer.timeout.connect(self.timed_start)
|
||||||
self.Timer.timeout.connect(self.set_silence)
|
self.Timer.timeout.connect(self.set_silence)
|
||||||
|
self.Timer.timeout.connect(self.check_power)
|
||||||
self.Timer.start(1000)
|
self.Timer.start(1000)
|
||||||
|
self.LongTimer = QTimer()
|
||||||
|
self.LongTimer.timeout.connect(self.long_timed_task)
|
||||||
|
self.LongTimer.start(3600000)
|
||||||
|
|
||||||
|
def long_timed_task(self):
|
||||||
|
"""长时间定期检定任务"""
|
||||||
|
|
||||||
|
Config.get_gameid()
|
||||||
|
Config.main_window.setting.show_notice()
|
||||||
|
if Config.get(Config.update_IfAutoUpdate):
|
||||||
|
Config.main_window.setting.check_update()
|
||||||
|
|
||||||
def timed_start(self):
|
def timed_start(self):
|
||||||
"""定时启动代理任务"""
|
"""定时启动代理任务"""
|
||||||
|
|
||||||
# 获取定时列表
|
for name, info in Config.queue_dict.items():
|
||||||
queue_list = self.search_queue()
|
|
||||||
|
|
||||||
for i in queue_list:
|
if not info["Config"].get(info["Config"].queueSet_Enabled):
|
||||||
|
|
||||||
name, info = i
|
|
||||||
|
|
||||||
if not info["QueueSet"]["Enabled"]:
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
history = Config.get_history(name)
|
data = info["Config"].toDict()
|
||||||
|
|
||||||
time_set = [
|
time_set = [
|
||||||
info["Time"][f"TimeSet_{_}"]
|
data["Time"][f"TimeSet_{_}"]
|
||||||
for _ in range(10)
|
for _ in range(10)
|
||||||
if info["Time"][f"TimeEnabled_{_}"]
|
if data["Time"][f"TimeEnabled_{_}"]
|
||||||
]
|
]
|
||||||
# 按时间调起代理任务
|
# 按时间调起代理任务
|
||||||
curtime = datetime.now().strftime("%Y-%m-%d %H:%M")
|
curtime = datetime.now().strftime("%Y-%m-%d %H:%M")
|
||||||
if (
|
if (
|
||||||
curtime[11:16] in time_set
|
curtime[11:16] in time_set
|
||||||
and curtime != history["Time"][:16]
|
and curtime
|
||||||
|
!= info["Config"].get(info["Config"].Data_LastProxyTime)[:16]
|
||||||
and name not in Config.running_list
|
and name not in Config.running_list
|
||||||
):
|
):
|
||||||
|
|
||||||
logger.info(f"定时任务:{name}")
|
logger.info(f"定时任务:{name}")
|
||||||
TaskManager.add_task("自动代理_新调度台", name, info)
|
TaskManager.add_task("自动代理_新调度台", name, data)
|
||||||
|
|
||||||
def set_silence(self):
|
def set_silence(self):
|
||||||
"""设置静默模式"""
|
"""设置静默模式"""
|
||||||
|
|
||||||
if (
|
if (
|
||||||
Config.global_config.get(Config.global_config.function_IfSilence)
|
not Config.if_ignore_silence
|
||||||
and Config.global_config.get(Config.global_config.function_BossKey) != ""
|
and Config.get(Config.function_IfSilence)
|
||||||
|
and Config.get(Config.function_BossKey) != ""
|
||||||
):
|
):
|
||||||
|
|
||||||
windows = System.get_window_info()
|
windows = System.get_window_info()
|
||||||
|
|
||||||
|
# 此处排除雷电名为新通知的窗口
|
||||||
if any(
|
if any(
|
||||||
str(emulator_path) in window
|
str(emulator_path) in window and window[0] != "新通知"
|
||||||
for window in windows
|
for window in windows
|
||||||
for emulator_path in Config.silence_list
|
for emulator_path in Config.silence_list
|
||||||
):
|
):
|
||||||
try:
|
try:
|
||||||
pyautogui.hotkey(
|
keyboard.press_and_release(
|
||||||
*[
|
"+".join(
|
||||||
_.strip().lower()
|
_.strip().lower()
|
||||||
for _ in Config.global_config.get(
|
for _ in Config.get(Config.function_BossKey).split("+")
|
||||||
Config.global_config.function_BossKey
|
)
|
||||||
).split("+")
|
|
||||||
]
|
|
||||||
)
|
)
|
||||||
except pyautogui.FailSafeException as e:
|
except Exception as e:
|
||||||
if not self.if_FailSafeException:
|
logger.error(f"模拟按键时出错:{e}")
|
||||||
logger.warning(f"FailSafeException: {e}")
|
|
||||||
self.if_FailSafeException = True
|
|
||||||
|
|
||||||
def search_queue(self) -> list:
|
def check_power(self):
|
||||||
"""搜索所有调度队列实例"""
|
|
||||||
|
|
||||||
queue_list = []
|
if Config.power_sign != "NoAction" and not Config.running_list:
|
||||||
|
|
||||||
if (Config.app_path / "config/QueueConfig").exists():
|
from app.ui import ProgressRingMessageBox
|
||||||
for json_file in (Config.app_path / "config/QueueConfig").glob("*.json"):
|
|
||||||
with json_file.open("r", encoding="utf-8") as f:
|
|
||||||
info = json.load(f)
|
|
||||||
queue_list.append([json_file.stem, info])
|
|
||||||
|
|
||||||
return queue_list
|
mode_book = {
|
||||||
|
"KillSelf": "退出软件",
|
||||||
|
"Sleep": "睡眠",
|
||||||
|
"Hibernate": "休眠",
|
||||||
|
"Shutdown": "关机",
|
||||||
|
}
|
||||||
|
|
||||||
|
choice = ProgressRingMessageBox(
|
||||||
|
Config.main_window, f"{mode_book[Config.power_sign]}倒计时"
|
||||||
|
)
|
||||||
|
if choice.exec():
|
||||||
|
System.set_power(Config.power_sign)
|
||||||
|
Config.set_power_sign("NoAction")
|
||||||
|
else:
|
||||||
|
Config.set_power_sign("NoAction")
|
||||||
|
|
||||||
|
|
||||||
MainTimer = _MainTimer()
|
MainTimer = _MainTimer()
|
||||||
|
|||||||
1722
app/models/MAA.py
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA模组包
|
AUTO_MAA模组包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA服务包
|
AUTO_MAA服务包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -32,5 +32,6 @@ __license__ = "GPL-3.0 license"
|
|||||||
from .notification import Notify
|
from .notification import Notify
|
||||||
from .security import Crypto
|
from .security import Crypto
|
||||||
from .system import System
|
from .system import System
|
||||||
|
from .skland import skland_sign_in
|
||||||
|
|
||||||
__all__ = ["Notify", "Crypto", "System"]
|
__all__ = ["Notify", "Crypto", "System", "skland_sign_in"]
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,34 +16,35 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA通知服务
|
AUTO_MAA通知服务
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from PySide6.QtWidgets import QWidget
|
|
||||||
from PySide6.QtCore import Signal
|
|
||||||
import requests
|
|
||||||
from loguru import logger
|
|
||||||
from plyer import notification
|
|
||||||
import re
|
import re
|
||||||
import smtplib
|
import smtplib
|
||||||
from email.mime.text import MIMEText
|
import time
|
||||||
from email.mime.multipart import MIMEMultipart
|
|
||||||
from email.header import Header
|
from email.header import Header
|
||||||
|
from email.mime.multipart import MIMEMultipart
|
||||||
|
from email.mime.text import MIMEText
|
||||||
from email.utils import formataddr
|
from email.utils import formataddr
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
from serverchan_sdk import sc_send
|
import requests
|
||||||
|
from PySide6.QtCore import QObject, Signal
|
||||||
|
from loguru import logger
|
||||||
|
from plyer import notification
|
||||||
|
|
||||||
from app.core import Config
|
from app.core import Config
|
||||||
from app.services.security import Crypto
|
from app.services.security import Crypto
|
||||||
|
from app.utils.ImageUtils import ImageUtils
|
||||||
|
|
||||||
|
|
||||||
class Notification(QWidget):
|
class Notification(QObject):
|
||||||
|
|
||||||
push_info_bar = Signal(str, str, str, int)
|
push_info_bar = Signal(str, str, str, int)
|
||||||
|
|
||||||
@@ -53,7 +54,7 @@ class Notification(QWidget):
|
|||||||
def push_plyer(self, title, message, ticker, t):
|
def push_plyer(self, title, message, ticker, t):
|
||||||
"""推送系统通知"""
|
"""推送系统通知"""
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.notify_IfPushPlyer):
|
if Config.get(Config.notify_IfPushPlyer):
|
||||||
|
|
||||||
notification.notify(
|
notification.notify(
|
||||||
title=title,
|
title=title,
|
||||||
@@ -67,181 +68,347 @@ class Notification(QWidget):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def send_mail(self, mode, title, content) -> None:
|
def send_mail(self, mode, title, content, to_address) -> None:
|
||||||
"""推送邮件通知"""
|
"""推送邮件通知"""
|
||||||
|
if (
|
||||||
|
Config.get(Config.notify_SMTPServerAddress) == ""
|
||||||
|
or Config.get(Config.notify_AuthorizationCode) == ""
|
||||||
|
or not bool(
|
||||||
|
re.match(
|
||||||
|
r"^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$",
|
||||||
|
Config.get(Config.notify_FromAddress),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
or not bool(
|
||||||
|
re.match(
|
||||||
|
r"^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$",
|
||||||
|
to_address,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
):
|
||||||
|
logger.error(
|
||||||
|
"请正确设置邮件通知的SMTP服务器地址、授权码、发件人地址和收件人地址"
|
||||||
|
)
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"邮件通知推送异常",
|
||||||
|
"请正确设置邮件通知的SMTP服务器地址、授权码、发件人地址和收件人地址",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.notify_IfSendMail):
|
try:
|
||||||
|
# 定义邮件正文
|
||||||
|
if mode == "文本":
|
||||||
|
message = MIMEText(content, "plain", "utf-8")
|
||||||
|
elif mode == "网页":
|
||||||
|
message = MIMEMultipart("alternative")
|
||||||
|
message["From"] = formataddr(
|
||||||
|
(
|
||||||
|
Header("AUTO_MAA通知服务", "utf-8").encode(),
|
||||||
|
Config.get(Config.notify_FromAddress),
|
||||||
|
)
|
||||||
|
) # 发件人显示的名字
|
||||||
|
message["To"] = formataddr(
|
||||||
|
(
|
||||||
|
Header("AUTO_MAA用户", "utf-8").encode(),
|
||||||
|
to_address,
|
||||||
|
)
|
||||||
|
) # 收件人显示的名字
|
||||||
|
message["Subject"] = Header(title, "utf-8")
|
||||||
|
|
||||||
if (
|
if mode == "网页":
|
||||||
Config.global_config.get(Config.global_config.notify_SMTPServerAddress)
|
message.attach(MIMEText(content, "html", "utf-8"))
|
||||||
== ""
|
|
||||||
or Config.global_config.get(
|
|
||||||
Config.global_config.notify_AuthorizationCode
|
|
||||||
)
|
|
||||||
== ""
|
|
||||||
or not bool(
|
|
||||||
re.match(
|
|
||||||
r"^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$",
|
|
||||||
Config.global_config.get(
|
|
||||||
Config.global_config.notify_FromAddress
|
|
||||||
),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
or not bool(
|
|
||||||
re.match(
|
|
||||||
r"^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$",
|
|
||||||
Config.global_config.get(Config.global_config.notify_ToAddress),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
):
|
|
||||||
logger.error(
|
|
||||||
"请正确设置邮件通知的SMTP服务器地址、授权码、发件人地址和收件人地址"
|
|
||||||
)
|
|
||||||
self.push_info_bar.emit(
|
|
||||||
"error",
|
|
||||||
"邮件通知推送异常",
|
|
||||||
"请正确设置邮件通知的SMTP服务器地址、授权码、发件人地址和收件人地址",
|
|
||||||
-1,
|
|
||||||
)
|
|
||||||
return None
|
|
||||||
|
|
||||||
try:
|
smtpObj = smtplib.SMTP_SSL(
|
||||||
# 定义邮件正文
|
Config.get(Config.notify_SMTPServerAddress),
|
||||||
if mode == "文本":
|
465,
|
||||||
message = MIMEText(content, "plain", "utf-8")
|
)
|
||||||
elif mode == "网页":
|
smtpObj.login(
|
||||||
message = MIMEMultipart("alternative")
|
Config.get(Config.notify_FromAddress),
|
||||||
message["From"] = formataddr(
|
Crypto.win_decryptor(Config.get(Config.notify_AuthorizationCode)),
|
||||||
(
|
)
|
||||||
Header("AUTO_MAA通知服务", "utf-8").encode(),
|
smtpObj.sendmail(
|
||||||
Config.global_config.get(
|
Config.get(Config.notify_FromAddress),
|
||||||
Config.global_config.notify_FromAddress
|
to_address,
|
||||||
),
|
message.as_string(),
|
||||||
)
|
)
|
||||||
) # 发件人显示的名字
|
smtpObj.quit()
|
||||||
message["To"] = formataddr(
|
logger.success("邮件发送成功")
|
||||||
(
|
return None
|
||||||
Header("AUTO_MAA用户", "utf-8").encode(),
|
except Exception as e:
|
||||||
Config.global_config.get(Config.global_config.notify_ToAddress),
|
logger.error(f"发送邮件时出错:\n{e}")
|
||||||
)
|
self.push_info_bar.emit("error", "发送邮件时出错", f"{e}", -1)
|
||||||
) # 收件人显示的名字
|
return None
|
||||||
message["Subject"] = Header(title, "utf-8")
|
return None
|
||||||
|
|
||||||
if mode == "网页":
|
def ServerChanPush(self, title, content, send_key, tag, channel):
|
||||||
message.attach(MIMEText(content, "html", "utf-8"))
|
|
||||||
|
|
||||||
smtpObj = smtplib.SMTP_SSL(
|
|
||||||
Config.global_config.get(
|
|
||||||
Config.global_config.notify_SMTPServerAddress
|
|
||||||
),
|
|
||||||
465,
|
|
||||||
)
|
|
||||||
smtpObj.login(
|
|
||||||
Config.global_config.get(Config.global_config.notify_FromAddress),
|
|
||||||
Crypto.win_decryptor(
|
|
||||||
Config.global_config.get(
|
|
||||||
Config.global_config.notify_AuthorizationCode
|
|
||||||
)
|
|
||||||
),
|
|
||||||
)
|
|
||||||
smtpObj.sendmail(
|
|
||||||
Config.global_config.get(Config.global_config.notify_FromAddress),
|
|
||||||
Config.global_config.get(Config.global_config.notify_ToAddress),
|
|
||||||
message.as_string(),
|
|
||||||
)
|
|
||||||
smtpObj.quit()
|
|
||||||
logger.success("邮件发送成功")
|
|
||||||
except Exception as e:
|
|
||||||
logger.error(f"发送邮件时出错:\n{e}")
|
|
||||||
self.push_info_bar.emit("error", "发送邮件时出错", f"{e}", -1)
|
|
||||||
|
|
||||||
def ServerChanPush(self, title, content):
|
|
||||||
"""使用Server酱推送通知"""
|
"""使用Server酱推送通知"""
|
||||||
|
if not send_key:
|
||||||
|
logger.error("请正确设置Server酱的SendKey")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error", "Server酱通知推送异常", "请正确设置Server酱的SendKey", -1
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.notify_IfServerChan):
|
try:
|
||||||
send_key = Config.global_config.get(
|
# 构造 URL
|
||||||
Config.global_config.notify_ServerChanKey
|
if send_key.startswith("sctp"):
|
||||||
)
|
match = re.match(r"^sctp(\d+)t", send_key)
|
||||||
option = {}
|
if match:
|
||||||
is_valid = lambda s: s == "" or (
|
url = f"https://{match.group(1)}.push.ft07.com/send/{send_key}.send"
|
||||||
s == "|".join(s.split("|")) and (s.count("|") == 0 or all(s.split("|")))
|
else:
|
||||||
)
|
raise ValueError("SendKey 格式错误(sctp)")
|
||||||
"""
|
|
||||||
is_valid => True, 如果启用的话需要正确设置Tag和Channel。
|
|
||||||
允许空的Tag和Channel即不启用,但不允许例如a||b,|a|b,a|b|,||||
|
|
||||||
"""
|
|
||||||
send_tag = Config.global_config.get(
|
|
||||||
Config.global_config.notify_ServerChanTag
|
|
||||||
)
|
|
||||||
send_channel = Config.global_config.get(
|
|
||||||
Config.global_config.notify_ServerChanChannel
|
|
||||||
)
|
|
||||||
|
|
||||||
if is_valid(send_tag):
|
|
||||||
option["tags"] = send_tag
|
|
||||||
else:
|
else:
|
||||||
option["tags"] = ""
|
url = f"https://sctapi.ftqq.com/{send_key}.send"
|
||||||
logger.warning("请正确设置Auto_MAA中ServerChan的Tag。")
|
|
||||||
|
# 构建 tags 和 channel
|
||||||
|
def is_valid(s):
|
||||||
|
return s == "" or (
|
||||||
|
s == "|".join(s.split("|"))
|
||||||
|
and (s.count("|") == 0 or all(s.split("|")))
|
||||||
|
)
|
||||||
|
|
||||||
|
tags = "|".join(_.strip() for _ in tag.split("|"))
|
||||||
|
channels = "|".join(_.strip() for _ in channel.split("|"))
|
||||||
|
|
||||||
|
options = {}
|
||||||
|
if is_valid(tags):
|
||||||
|
options["tags"] = tags
|
||||||
|
else:
|
||||||
|
logger.warning("Server酱 Tag 配置不正确,将被忽略")
|
||||||
self.push_info_bar.emit(
|
self.push_info_bar.emit(
|
||||||
"warning",
|
"warning",
|
||||||
"Server酱通知推送异常",
|
"Server酱通知推送异常",
|
||||||
"请正确设置Auto_MAA中ServerChan的Tag。",
|
"请正确设置 ServerChan 的 Tag",
|
||||||
-1,
|
-1,
|
||||||
)
|
)
|
||||||
|
|
||||||
if is_valid(send_channel):
|
if is_valid(channels):
|
||||||
option["channel"] = send_channel
|
options["channel"] = channels
|
||||||
else:
|
else:
|
||||||
option["channel"] = ""
|
logger.warning("Server酱 Channel 配置不正确,将被忽略")
|
||||||
logger.warning("请正确设置Auto_MAA中ServerChan的Channel。")
|
|
||||||
self.push_info_bar.emit(
|
self.push_info_bar.emit(
|
||||||
"warning",
|
"warning",
|
||||||
"Server酱通知推送异常",
|
"Server酱通知推送异常",
|
||||||
"请正确设置Auto_MAA中ServerChan的Channel。",
|
"请正确设置 ServerChan 的 Channel",
|
||||||
-1,
|
-1,
|
||||||
)
|
)
|
||||||
|
|
||||||
response = sc_send(send_key, title, content, option)
|
# 请求发送
|
||||||
if response["code"] == 0:
|
params = {"title": title, "desp": content, **options}
|
||||||
|
headers = {"Content-Type": "application/json;charset=utf-8"}
|
||||||
|
|
||||||
|
response = requests.post(url, json=params, headers=headers, timeout=10)
|
||||||
|
result = response.json()
|
||||||
|
|
||||||
|
if result.get("code") == 0:
|
||||||
logger.info("Server酱推送通知成功")
|
logger.info("Server酱推送通知成功")
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
logger.info("Server酱推送通知失败")
|
error_code = result.get("code", "-1")
|
||||||
logger.error(response)
|
logger.error(f"Server酱通知推送失败:响应码:{error_code}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error", "Server酱通知推送失败", f"响应码:{error_code}", -1
|
||||||
|
)
|
||||||
|
return f"Server酱通知推送失败:{error_code}"
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logger.exception("Server酱通知推送异常")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"Server酱通知推送异常",
|
||||||
|
"请检查相关设置和网络连接。如全部配置正确,请稍后再试。",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return f"Server酱通知推送异常:{str(e)}"
|
||||||
|
|
||||||
|
def CompanyWebHookBotPush(self, title, content, webhook_url):
|
||||||
|
"""使用企业微信群机器人推送通知"""
|
||||||
|
if webhook_url == "":
|
||||||
|
logger.error("请正确设置企业微信群机器人的WebHook地址")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人通知推送异常",
|
||||||
|
"请正确设置企业微信群机器人的WebHook地址",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
content = f"{title}\n{content}"
|
||||||
|
data = {"msgtype": "text", "text": {"content": content}}
|
||||||
|
|
||||||
|
for _ in range(3):
|
||||||
|
try:
|
||||||
|
response = requests.post(
|
||||||
|
url=webhook_url,
|
||||||
|
json=data,
|
||||||
|
timeout=10,
|
||||||
|
)
|
||||||
|
info = response.json()
|
||||||
|
break
|
||||||
|
except Exception as e:
|
||||||
|
err = e
|
||||||
|
time.sleep(0.1)
|
||||||
|
else:
|
||||||
|
logger.error(f"推送企业微信群机器人时出错:{err}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人通知推送失败",
|
||||||
|
f"使用企业微信群机器人推送通知时出错:{err}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if info["errcode"] == 0:
|
||||||
|
logger.info("企业微信群机器人推送通知成功")
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
logger.error(f"企业微信群机器人推送通知失败:{info}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人通知推送失败",
|
||||||
|
f"使用企业微信群机器人推送通知时出错:{err}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return f"使用企业微信群机器人推送通知时出错:{err}"
|
||||||
|
|
||||||
|
def CompanyWebHookBotPushImage(self, image_path: Path, webhook_url: str) -> bool:
|
||||||
|
"""使用企业微信群机器人推送图片通知"""
|
||||||
|
try:
|
||||||
|
# 压缩图片
|
||||||
|
ImageUtils.compress_image_if_needed(image_path)
|
||||||
|
|
||||||
|
# 检查图片是否存在
|
||||||
|
if not image_path.exists():
|
||||||
|
logger.error(
|
||||||
|
"图片推送异常 | 图片不存在或者压缩失败,请检查图片路径是否正确"
|
||||||
|
)
|
||||||
self.push_info_bar.emit(
|
self.push_info_bar.emit(
|
||||||
"error",
|
"error",
|
||||||
"Server酱通知推送失败",
|
"企业微信群机器人通知推送异常",
|
||||||
f'使用Server酱推送通知时出错:\n{response["data"]['error']}',
|
"图片不存在或者压缩失败,请检查图片路径是否正确",
|
||||||
-1,
|
-1,
|
||||||
)
|
)
|
||||||
return f'使用Server酱推送通知时出错:\n{response["data"]['error']}'
|
return False
|
||||||
|
|
||||||
def CompanyWebHookBotPush(self, title, content):
|
if not webhook_url:
|
||||||
"""使用企业微信群机器人推送通知"""
|
logger.error("请正确设置企业微信群机器人的WebHook地址")
|
||||||
if Config.global_config.get(Config.global_config.notify_IfCompanyWebHookBot):
|
self.push_info_bar.emit(
|
||||||
content = f"{title}\n{content}"
|
"error",
|
||||||
data = {"msgtype": "text", "text": {"content": content}}
|
"企业微信群机器人通知推送异常",
|
||||||
response = requests.post(
|
"请正确设置企业微信群机器人的WebHook地址",
|
||||||
url=Config.global_config.get(
|
-1,
|
||||||
Config.global_config.notify_CompanyWebHookBotUrl
|
)
|
||||||
),
|
return False
|
||||||
json=data,
|
|
||||||
)
|
# 获取图片base64和md5
|
||||||
if response.json()["errcode"] == 0:
|
try:
|
||||||
logger.info("企业微信群机器人推送通知成功")
|
image_base64 = ImageUtils.get_base64_from_file(str(image_path))
|
||||||
|
image_md5 = ImageUtils.calculate_md5_from_file(str(image_path))
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"图片编码或MD5计算失败:{e}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人通知推送异常",
|
||||||
|
f"图片编码或MD5计算失败:{e}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
|
||||||
|
data = {
|
||||||
|
"msgtype": "image",
|
||||||
|
"image": {"base64": image_base64, "md5": image_md5},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _ in range(3):
|
||||||
|
try:
|
||||||
|
response = requests.post(
|
||||||
|
url=webhook_url,
|
||||||
|
json=data,
|
||||||
|
timeout=10,
|
||||||
|
)
|
||||||
|
info = response.json()
|
||||||
|
break
|
||||||
|
except requests.RequestException as e:
|
||||||
|
err = e
|
||||||
|
logger.warning(f"推送企业微信群机器人图片第{_+1}次失败:{e}")
|
||||||
|
time.sleep(0.1)
|
||||||
|
else:
|
||||||
|
logger.error(f"推送企业微信群机器人图片时出错:{err}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人图片推送失败",
|
||||||
|
f"使用企业微信群机器人推送图片时出错:{err}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
|
||||||
|
if info.get("errcode") == 0:
|
||||||
|
logger.info("企业微信群机器人推送图片成功")
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
logger.info("企业微信群机器人推送通知失败")
|
logger.error(f"企业微信群机器人推送图片失败:{info}")
|
||||||
logger.error(response.json())
|
|
||||||
self.push_info_bar.emit(
|
self.push_info_bar.emit(
|
||||||
"error",
|
"error",
|
||||||
"企业微信群机器人通知推送失败",
|
"企业微信群机器人图片推送失败",
|
||||||
f'使用企业微信群机器人推送通知时出错:\n{response.json()["errmsg"]}',
|
f"使用企业微信群机器人推送图片时出错:{info}",
|
||||||
-1,
|
-1,
|
||||||
)
|
)
|
||||||
return (
|
return False
|
||||||
f'使用企业微信群机器人推送通知时出错:\n{response.json()["errmsg"]}'
|
|
||||||
)
|
except Exception as e:
|
||||||
|
logger.error(f"推送企业微信群机器人图片时发生未知异常:{e}")
|
||||||
|
self.push_info_bar.emit(
|
||||||
|
"error",
|
||||||
|
"企业微信群机器人图片推送失败",
|
||||||
|
f"发生未知异常:{e}",
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
|
||||||
|
def send_test_notification(self):
|
||||||
|
"""发送测试通知到所有已启用的通知渠道"""
|
||||||
|
# 发送系统通知
|
||||||
|
self.push_plyer(
|
||||||
|
"测试通知",
|
||||||
|
"这是 AUTO_MAA 外部通知测试信息。如果你看到了这段内容,说明 AUTO_MAA 的通知功能已经正确配置且可以正常工作!",
|
||||||
|
"测试通知",
|
||||||
|
3,
|
||||||
|
)
|
||||||
|
|
||||||
|
# 发送邮件通知
|
||||||
|
if Config.get(Config.notify_IfSendMail):
|
||||||
|
self.send_mail(
|
||||||
|
"文本",
|
||||||
|
"AUTO_MAA测试通知",
|
||||||
|
"这是 AUTO_MAA 外部通知测试信息。如果你看到了这段内容,说明 AUTO_MAA 的通知功能已经正确配置且可以正常工作!",
|
||||||
|
Config.get(Config.notify_ToAddress),
|
||||||
|
)
|
||||||
|
|
||||||
|
# 发送Server酱通知
|
||||||
|
if Config.get(Config.notify_IfServerChan):
|
||||||
|
self.ServerChanPush(
|
||||||
|
"AUTO_MAA测试通知",
|
||||||
|
"这是 AUTO_MAA 外部通知测试信息。如果你看到了这段内容,说明 AUTO_MAA 的通知功能已经正确配置且可以正常工作!",
|
||||||
|
Config.get(Config.notify_ServerChanKey),
|
||||||
|
Config.get(Config.notify_ServerChanTag),
|
||||||
|
Config.get(Config.notify_ServerChanChannel),
|
||||||
|
)
|
||||||
|
|
||||||
|
# 发送企业微信机器人通知
|
||||||
|
if Config.get(Config.notify_IfCompanyWebHookBot):
|
||||||
|
self.CompanyWebHookBotPush(
|
||||||
|
"AUTO_MAA测试通知",
|
||||||
|
"这是 AUTO_MAA 外部通知测试信息。如果你看到了这段内容,说明 AUTO_MAA 的通知功能已经正确配置且可以正常工作!",
|
||||||
|
Config.get(Config.notify_CompanyWebHookBotUrl),
|
||||||
|
)
|
||||||
|
Notify.CompanyWebHookBotPushImage(
|
||||||
|
Config.app_path / "resources/images/notification/test_notify.png",
|
||||||
|
Config.get(Config.notify_CompanyWebHookBotUrl),
|
||||||
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
Notify = Notification()
|
Notify = Notification()
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,17 +16,16 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA安全服务
|
AUTO_MAA安全服务
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
import sqlite3
|
|
||||||
import hashlib
|
import hashlib
|
||||||
import random
|
import random
|
||||||
import secrets
|
import secrets
|
||||||
@@ -85,9 +84,12 @@ class CryptoHandler:
|
|||||||
private_key_local = AES_key.encrypt(pad(private_key.exportKey(), 32))
|
private_key_local = AES_key.encrypt(pad(private_key.exportKey(), 32))
|
||||||
(Config.app_path / "data/key/private_key.bin").write_bytes(private_key_local)
|
(Config.app_path / "data/key/private_key.bin").write_bytes(private_key_local)
|
||||||
|
|
||||||
def AUTO_encryptor(self, note: str) -> bytes:
|
def AUTO_encryptor(self, note: str) -> str:
|
||||||
"""使用AUTO_MAA的算法加密数据"""
|
"""使用AUTO_MAA的算法加密数据"""
|
||||||
|
|
||||||
|
if note == "":
|
||||||
|
return ""
|
||||||
|
|
||||||
# 读取RSA公钥
|
# 读取RSA公钥
|
||||||
public_key_local = RSA.import_key(
|
public_key_local = RSA.import_key(
|
||||||
(Config.app_path / "data/key/public_key.pem").read_bytes()
|
(Config.app_path / "data/key/public_key.pem").read_bytes()
|
||||||
@@ -95,11 +97,14 @@ class CryptoHandler:
|
|||||||
# 使用RSA公钥对数据进行加密
|
# 使用RSA公钥对数据进行加密
|
||||||
cipher = PKCS1_OAEP.new(public_key_local)
|
cipher = PKCS1_OAEP.new(public_key_local)
|
||||||
encrypted = cipher.encrypt(note.encode("utf-8"))
|
encrypted = cipher.encrypt(note.encode("utf-8"))
|
||||||
return encrypted
|
return base64.b64encode(encrypted).decode("utf-8")
|
||||||
|
|
||||||
def AUTO_decryptor(self, note: bytes, PASSWORD: str) -> str:
|
def AUTO_decryptor(self, note: str, PASSWORD: str) -> str:
|
||||||
"""使用AUTO_MAA的算法解密数据"""
|
"""使用AUTO_MAA的算法解密数据"""
|
||||||
|
|
||||||
|
if note == "":
|
||||||
|
return ""
|
||||||
|
|
||||||
# 读入RSA私钥密文、盐与校验哈希值
|
# 读入RSA私钥密文、盐与校验哈希值
|
||||||
private_key_local = (
|
private_key_local = (
|
||||||
(Config.app_path / "data/key/private_key.bin").read_bytes().strip()
|
(Config.app_path / "data/key/private_key.bin").read_bytes().strip()
|
||||||
@@ -133,63 +138,40 @@ class CryptoHandler:
|
|||||||
private_key = RSA.import_key(private_key_pem)
|
private_key = RSA.import_key(private_key_pem)
|
||||||
# 使用RSA私钥解密数据
|
# 使用RSA私钥解密数据
|
||||||
decrypter = PKCS1_OAEP.new(private_key)
|
decrypter = PKCS1_OAEP.new(private_key)
|
||||||
note = decrypter.decrypt(note)
|
note = decrypter.decrypt(base64.b64decode(note)).decode("utf-8")
|
||||||
return note.decode("utf-8")
|
return note
|
||||||
|
|
||||||
def change_PASSWORD(self, PASSWORD_old: str, PASSWORD_new: str) -> None:
|
def change_PASSWORD(self, PASSWORD_old: str, PASSWORD_new: str) -> None:
|
||||||
"""修改管理密钥"""
|
"""修改管理密钥"""
|
||||||
|
|
||||||
member_list = self.search_member()
|
for member in Config.member_dict.values():
|
||||||
|
|
||||||
for user_data in member_list:
|
|
||||||
|
|
||||||
# 读取用户数据
|
|
||||||
db = sqlite3.connect(user_data["Path"])
|
|
||||||
cur = db.cursor()
|
|
||||||
cur.execute("SELECT * FROM adminx WHERE True")
|
|
||||||
data = cur.fetchall()
|
|
||||||
|
|
||||||
# 使用旧管理密钥解密
|
# 使用旧管理密钥解密
|
||||||
user_data["Password"] = []
|
for user in member["UserData"].values():
|
||||||
for i in range(len(data)):
|
user["Password"] = self.AUTO_decryptor(
|
||||||
user_data["Password"].append(
|
user["Config"].get(user["Config"].Info_Password), PASSWORD_old
|
||||||
self.AUTO_decryptor(data[i][12], PASSWORD_old)
|
|
||||||
)
|
)
|
||||||
cur.close()
|
|
||||||
db.close()
|
|
||||||
|
|
||||||
self.get_PASSWORD(PASSWORD_new)
|
self.get_PASSWORD(PASSWORD_new)
|
||||||
|
|
||||||
for user_data in member_list:
|
for member in Config.member_dict.values():
|
||||||
|
|
||||||
# 读取用户数据
|
|
||||||
db = sqlite3.connect(user_data["Path"])
|
|
||||||
cur = db.cursor()
|
|
||||||
cur.execute("SELECT * FROM adminx WHERE True")
|
|
||||||
data = cur.fetchall()
|
|
||||||
|
|
||||||
# 使用新管理密钥重新加密
|
# 使用新管理密钥重新加密
|
||||||
for i in range(len(data)):
|
for user in member["UserData"].values():
|
||||||
cur.execute(
|
user["Config"].set(
|
||||||
"UPDATE adminx SET password = ? WHERE mode = ? AND uid = ?",
|
user["Config"].Info_Password, self.AUTO_encryptor(user["Password"])
|
||||||
(
|
|
||||||
self.AUTO_encryptor(user_data["Password"][i]),
|
|
||||||
data[i][15],
|
|
||||||
data[i][16],
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
db.commit()
|
user["Password"] = None
|
||||||
user_data["Password"][i] = None
|
del user["Password"]
|
||||||
del user_data["Password"]
|
|
||||||
|
|
||||||
cur.close()
|
|
||||||
db.close()
|
|
||||||
|
|
||||||
def win_encryptor(
|
def win_encryptor(
|
||||||
self, note: str, description: str = None, entropy: bytes = None
|
self, note: str, description: str = None, entropy: bytes = None
|
||||||
) -> str:
|
) -> str:
|
||||||
"""使用Windows DPAPI加密数据"""
|
"""使用Windows DPAPI加密数据"""
|
||||||
|
|
||||||
|
if note == "":
|
||||||
|
return ""
|
||||||
|
|
||||||
encrypted = win32crypt.CryptProtectData(
|
encrypted = win32crypt.CryptProtectData(
|
||||||
note.encode("utf-8"), description, entropy, None, None, 0
|
note.encode("utf-8"), description, entropy, None, None, 0
|
||||||
)
|
)
|
||||||
@@ -223,7 +205,7 @@ class CryptoHandler:
|
|||||||
"""验证管理密钥"""
|
"""验证管理密钥"""
|
||||||
|
|
||||||
return bool(
|
return bool(
|
||||||
self.AUTO_decryptor(self.AUTO_encryptor(""), PASSWORD) != "管理密钥错误"
|
self.AUTO_decryptor(self.AUTO_encryptor("-"), PASSWORD) != "管理密钥错误"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
241
app/services/skland.py
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
|
# This file incorporates work covered by the following copyright and
|
||||||
|
# permission notice:
|
||||||
|
#
|
||||||
|
# skland-checkin-ghaction Copyright © 2023 Yanstory
|
||||||
|
# https://github.com/Yanstory/skland-checkin-ghaction
|
||||||
|
|
||||||
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
|
# AUTO_MAA is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published
|
||||||
|
# by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
|
||||||
|
# AUTO_MAA is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
|
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||||||
|
# the GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
|
|
||||||
|
"""
|
||||||
|
AUTO_MAA
|
||||||
|
AUTO_MAA森空岛服务
|
||||||
|
v4.3
|
||||||
|
作者:DLmaster_361、ClozyA
|
||||||
|
"""
|
||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
import time
|
||||||
|
import json
|
||||||
|
import hmac
|
||||||
|
import hashlib
|
||||||
|
import requests
|
||||||
|
from urllib import parse
|
||||||
|
|
||||||
|
|
||||||
|
def skland_sign_in(token) -> dict:
|
||||||
|
"""森空岛签到"""
|
||||||
|
|
||||||
|
app_code = "4ca99fa6b56cc2ba"
|
||||||
|
# 用于获取grant code
|
||||||
|
grant_code_url = "https://as.hypergryph.com/user/oauth2/v2/grant"
|
||||||
|
# 用于获取cred
|
||||||
|
cred_code_url = "https://zonai.skland.com/api/v1/user/auth/generate_cred_by_code"
|
||||||
|
# 查询角色绑定
|
||||||
|
binding_url = "https://zonai.skland.com/api/v1/game/player/binding"
|
||||||
|
# 签到接口
|
||||||
|
sign_url = "https://zonai.skland.com/api/v1/game/attendance"
|
||||||
|
|
||||||
|
# 基础请求头
|
||||||
|
header = {
|
||||||
|
"cred": "",
|
||||||
|
"User-Agent": "Skland/1.5.1 (com.hypergryph.skland; build:100501001; Android 34;) Okhttp/4.11.0",
|
||||||
|
"Accept-Encoding": "gzip",
|
||||||
|
"Connection": "close",
|
||||||
|
}
|
||||||
|
header_login = header.copy()
|
||||||
|
header_for_sign = {
|
||||||
|
"platform": "1",
|
||||||
|
"timestamp": "",
|
||||||
|
"dId": "",
|
||||||
|
"vName": "1.5.1",
|
||||||
|
}
|
||||||
|
|
||||||
|
# 生成签名
|
||||||
|
def generate_signature(token_for_sign: str, path, body_or_query):
|
||||||
|
"""
|
||||||
|
生成请求签名
|
||||||
|
:param token_for_sign: 用于加密的token
|
||||||
|
:param path: 请求路径(如 /api/v1/game/player/binding)
|
||||||
|
:param body_or_query: GET用query字符串,POST用body字符串
|
||||||
|
:return: (sign, 新的header_for_sign字典)
|
||||||
|
"""
|
||||||
|
t = str(int(time.time()) - 2) # 时间戳,-2秒以防服务器时间不一致
|
||||||
|
token_bytes = token_for_sign.encode("utf-8")
|
||||||
|
header_ca = dict(header_for_sign)
|
||||||
|
header_ca["timestamp"] = t
|
||||||
|
header_ca_str = json.dumps(header_ca, separators=(",", ":"))
|
||||||
|
s = path + body_or_query + t + header_ca_str # 拼接原始字符串
|
||||||
|
# HMAC-SHA256 + MD5得到最终sign
|
||||||
|
hex_s = hmac.new(token_bytes, s.encode("utf-8"), hashlib.sha256).hexdigest()
|
||||||
|
md5 = hashlib.md5(hex_s.encode("utf-8")).hexdigest()
|
||||||
|
return md5, header_ca
|
||||||
|
|
||||||
|
# 获取带签名的header
|
||||||
|
def get_sign_header(url: str, method, body, old_header, sign_token):
|
||||||
|
"""
|
||||||
|
获取带签名的请求头
|
||||||
|
:param url: 请求完整url
|
||||||
|
:param method: 请求方式 GET/POST
|
||||||
|
:param body: POST请求体或GET时为None
|
||||||
|
:param old_header: 原始请求头
|
||||||
|
:param sign_token: 当前会话的签名token
|
||||||
|
:return: 新请求头
|
||||||
|
"""
|
||||||
|
h = json.loads(json.dumps(old_header))
|
||||||
|
p = parse.urlparse(url)
|
||||||
|
if method.lower() == "get":
|
||||||
|
sign, header_ca = generate_signature(sign_token, p.path, p.query)
|
||||||
|
else:
|
||||||
|
sign, header_ca = generate_signature(
|
||||||
|
sign_token, p.path, json.dumps(body) if body else ""
|
||||||
|
)
|
||||||
|
h["sign"] = sign
|
||||||
|
for i in header_ca:
|
||||||
|
h[i] = header_ca[i]
|
||||||
|
return h
|
||||||
|
|
||||||
|
# 复制请求头并添加cred
|
||||||
|
def copy_header(cred):
|
||||||
|
v = json.loads(json.dumps(header))
|
||||||
|
v["cred"] = cred
|
||||||
|
return v
|
||||||
|
|
||||||
|
# 使用token一步步拿到cred和sign_token
|
||||||
|
def login_by_token(token_code):
|
||||||
|
"""
|
||||||
|
:param token_code: 你的skyland token
|
||||||
|
:return: (cred, sign_token)
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# token为json对象时提取data.content
|
||||||
|
t = json.loads(token_code)
|
||||||
|
token_code = t["data"]["content"]
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
grant_code = get_grant_code(token_code)
|
||||||
|
return get_cred(grant_code)
|
||||||
|
|
||||||
|
# 通过grant code换cred和sign_token
|
||||||
|
def get_cred(grant):
|
||||||
|
rsp = requests.post(
|
||||||
|
cred_code_url, json={"code": grant, "kind": 1}, headers=header_login
|
||||||
|
).json()
|
||||||
|
if rsp["code"] != 0:
|
||||||
|
raise Exception(f'获得cred失败:{rsp.get("messgae")}')
|
||||||
|
sign_token = rsp["data"]["token"]
|
||||||
|
cred = rsp["data"]["cred"]
|
||||||
|
return cred, sign_token
|
||||||
|
|
||||||
|
# 通过token换grant code
|
||||||
|
def get_grant_code(token):
|
||||||
|
rsp = requests.post(
|
||||||
|
grant_code_url,
|
||||||
|
json={"appCode": app_code, "token": token, "type": 0},
|
||||||
|
headers=header_login,
|
||||||
|
).json()
|
||||||
|
if rsp["status"] != 0:
|
||||||
|
raise Exception(
|
||||||
|
f'使用token: {token[:3]}******{token[-3:]} 获得认证代码失败:{rsp.get("msg")}'
|
||||||
|
)
|
||||||
|
return rsp["data"]["code"]
|
||||||
|
|
||||||
|
# 获取已绑定的角色列表
|
||||||
|
def get_binding_list(cred, sign_token):
|
||||||
|
"""
|
||||||
|
查询绑定的角色
|
||||||
|
:param cred: 当前cred
|
||||||
|
:param sign_token: 当前sign_token
|
||||||
|
:return: 角色列表
|
||||||
|
"""
|
||||||
|
v = []
|
||||||
|
rsp = requests.get(
|
||||||
|
binding_url,
|
||||||
|
headers=get_sign_header(
|
||||||
|
binding_url, "get", None, copy_header(cred), sign_token
|
||||||
|
),
|
||||||
|
).json()
|
||||||
|
if rsp["code"] != 0:
|
||||||
|
logger.error(f"森空岛服务 | 请求角色列表出现问题:{rsp['message']}")
|
||||||
|
if rsp.get("message") == "用户未登录":
|
||||||
|
logger.error(f"森空岛服务 | 用户登录可能失效了,请重新登录!")
|
||||||
|
return v
|
||||||
|
# 只取明日方舟(arknights)的绑定账号
|
||||||
|
for i in rsp["data"]["list"]:
|
||||||
|
if i.get("appCode") != "arknights":
|
||||||
|
continue
|
||||||
|
v.extend(i.get("bindingList"))
|
||||||
|
return v
|
||||||
|
|
||||||
|
# 执行签到
|
||||||
|
def do_sign(cred, sign_token) -> dict:
|
||||||
|
"""
|
||||||
|
对所有绑定的角色进行签到
|
||||||
|
:param cred: 当前cred
|
||||||
|
:param sign_token: 当前sign_token
|
||||||
|
:return: 签到结果字典
|
||||||
|
"""
|
||||||
|
|
||||||
|
characters = get_binding_list(cred, sign_token)
|
||||||
|
result = {"成功": [], "重复": [], "失败": [], "总计": len(characters)}
|
||||||
|
|
||||||
|
for character in characters:
|
||||||
|
|
||||||
|
body = {
|
||||||
|
"uid": character.get("uid"),
|
||||||
|
"gameId": character.get("channelMasterId"),
|
||||||
|
}
|
||||||
|
rsp = requests.post(
|
||||||
|
sign_url,
|
||||||
|
headers=get_sign_header(
|
||||||
|
sign_url, "post", body, copy_header(cred), sign_token
|
||||||
|
),
|
||||||
|
json=body,
|
||||||
|
).json()
|
||||||
|
|
||||||
|
if rsp["code"] != 0:
|
||||||
|
|
||||||
|
result[
|
||||||
|
"重复" if rsp.get("message") == "请勿重复签到!" else "失败"
|
||||||
|
].append(
|
||||||
|
f"{character.get("nickName")}({character.get("channelName")})"
|
||||||
|
)
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
result["成功"].append(
|
||||||
|
f"{character.get("nickName")}({character.get("channelName")})"
|
||||||
|
)
|
||||||
|
|
||||||
|
time.sleep(3)
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
# 主流程
|
||||||
|
try:
|
||||||
|
# 拿到cred和sign_token
|
||||||
|
cred, sign_token = login_by_token(token)
|
||||||
|
time.sleep(1)
|
||||||
|
# 依次签到
|
||||||
|
return do_sign(cred, sign_token)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"森空岛服务 | 森空岛签到失败: {e}")
|
||||||
|
return {"成功": [], "重复": [], "失败": [], "总计": 0}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,17 +16,17 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA系统服务
|
AUTO_MAA系统服务
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtWidgets import QWidget
|
from PySide6.QtWidgets import QApplication
|
||||||
import sys
|
import sys
|
||||||
import ctypes
|
import ctypes
|
||||||
import win32gui
|
import win32gui
|
||||||
@@ -44,9 +44,7 @@ class _SystemHandler:
|
|||||||
ES_CONTINUOUS = 0x80000000
|
ES_CONTINUOUS = 0x80000000
|
||||||
ES_SYSTEM_REQUIRED = 0x00000001
|
ES_SYSTEM_REQUIRED = 0x00000001
|
||||||
|
|
||||||
def __init__(self, main_window: QWidget = None):
|
def __init__(self):
|
||||||
|
|
||||||
self.main_window = main_window
|
|
||||||
|
|
||||||
self.set_Sleep()
|
self.set_Sleep()
|
||||||
self.set_SelfStart()
|
self.set_SelfStart()
|
||||||
@@ -54,7 +52,7 @@ class _SystemHandler:
|
|||||||
def set_Sleep(self) -> None:
|
def set_Sleep(self) -> None:
|
||||||
"""同步系统休眠状态"""
|
"""同步系统休眠状态"""
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.function_IfAllowSleep):
|
if Config.get(Config.function_IfAllowSleep):
|
||||||
# 设置系统电源状态
|
# 设置系统电源状态
|
||||||
ctypes.windll.kernel32.SetThreadExecutionState(
|
ctypes.windll.kernel32.SetThreadExecutionState(
|
||||||
self.ES_CONTINUOUS | self.ES_SYSTEM_REQUIRED
|
self.ES_CONTINUOUS | self.ES_SYSTEM_REQUIRED
|
||||||
@@ -66,10 +64,7 @@ class _SystemHandler:
|
|||||||
def set_SelfStart(self) -> None:
|
def set_SelfStart(self) -> None:
|
||||||
"""同步开机自启"""
|
"""同步开机自启"""
|
||||||
|
|
||||||
if (
|
if Config.get(Config.start_IfSelfStart) and not self.is_startup():
|
||||||
Config.global_config.get(Config.global_config.start_IfSelfStart)
|
|
||||||
and not self.is_startup()
|
|
||||||
):
|
|
||||||
key = winreg.OpenKey(
|
key = winreg.OpenKey(
|
||||||
winreg.HKEY_CURRENT_USER,
|
winreg.HKEY_CURRENT_USER,
|
||||||
r"Software\Microsoft\Windows\CurrentVersion\Run",
|
r"Software\Microsoft\Windows\CurrentVersion\Run",
|
||||||
@@ -78,10 +73,7 @@ class _SystemHandler:
|
|||||||
)
|
)
|
||||||
winreg.SetValueEx(key, "AUTO_MAA", 0, winreg.REG_SZ, Config.app_path_sys)
|
winreg.SetValueEx(key, "AUTO_MAA", 0, winreg.REG_SZ, Config.app_path_sys)
|
||||||
winreg.CloseKey(key)
|
winreg.CloseKey(key)
|
||||||
elif (
|
elif not Config.get(Config.start_IfSelfStart) and self.is_startup():
|
||||||
not Config.global_config.get(Config.global_config.start_IfSelfStart)
|
|
||||||
and self.is_startup()
|
|
||||||
):
|
|
||||||
key = winreg.OpenKey(
|
key = winreg.OpenKey(
|
||||||
winreg.HKEY_CURRENT_USER,
|
winreg.HKEY_CURRENT_USER,
|
||||||
r"Software\Microsoft\Windows\CurrentVersion\Run",
|
r"Software\Microsoft\Windows\CurrentVersion\Run",
|
||||||
@@ -95,7 +87,7 @@ class _SystemHandler:
|
|||||||
|
|
||||||
if sys.platform.startswith("win"):
|
if sys.platform.startswith("win"):
|
||||||
|
|
||||||
if mode == "None":
|
if mode == "NoAction":
|
||||||
|
|
||||||
logger.info("不执行系统电源操作")
|
logger.info("不执行系统电源操作")
|
||||||
|
|
||||||
@@ -118,11 +110,13 @@ class _SystemHandler:
|
|||||||
|
|
||||||
elif mode == "KillSelf":
|
elif mode == "KillSelf":
|
||||||
|
|
||||||
self.main_window.close()
|
Config.main_window.close()
|
||||||
|
QApplication.quit()
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
elif sys.platform.startswith("linux"):
|
elif sys.platform.startswith("linux"):
|
||||||
|
|
||||||
if mode == "None":
|
if mode == "NoAction":
|
||||||
|
|
||||||
logger.info("不执行系统电源操作")
|
logger.info("不执行系统电源操作")
|
||||||
|
|
||||||
@@ -143,7 +137,9 @@ class _SystemHandler:
|
|||||||
|
|
||||||
elif mode == "KillSelf":
|
elif mode == "KillSelf":
|
||||||
|
|
||||||
self.main_window.close()
|
Config.main_window.close()
|
||||||
|
QApplication.quit()
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
def is_startup(self) -> bool:
|
def is_startup(self) -> bool:
|
||||||
"""判断程序是否已经开机自启"""
|
"""判断程序是否已经开机自启"""
|
||||||
|
|||||||
1631
app/ui/Widget.py
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA图形化界面包
|
AUTO_MAA图形化界面包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -30,5 +30,6 @@ __author__ = "DLmaster361 <DLmaster_361@163.com>"
|
|||||||
__license__ = "GPL-3.0 license"
|
__license__ = "GPL-3.0 license"
|
||||||
|
|
||||||
from .main_window import AUTO_MAA
|
from .main_window import AUTO_MAA
|
||||||
|
from .Widget import ProgressRingMessageBox
|
||||||
|
|
||||||
__all__ = ["AUTO_MAA"]
|
__all__ = ["AUTO_MAA", "ProgressRingMessageBox"]
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA调度中枢界面
|
AUTO_MAA调度中枢界面
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -33,8 +33,8 @@ from PySide6.QtWidgets import (
|
|||||||
QHBoxLayout,
|
QHBoxLayout,
|
||||||
)
|
)
|
||||||
from qfluentwidgets import (
|
from qfluentwidgets import (
|
||||||
|
BodyLabel,
|
||||||
CardWidget,
|
CardWidget,
|
||||||
Pivot,
|
|
||||||
ScrollArea,
|
ScrollArea,
|
||||||
FluentIcon,
|
FluentIcon,
|
||||||
HeaderCardWidget,
|
HeaderCardWidget,
|
||||||
@@ -44,14 +44,12 @@ from qfluentwidgets import (
|
|||||||
SubtitleLabel,
|
SubtitleLabel,
|
||||||
PushButton,
|
PushButton,
|
||||||
)
|
)
|
||||||
from PySide6.QtCore import Qt
|
|
||||||
from PySide6.QtGui import QTextCursor
|
from PySide6.QtGui import QTextCursor
|
||||||
from typing import List, Dict
|
from typing import List, Dict
|
||||||
import json
|
|
||||||
|
|
||||||
|
|
||||||
from app.core import Config, TaskManager, Task, MainInfoBar
|
from app.core import Config, TaskManager, Task, MainInfoBar, SoundPlayer
|
||||||
from .Widget import StatefulItemCard
|
from .Widget import StatefulItemCard, ComboBoxMessageBox, PivotArea
|
||||||
|
|
||||||
|
|
||||||
class DispatchCenter(QWidget):
|
class DispatchCenter(QWidget):
|
||||||
@@ -61,13 +59,29 @@ class DispatchCenter(QWidget):
|
|||||||
|
|
||||||
self.setObjectName("调度中枢")
|
self.setObjectName("调度中枢")
|
||||||
|
|
||||||
self.pivot = Pivot(self)
|
self.multi_button = PushButton(FluentIcon.ADD, "添加任务", self)
|
||||||
|
self.multi_button.setToolTip("添加任务")
|
||||||
|
self.multi_button.clicked.connect(self.start_multi_task)
|
||||||
|
|
||||||
|
self.power_combox = ComboBox()
|
||||||
|
self.power_combox.addItem("无动作", userData="NoAction")
|
||||||
|
self.power_combox.addItem("退出软件", userData="KillSelf")
|
||||||
|
self.power_combox.addItem("睡眠", userData="Sleep")
|
||||||
|
self.power_combox.addItem("休眠", userData="Hibernate")
|
||||||
|
self.power_combox.addItem("关机", userData="Shutdown")
|
||||||
|
self.power_combox.setCurrentText("无动作")
|
||||||
|
self.power_combox.currentIndexChanged.connect(self.set_power_sign)
|
||||||
|
|
||||||
|
self.pivotArea = PivotArea(self)
|
||||||
|
self.pivot = self.pivotArea.pivot
|
||||||
|
|
||||||
self.stackedWidget = QStackedWidget(self)
|
self.stackedWidget = QStackedWidget(self)
|
||||||
self.Layout = QVBoxLayout(self)
|
self.stackedWidget.setContentsMargins(0, 0, 0, 0)
|
||||||
|
self.stackedWidget.setStyleSheet("background: transparent; border: none;")
|
||||||
|
|
||||||
self.script_list: Dict[str, DispatchBox] = {}
|
self.script_list: Dict[str, DispatchCenter.DispatchBox] = {}
|
||||||
|
|
||||||
dispatch_box = DispatchBox("主调度台", self)
|
dispatch_box = self.DispatchBox("主调度台", self)
|
||||||
self.script_list["主调度台"] = dispatch_box
|
self.script_list["主调度台"] = dispatch_box
|
||||||
self.stackedWidget.addWidget(self.script_list["主调度台"])
|
self.stackedWidget.addWidget(self.script_list["主调度台"])
|
||||||
self.pivot.addItem(
|
self.pivot.addItem(
|
||||||
@@ -77,7 +91,15 @@ class DispatchCenter(QWidget):
|
|||||||
icon=FluentIcon.CAFE,
|
icon=FluentIcon.CAFE,
|
||||||
)
|
)
|
||||||
|
|
||||||
self.Layout.addWidget(self.pivot, 0, Qt.AlignHCenter)
|
h_layout = QHBoxLayout()
|
||||||
|
h_layout.addWidget(self.multi_button)
|
||||||
|
h_layout.addWidget(self.pivotArea)
|
||||||
|
h_layout.addWidget(BodyLabel("全部完成后", self))
|
||||||
|
h_layout.addWidget(self.power_combox)
|
||||||
|
h_layout.setContentsMargins(11, 5, 11, 0)
|
||||||
|
|
||||||
|
self.Layout = QVBoxLayout(self)
|
||||||
|
self.Layout.addLayout(h_layout)
|
||||||
self.Layout.addWidget(self.stackedWidget)
|
self.Layout.addWidget(self.stackedWidget)
|
||||||
self.Layout.setContentsMargins(0, 0, 0, 0)
|
self.Layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
|
||||||
@@ -88,9 +110,9 @@ class DispatchCenter(QWidget):
|
|||||||
def add_board(self, task: Task) -> None:
|
def add_board(self, task: Task) -> None:
|
||||||
"""添加一个调度台界面"""
|
"""添加一个调度台界面"""
|
||||||
|
|
||||||
dispatch_box = DispatchBox(task.name, self)
|
dispatch_box = self.DispatchBox(task.name, self)
|
||||||
|
|
||||||
dispatch_box.top_bar.button.clicked.connect(
|
dispatch_box.top_bar.main_button.clicked.connect(
|
||||||
lambda: TaskManager.stop_task(task.name)
|
lambda: TaskManager.stop_task(task.name)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -124,9 +146,9 @@ class DispatchCenter(QWidget):
|
|||||||
self.script_list["主调度台"].top_bar.Lable.show()
|
self.script_list["主调度台"].top_bar.Lable.show()
|
||||||
self.script_list["主调度台"].top_bar.object.hide()
|
self.script_list["主调度台"].top_bar.object.hide()
|
||||||
self.script_list["主调度台"].top_bar.mode.hide()
|
self.script_list["主调度台"].top_bar.mode.hide()
|
||||||
self.script_list["主调度台"].top_bar.button.clicked.disconnect()
|
self.script_list["主调度台"].top_bar.main_button.clicked.disconnect()
|
||||||
self.script_list["主调度台"].top_bar.button.setText("中止任务")
|
self.script_list["主调度台"].top_bar.main_button.setText("中止任务")
|
||||||
self.script_list["主调度台"].top_bar.button.clicked.connect(
|
self.script_list["主调度台"].top_bar.main_button.clicked.connect(
|
||||||
lambda: TaskManager.stop_task(task.name)
|
lambda: TaskManager.stop_task(task.name)
|
||||||
)
|
)
|
||||||
task.create_task_list.connect(
|
task.create_task_list.connect(
|
||||||
@@ -144,277 +166,386 @@ class DispatchCenter(QWidget):
|
|||||||
task.update_log_text.connect(
|
task.update_log_text.connect(
|
||||||
self.script_list["主调度台"].info.log_text.text.setText
|
self.script_list["主调度台"].info.log_text.text.setText
|
||||||
)
|
)
|
||||||
task.accomplish.connect(lambda: self.disconnect_main_board(task.name))
|
task.accomplish.connect(
|
||||||
|
lambda logs: self.disconnect_main_board(task.name, logs)
|
||||||
|
)
|
||||||
|
|
||||||
def disconnect_main_board(self, name: str) -> None:
|
def disconnect_main_board(self, name: str, logs: list) -> None:
|
||||||
"""断开主调度台"""
|
"""断开主调度台"""
|
||||||
|
|
||||||
self.script_list["主调度台"].top_bar.Lable.hide()
|
self.script_list["主调度台"].top_bar.Lable.hide()
|
||||||
self.script_list["主调度台"].top_bar.object.show()
|
self.script_list["主调度台"].top_bar.object.show()
|
||||||
self.script_list["主调度台"].top_bar.mode.show()
|
self.script_list["主调度台"].top_bar.mode.show()
|
||||||
self.script_list["主调度台"].top_bar.button.clicked.disconnect()
|
self.script_list["主调度台"].top_bar.main_button.clicked.disconnect()
|
||||||
self.script_list["主调度台"].top_bar.button.setText("开始任务")
|
self.script_list["主调度台"].top_bar.main_button.setText("开始任务")
|
||||||
self.script_list["主调度台"].top_bar.button.clicked.connect(
|
self.script_list["主调度台"].top_bar.main_button.clicked.connect(
|
||||||
self.script_list["主调度台"].top_bar.start_task
|
self.script_list["主调度台"].top_bar.start_main_task
|
||||||
)
|
|
||||||
self.script_list["主调度台"].info.log_text.text.setText(
|
|
||||||
Config.get_history(name)["History"]
|
|
||||||
)
|
)
|
||||||
|
if len(logs) > 0:
|
||||||
|
history = ""
|
||||||
|
for log in logs:
|
||||||
|
history += (
|
||||||
|
f"任务名称:{log[0]},{log[1]["History"].replace("\n","\n ")}\n"
|
||||||
|
)
|
||||||
|
self.script_list["主调度台"].info.log_text.text.setText(history)
|
||||||
|
else:
|
||||||
|
self.script_list["主调度台"].info.log_text.text.setText("没有任务被执行")
|
||||||
|
|
||||||
def update_top_bar(self):
|
def update_top_bar(self):
|
||||||
"""更新顶栏"""
|
"""更新顶栏"""
|
||||||
|
|
||||||
list = []
|
|
||||||
queue_numb, member_numb = 0, 0
|
|
||||||
|
|
||||||
if (Config.app_path / "config/QueueConfig").exists():
|
|
||||||
for json_file in (Config.app_path / "config/QueueConfig").glob("*.json"):
|
|
||||||
list.append(f"队列 - {json_file.stem}")
|
|
||||||
queue_numb += 1
|
|
||||||
|
|
||||||
if (Config.app_path / "config/MaaConfig").exists():
|
|
||||||
for subdir in (Config.app_path / "config/MaaConfig").iterdir():
|
|
||||||
if subdir.is_dir():
|
|
||||||
list.append(f"实例 - Maa - {subdir.name}")
|
|
||||||
member_numb += 1
|
|
||||||
|
|
||||||
self.script_list["主调度台"].top_bar.object.clear()
|
self.script_list["主调度台"].top_bar.object.clear()
|
||||||
self.script_list["主调度台"].top_bar.object.addItems(list)
|
|
||||||
self.script_list["主调度台"].top_bar.mode.clear()
|
|
||||||
self.script_list["主调度台"].top_bar.mode.addItems(["自动代理", "人工排查"])
|
|
||||||
|
|
||||||
if queue_numb == 1:
|
for name, info in Config.queue_dict.items():
|
||||||
|
self.script_list["主调度台"].top_bar.object.addItem(
|
||||||
|
(
|
||||||
|
"队列"
|
||||||
|
if info["Config"].get(info["Config"].queueSet_Name) == ""
|
||||||
|
else f"队列 - {info["Config"].get(info["Config"].queueSet_Name)}"
|
||||||
|
),
|
||||||
|
userData=name,
|
||||||
|
)
|
||||||
|
|
||||||
|
for name, info in Config.member_dict.items():
|
||||||
|
self.script_list["主调度台"].top_bar.object.addItem(
|
||||||
|
(
|
||||||
|
f"实例 - {info['Type']}"
|
||||||
|
if info["Config"].get(info["Config"].MaaSet_Name) == ""
|
||||||
|
else f"实例 - {info['Type']} - {info["Config"].get(info["Config"].MaaSet_Name)}"
|
||||||
|
),
|
||||||
|
userData=name,
|
||||||
|
)
|
||||||
|
|
||||||
|
if len(Config.queue_dict) == 1:
|
||||||
self.script_list["主调度台"].top_bar.object.setCurrentIndex(0)
|
self.script_list["主调度台"].top_bar.object.setCurrentIndex(0)
|
||||||
elif member_numb == 1:
|
elif len(Config.member_dict) == 1:
|
||||||
self.script_list["主调度台"].top_bar.object.setCurrentIndex(queue_numb)
|
self.script_list["主调度台"].top_bar.object.setCurrentIndex(
|
||||||
|
len(Config.queue_dict)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
self.script_list["主调度台"].top_bar.object.setCurrentIndex(-1)
|
self.script_list["主调度台"].top_bar.object.setCurrentIndex(-1)
|
||||||
|
|
||||||
|
self.script_list["主调度台"].top_bar.mode.clear()
|
||||||
|
self.script_list["主调度台"].top_bar.mode.addItems(["自动代理", "人工排查"])
|
||||||
self.script_list["主调度台"].top_bar.mode.setCurrentIndex(0)
|
self.script_list["主调度台"].top_bar.mode.setCurrentIndex(0)
|
||||||
|
|
||||||
|
def update_power_sign(self) -> None:
|
||||||
|
"""更新电源设置"""
|
||||||
|
|
||||||
class DispatchBox(QWidget):
|
mode_book = {
|
||||||
|
"NoAction": "无动作",
|
||||||
|
"KillSelf": "退出软件",
|
||||||
|
"Sleep": "睡眠",
|
||||||
|
"Hibernate": "休眠",
|
||||||
|
"Shutdown": "关机",
|
||||||
|
}
|
||||||
|
self.power_combox.currentIndexChanged.disconnect()
|
||||||
|
self.power_combox.setCurrentText(mode_book[Config.power_sign])
|
||||||
|
self.power_combox.currentIndexChanged.connect(self.set_power_sign)
|
||||||
|
|
||||||
def __init__(self, name: str, parent=None):
|
def set_power_sign(self) -> None:
|
||||||
super().__init__(parent)
|
"""设置所有任务完成后动作"""
|
||||||
|
|
||||||
self.setObjectName(name)
|
if not Config.running_list:
|
||||||
|
|
||||||
layout = QVBoxLayout()
|
self.power_combox.currentIndexChanged.disconnect()
|
||||||
|
self.power_combox.setCurrentText("无动作")
|
||||||
|
self.power_combox.currentIndexChanged.connect(self.set_power_sign)
|
||||||
|
logger.warning("没有正在运行的任务,无法设置任务完成后动作")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning",
|
||||||
|
"没有正在运行的任务",
|
||||||
|
"无法设置任务完成后动作",
|
||||||
|
5000,
|
||||||
|
)
|
||||||
|
|
||||||
scrollArea = ScrollArea()
|
else:
|
||||||
scrollArea.setWidgetResizable(True)
|
|
||||||
|
|
||||||
content_widget = QWidget()
|
Config.set_power_sign(self.power_combox.currentData())
|
||||||
content_layout = QVBoxLayout(content_widget)
|
|
||||||
|
|
||||||
self.top_bar = self.DispatchTopBar(self, name)
|
def start_multi_task(self) -> None:
|
||||||
self.info = self.DispatchInfoCard(self)
|
"""开始任务"""
|
||||||
|
|
||||||
content_layout.addWidget(self.top_bar)
|
|
||||||
content_layout.addWidget(self.info)
|
|
||||||
|
|
||||||
scrollArea.setWidget(content_widget)
|
|
||||||
|
|
||||||
layout.addWidget(scrollArea)
|
|
||||||
|
|
||||||
self.setLayout(layout)
|
|
||||||
|
|
||||||
class DispatchTopBar(CardWidget):
|
|
||||||
|
|
||||||
def __init__(self, parent=None, name: str = None):
|
|
||||||
super().__init__(parent)
|
|
||||||
|
|
||||||
Layout = QHBoxLayout(self)
|
|
||||||
|
|
||||||
if name == "主调度台":
|
|
||||||
|
|
||||||
self.Lable = SubtitleLabel("", self)
|
|
||||||
self.Lable.hide()
|
|
||||||
self.object = ComboBox()
|
|
||||||
self.object.setPlaceholderText("请选择调度对象")
|
|
||||||
self.mode = ComboBox()
|
|
||||||
self.mode.setPlaceholderText("请选择调度模式")
|
|
||||||
|
|
||||||
self.button = PushButton("开始任务")
|
|
||||||
self.button.clicked.connect(self.start_task)
|
|
||||||
|
|
||||||
Layout.addWidget(self.Lable)
|
|
||||||
Layout.addWidget(self.object)
|
|
||||||
Layout.addWidget(self.mode)
|
|
||||||
Layout.addStretch(1)
|
|
||||||
Layout.addWidget(self.button)
|
|
||||||
|
|
||||||
else:
|
|
||||||
|
|
||||||
self.Lable = SubtitleLabel(name, self)
|
|
||||||
self.button = PushButton("中止任务")
|
|
||||||
|
|
||||||
Layout.addWidget(self.Lable)
|
|
||||||
Layout.addStretch(1)
|
|
||||||
Layout.addWidget(self.button)
|
|
||||||
|
|
||||||
def start_task(self):
|
|
||||||
"""开始任务"""
|
|
||||||
|
|
||||||
if self.object.currentIndex() == -1:
|
|
||||||
logger.warning("未选择调度对象")
|
|
||||||
MainInfoBar.push_info_bar(
|
|
||||||
"warning", "未选择调度对象", "请选择后再开始任务", 5000
|
|
||||||
)
|
|
||||||
return None
|
|
||||||
|
|
||||||
if self.mode.currentIndex() == -1:
|
|
||||||
logger.warning("未选择调度模式")
|
|
||||||
MainInfoBar.push_info_bar(
|
|
||||||
"warning", "未选择调度模式", "请选择后再开始任务", 5000
|
|
||||||
)
|
|
||||||
return None
|
|
||||||
|
|
||||||
name = self.object.currentText().split(" - ")[-1]
|
|
||||||
|
|
||||||
|
# 获取所有可用的队列和实例
|
||||||
|
text_list = []
|
||||||
|
data_list = []
|
||||||
|
for name, info in Config.queue_dict.items():
|
||||||
if name in Config.running_list:
|
if name in Config.running_list:
|
||||||
logger.warning(f"任务已存在:{name}")
|
continue
|
||||||
MainInfoBar.push_info_bar("warning", "任务已存在", name, 5000)
|
text_list.append(
|
||||||
|
"队列"
|
||||||
|
if info["Config"].get(info["Config"].queueSet_Name) == ""
|
||||||
|
else f"队列 - {info["Config"].get(info["Config"].queueSet_Name)}"
|
||||||
|
)
|
||||||
|
data_list.append(name)
|
||||||
|
|
||||||
|
for name, info in Config.member_dict.items():
|
||||||
|
if name in Config.running_list:
|
||||||
|
continue
|
||||||
|
text_list.append(
|
||||||
|
f"实例 - {info['Type']}"
|
||||||
|
if info["Config"].get(info["Config"].MaaSet_Name) == ""
|
||||||
|
else f"实例 - {info['Type']} - {info["Config"].get(info["Config"].MaaSet_Name)}"
|
||||||
|
)
|
||||||
|
data_list.append(name)
|
||||||
|
|
||||||
|
choice = ComboBoxMessageBox(
|
||||||
|
self.window(),
|
||||||
|
"选择一个对象以添加相应多开任务",
|
||||||
|
["选择调度对象"],
|
||||||
|
[text_list],
|
||||||
|
[data_list],
|
||||||
|
)
|
||||||
|
|
||||||
|
if choice.exec() and choice.input[0].currentIndex() != -1:
|
||||||
|
|
||||||
|
if choice.input[0].currentData() in Config.running_list:
|
||||||
|
logger.warning(f"任务已存在:{choice.input[0].currentData()}")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "任务已存在", choice.input[0].currentData(), 5000
|
||||||
|
)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
if self.object.currentText().split(" - ")[0] == "队列":
|
if "调度队列" in choice.input[0].currentData():
|
||||||
|
|
||||||
with (Config.app_path / f"config/QueueConfig/{name}.json").open(
|
logger.info(f"用户添加任务:{choice.input[0].currentData()}")
|
||||||
mode="r", encoding="utf-8"
|
TaskManager.add_task(
|
||||||
) as f:
|
"自动代理_新调度台",
|
||||||
info = json.load(f)
|
choice.input[0].currentData(),
|
||||||
|
Config.queue_dict[choice.input[0].currentData()]["Config"].toDict(),
|
||||||
|
)
|
||||||
|
|
||||||
logger.info(f"用户添加任务:{name}")
|
elif "脚本" in choice.input[0].currentData():
|
||||||
TaskManager.add_task(f"{self.mode.currentText()}_主调度台", name, info)
|
|
||||||
|
|
||||||
elif self.object.currentText().split(" - ")[0] == "实例":
|
if Config.member_dict[choice.input[0].currentData()]["Type"] == "Maa":
|
||||||
|
|
||||||
if self.object.currentText().split(" - ")[1] == "Maa":
|
logger.info(f"用户添加任务:{choice.input[0].currentData()}")
|
||||||
|
|
||||||
info = {"Queue": {"Member_1": name}}
|
|
||||||
|
|
||||||
logger.info(f"用户添加任务:{name}")
|
|
||||||
TaskManager.add_task(
|
TaskManager.add_task(
|
||||||
f"{self.mode.currentText()}_主调度台", "自定义队列", info
|
"自动代理_新调度台",
|
||||||
|
f"自定义队列 - {choice.input[0].currentData()}",
|
||||||
|
{"Queue": {"Member_1": choice.input[0].currentData()}},
|
||||||
)
|
)
|
||||||
|
|
||||||
class DispatchInfoCard(HeaderCardWidget):
|
class DispatchBox(QWidget):
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, name: str, parent=None):
|
||||||
super().__init__(parent)
|
super().__init__(parent)
|
||||||
|
|
||||||
self.setTitle("调度信息")
|
self.setObjectName(name)
|
||||||
|
|
||||||
self.task = self.TaskInfoCard(self)
|
self.top_bar = self.DispatchTopBar(self, name)
|
||||||
self.user = self.UserInfoCard(self)
|
self.info = self.DispatchInfoCard(self)
|
||||||
self.log_text = self.LogCard(self)
|
|
||||||
|
|
||||||
self.viewLayout.addWidget(self.task)
|
content_widget = QWidget()
|
||||||
self.viewLayout.addWidget(self.user)
|
content_layout = QVBoxLayout(content_widget)
|
||||||
self.viewLayout.addWidget(self.log_text)
|
content_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
content_layout.addWidget(self.top_bar)
|
||||||
|
content_layout.addWidget(self.info)
|
||||||
|
|
||||||
self.viewLayout.setStretch(0, 1)
|
scrollArea = ScrollArea()
|
||||||
self.viewLayout.setStretch(1, 1)
|
scrollArea.setWidgetResizable(True)
|
||||||
self.viewLayout.setStretch(2, 5)
|
scrollArea.setContentsMargins(0, 0, 0, 0)
|
||||||
|
scrollArea.setStyleSheet("background: transparent; border: none;")
|
||||||
|
scrollArea.setWidget(content_widget)
|
||||||
|
|
||||||
def update_board(self, task_list: list, user_list: list, log: str):
|
layout = QVBoxLayout(self)
|
||||||
"""更新调度信息"""
|
layout.addWidget(scrollArea)
|
||||||
|
|
||||||
self.task.update_task(task_list)
|
class DispatchTopBar(CardWidget):
|
||||||
self.user.update_user(user_list)
|
|
||||||
self.log_text.text.setText(log)
|
|
||||||
|
|
||||||
class TaskInfoCard(HeaderCardWidget):
|
def __init__(self, parent=None, name: str = None):
|
||||||
|
super().__init__(parent)
|
||||||
|
|
||||||
|
Layout = QHBoxLayout(self)
|
||||||
|
|
||||||
|
if name == "主调度台":
|
||||||
|
|
||||||
|
self.Lable = SubtitleLabel("", self)
|
||||||
|
self.Lable.hide()
|
||||||
|
self.object = ComboBox()
|
||||||
|
self.object.setPlaceholderText("请选择调度对象")
|
||||||
|
self.mode = ComboBox()
|
||||||
|
self.mode.setPlaceholderText("请选择调度模式")
|
||||||
|
|
||||||
|
self.main_button = PushButton("开始任务")
|
||||||
|
self.main_button.clicked.connect(self.start_main_task)
|
||||||
|
|
||||||
|
Layout.addWidget(self.Lable)
|
||||||
|
Layout.addWidget(self.object)
|
||||||
|
Layout.addWidget(self.mode)
|
||||||
|
Layout.addStretch(1)
|
||||||
|
Layout.addWidget(self.main_button)
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
self.Lable = SubtitleLabel(name, self)
|
||||||
|
self.main_button = PushButton("中止任务")
|
||||||
|
|
||||||
|
Layout.addWidget(self.Lable)
|
||||||
|
Layout.addStretch(1)
|
||||||
|
Layout.addWidget(self.main_button)
|
||||||
|
|
||||||
|
def start_main_task(self):
|
||||||
|
"""开始任务"""
|
||||||
|
|
||||||
|
if self.object.currentIndex() == -1:
|
||||||
|
logger.warning("未选择调度对象")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "未选择调度对象", "请选择后再开始任务", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if self.mode.currentIndex() == -1:
|
||||||
|
logger.warning("未选择调度模式")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "未选择调度模式", "请选择后再开始任务", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if self.object.currentData() in Config.running_list:
|
||||||
|
logger.warning(f"任务已存在:{self.object.currentData()}")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "任务已存在", self.object.currentData(), 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if "调度队列" in self.object.currentData():
|
||||||
|
|
||||||
|
logger.info(f"用户添加任务:{self.object.currentData()}")
|
||||||
|
TaskManager.add_task(
|
||||||
|
f"{self.mode.currentText()}_主调度台",
|
||||||
|
self.object.currentData(),
|
||||||
|
Config.queue_dict[self.object.currentData()]["Config"].toDict(),
|
||||||
|
)
|
||||||
|
|
||||||
|
elif "脚本" in self.object.currentData():
|
||||||
|
|
||||||
|
if Config.member_dict[self.object.currentData()]["Type"] == "Maa":
|
||||||
|
|
||||||
|
logger.info(f"用户添加任务:{self.object.currentData()}")
|
||||||
|
TaskManager.add_task(
|
||||||
|
f"{self.mode.currentText()}_主调度台",
|
||||||
|
"自定义队列",
|
||||||
|
{"Queue": {"Member_1": self.object.currentData()}},
|
||||||
|
)
|
||||||
|
|
||||||
|
class DispatchInfoCard(HeaderCardWidget):
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super().__init__(parent)
|
super().__init__(parent)
|
||||||
self.setTitle("任务队列")
|
|
||||||
|
|
||||||
self.Layout = QVBoxLayout()
|
self.setTitle("调度信息")
|
||||||
self.viewLayout.addLayout(self.Layout)
|
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
|
||||||
|
|
||||||
self.task_cards: List[StatefulItemCard] = []
|
self.task = self.TaskInfoCard(self)
|
||||||
|
self.user = self.UserInfoCard(self)
|
||||||
|
self.log_text = self.LogCard(self)
|
||||||
|
|
||||||
def create_task(self, task_list: list):
|
self.viewLayout.addWidget(self.task)
|
||||||
"""创建任务队列"""
|
self.viewLayout.addWidget(self.user)
|
||||||
|
self.viewLayout.addWidget(self.log_text)
|
||||||
|
|
||||||
while self.Layout.count() > 0:
|
self.viewLayout.setStretch(0, 1)
|
||||||
item = self.Layout.takeAt(0)
|
self.viewLayout.setStretch(1, 1)
|
||||||
if item.spacerItem():
|
self.viewLayout.setStretch(2, 5)
|
||||||
self.Layout.removeItem(item.spacerItem())
|
|
||||||
elif item.widget():
|
|
||||||
item.widget().deleteLater()
|
|
||||||
|
|
||||||
self.task_cards = []
|
def update_board(self, task_list: list, user_list: list, log: str):
|
||||||
|
"""更新调度信息"""
|
||||||
|
|
||||||
for task in task_list:
|
self.task.update_task(task_list)
|
||||||
|
self.user.update_user(user_list)
|
||||||
|
self.log_text.text.setText(log)
|
||||||
|
|
||||||
self.task_cards.append(StatefulItemCard(task))
|
class TaskInfoCard(HeaderCardWidget):
|
||||||
self.Layout.addWidget(self.task_cards[-1])
|
|
||||||
|
|
||||||
self.Layout.addStretch(1)
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle("任务队列")
|
||||||
|
|
||||||
def update_task(self, task_list: list):
|
self.Layout = QVBoxLayout()
|
||||||
"""更新任务队列"""
|
self.viewLayout.addLayout(self.Layout)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
|
||||||
for i in range(len(task_list)):
|
self.task_cards: List[StatefulItemCard] = []
|
||||||
|
|
||||||
self.task_cards[i].update_status(task_list[i][1])
|
def create_task(self, task_list: list):
|
||||||
|
"""创建任务队列"""
|
||||||
|
|
||||||
class UserInfoCard(HeaderCardWidget):
|
while self.Layout.count() > 0:
|
||||||
|
item = self.Layout.takeAt(0)
|
||||||
|
if item.spacerItem():
|
||||||
|
self.Layout.removeItem(item.spacerItem())
|
||||||
|
elif item.widget():
|
||||||
|
item.widget().deleteLater()
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
self.task_cards = []
|
||||||
super().__init__(parent)
|
|
||||||
self.setTitle("用户队列")
|
|
||||||
|
|
||||||
self.Layout = QVBoxLayout()
|
for task in task_list:
|
||||||
self.viewLayout.addLayout(self.Layout)
|
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
|
||||||
|
|
||||||
self.user_cards: List[StatefulItemCard] = []
|
self.task_cards.append(StatefulItemCard(task))
|
||||||
|
self.Layout.addWidget(self.task_cards[-1])
|
||||||
|
|
||||||
def create_user(self, user_list: list):
|
self.Layout.addStretch(1)
|
||||||
"""创建用户队列"""
|
|
||||||
|
|
||||||
while self.Layout.count() > 0:
|
def update_task(self, task_list: list):
|
||||||
item = self.Layout.takeAt(0)
|
"""更新任务队列"""
|
||||||
if item.spacerItem():
|
|
||||||
self.Layout.removeItem(item.spacerItem())
|
|
||||||
elif item.widget():
|
|
||||||
item.widget().deleteLater()
|
|
||||||
|
|
||||||
self.user_cards = []
|
for i in range(len(task_list)):
|
||||||
|
|
||||||
for user in user_list:
|
self.task_cards[i].update_status(task_list[i][1])
|
||||||
|
|
||||||
self.user_cards.append(StatefulItemCard(user))
|
class UserInfoCard(HeaderCardWidget):
|
||||||
self.Layout.addWidget(self.user_cards[-1])
|
|
||||||
|
|
||||||
self.Layout.addStretch(1)
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle("用户队列")
|
||||||
|
|
||||||
def update_user(self, user_list: list):
|
self.Layout = QVBoxLayout()
|
||||||
"""更新用户队列"""
|
self.viewLayout.addLayout(self.Layout)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
|
||||||
for i in range(len(user_list)):
|
self.user_cards: List[StatefulItemCard] = []
|
||||||
|
|
||||||
self.user_cards[i].Label.setText(user_list[i][0])
|
def create_user(self, user_list: list):
|
||||||
self.user_cards[i].update_status(user_list[i][1])
|
"""创建用户队列"""
|
||||||
|
|
||||||
class LogCard(HeaderCardWidget):
|
while self.Layout.count() > 0:
|
||||||
|
item = self.Layout.takeAt(0)
|
||||||
|
if item.spacerItem():
|
||||||
|
self.Layout.removeItem(item.spacerItem())
|
||||||
|
elif item.widget():
|
||||||
|
item.widget().deleteLater()
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
self.user_cards = []
|
||||||
super().__init__(parent)
|
|
||||||
self.setTitle("日志")
|
|
||||||
|
|
||||||
self.text = TextBrowser()
|
for user in user_list:
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
|
||||||
self.viewLayout.addWidget(self.text)
|
|
||||||
|
|
||||||
self.text.textChanged.connect(self.to_end)
|
self.user_cards.append(StatefulItemCard(user))
|
||||||
|
self.Layout.addWidget(self.user_cards[-1])
|
||||||
|
|
||||||
def to_end(self):
|
self.Layout.addStretch(1)
|
||||||
"""滚动到底部"""
|
|
||||||
|
|
||||||
self.text.moveCursor(QTextCursor.End)
|
def update_user(self, user_list: list):
|
||||||
self.text.ensureCursorVisible()
|
"""更新用户队列"""
|
||||||
|
|
||||||
|
for i in range(len(user_list)):
|
||||||
|
|
||||||
|
self.user_cards[i].Label.setText(user_list[i][0])
|
||||||
|
self.user_cards[i].update_status(user_list[i][1])
|
||||||
|
|
||||||
|
class LogCard(HeaderCardWidget):
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle("日志")
|
||||||
|
|
||||||
|
self.text = TextBrowser()
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
self.viewLayout.addWidget(self.text)
|
||||||
|
|
||||||
|
self.text.textChanged.connect(self.to_end)
|
||||||
|
|
||||||
|
def to_end(self):
|
||||||
|
"""滚动到底部"""
|
||||||
|
|
||||||
|
self.text.moveCursor(QTextCursor.End)
|
||||||
|
self.text.ensureCursorVisible()
|
||||||
|
|||||||
608
app/ui/downloader.py
Normal file
@@ -0,0 +1,608 @@
|
|||||||
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
|
# AUTO_MAA is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published
|
||||||
|
# by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
|
||||||
|
# AUTO_MAA is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
|
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||||||
|
# the GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
|
"""
|
||||||
|
AUTO_MAA
|
||||||
|
AUTO_MAA更新器
|
||||||
|
v4.3
|
||||||
|
作者:DLmaster_361
|
||||||
|
"""
|
||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
import zipfile
|
||||||
|
import requests
|
||||||
|
import subprocess
|
||||||
|
import time
|
||||||
|
import psutil
|
||||||
|
from functools import partial
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from PySide6.QtWidgets import QDialog, QVBoxLayout
|
||||||
|
from qfluentwidgets import (
|
||||||
|
ProgressBar,
|
||||||
|
IndeterminateProgressBar,
|
||||||
|
BodyLabel,
|
||||||
|
setTheme,
|
||||||
|
Theme,
|
||||||
|
)
|
||||||
|
from PySide6.QtGui import QCloseEvent
|
||||||
|
from PySide6.QtCore import QThread, Signal, QTimer, QEventLoop
|
||||||
|
|
||||||
|
from typing import List, Dict, Union
|
||||||
|
|
||||||
|
|
||||||
|
def version_text(version_numb: list) -> str:
|
||||||
|
"""将版本号列表转为可读的文本信息"""
|
||||||
|
|
||||||
|
while len(version_numb) < 4:
|
||||||
|
version_numb.append(0)
|
||||||
|
|
||||||
|
if version_numb[3] == 0:
|
||||||
|
version = f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
||||||
|
else:
|
||||||
|
version = (
|
||||||
|
f"v{'.'.join(str(_) for _ in version_numb[0:3])}-beta.{version_numb[3]}"
|
||||||
|
)
|
||||||
|
return version
|
||||||
|
|
||||||
|
|
||||||
|
class DownloadProcess(QThread):
|
||||||
|
"""分段下载子线程"""
|
||||||
|
|
||||||
|
progress = Signal(int)
|
||||||
|
accomplish = Signal(float)
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
url: str,
|
||||||
|
start_byte: int,
|
||||||
|
end_byte: int,
|
||||||
|
download_path: Path,
|
||||||
|
check_times: int = -1,
|
||||||
|
) -> None:
|
||||||
|
super(DownloadProcess, self).__init__()
|
||||||
|
|
||||||
|
self.setObjectName(f"DownloadProcess-{url}-{start_byte}-{end_byte}")
|
||||||
|
|
||||||
|
self.url = url
|
||||||
|
self.start_byte = start_byte
|
||||||
|
self.end_byte = end_byte
|
||||||
|
self.download_path = download_path
|
||||||
|
self.check_times = check_times
|
||||||
|
|
||||||
|
@logger.catch
|
||||||
|
def run(self) -> None:
|
||||||
|
|
||||||
|
# 清理可能存在的临时文件
|
||||||
|
if self.download_path.exists():
|
||||||
|
self.download_path.unlink()
|
||||||
|
|
||||||
|
headers = {"Range": f"bytes={self.start_byte}-{self.end_byte}"}
|
||||||
|
|
||||||
|
while not self.isInterruptionRequested() and self.check_times != 0:
|
||||||
|
|
||||||
|
try:
|
||||||
|
|
||||||
|
start_time = time.time()
|
||||||
|
|
||||||
|
response = requests.get(
|
||||||
|
self.url, headers=headers, timeout=10, stream=True
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code != 206:
|
||||||
|
|
||||||
|
if self.check_times != -1:
|
||||||
|
self.check_times -= 1
|
||||||
|
|
||||||
|
time.sleep(1)
|
||||||
|
continue
|
||||||
|
|
||||||
|
downloaded_size = 0
|
||||||
|
with self.download_path.open(mode="wb") as f:
|
||||||
|
|
||||||
|
for chunk in response.iter_content(chunk_size=8192):
|
||||||
|
|
||||||
|
if self.isInterruptionRequested():
|
||||||
|
break
|
||||||
|
|
||||||
|
f.write(chunk)
|
||||||
|
downloaded_size += len(chunk)
|
||||||
|
|
||||||
|
self.progress.emit(downloaded_size)
|
||||||
|
|
||||||
|
if self.isInterruptionRequested():
|
||||||
|
|
||||||
|
if self.download_path.exists():
|
||||||
|
self.download_path.unlink()
|
||||||
|
self.accomplish.emit(0)
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
self.accomplish.emit(time.time() - start_time)
|
||||||
|
|
||||||
|
break
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
|
||||||
|
if self.check_times != -1:
|
||||||
|
self.check_times -= 1
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
if self.download_path.exists():
|
||||||
|
self.download_path.unlink()
|
||||||
|
self.accomplish.emit(0)
|
||||||
|
|
||||||
|
|
||||||
|
class ZipExtractProcess(QThread):
|
||||||
|
"""解压子线程"""
|
||||||
|
|
||||||
|
info = Signal(str)
|
||||||
|
accomplish = Signal()
|
||||||
|
|
||||||
|
def __init__(self, name: str, app_path: Path, download_path: Path) -> None:
|
||||||
|
super(ZipExtractProcess, self).__init__()
|
||||||
|
|
||||||
|
self.setObjectName(f"ZipExtractProcess-{name}")
|
||||||
|
|
||||||
|
self.name = name
|
||||||
|
self.app_path = app_path
|
||||||
|
self.download_path = download_path
|
||||||
|
|
||||||
|
@logger.catch
|
||||||
|
def run(self) -> None:
|
||||||
|
|
||||||
|
try:
|
||||||
|
|
||||||
|
while True:
|
||||||
|
|
||||||
|
if self.isInterruptionRequested():
|
||||||
|
self.download_path.unlink()
|
||||||
|
return None
|
||||||
|
try:
|
||||||
|
with zipfile.ZipFile(self.download_path, "r") as zip_ref:
|
||||||
|
zip_ref.extractall(self.app_path)
|
||||||
|
self.accomplish.emit()
|
||||||
|
break
|
||||||
|
except PermissionError:
|
||||||
|
if self.name == "AUTO_MAA":
|
||||||
|
self.info.emit(f"解压出错:AUTO_MAA正在运行,正在尝试将其关闭")
|
||||||
|
self.kill_process(self.app_path / "AUTO_MAA.exe")
|
||||||
|
else:
|
||||||
|
self.info.emit(f"解压出错:{self.name}正在运行,正在等待其关闭")
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
|
||||||
|
e = str(e)
|
||||||
|
e = "\n".join([e[_ : _ + 75] for _ in range(0, len(e), 75)])
|
||||||
|
self.info.emit(f"解压更新时出错:\n{e}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
def kill_process(self, path: Path) -> None:
|
||||||
|
"""根据路径中止进程"""
|
||||||
|
|
||||||
|
for pid in self.search_pids(path):
|
||||||
|
killprocess = subprocess.Popen(
|
||||||
|
f"taskkill /F /PID {pid}",
|
||||||
|
shell=True,
|
||||||
|
creationflags=subprocess.CREATE_NO_WINDOW,
|
||||||
|
)
|
||||||
|
killprocess.wait()
|
||||||
|
|
||||||
|
def search_pids(self, path: Path) -> list:
|
||||||
|
"""根据路径查找进程PID"""
|
||||||
|
|
||||||
|
pids = []
|
||||||
|
for proc in psutil.process_iter(["pid", "exe"]):
|
||||||
|
try:
|
||||||
|
if proc.info["exe"] and proc.info["exe"].lower() == str(path).lower():
|
||||||
|
pids.append(proc.info["pid"])
|
||||||
|
except (psutil.NoSuchProcess, psutil.AccessDenied, psutil.ZombieProcess):
|
||||||
|
# 进程可能在此期间已结束或无法访问,忽略这些异常
|
||||||
|
pass
|
||||||
|
return pids
|
||||||
|
|
||||||
|
|
||||||
|
class DownloadManager(QDialog):
|
||||||
|
"""下载管理器"""
|
||||||
|
|
||||||
|
speed_test_accomplish = Signal()
|
||||||
|
download_accomplish = Signal()
|
||||||
|
download_process_clear = Signal()
|
||||||
|
|
||||||
|
isInterruptionRequested = False
|
||||||
|
|
||||||
|
def __init__(self, app_path: Path, name: str, version: list, config: dict) -> None:
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
|
self.app_path = app_path
|
||||||
|
self.name = name
|
||||||
|
self.version = version
|
||||||
|
self.config = config
|
||||||
|
self.download_path = app_path / "DOWNLOAD_TEMP.zip" # 临时下载文件的路径
|
||||||
|
self.download_process_dict: Dict[str, DownloadProcess] = {}
|
||||||
|
self.timer_dict: Dict[str, QTimer] = {}
|
||||||
|
self.if_speed_test_accomplish = False
|
||||||
|
|
||||||
|
self.resize(700, 70)
|
||||||
|
|
||||||
|
setTheme(Theme.AUTO, lazy=True)
|
||||||
|
|
||||||
|
# 创建垂直布局
|
||||||
|
self.Layout = QVBoxLayout(self)
|
||||||
|
|
||||||
|
self.info = BodyLabel("正在初始化", self)
|
||||||
|
self.progress_1 = IndeterminateProgressBar(self)
|
||||||
|
self.progress_2 = ProgressBar(self)
|
||||||
|
|
||||||
|
self.update_progress(0, 0, 0)
|
||||||
|
|
||||||
|
self.Layout.addWidget(self.info)
|
||||||
|
self.Layout.addStretch(1)
|
||||||
|
self.Layout.addWidget(self.progress_1)
|
||||||
|
self.Layout.addWidget(self.progress_2)
|
||||||
|
self.Layout.addStretch(1)
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
|
||||||
|
if self.name == "AUTO_MAA":
|
||||||
|
if self.config["mode"] == "Proxy":
|
||||||
|
self.test_speed_task1()
|
||||||
|
self.speed_test_accomplish.connect(self.download_task1)
|
||||||
|
elif self.config["mode"] == "MirrorChyan":
|
||||||
|
self.download_task1()
|
||||||
|
elif self.config["mode"] == "MirrorChyan":
|
||||||
|
self.download_task1()
|
||||||
|
|
||||||
|
def get_download_url(self, mode: str) -> Union[str, Dict[str, str]]:
|
||||||
|
"""获取下载链接"""
|
||||||
|
|
||||||
|
url_dict = {}
|
||||||
|
|
||||||
|
if mode == "测速":
|
||||||
|
|
||||||
|
url_dict["GitHub站"] = (
|
||||||
|
f"https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
)
|
||||||
|
url_dict["官方镜像站"] = (
|
||||||
|
f"https://gitee.com/DLmaster_361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
)
|
||||||
|
for name, download_url_head in self.config["download_dict"].items():
|
||||||
|
url_dict[name] = (
|
||||||
|
f"{download_url_head}AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
)
|
||||||
|
for proxy_url in self.config["proxy_list"]:
|
||||||
|
url_dict[proxy_url] = (
|
||||||
|
f"{proxy_url}https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
)
|
||||||
|
return url_dict
|
||||||
|
|
||||||
|
elif mode == "下载":
|
||||||
|
|
||||||
|
if self.name == "AUTO_MAA":
|
||||||
|
|
||||||
|
if self.config["mode"] == "Proxy":
|
||||||
|
|
||||||
|
if "selected" in self.config:
|
||||||
|
selected_url = self.config["selected"]
|
||||||
|
elif "speed_result" in self.config:
|
||||||
|
selected_url = max(
|
||||||
|
self.config["speed_result"],
|
||||||
|
key=self.config["speed_result"].get,
|
||||||
|
)
|
||||||
|
|
||||||
|
if selected_url == "GitHub站":
|
||||||
|
return f"https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
elif selected_url == "官方镜像站":
|
||||||
|
return f"https://gitee.com/DLmaster_361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
elif selected_url in self.config["download_dict"].keys():
|
||||||
|
return f"{self.config["download_dict"][selected_url]}AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
else:
|
||||||
|
return f"{selected_url}https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.version)}/AUTO_MAA_{version_text(self.version)}.zip"
|
||||||
|
|
||||||
|
elif self.config["mode"] == "MirrorChyan":
|
||||||
|
|
||||||
|
with requests.get(
|
||||||
|
self.config["url"],
|
||||||
|
allow_redirects=True,
|
||||||
|
timeout=10,
|
||||||
|
stream=True,
|
||||||
|
) as response:
|
||||||
|
if response.status_code == 200:
|
||||||
|
return response.url
|
||||||
|
|
||||||
|
elif self.config["mode"] == "MirrorChyan":
|
||||||
|
|
||||||
|
with requests.get(
|
||||||
|
self.config["url"], allow_redirects=True, timeout=10, stream=True
|
||||||
|
) as response:
|
||||||
|
if response.status_code == 200:
|
||||||
|
return response.url
|
||||||
|
|
||||||
|
def test_speed_task1(self) -> None:
|
||||||
|
|
||||||
|
if self.isInterruptionRequested:
|
||||||
|
return None
|
||||||
|
|
||||||
|
url_dict = self.get_download_url("测速")
|
||||||
|
self.test_speed_result: Dict[str, float] = {}
|
||||||
|
|
||||||
|
for name, url in url_dict.items():
|
||||||
|
|
||||||
|
if self.isInterruptionRequested:
|
||||||
|
break
|
||||||
|
|
||||||
|
# 创建测速线程,下载4MB文件以测试下载速度
|
||||||
|
self.download_process_dict[name] = DownloadProcess(
|
||||||
|
url,
|
||||||
|
0,
|
||||||
|
4194304,
|
||||||
|
self.app_path / f"{name.replace('/','').replace(':','')}.zip",
|
||||||
|
10,
|
||||||
|
)
|
||||||
|
self.test_speed_result[name] = -1
|
||||||
|
self.download_process_dict[name].accomplish.connect(
|
||||||
|
partial(self.test_speed_task2, name)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.download_process_dict[name].start()
|
||||||
|
timer = QTimer(self)
|
||||||
|
timer.setSingleShot(True)
|
||||||
|
timer.timeout.connect(partial(self.kill_speed_test, name))
|
||||||
|
timer.start(30000)
|
||||||
|
self.timer_dict[name] = timer
|
||||||
|
|
||||||
|
self.update_info("正在测速,预计用时30秒")
|
||||||
|
self.update_progress(0, 1, 0)
|
||||||
|
|
||||||
|
def kill_speed_test(self, name: str) -> None:
|
||||||
|
|
||||||
|
if name in self.download_process_dict:
|
||||||
|
self.download_process_dict[name].requestInterruption()
|
||||||
|
|
||||||
|
def test_speed_task2(self, name: str, t: float) -> None:
|
||||||
|
|
||||||
|
# 计算下载速度
|
||||||
|
if self.isInterruptionRequested:
|
||||||
|
self.update_info(f"已中止测速进程:{name}")
|
||||||
|
self.test_speed_result[name] = 0
|
||||||
|
elif t != 0:
|
||||||
|
self.update_info(f"{name}:{ 4 / t:.2f} MB/s")
|
||||||
|
self.test_speed_result[name] = 4 / t
|
||||||
|
else:
|
||||||
|
self.update_info(f"{name}:{ 0:.2f} MB/s")
|
||||||
|
self.test_speed_result[name] = 0
|
||||||
|
self.update_progress(
|
||||||
|
0,
|
||||||
|
len(self.test_speed_result),
|
||||||
|
sum(1 for speed in self.test_speed_result.values() if speed != -1),
|
||||||
|
)
|
||||||
|
|
||||||
|
# 删除临时文件
|
||||||
|
if (self.app_path / f"{name.replace('/','').replace(':','')}.zip").exists():
|
||||||
|
(self.app_path / f"{name.replace('/','').replace(':','')}.zip").unlink()
|
||||||
|
|
||||||
|
# 清理下载线程
|
||||||
|
self.timer_dict[name].stop()
|
||||||
|
self.timer_dict[name].deleteLater()
|
||||||
|
self.timer_dict.pop(name)
|
||||||
|
self.download_process_dict[name].requestInterruption()
|
||||||
|
self.download_process_dict[name].quit()
|
||||||
|
self.download_process_dict[name].wait()
|
||||||
|
self.download_process_dict[name].deleteLater()
|
||||||
|
self.download_process_dict.pop(name)
|
||||||
|
if not self.download_process_dict:
|
||||||
|
self.download_process_clear.emit()
|
||||||
|
|
||||||
|
# 当有速度大于1 MB/s的链接或存在3个即以上链接测速完成时,停止其他测速
|
||||||
|
if not self.if_speed_test_accomplish and (
|
||||||
|
sum(1 for speed in self.test_speed_result.values() if speed > 0) >= 3
|
||||||
|
or any(speed > 1 for speed in self.test_speed_result.values())
|
||||||
|
):
|
||||||
|
self.if_speed_test_accomplish = True
|
||||||
|
for timer in self.timer_dict.values():
|
||||||
|
timer.timeout.emit()
|
||||||
|
|
||||||
|
if any(speed == -1 for _, speed in self.test_speed_result.items()):
|
||||||
|
return None
|
||||||
|
|
||||||
|
# 保存测速结果
|
||||||
|
self.config["speed_result"] = self.test_speed_result
|
||||||
|
|
||||||
|
self.update_info("测速完成!")
|
||||||
|
self.speed_test_accomplish.emit()
|
||||||
|
|
||||||
|
def download_task1(self) -> None:
|
||||||
|
|
||||||
|
if self.isInterruptionRequested:
|
||||||
|
return None
|
||||||
|
|
||||||
|
url = self.get_download_url("下载")
|
||||||
|
self.downloaded_size_list: List[List[int, bool]] = []
|
||||||
|
|
||||||
|
response = requests.head(url, timeout=10)
|
||||||
|
|
||||||
|
self.file_size = int(response.headers.get("content-length", 0))
|
||||||
|
part_size = self.file_size // self.config["thread_numb"]
|
||||||
|
self.downloaded_size = 0
|
||||||
|
self.last_download_size = 0
|
||||||
|
self.last_time = time.time()
|
||||||
|
self.speed = 0
|
||||||
|
|
||||||
|
# 拆分下载任务,启用多线程下载
|
||||||
|
for i in range(self.config["thread_numb"]):
|
||||||
|
|
||||||
|
if self.isInterruptionRequested:
|
||||||
|
break
|
||||||
|
|
||||||
|
# 计算单任务下载范围
|
||||||
|
start_byte = i * part_size
|
||||||
|
end_byte = (
|
||||||
|
(i + 1) * part_size - 1
|
||||||
|
if (i != self.config["thread_numb"] - 1)
|
||||||
|
else self.file_size - 1
|
||||||
|
)
|
||||||
|
|
||||||
|
# 创建下载子线程
|
||||||
|
self.download_process_dict[f"part{i}"] = DownloadProcess(
|
||||||
|
url,
|
||||||
|
start_byte,
|
||||||
|
end_byte,
|
||||||
|
self.download_path.with_suffix(f".part{i}"),
|
||||||
|
1 if self.config["mode"] == "MirrorChyan" else -1,
|
||||||
|
)
|
||||||
|
self.downloaded_size_list.append([0, False])
|
||||||
|
self.download_process_dict[f"part{i}"].progress.connect(
|
||||||
|
partial(self.download_task2, i)
|
||||||
|
)
|
||||||
|
self.download_process_dict[f"part{i}"].accomplish.connect(
|
||||||
|
partial(self.download_task3, i)
|
||||||
|
)
|
||||||
|
self.download_process_dict[f"part{i}"].start()
|
||||||
|
|
||||||
|
def download_task2(self, index: str, current: int) -> None:
|
||||||
|
"""更新下载进度"""
|
||||||
|
|
||||||
|
self.downloaded_size_list[index][0] = current
|
||||||
|
self.downloaded_size = sum([_[0] for _ in self.downloaded_size_list])
|
||||||
|
self.update_progress(0, self.file_size, self.downloaded_size)
|
||||||
|
|
||||||
|
if time.time() - self.last_time >= 1.0:
|
||||||
|
self.speed = (
|
||||||
|
(self.downloaded_size - self.last_download_size)
|
||||||
|
/ (time.time() - self.last_time)
|
||||||
|
/ 1024
|
||||||
|
)
|
||||||
|
self.last_download_size = self.downloaded_size
|
||||||
|
self.last_time = time.time()
|
||||||
|
|
||||||
|
if self.speed >= 1024:
|
||||||
|
self.update_info(
|
||||||
|
f"正在下载:{self.name} 已下载:{self.downloaded_size / 1048576:.2f}/{self.file_size / 1048576:.2f} MB ({self.downloaded_size / self.file_size * 100:.2f}%) 下载速度:{self.speed / 1024:.2f} MB/s",
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.update_info(
|
||||||
|
f"正在下载:{self.name} 已下载:{self.downloaded_size / 1048576:.2f}/{self.file_size / 1048576:.2f} MB ({self.downloaded_size / self.file_size * 100:.2f}%) 下载速度:{self.speed:.2f} KB/s",
|
||||||
|
)
|
||||||
|
|
||||||
|
def download_task3(self, index: str, t: float) -> None:
|
||||||
|
|
||||||
|
# 标记下载线程完成
|
||||||
|
self.downloaded_size_list[index][1] = True
|
||||||
|
|
||||||
|
# 清理下载线程
|
||||||
|
self.download_process_dict[f"part{index}"].requestInterruption()
|
||||||
|
self.download_process_dict[f"part{index}"].quit()
|
||||||
|
self.download_process_dict[f"part{index}"].wait()
|
||||||
|
self.download_process_dict[f"part{index}"].deleteLater()
|
||||||
|
self.download_process_dict.pop(f"part{index}")
|
||||||
|
if not self.download_process_dict:
|
||||||
|
self.download_process_clear.emit()
|
||||||
|
|
||||||
|
if (
|
||||||
|
any([not _[1] for _ in self.downloaded_size_list])
|
||||||
|
or self.isInterruptionRequested
|
||||||
|
):
|
||||||
|
return None
|
||||||
|
|
||||||
|
# 合并下载的分段文件
|
||||||
|
with self.download_path.open(mode="wb") as outfile:
|
||||||
|
for i in range(self.config["thread_numb"]):
|
||||||
|
with self.download_path.with_suffix(f".part{i}").open(
|
||||||
|
mode="rb"
|
||||||
|
) as infile:
|
||||||
|
outfile.write(infile.read())
|
||||||
|
self.download_path.with_suffix(f".part{i}").unlink()
|
||||||
|
|
||||||
|
self.update_info("正在解压更新文件")
|
||||||
|
self.update_progress(0, 0, 0)
|
||||||
|
|
||||||
|
# 创建解压线程
|
||||||
|
self.zip_extract = ZipExtractProcess(
|
||||||
|
self.name, self.app_path, self.download_path
|
||||||
|
)
|
||||||
|
self.zip_loop = QEventLoop()
|
||||||
|
self.zip_extract.info.connect(self.update_info)
|
||||||
|
self.zip_extract.accomplish.connect(self.zip_loop.quit)
|
||||||
|
self.zip_extract.start()
|
||||||
|
self.zip_loop.exec()
|
||||||
|
|
||||||
|
self.update_info("正在删除临时文件")
|
||||||
|
self.update_progress(0, 0, 0)
|
||||||
|
if (self.app_path / "changes.json").exists():
|
||||||
|
(self.app_path / "changes.json").unlink()
|
||||||
|
if self.download_path.exists():
|
||||||
|
self.download_path.unlink()
|
||||||
|
|
||||||
|
# 下载完成后打开对应程序
|
||||||
|
if not self.isInterruptionRequested and self.name == "MAA":
|
||||||
|
subprocess.Popen(
|
||||||
|
[self.app_path / "MAA.exe"],
|
||||||
|
creationflags=subprocess.CREATE_NEW_PROCESS_GROUP
|
||||||
|
| subprocess.DETACHED_PROCESS
|
||||||
|
| subprocess.CREATE_NO_WINDOW,
|
||||||
|
)
|
||||||
|
if self.name == "AUTO_MAA":
|
||||||
|
self.update_info(f"即将安装{self.name}")
|
||||||
|
else:
|
||||||
|
self.update_info(f"{self.name}下载成功!")
|
||||||
|
self.update_progress(0, 100, 100)
|
||||||
|
self.download_accomplish.emit()
|
||||||
|
|
||||||
|
def update_info(self, text: str) -> None:
|
||||||
|
self.info.setText(text)
|
||||||
|
|
||||||
|
def update_progress(self, begin: int, end: int, current: int) -> None:
|
||||||
|
|
||||||
|
if begin == 0 and end == 0:
|
||||||
|
self.progress_2.setVisible(False)
|
||||||
|
self.progress_1.setVisible(True)
|
||||||
|
else:
|
||||||
|
self.progress_1.setVisible(False)
|
||||||
|
self.progress_2.setVisible(True)
|
||||||
|
self.progress_2.setRange(begin, end)
|
||||||
|
self.progress_2.setValue(current)
|
||||||
|
|
||||||
|
def requestInterruption(self) -> None:
|
||||||
|
|
||||||
|
self.isInterruptionRequested = True
|
||||||
|
|
||||||
|
if hasattr(self, "zip_extract") and self.zip_extract:
|
||||||
|
self.zip_extract.requestInterruption()
|
||||||
|
|
||||||
|
if hasattr(self, "zip_loop") and self.zip_loop:
|
||||||
|
self.zip_loop.quit()
|
||||||
|
|
||||||
|
for process in self.download_process_dict.values():
|
||||||
|
process.requestInterruption()
|
||||||
|
|
||||||
|
if self.download_process_dict:
|
||||||
|
loop = QEventLoop()
|
||||||
|
self.download_process_clear.connect(loop.quit)
|
||||||
|
loop.exec()
|
||||||
|
|
||||||
|
def closeEvent(self, event: QCloseEvent):
|
||||||
|
"""清理残余进程"""
|
||||||
|
|
||||||
|
self.requestInterruption()
|
||||||
|
|
||||||
|
event.accept()
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA历史记录界面
|
AUTO_MAA历史记录界面
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -36,18 +36,23 @@ from qfluentwidgets import (
|
|||||||
FluentIcon,
|
FluentIcon,
|
||||||
HeaderCardWidget,
|
HeaderCardWidget,
|
||||||
PushButton,
|
PushButton,
|
||||||
ExpandGroupSettingCard,
|
|
||||||
TextBrowser,
|
TextBrowser,
|
||||||
|
CardWidget,
|
||||||
|
ComboBox,
|
||||||
|
ZhDatePicker,
|
||||||
|
SubtitleLabel,
|
||||||
)
|
)
|
||||||
from PySide6.QtCore import Signal
|
from PySide6.QtCore import Signal, QDate
|
||||||
import os
|
import os
|
||||||
|
import subprocess
|
||||||
|
from datetime import datetime, timedelta
|
||||||
from functools import partial
|
from functools import partial
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import List
|
from typing import Union, List, Dict
|
||||||
|
|
||||||
|
|
||||||
from app.core import Config
|
from app.core import Config, SoundPlayer
|
||||||
from .Widget import StatefulItemCard, QuantifiedItemCard
|
from .Widget import StatefulItemCard, QuantifiedItemCard, QuickExpandGroupCard
|
||||||
|
|
||||||
|
|
||||||
class History(QWidget):
|
class History(QWidget):
|
||||||
@@ -56,22 +61,29 @@ class History(QWidget):
|
|||||||
super().__init__(parent)
|
super().__init__(parent)
|
||||||
self.setObjectName("历史记录")
|
self.setObjectName("历史记录")
|
||||||
|
|
||||||
|
self.history_top_bar = self.HistoryTopBar(self)
|
||||||
|
self.history_top_bar.search_history.connect(self.reload_history)
|
||||||
|
|
||||||
content_widget = QWidget()
|
content_widget = QWidget()
|
||||||
self.content_layout = QVBoxLayout(content_widget)
|
self.content_layout = QVBoxLayout(content_widget)
|
||||||
|
self.content_layout.setContentsMargins(0, 0, 11, 0)
|
||||||
|
|
||||||
scrollArea = ScrollArea()
|
scrollArea = ScrollArea()
|
||||||
scrollArea.setWidgetResizable(True)
|
scrollArea.setWidgetResizable(True)
|
||||||
|
scrollArea.setContentsMargins(0, 0, 0, 0)
|
||||||
|
scrollArea.setStyleSheet("background: transparent; border: none;")
|
||||||
scrollArea.setWidget(content_widget)
|
scrollArea.setWidget(content_widget)
|
||||||
layout = QVBoxLayout()
|
|
||||||
|
layout = QVBoxLayout(self)
|
||||||
|
layout.addWidget(self.history_top_bar)
|
||||||
layout.addWidget(scrollArea)
|
layout.addWidget(scrollArea)
|
||||||
self.setLayout(layout)
|
|
||||||
|
|
||||||
self.history_card_list = []
|
self.history_card_list = []
|
||||||
|
|
||||||
self.refresh()
|
def reload_history(self, mode: str, start_date: QDate, end_date: QDate) -> None:
|
||||||
|
"""加载历史记录界面"""
|
||||||
|
|
||||||
def refresh(self):
|
SoundPlayer.play("历史记录查询")
|
||||||
"""刷新脚本实例界面"""
|
|
||||||
|
|
||||||
while self.content_layout.count() > 0:
|
while self.content_layout.count() > 0:
|
||||||
item = self.content_layout.takeAt(0)
|
item = self.content_layout.takeAt(0)
|
||||||
@@ -82,177 +94,300 @@ class History(QWidget):
|
|||||||
|
|
||||||
self.history_card_list = []
|
self.history_card_list = []
|
||||||
|
|
||||||
history_dict = Config.search_history()
|
history_dict = Config.search_history(
|
||||||
|
mode,
|
||||||
|
datetime(start_date.year(), start_date.month(), start_date.day()),
|
||||||
|
datetime(end_date.year(), end_date.month(), end_date.day()),
|
||||||
|
)
|
||||||
|
|
||||||
for date, user_list in history_dict.items():
|
for date, user in history_dict.items():
|
||||||
|
|
||||||
self.history_card_list.append(HistoryCard(date, user_list, self))
|
self.history_card_list.append(self.HistoryCard(mode, date, user, self))
|
||||||
self.content_layout.addWidget(self.history_card_list[-1])
|
self.content_layout.addWidget(self.history_card_list[-1])
|
||||||
|
|
||||||
self.content_layout.addStretch(1)
|
self.content_layout.addStretch(1)
|
||||||
|
|
||||||
|
class HistoryTopBar(CardWidget):
|
||||||
|
"""历史记录顶部工具栏"""
|
||||||
|
|
||||||
class HistoryCard(ExpandGroupSettingCard):
|
search_history = Signal(str, QDate, QDate)
|
||||||
|
|
||||||
def __init__(self, date: str, user_list: List[Path], parent=None):
|
def __init__(self, parent=None):
|
||||||
super().__init__(
|
super().__init__(parent)
|
||||||
FluentIcon.HISTORY, date, f"{date}的历史运行记录与统计信息", parent
|
|
||||||
)
|
|
||||||
|
|
||||||
widget = QWidget()
|
Layout = QHBoxLayout(self)
|
||||||
Layout = QVBoxLayout(widget)
|
|
||||||
self.viewLayout.setContentsMargins(0, 0, 0, 0)
|
|
||||||
self.viewLayout.setSpacing(0)
|
|
||||||
self.addGroupWidget(widget)
|
|
||||||
|
|
||||||
self.user_history_card_list = []
|
self.lable_1 = SubtitleLabel("查询范围:")
|
||||||
|
self.start_date = ZhDatePicker()
|
||||||
|
self.start_date.setDate(QDate(2019, 5, 1))
|
||||||
|
self.lable_2 = SubtitleLabel("→")
|
||||||
|
self.end_date = ZhDatePicker()
|
||||||
|
server_date = Config.server_date()
|
||||||
|
self.end_date.setDate(
|
||||||
|
QDate(server_date.year, server_date.month, server_date.day)
|
||||||
|
)
|
||||||
|
self.mode = ComboBox()
|
||||||
|
self.mode.setPlaceholderText("请选择查询模式")
|
||||||
|
self.mode.addItems(["按日合并", "按周合并", "按月合并"])
|
||||||
|
|
||||||
for user_path in user_list:
|
self.select_month = PushButton(FluentIcon.TAG, "最近一月")
|
||||||
|
self.select_week = PushButton(FluentIcon.TAG, "最近一周")
|
||||||
|
self.search = PushButton(FluentIcon.SEARCH, "查询")
|
||||||
|
self.select_month.clicked.connect(lambda: self.select_date("month"))
|
||||||
|
self.select_week.clicked.connect(lambda: self.select_date("week"))
|
||||||
|
self.search.clicked.connect(
|
||||||
|
lambda: self.search_history.emit(
|
||||||
|
self.mode.currentText(),
|
||||||
|
self.start_date.getDate(),
|
||||||
|
self.end_date.getDate(),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
self.user_history_card_list.append(self.UserHistoryCard(user_path, self))
|
Layout.addWidget(self.lable_1)
|
||||||
Layout.addWidget(self.user_history_card_list[-1])
|
Layout.addWidget(self.start_date)
|
||||||
|
Layout.addWidget(self.lable_2)
|
||||||
|
Layout.addWidget(self.end_date)
|
||||||
|
Layout.addWidget(self.mode)
|
||||||
|
Layout.addStretch(1)
|
||||||
|
Layout.addWidget(self.select_month)
|
||||||
|
Layout.addWidget(self.select_week)
|
||||||
|
Layout.addWidget(self.search)
|
||||||
|
|
||||||
class UserHistoryCard(HeaderCardWidget):
|
def select_date(self, date: str) -> None:
|
||||||
|
"""选中最近一段时间并启动查询"""
|
||||||
|
|
||||||
|
server_date = Config.server_date()
|
||||||
|
if date == "week":
|
||||||
|
begin_date = server_date - timedelta(weeks=1)
|
||||||
|
elif date == "month":
|
||||||
|
begin_date = server_date - timedelta(days=30)
|
||||||
|
|
||||||
|
self.start_date.setDate(
|
||||||
|
QDate(begin_date.year, begin_date.month, begin_date.day)
|
||||||
|
)
|
||||||
|
self.end_date.setDate(
|
||||||
|
QDate(server_date.year, server_date.month, server_date.day)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.search.clicked.emit()
|
||||||
|
|
||||||
|
class HistoryCard(QuickExpandGroupCard):
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
user_history_path: Path,
|
mode: str,
|
||||||
|
date: str,
|
||||||
|
user: Union[List[Path], Dict[str, List[Path]]],
|
||||||
parent=None,
|
parent=None,
|
||||||
):
|
):
|
||||||
super().__init__(parent)
|
super().__init__(
|
||||||
|
FluentIcon.HISTORY, date, f"{date}的历史运行记录与统计信息", parent
|
||||||
|
)
|
||||||
|
|
||||||
self.setTitle(user_history_path.name.replace(".json", ""))
|
widget = QWidget()
|
||||||
|
Layout = QVBoxLayout(widget)
|
||||||
self.user_history_path = user_history_path
|
|
||||||
self.main_history = Config.load_maa_logs("总览", user_history_path)
|
|
||||||
|
|
||||||
self.index_card = self.IndexCard(self.main_history["条目索引"], self)
|
|
||||||
self.statistics_card = QHBoxLayout()
|
|
||||||
self.log_card = self.LogCard(self)
|
|
||||||
|
|
||||||
self.index_card.index_changed.connect(self.update_info)
|
|
||||||
|
|
||||||
self.viewLayout.addWidget(self.index_card)
|
|
||||||
self.viewLayout.addLayout(self.statistics_card)
|
|
||||||
self.viewLayout.addWidget(self.log_card)
|
|
||||||
self.viewLayout.setContentsMargins(0, 0, 0, 0)
|
self.viewLayout.setContentsMargins(0, 0, 0, 0)
|
||||||
self.viewLayout.setSpacing(0)
|
self.viewLayout.setSpacing(0)
|
||||||
self.viewLayout.setStretch(0, 1)
|
self.addGroupWidget(widget)
|
||||||
self.viewLayout.setStretch(2, 4)
|
|
||||||
|
|
||||||
self.update_info("数据总览")
|
self.user_history_card_list = []
|
||||||
|
|
||||||
def update_info(self, index: str) -> None:
|
if mode == "按日合并":
|
||||||
"""更新信息"""
|
|
||||||
|
|
||||||
if index == "数据总览":
|
for user_path in user:
|
||||||
|
self.user_history_card_list.append(
|
||||||
while self.statistics_card.count() > 0:
|
self.UserHistoryCard(mode, user_path.stem, user_path, self)
|
||||||
item = self.statistics_card.takeAt(0)
|
|
||||||
if item.spacerItem():
|
|
||||||
self.statistics_card.removeItem(item.spacerItem())
|
|
||||||
elif item.widget():
|
|
||||||
item.widget().deleteLater()
|
|
||||||
|
|
||||||
for name, item_list in self.main_history["统计数据"].items():
|
|
||||||
|
|
||||||
statistics_card = self.StatisticsCard(name, item_list, self)
|
|
||||||
self.statistics_card.addWidget(statistics_card)
|
|
||||||
|
|
||||||
self.log_card.hide()
|
|
||||||
|
|
||||||
else:
|
|
||||||
|
|
||||||
single_history = Config.load_maa_logs(
|
|
||||||
"单项",
|
|
||||||
self.user_history_path.with_suffix("")
|
|
||||||
/ f"{index.replace(":","-")}.json",
|
|
||||||
)
|
|
||||||
|
|
||||||
while self.statistics_card.count() > 0:
|
|
||||||
item = self.statistics_card.takeAt(0)
|
|
||||||
if item.spacerItem():
|
|
||||||
self.statistics_card.removeItem(item.spacerItem())
|
|
||||||
elif item.widget():
|
|
||||||
item.widget().deleteLater()
|
|
||||||
|
|
||||||
for name, item_list in single_history["统计数据"].items():
|
|
||||||
|
|
||||||
statistics_card = self.StatisticsCard(name, item_list, self)
|
|
||||||
self.statistics_card.addWidget(statistics_card)
|
|
||||||
|
|
||||||
self.log_card.text.setText(single_history["日志信息"])
|
|
||||||
self.log_card.button.clicked.disconnect()
|
|
||||||
self.log_card.button.clicked.connect(
|
|
||||||
lambda: os.startfile(
|
|
||||||
self.user_history_path.with_suffix("")
|
|
||||||
/ f"{index.replace(":","-")}.log"
|
|
||||||
)
|
)
|
||||||
)
|
Layout.addWidget(self.user_history_card_list[-1])
|
||||||
self.log_card.show()
|
|
||||||
|
|
||||||
self.viewLayout.setStretch(1, self.statistics_card.count())
|
elif mode in ["按周合并", "按月合并"]:
|
||||||
|
|
||||||
self.setMinimumHeight(300)
|
for user, info in user.items():
|
||||||
|
self.user_history_card_list.append(
|
||||||
class IndexCard(HeaderCardWidget):
|
self.UserHistoryCard(mode, user, info, self)
|
||||||
|
|
||||||
index_changed = Signal(str)
|
|
||||||
|
|
||||||
def __init__(self, index_list: list, parent=None):
|
|
||||||
super().__init__(parent)
|
|
||||||
self.setTitle("记录条目")
|
|
||||||
|
|
||||||
self.Layout = QVBoxLayout()
|
|
||||||
self.viewLayout.addLayout(self.Layout)
|
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
|
||||||
|
|
||||||
self.index_cards: List[StatefulItemCard] = []
|
|
||||||
|
|
||||||
for index in index_list:
|
|
||||||
|
|
||||||
self.index_cards.append(StatefulItemCard(index))
|
|
||||||
self.index_cards[-1].clicked.connect(
|
|
||||||
partial(self.index_changed.emit, index[0])
|
|
||||||
)
|
)
|
||||||
self.Layout.addWidget(self.index_cards[-1])
|
Layout.addWidget(self.user_history_card_list[-1])
|
||||||
|
|
||||||
self.Layout.addStretch(1)
|
class UserHistoryCard(HeaderCardWidget):
|
||||||
|
"""用户历史记录卡片"""
|
||||||
|
|
||||||
class StatisticsCard(HeaderCardWidget):
|
def __init__(
|
||||||
|
self,
|
||||||
def __init__(self, name: str, item_list: list, parent=None):
|
mode: str,
|
||||||
|
name: str,
|
||||||
|
user_history: Union[Path, List[Path]],
|
||||||
|
parent=None,
|
||||||
|
):
|
||||||
super().__init__(parent)
|
super().__init__(parent)
|
||||||
|
|
||||||
self.setTitle(name)
|
self.setTitle(name)
|
||||||
|
|
||||||
self.Layout = QVBoxLayout()
|
if mode == "按日合并":
|
||||||
self.viewLayout.addLayout(self.Layout)
|
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
|
||||||
|
|
||||||
self.item_cards: List[QuantifiedItemCard] = []
|
self.user_history_path = user_history
|
||||||
|
self.main_history = Config.load_maa_logs("总览", user_history)
|
||||||
|
|
||||||
for item in item_list:
|
self.index_card = self.IndexCard(
|
||||||
|
self.main_history["条目索引"], self
|
||||||
|
)
|
||||||
|
self.index_card.index_changed.connect(self.update_info)
|
||||||
|
self.viewLayout.addWidget(self.index_card)
|
||||||
|
|
||||||
self.item_cards.append(QuantifiedItemCard(item))
|
elif mode in ["按周合并", "按月合并"]:
|
||||||
self.Layout.addWidget(self.item_cards[-1])
|
|
||||||
|
|
||||||
if len(item_list) == 0:
|
history = Config.merge_maa_logs("指定项", user_history)
|
||||||
self.Layout.addWidget(QuantifiedItemCard(["暂无记录", ""]))
|
|
||||||
|
|
||||||
self.Layout.addStretch(1)
|
self.main_history = {}
|
||||||
|
self.main_history["统计数据"] = {
|
||||||
|
"公招统计": list(history["recruit_statistics"].items())
|
||||||
|
}
|
||||||
|
|
||||||
class LogCard(HeaderCardWidget):
|
for game_id, drops in history["drop_statistics"].items():
|
||||||
|
self.main_history["统计数据"][f"掉落统计:{game_id}"] = list(
|
||||||
|
drops.items()
|
||||||
|
)
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
self.statistics_card = QHBoxLayout()
|
||||||
super().__init__(parent)
|
self.log_card = self.LogCard(self)
|
||||||
self.setTitle("日志")
|
|
||||||
|
|
||||||
self.text = TextBrowser(self)
|
self.viewLayout.addLayout(self.statistics_card)
|
||||||
self.button = PushButton("打开日志文件", self)
|
self.viewLayout.addWidget(self.log_card)
|
||||||
self.button.clicked.connect(lambda: print("打开日志文件"))
|
self.viewLayout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
self.viewLayout.setSpacing(0)
|
||||||
|
self.viewLayout.setStretch(0, 1)
|
||||||
|
self.viewLayout.setStretch(2, 4)
|
||||||
|
|
||||||
Layout = QVBoxLayout()
|
self.update_info("数据总览")
|
||||||
Layout.addWidget(self.text)
|
|
||||||
Layout.addWidget(self.button)
|
def update_info(self, index: str) -> None:
|
||||||
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
"""更新信息"""
|
||||||
self.viewLayout.addLayout(Layout)
|
|
||||||
|
if index == "数据总览":
|
||||||
|
|
||||||
|
while self.statistics_card.count() > 0:
|
||||||
|
item = self.statistics_card.takeAt(0)
|
||||||
|
if item.spacerItem():
|
||||||
|
self.statistics_card.removeItem(item.spacerItem())
|
||||||
|
elif item.widget():
|
||||||
|
item.widget().deleteLater()
|
||||||
|
|
||||||
|
for name, item_list in self.main_history["统计数据"].items():
|
||||||
|
|
||||||
|
statistics_card = self.StatisticsCard(name, item_list, self)
|
||||||
|
self.statistics_card.addWidget(statistics_card)
|
||||||
|
|
||||||
|
self.log_card.hide()
|
||||||
|
|
||||||
|
else:
|
||||||
|
|
||||||
|
single_history = Config.load_maa_logs(
|
||||||
|
"单项",
|
||||||
|
self.user_history_path.with_suffix("")
|
||||||
|
/ f"{index.replace(":","-")}.json",
|
||||||
|
)
|
||||||
|
|
||||||
|
while self.statistics_card.count() > 0:
|
||||||
|
item = self.statistics_card.takeAt(0)
|
||||||
|
if item.spacerItem():
|
||||||
|
self.statistics_card.removeItem(item.spacerItem())
|
||||||
|
elif item.widget():
|
||||||
|
item.widget().deleteLater()
|
||||||
|
|
||||||
|
for name, item_list in single_history["统计数据"].items():
|
||||||
|
|
||||||
|
statistics_card = self.StatisticsCard(name, item_list, self)
|
||||||
|
self.statistics_card.addWidget(statistics_card)
|
||||||
|
|
||||||
|
self.log_card.text.setText(single_history["日志信息"])
|
||||||
|
self.log_card.open_file.clicked.disconnect()
|
||||||
|
self.log_card.open_file.clicked.connect(
|
||||||
|
lambda: os.startfile(
|
||||||
|
self.user_history_path.with_suffix("")
|
||||||
|
/ f"{index.replace(":","-")}.log"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self.log_card.open_dir.clicked.disconnect()
|
||||||
|
self.log_card.open_dir.clicked.connect(
|
||||||
|
lambda: subprocess.Popen(
|
||||||
|
[
|
||||||
|
"explorer",
|
||||||
|
"/select,",
|
||||||
|
str(
|
||||||
|
self.user_history_path.with_suffix("")
|
||||||
|
/ f"{index.replace(":","-")}.log"
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self.log_card.show()
|
||||||
|
|
||||||
|
self.viewLayout.setStretch(1, self.statistics_card.count())
|
||||||
|
|
||||||
|
self.setMinimumHeight(300)
|
||||||
|
|
||||||
|
class IndexCard(HeaderCardWidget):
|
||||||
|
|
||||||
|
index_changed = Signal(str)
|
||||||
|
|
||||||
|
def __init__(self, index_list: list, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle("记录条目")
|
||||||
|
|
||||||
|
self.Layout = QVBoxLayout()
|
||||||
|
self.viewLayout.addLayout(self.Layout)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
|
||||||
|
self.index_cards: List[StatefulItemCard] = []
|
||||||
|
|
||||||
|
for index in index_list:
|
||||||
|
|
||||||
|
self.index_cards.append(StatefulItemCard(index))
|
||||||
|
self.index_cards[-1].clicked.connect(
|
||||||
|
partial(self.index_changed.emit, index[0])
|
||||||
|
)
|
||||||
|
self.Layout.addWidget(self.index_cards[-1])
|
||||||
|
|
||||||
|
self.Layout.addStretch(1)
|
||||||
|
|
||||||
|
class StatisticsCard(HeaderCardWidget):
|
||||||
|
|
||||||
|
def __init__(self, name: str, item_list: list, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle(name)
|
||||||
|
|
||||||
|
self.Layout = QVBoxLayout()
|
||||||
|
self.viewLayout.addLayout(self.Layout)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
|
||||||
|
self.item_cards: List[QuantifiedItemCard] = []
|
||||||
|
|
||||||
|
for item in item_list:
|
||||||
|
|
||||||
|
self.item_cards.append(QuantifiedItemCard(item))
|
||||||
|
self.Layout.addWidget(self.item_cards[-1])
|
||||||
|
|
||||||
|
if len(item_list) == 0:
|
||||||
|
self.Layout.addWidget(QuantifiedItemCard(["暂无记录", ""]))
|
||||||
|
|
||||||
|
self.Layout.addStretch(1)
|
||||||
|
|
||||||
|
class LogCard(HeaderCardWidget):
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setTitle("日志")
|
||||||
|
|
||||||
|
self.text = TextBrowser(self)
|
||||||
|
self.open_file = PushButton("打开日志文件", self)
|
||||||
|
self.open_file.clicked.connect(lambda: print("打开日志文件"))
|
||||||
|
self.open_dir = PushButton("打开所在目录", self)
|
||||||
|
self.open_dir.clicked.connect(lambda: print("打开所在文件"))
|
||||||
|
|
||||||
|
Layout = QVBoxLayout()
|
||||||
|
h_layout = QHBoxLayout()
|
||||||
|
h_layout.addWidget(self.open_file)
|
||||||
|
h_layout.addWidget(self.open_dir)
|
||||||
|
Layout.addWidget(self.text)
|
||||||
|
Layout.addLayout(h_layout)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
self.viewLayout.addLayout(Layout)
|
||||||
|
|||||||
149
app/ui/home.py
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA主界面
|
AUTO_MAA主界面
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -45,13 +45,11 @@ from qfluentwidgets import (
|
|||||||
)
|
)
|
||||||
import re
|
import re
|
||||||
import shutil
|
import shutil
|
||||||
import requests
|
|
||||||
import json
|
import json
|
||||||
import time
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
from app.core import Config, MainInfoBar
|
from app.core import Config, MainInfoBar, Network
|
||||||
from .Widget import Banner, IconButton
|
from .Widget import Banner, IconButton
|
||||||
|
|
||||||
|
|
||||||
@@ -64,14 +62,6 @@ class Home(QWidget):
|
|||||||
self.banner = Banner()
|
self.banner = Banner()
|
||||||
self.banner_text = TextBrowser()
|
self.banner_text = TextBrowser()
|
||||||
|
|
||||||
widget = QWidget()
|
|
||||||
Layout = QVBoxLayout(widget)
|
|
||||||
|
|
||||||
Layout.addWidget(self.banner)
|
|
||||||
Layout.addWidget(self.banner_text)
|
|
||||||
Layout.setStretch(0, 2)
|
|
||||||
Layout.setStretch(1, 3)
|
|
||||||
|
|
||||||
v_layout = QVBoxLayout(self.banner)
|
v_layout = QVBoxLayout(self.banner)
|
||||||
v_layout.setContentsMargins(0, 0, 0, 15)
|
v_layout.setContentsMargins(0, 0, 0, 15)
|
||||||
v_layout.setSpacing(5)
|
v_layout.setSpacing(5)
|
||||||
@@ -148,27 +138,31 @@ class Home(QWidget):
|
|||||||
# 将底部水平布局添加到垂直布局
|
# 将底部水平布局添加到垂直布局
|
||||||
v_layout.addLayout(h2_layout)
|
v_layout.addLayout(h2_layout)
|
||||||
|
|
||||||
layout = QVBoxLayout()
|
content_widget = QWidget()
|
||||||
|
content_layout = QVBoxLayout(content_widget)
|
||||||
|
content_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
content_layout.addWidget(self.banner)
|
||||||
|
content_layout.addWidget(self.banner_text)
|
||||||
|
content_layout.setStretch(0, 2)
|
||||||
|
content_layout.setStretch(1, 3)
|
||||||
|
|
||||||
scrollArea = ScrollArea()
|
scrollArea = ScrollArea()
|
||||||
scrollArea.setWidgetResizable(True)
|
scrollArea.setWidgetResizable(True)
|
||||||
scrollArea.setWidget(widget)
|
scrollArea.setContentsMargins(0, 0, 0, 0)
|
||||||
|
scrollArea.setStyleSheet("background: transparent; border: none;")
|
||||||
|
scrollArea.setWidget(content_widget)
|
||||||
|
|
||||||
|
layout = QVBoxLayout(self)
|
||||||
layout.addWidget(scrollArea)
|
layout.addWidget(scrollArea)
|
||||||
self.setLayout(layout)
|
|
||||||
|
|
||||||
self.set_banner()
|
self.set_banner()
|
||||||
|
|
||||||
def get_home_image(self) -> None:
|
def get_home_image(self) -> None:
|
||||||
"""获取主页图片"""
|
"""获取主页图片"""
|
||||||
|
|
||||||
if (
|
if Config.get(Config.function_HomeImageMode) == "默认":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "默认"
|
|
||||||
):
|
|
||||||
pass
|
pass
|
||||||
elif (
|
elif Config.get(Config.function_HomeImageMode) == "自定义":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "自定义"
|
|
||||||
):
|
|
||||||
|
|
||||||
file_path, _ = QFileDialog.getOpenFileName(
|
file_path, _ = QFileDialog.getOpenFileName(
|
||||||
self, "打开自定义主页图片", "", "图片文件 (*.png *.jpg *.bmp)"
|
self, "打开自定义主页图片", "", "图片文件 (*.png *.jpg *.bmp)"
|
||||||
@@ -202,29 +196,26 @@ class Home(QWidget):
|
|||||||
"未选择图片文件!",
|
"未选择图片文件!",
|
||||||
5000,
|
5000,
|
||||||
)
|
)
|
||||||
elif (
|
elif Config.get(Config.function_HomeImageMode) == "主题图像":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "主题图像"
|
|
||||||
):
|
|
||||||
|
|
||||||
# 从远程服务器获取最新主题图像
|
# 从远程服务器获取最新主题图像
|
||||||
for _ in range(3):
|
network = Network.add_task(
|
||||||
try:
|
mode="get",
|
||||||
response = requests.get(
|
url="https://gitee.com/DLmaster_361/AUTO_MAA/raw/server/theme_image.json",
|
||||||
"https://gitee.com/DLmaster_361/AUTO_MAA/raw/server/theme_image.json"
|
)
|
||||||
)
|
network.loop.exec()
|
||||||
theme_image = response.json()
|
network_result = Network.get_result(network)
|
||||||
break
|
if network_result["status_code"] == 200:
|
||||||
except Exception as e:
|
theme_image = network_result["response_json"]
|
||||||
err = e
|
|
||||||
time.sleep(0.1)
|
|
||||||
else:
|
else:
|
||||||
logger.error(f"获取最新主题图像时出错:\n{err}")
|
logger.warning(
|
||||||
|
f"获取最新主题图像时出错:{network_result['error_message']}"
|
||||||
|
)
|
||||||
MainInfoBar.push_info_bar(
|
MainInfoBar.push_info_bar(
|
||||||
"error",
|
"warning",
|
||||||
"主题图像获取失败",
|
"获取最新主题图像时出错",
|
||||||
f"获取最新主题图像信息时出错:\n{err}",
|
f"网络错误:{network_result['status_code']}",
|
||||||
-1,
|
5000,
|
||||||
)
|
)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@@ -244,19 +235,25 @@ class Home(QWidget):
|
|||||||
).exists() or (
|
).exists() or (
|
||||||
datetime.now()
|
datetime.now()
|
||||||
> datetime.strptime(theme_image["time"], "%Y-%m-%d %H:%M")
|
> datetime.strptime(theme_image["time"], "%Y-%m-%d %H:%M")
|
||||||
and datetime.strptime(theme_image["time"], "%Y-%m-%d %H:%M")
|
|
||||||
> time_local
|
> time_local
|
||||||
):
|
):
|
||||||
|
|
||||||
response = requests.get(theme_image["url"])
|
network = Network.add_task(
|
||||||
if response.status_code == 200:
|
mode="get_file",
|
||||||
|
url=theme_image["url"],
|
||||||
|
path=Config.app_path / "resources/images/Home/BannerTheme.jpg",
|
||||||
|
)
|
||||||
|
network.loop.exec()
|
||||||
|
network_result = Network.get_result(network)
|
||||||
|
|
||||||
with open(
|
if network_result["status_code"] == 200:
|
||||||
Config.app_path / "resources/images/Home/BannerTheme.jpg", "wb"
|
|
||||||
) as file:
|
|
||||||
file.write(response.content)
|
|
||||||
|
|
||||||
logger.info(f"主题图像「{theme_image["name"]}」下载成功")
|
with (Config.app_path / "resources/theme_image.json").open(
|
||||||
|
mode="w", encoding="utf-8"
|
||||||
|
) as f:
|
||||||
|
json.dump(theme_image, f, ensure_ascii=False, indent=4)
|
||||||
|
|
||||||
|
logger.success(f"主题图像「{theme_image["name"]}」下载成功")
|
||||||
MainInfoBar.push_info_bar(
|
MainInfoBar.push_info_bar(
|
||||||
"success",
|
"success",
|
||||||
"主题图像下载成功",
|
"主题图像下载成功",
|
||||||
@@ -266,18 +263,15 @@ class Home(QWidget):
|
|||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
||||||
logger.error("主题图像下载失败")
|
logger.warning(
|
||||||
MainInfoBar.push_info_bar(
|
f"下载最新主题图像时出错:{network_result['error_message']}"
|
||||||
"error",
|
)
|
||||||
"主题图像下载失败",
|
MainInfoBar.push_info_bar(
|
||||||
f"主题图像下载失败:{response.status_code}",
|
"warning",
|
||||||
-1,
|
"下载最新主题图像时出错",
|
||||||
|
f"网络错误:{network_result['status_code']}",
|
||||||
|
5000,
|
||||||
)
|
)
|
||||||
|
|
||||||
with (Config.app_path / "resources/theme_image.json").open(
|
|
||||||
mode="w", encoding="utf-8"
|
|
||||||
) as f:
|
|
||||||
json.dump(theme_image, f, ensure_ascii=False, indent=4)
|
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
||||||
@@ -293,28 +287,19 @@ class Home(QWidget):
|
|||||||
|
|
||||||
def set_banner(self):
|
def set_banner(self):
|
||||||
"""设置主页图像"""
|
"""设置主页图像"""
|
||||||
if (
|
if Config.get(Config.function_HomeImageMode) == "默认":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "默认"
|
|
||||||
):
|
|
||||||
self.banner.set_banner_image(
|
self.banner.set_banner_image(
|
||||||
str(Config.app_path / "resources/images/Home/BannerDefault.png")
|
str(Config.app_path / "resources/images/Home/BannerDefault.png")
|
||||||
)
|
)
|
||||||
self.imageButton.hide()
|
self.imageButton.hide()
|
||||||
self.banner_text.setVisible(False)
|
self.banner_text.setVisible(False)
|
||||||
elif (
|
elif Config.get(Config.function_HomeImageMode) == "自定义":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "自定义"
|
|
||||||
):
|
|
||||||
for file in Config.app_path.glob("resources/images/Home/BannerCustomize.*"):
|
for file in Config.app_path.glob("resources/images/Home/BannerCustomize.*"):
|
||||||
self.banner.set_banner_image(str(file))
|
self.banner.set_banner_image(str(file))
|
||||||
break
|
break
|
||||||
self.imageButton.show()
|
self.imageButton.show()
|
||||||
self.banner_text.setVisible(False)
|
self.banner_text.setVisible(False)
|
||||||
elif (
|
elif Config.get(Config.function_HomeImageMode) == "主题图像":
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "主题图像"
|
|
||||||
):
|
|
||||||
self.banner.set_banner_image(
|
self.banner.set_banner_image(
|
||||||
str(Config.app_path / "resources/images/Home/BannerTheme.jpg")
|
str(Config.app_path / "resources/images/Home/BannerTheme.jpg")
|
||||||
)
|
)
|
||||||
@@ -388,11 +373,11 @@ class ButtonGroup(SimpleCardWidget):
|
|||||||
doc_button.clicked.connect(self.open_chat)
|
doc_button.clicked.connect(self.open_chat)
|
||||||
layout.addWidget(doc_button)
|
layout.addWidget(doc_button)
|
||||||
|
|
||||||
# 创建 官方店铺 按钮 (当然没有)
|
# 创建 MirrorChyan 按钮
|
||||||
doc_button = IconButton(
|
doc_button = IconButton(
|
||||||
FluentIcon.SHOPPING_CART.icon(color=QColor("#fff")),
|
FluentIcon.SHOPPING_CART.icon(color=QColor("#fff")),
|
||||||
tip_title="官方店铺",
|
tip_title="非官方店铺",
|
||||||
tip_content="暂时没有官方店铺,但是可以加入官方群聊哦~",
|
tip_content="获取 MirrorChyan CDK,更新快人一步",
|
||||||
isTooltip=True,
|
isTooltip=True,
|
||||||
)
|
)
|
||||||
doc_button.setIconSize(QSize(32, 32))
|
doc_button.setIconSize(QSize(32, 32))
|
||||||
@@ -419,5 +404,7 @@ class ButtonGroup(SimpleCardWidget):
|
|||||||
QDesktopServices.openUrl(QUrl("https://clozya.github.io/AUTOMAA_docs"))
|
QDesktopServices.openUrl(QUrl("https://clozya.github.io/AUTOMAA_docs"))
|
||||||
|
|
||||||
def open_sales(self):
|
def open_sales(self):
|
||||||
"""其实还是打开 Q群 链接"""
|
"""打开 MirrorChyan 链接"""
|
||||||
QDesktopServices.openUrl(QUrl("https://qm.qq.com/q/bd9fISNoME"))
|
QDesktopServices.openUrl(
|
||||||
|
QUrl("https://mirrorchyan.com/zh/get-start?source=auto_maa-home")
|
||||||
|
)
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,43 +16,40 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA主界面
|
AUTO_MAA主界面
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtWidgets import QSystemTrayIcon
|
from PySide6.QtWidgets import QApplication, QSystemTrayIcon
|
||||||
from qfluentwidgets import (
|
from qfluentwidgets import (
|
||||||
Action,
|
Action,
|
||||||
PushButton,
|
|
||||||
SystemTrayMenu,
|
SystemTrayMenu,
|
||||||
SplashScreen,
|
SplashScreen,
|
||||||
FluentIcon,
|
FluentIcon,
|
||||||
InfoBar,
|
|
||||||
InfoBarPosition,
|
|
||||||
setTheme,
|
setTheme,
|
||||||
isDarkTheme,
|
isDarkTheme,
|
||||||
SystemThemeListener,
|
SystemThemeListener,
|
||||||
Theme,
|
Theme,
|
||||||
MSFluentWindow,
|
MSFluentWindow,
|
||||||
NavigationItemPosition,
|
NavigationItemPosition,
|
||||||
qconfig,
|
|
||||||
)
|
)
|
||||||
from PySide6.QtGui import QIcon, QCloseEvent
|
from PySide6.QtGui import QIcon, QCloseEvent
|
||||||
from PySide6.QtCore import Qt, QTimer
|
from PySide6.QtCore import QTimer
|
||||||
import json
|
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
import shutil
|
import shutil
|
||||||
|
import darkdetect
|
||||||
|
|
||||||
from app.core import Config, TaskManager, MainTimer, MainInfoBar
|
from app.core import Config, TaskManager, MainTimer, MainInfoBar, SoundPlayer
|
||||||
from app.services import Notify, Crypto, System
|
from app.services import Notify, Crypto, System
|
||||||
from .home import Home
|
from .home import Home
|
||||||
from .member_manager import MemberManager
|
from .member_manager import MemberManager
|
||||||
|
from .plan_manager import PlanManager
|
||||||
from .queue_manager import QueueManager
|
from .queue_manager import QueueManager
|
||||||
from .dispatch_center import DispatchCenter
|
from .dispatch_center import DispatchCenter
|
||||||
from .history import History
|
from .history import History
|
||||||
@@ -65,18 +62,26 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
super().__init__()
|
super().__init__()
|
||||||
|
|
||||||
self.setWindowIcon(QIcon(str(Config.app_path / "resources/icons/AUTO_MAA.ico")))
|
self.setWindowIcon(QIcon(str(Config.app_path / "resources/icons/AUTO_MAA.ico")))
|
||||||
self.setWindowTitle("AUTO_MAA")
|
|
||||||
|
|
||||||
setTheme(Theme.AUTO, lazy=True)
|
version_numb = list(map(int, Config.VERSION.split(".")))
|
||||||
|
version_text = (
|
||||||
|
f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
||||||
|
if version_numb[3] == 0
|
||||||
|
else f"v{'.'.join(str(_) for _ in version_numb[0:3])}-beta.{version_numb[3]}"
|
||||||
|
)
|
||||||
|
|
||||||
|
self.setWindowTitle(f"AUTO_MAA - {version_text}")
|
||||||
|
|
||||||
|
self.switch_theme()
|
||||||
|
|
||||||
self.splashScreen = SplashScreen(self.windowIcon(), self)
|
self.splashScreen = SplashScreen(self.windowIcon(), self)
|
||||||
self.show_ui("显示主窗口", if_quick=True)
|
self.show_ui("显示主窗口", if_quick=True)
|
||||||
|
|
||||||
MainInfoBar.main_window = self.window()
|
Config.main_window = self.window()
|
||||||
System.main_window = self.window()
|
|
||||||
|
|
||||||
# 创建主窗口
|
# 创建主窗口
|
||||||
self.home = Home(self)
|
self.home = Home(self)
|
||||||
|
self.plan_manager = PlanManager(self)
|
||||||
self.member_manager = MemberManager(self)
|
self.member_manager = MemberManager(self)
|
||||||
self.queue_manager = QueueManager(self)
|
self.queue_manager = QueueManager(self)
|
||||||
self.dispatch_center = DispatchCenter(self)
|
self.dispatch_center = DispatchCenter(self)
|
||||||
@@ -97,6 +102,13 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
FluentIcon.ROBOT,
|
FluentIcon.ROBOT,
|
||||||
NavigationItemPosition.TOP,
|
NavigationItemPosition.TOP,
|
||||||
)
|
)
|
||||||
|
self.addSubInterface(
|
||||||
|
self.plan_manager,
|
||||||
|
FluentIcon.CALENDAR,
|
||||||
|
"计划管理",
|
||||||
|
FluentIcon.CALENDAR,
|
||||||
|
NavigationItemPosition.TOP,
|
||||||
|
)
|
||||||
self.addSubInterface(
|
self.addSubInterface(
|
||||||
self.queue_manager,
|
self.queue_manager,
|
||||||
FluentIcon.BOOK_SHELF,
|
FluentIcon.BOOK_SHELF,
|
||||||
@@ -125,27 +137,7 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
FluentIcon.SETTING,
|
FluentIcon.SETTING,
|
||||||
NavigationItemPosition.BOTTOM,
|
NavigationItemPosition.BOTTOM,
|
||||||
)
|
)
|
||||||
self.stackedWidget.currentChanged.connect(
|
self.stackedWidget.currentChanged.connect(self.__currentChanged)
|
||||||
lambda index: (self.member_manager.refresh() if index == 1 else None)
|
|
||||||
)
|
|
||||||
self.stackedWidget.currentChanged.connect(
|
|
||||||
lambda index: self.queue_manager.refresh() if index == 2 else None
|
|
||||||
)
|
|
||||||
self.stackedWidget.currentChanged.connect(
|
|
||||||
lambda index: (
|
|
||||||
self.dispatch_center.pivot.setCurrentItem("主调度台")
|
|
||||||
if index == 3
|
|
||||||
else None
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.stackedWidget.currentChanged.connect(
|
|
||||||
lambda index: (
|
|
||||||
self.dispatch_center.update_top_bar() if index == 3 else None
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.stackedWidget.currentChanged.connect(
|
|
||||||
lambda index: (self.history.refresh() if index == 4 else None)
|
|
||||||
)
|
|
||||||
|
|
||||||
# 创建系统托盘及其菜单
|
# 创建系统托盘及其菜单
|
||||||
self.tray = QSystemTrayIcon(
|
self.tray = QSystemTrayIcon(
|
||||||
@@ -179,13 +171,21 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
|
|
||||||
# 退出主程序菜单项
|
# 退出主程序菜单项
|
||||||
self.tray_menu.addAction(
|
self.tray_menu.addAction(
|
||||||
Action(FluentIcon.POWER_BUTTON, "退出主程序", triggered=self.window().close)
|
Action(
|
||||||
|
FluentIcon.POWER_BUTTON,
|
||||||
|
"退出主程序",
|
||||||
|
triggered=lambda: System.set_power("KillSelf"),
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
# 设置托盘菜单
|
# 设置托盘菜单
|
||||||
self.tray.setContextMenu(self.tray_menu)
|
self.tray.setContextMenu(self.tray_menu)
|
||||||
self.tray.activated.connect(self.on_tray_activated)
|
self.tray.activated.connect(self.on_tray_activated)
|
||||||
|
|
||||||
|
self.set_min_method()
|
||||||
|
|
||||||
|
Config.user_info_changed.connect(self.member_manager.refresh_dashboard)
|
||||||
|
Config.power_sign_changed.connect(self.dispatch_center.update_power_sign)
|
||||||
TaskManager.create_gui.connect(self.dispatch_center.add_board)
|
TaskManager.create_gui.connect(self.dispatch_center.add_board)
|
||||||
TaskManager.connect_gui.connect(self.dispatch_center.connect_main_board)
|
TaskManager.connect_gui.connect(self.dispatch_center.connect_main_board)
|
||||||
Notify.push_info_bar.connect(MainInfoBar.push_info_bar)
|
Notify.push_info_bar.connect(MainInfoBar.push_info_bar)
|
||||||
@@ -205,77 +205,39 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
self.themeListener.systemThemeChanged.connect(self.switch_theme)
|
self.themeListener.systemThemeChanged.connect(self.switch_theme)
|
||||||
self.themeListener.start()
|
self.themeListener.start()
|
||||||
|
|
||||||
def switch_theme(self):
|
def switch_theme(self) -> None:
|
||||||
"""切换主题"""
|
"""切换主题"""
|
||||||
|
|
||||||
setTheme(Theme.AUTO, lazy=True)
|
setTheme(
|
||||||
QTimer.singleShot(100, lambda: setTheme(Theme.AUTO, lazy=True))
|
Theme(darkdetect.theme()) if darkdetect.theme() else Theme.LIGHT, lazy=True
|
||||||
|
)
|
||||||
|
QTimer.singleShot(300, lambda: setTheme(Theme.AUTO, lazy=True))
|
||||||
|
|
||||||
# 云母特效启用时需要增加重试机制
|
# 云母特效启用时需要增加重试机制
|
||||||
|
# 云母特效不兼容Win10,如果True则通过云母进行主题转换,False则根据当前主题设置背景颜色
|
||||||
if self.isMicaEffectEnabled():
|
if self.isMicaEffectEnabled():
|
||||||
QTimer.singleShot(
|
QTimer.singleShot(
|
||||||
100,
|
300,
|
||||||
lambda: self.windowEffect.setMicaEffect(self.winId(), isDarkTheme()),
|
lambda: self.windowEffect.setMicaEffect(self.winId(), isDarkTheme()),
|
||||||
)
|
)
|
||||||
|
|
||||||
def start_up_task(self) -> None:
|
else:
|
||||||
"""启动时任务"""
|
# 根据当前主题设置背景颜色
|
||||||
|
if isDarkTheme():
|
||||||
# 加载配置
|
self.setStyleSheet(
|
||||||
qconfig.load(Config.config_path, Config.global_config)
|
"""
|
||||||
Config.global_config.save()
|
CardWidget {background-color: #313131;}
|
||||||
|
HeaderCardWidget {background-color: #313131;}
|
||||||
# 清理旧日志
|
background-color: #313131;
|
||||||
self.clean_old_logs()
|
"""
|
||||||
|
|
||||||
# 检查密码
|
|
||||||
self.setting.check_PASSWORD()
|
|
||||||
|
|
||||||
# 获取主题图像
|
|
||||||
if (
|
|
||||||
Config.global_config.get(Config.global_config.function_HomeImageMode)
|
|
||||||
== "主题图像"
|
|
||||||
):
|
|
||||||
self.home.get_home_image()
|
|
||||||
|
|
||||||
# 获取公告
|
|
||||||
self.setting.show_notice(if_show=False)
|
|
||||||
|
|
||||||
# 检查更新
|
|
||||||
if Config.global_config.get(Config.global_config.update_IfAutoUpdate):
|
|
||||||
result = self.setting.get_update_info()
|
|
||||||
if result == "已是最新版本~":
|
|
||||||
MainInfoBar.push_info_bar("success", "更新检查", result, 3000)
|
|
||||||
else:
|
|
||||||
info = InfoBar.info(
|
|
||||||
title="更新检查",
|
|
||||||
content=result,
|
|
||||||
orient=Qt.Horizontal,
|
|
||||||
isClosable=True,
|
|
||||||
position=InfoBarPosition.BOTTOM_LEFT,
|
|
||||||
duration=-1,
|
|
||||||
parent=self,
|
|
||||||
)
|
)
|
||||||
Up = PushButton("更新")
|
else:
|
||||||
Up.clicked.connect(lambda: self.setting.get_update(if_question=False))
|
self.setStyleSheet("background-color: #ffffff;")
|
||||||
Up.clicked.connect(info.close)
|
|
||||||
info.addWidget(Up)
|
|
||||||
info.show()
|
|
||||||
|
|
||||||
# 直接运行主任务
|
|
||||||
if Config.global_config.get(Config.global_config.start_IfRunDirectly):
|
|
||||||
|
|
||||||
self.start_main_task()
|
|
||||||
|
|
||||||
# 直接最小化
|
|
||||||
if Config.global_config.get(Config.global_config.start_IfMinimizeDirectly):
|
|
||||||
|
|
||||||
self.titleBar.minBtn.click()
|
|
||||||
|
|
||||||
def set_min_method(self) -> None:
|
def set_min_method(self) -> None:
|
||||||
"""设置最小化方法"""
|
"""设置最小化方法"""
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.ui_IfToTray):
|
if Config.get(Config.ui_IfToTray):
|
||||||
|
|
||||||
self.titleBar.minBtn.clicked.disconnect()
|
self.titleBar.minBtn.clicked.disconnect()
|
||||||
self.titleBar.minBtn.clicked.connect(lambda: self.show_ui("隐藏到托盘"))
|
self.titleBar.minBtn.clicked.connect(lambda: self.show_ui("隐藏到托盘"))
|
||||||
@@ -290,15 +252,177 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
if reason == QSystemTrayIcon.DoubleClick:
|
if reason == QSystemTrayIcon.DoubleClick:
|
||||||
self.show_ui("显示主窗口")
|
self.show_ui("显示主窗口")
|
||||||
|
|
||||||
|
def show_ui(
|
||||||
|
self, mode: str, if_quick: bool = False, if_start: bool = False
|
||||||
|
) -> None:
|
||||||
|
"""配置窗口状态"""
|
||||||
|
|
||||||
|
if Config.args.mode != "gui":
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.switch_theme()
|
||||||
|
|
||||||
|
if mode == "显示主窗口":
|
||||||
|
|
||||||
|
# 配置主窗口
|
||||||
|
if not self.window().isVisible():
|
||||||
|
size = list(
|
||||||
|
map(
|
||||||
|
int,
|
||||||
|
Config.get(Config.ui_size).split("x"),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
location = list(
|
||||||
|
map(
|
||||||
|
int,
|
||||||
|
Config.get(Config.ui_location).split("x"),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if self.window().isMaximized():
|
||||||
|
self.window().showNormal()
|
||||||
|
self.window().setGeometry(location[0], location[1], size[0], size[1])
|
||||||
|
self.window().show()
|
||||||
|
if not if_quick:
|
||||||
|
if (
|
||||||
|
Config.get(Config.ui_maximized)
|
||||||
|
and not self.window().isMaximized()
|
||||||
|
):
|
||||||
|
self.titleBar.maxBtn.click()
|
||||||
|
SoundPlayer.play("欢迎回来")
|
||||||
|
self.show_ui("配置托盘")
|
||||||
|
elif if_start:
|
||||||
|
if Config.get(Config.ui_maximized) and not self.window().isMaximized():
|
||||||
|
self.titleBar.maxBtn.click()
|
||||||
|
self.show_ui("配置托盘")
|
||||||
|
|
||||||
|
# 如果窗口不在屏幕内,则重置窗口位置
|
||||||
|
if not any(
|
||||||
|
self.window().geometry().intersects(screen.availableGeometry())
|
||||||
|
for screen in QApplication.screens()
|
||||||
|
):
|
||||||
|
self.window().showNormal()
|
||||||
|
self.window().setGeometry(100, 100, 1200, 700)
|
||||||
|
|
||||||
|
self.window().raise_()
|
||||||
|
self.window().activateWindow()
|
||||||
|
|
||||||
|
while Config.info_bar_list:
|
||||||
|
info_bar_item = Config.info_bar_list.pop(0)
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
info_bar_item["mode"],
|
||||||
|
info_bar_item["title"],
|
||||||
|
info_bar_item["content"],
|
||||||
|
info_bar_item["time"],
|
||||||
|
)
|
||||||
|
|
||||||
|
elif mode == "配置托盘":
|
||||||
|
|
||||||
|
if Config.get(Config.ui_IfShowTray):
|
||||||
|
self.tray.show()
|
||||||
|
else:
|
||||||
|
self.tray.hide()
|
||||||
|
|
||||||
|
elif mode == "隐藏到托盘":
|
||||||
|
|
||||||
|
# 保存窗口相关属性
|
||||||
|
if not self.window().isMaximized():
|
||||||
|
|
||||||
|
Config.set(
|
||||||
|
Config.ui_size,
|
||||||
|
f"{self.geometry().width()}x{self.geometry().height()}",
|
||||||
|
)
|
||||||
|
Config.set(
|
||||||
|
Config.ui_location,
|
||||||
|
f"{self.geometry().x()}x{self.geometry().y()}",
|
||||||
|
)
|
||||||
|
|
||||||
|
Config.set(Config.ui_maximized, self.window().isMaximized())
|
||||||
|
Config.save()
|
||||||
|
|
||||||
|
# 隐藏主窗口
|
||||||
|
if not if_quick:
|
||||||
|
|
||||||
|
self.window().hide()
|
||||||
|
self.tray.show()
|
||||||
|
|
||||||
|
def start_up_task(self) -> None:
|
||||||
|
"""启动时任务"""
|
||||||
|
|
||||||
|
# 清理旧日志
|
||||||
|
self.clean_old_logs()
|
||||||
|
|
||||||
|
# 清理安装包
|
||||||
|
if (Config.app_path / "AUTO_MAA-Setup.exe").exists():
|
||||||
|
try:
|
||||||
|
(Config.app_path / "AUTO_MAA-Setup.exe").unlink()
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
# 检查密码
|
||||||
|
self.setting.check_PASSWORD()
|
||||||
|
|
||||||
|
# 获取主题图像
|
||||||
|
if Config.get(Config.function_HomeImageMode) == "主题图像":
|
||||||
|
self.home.get_home_image()
|
||||||
|
|
||||||
|
# 直接运行主任务
|
||||||
|
if Config.get(Config.start_IfRunDirectly):
|
||||||
|
|
||||||
|
self.start_main_task()
|
||||||
|
|
||||||
|
# 获取公告
|
||||||
|
self.setting.show_notice(if_first=True)
|
||||||
|
|
||||||
|
# 检查更新
|
||||||
|
if Config.get(Config.update_IfAutoUpdate):
|
||||||
|
self.setting.check_update(if_first=True)
|
||||||
|
|
||||||
|
# 直接最小化
|
||||||
|
if Config.get(Config.start_IfMinimizeDirectly):
|
||||||
|
|
||||||
|
self.titleBar.minBtn.click()
|
||||||
|
|
||||||
|
if Config.args.config:
|
||||||
|
|
||||||
|
for config in [_ for _ in Config.args.config if _ in Config.queue_dict]:
|
||||||
|
|
||||||
|
TaskManager.add_task(
|
||||||
|
"自动代理_新调度台",
|
||||||
|
config,
|
||||||
|
Config.queue_dict["调度队列_1"]["Config"].toDict(),
|
||||||
|
)
|
||||||
|
|
||||||
|
for config in [_ for _ in Config.args.config if _ in Config.member_dict]:
|
||||||
|
|
||||||
|
TaskManager.add_task(
|
||||||
|
"自动代理_新调度台",
|
||||||
|
"自定义队列",
|
||||||
|
{"Queue": {"Member_1": config}},
|
||||||
|
)
|
||||||
|
|
||||||
|
if not any(
|
||||||
|
_ in (list(Config.member_dict.keys()) + list(Config.queue_dict.keys()))
|
||||||
|
for _ in Config.args.config
|
||||||
|
):
|
||||||
|
|
||||||
|
logger.warning(
|
||||||
|
"当前运行模式为命令行模式,由于您使用了错误的 --config 参数进行配置,程序自动退出"
|
||||||
|
)
|
||||||
|
System.set_power("KillSelf")
|
||||||
|
|
||||||
|
elif Config.args.mode == "cli":
|
||||||
|
|
||||||
|
logger.warning(
|
||||||
|
"当前运行模式为命令行模式,由于您未使用 --config 参数进行配置,程序自动退出"
|
||||||
|
)
|
||||||
|
System.set_power("KillSelf")
|
||||||
|
|
||||||
def clean_old_logs(self):
|
def clean_old_logs(self):
|
||||||
"""
|
"""
|
||||||
删除超过用户设定天数的日志文件(基于目录日期)
|
删除超过用户设定天数的日志文件(基于目录日期)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if (
|
if Config.get(Config.function_HistoryRetentionTime) == 0:
|
||||||
Config.global_config.get(Config.global_config.function_HistoryRetentionTime)
|
|
||||||
== 0
|
|
||||||
):
|
|
||||||
logger.info("由于用户设置日志永久保留,跳过日志清理")
|
logger.info("由于用户设置日志永久保留,跳过日志清理")
|
||||||
return
|
return
|
||||||
|
|
||||||
@@ -312,9 +436,7 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
# 只检查 `YYYY-MM-DD` 格式的文件夹
|
# 只检查 `YYYY-MM-DD` 格式的文件夹
|
||||||
folder_date = datetime.strptime(date_folder.name, "%Y-%m-%d")
|
folder_date = datetime.strptime(date_folder.name, "%Y-%m-%d")
|
||||||
if datetime.now() - folder_date > timedelta(
|
if datetime.now() - folder_date > timedelta(
|
||||||
days=Config.global_config.get(
|
days=Config.get(Config.function_HistoryRetentionTime)
|
||||||
Config.global_config.function_HistoryRetentionTime
|
|
||||||
)
|
|
||||||
):
|
):
|
||||||
shutil.rmtree(date_folder, ignore_errors=True)
|
shutil.rmtree(date_folder, ignore_errors=True)
|
||||||
deleted_count += 1
|
deleted_count += 1
|
||||||
@@ -327,90 +449,39 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
def start_main_task(self) -> None:
|
def start_main_task(self) -> None:
|
||||||
"""启动主任务"""
|
"""启动主任务"""
|
||||||
|
|
||||||
if (Config.app_path / "config/QueueConfig/调度队列_1.json").exists():
|
if "调度队列_1" in Config.queue_dict:
|
||||||
|
|
||||||
with (Config.app_path / "config/QueueConfig/调度队列_1.json").open(
|
|
||||||
mode="r", encoding="utf-8"
|
|
||||||
) as f:
|
|
||||||
info = json.load(f)
|
|
||||||
|
|
||||||
logger.info("自动添加任务:调度队列_1")
|
logger.info("自动添加任务:调度队列_1")
|
||||||
TaskManager.add_task("自动代理_主调度台", "主任务队列", info)
|
TaskManager.add_task(
|
||||||
|
"自动代理_主调度台",
|
||||||
|
"调度队列_1",
|
||||||
|
Config.queue_dict["调度队列_1"]["Config"].toDict(),
|
||||||
|
)
|
||||||
|
|
||||||
elif (Config.app_path / "config/MaaConfig/脚本_1").exists():
|
elif "脚本_1" in Config.member_dict:
|
||||||
|
|
||||||
info = {"Queue": {"Member_1": "脚本_1"}}
|
|
||||||
|
|
||||||
logger.info("自动添加任务:脚本_1")
|
logger.info("自动添加任务:脚本_1")
|
||||||
TaskManager.add_task("自动代理_主调度台", "主任务队列", info)
|
TaskManager.add_task(
|
||||||
|
"自动代理_主调度台", "自定义队列", {"Queue": {"Member_1": "脚本_1"}}
|
||||||
|
)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
||||||
logger.worning("启动主任务失败:未找到有效的主任务配置文件")
|
logger.warning("启动主任务失败:未找到有效的主任务配置文件")
|
||||||
MainInfoBar.push_info_bar(
|
MainInfoBar.push_info_bar(
|
||||||
"warning", "启动主任务失败", "“调度队列_1”与“脚本_1”均不存在", -1
|
"warning", "启动主任务失败", "“调度队列_1”与“脚本_1”均不存在", -1
|
||||||
)
|
)
|
||||||
|
|
||||||
def show_ui(self, mode: str, if_quick: bool = False) -> None:
|
def __currentChanged(self, index: int) -> None:
|
||||||
"""配置窗口状态"""
|
"""切换界面时任务"""
|
||||||
|
|
||||||
if mode == "显示主窗口":
|
if index == 1:
|
||||||
|
self.member_manager.reload_plan_name()
|
||||||
# 配置主窗口
|
elif index == 3:
|
||||||
size = list(
|
self.queue_manager.reload_member_name()
|
||||||
map(
|
elif index == 4:
|
||||||
int,
|
self.dispatch_center.pivot.setCurrentItem("主调度台")
|
||||||
Config.global_config.get(Config.global_config.ui_size).split("x"),
|
self.dispatch_center.update_top_bar()
|
||||||
)
|
|
||||||
)
|
|
||||||
location = list(
|
|
||||||
map(
|
|
||||||
int,
|
|
||||||
Config.global_config.get(Config.global_config.ui_location).split(
|
|
||||||
"x"
|
|
||||||
),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.window().setGeometry(location[0], location[1], size[0], size[1])
|
|
||||||
self.window().show()
|
|
||||||
self.window().raise_()
|
|
||||||
self.window().activateWindow()
|
|
||||||
if not if_quick:
|
|
||||||
if Config.global_config.get(Config.global_config.ui_maximized):
|
|
||||||
self.window().showMaximized()
|
|
||||||
self.set_min_method()
|
|
||||||
self.show_ui("配置托盘")
|
|
||||||
|
|
||||||
elif mode == "配置托盘":
|
|
||||||
|
|
||||||
if Config.global_config.get(Config.global_config.ui_IfShowTray):
|
|
||||||
self.tray.show()
|
|
||||||
else:
|
|
||||||
self.tray.hide()
|
|
||||||
|
|
||||||
elif mode == "隐藏到托盘":
|
|
||||||
|
|
||||||
# 保存窗口相关属性
|
|
||||||
if not self.window().isMaximized():
|
|
||||||
|
|
||||||
Config.global_config.set(
|
|
||||||
Config.global_config.ui_size,
|
|
||||||
f"{self.geometry().width()}x{self.geometry().height()}",
|
|
||||||
)
|
|
||||||
Config.global_config.set(
|
|
||||||
Config.global_config.ui_location,
|
|
||||||
f"{self.geometry().x()}x{self.geometry().y()}",
|
|
||||||
)
|
|
||||||
Config.global_config.set(
|
|
||||||
Config.global_config.ui_maximized, self.window().isMaximized()
|
|
||||||
)
|
|
||||||
Config.global_config.save()
|
|
||||||
|
|
||||||
# 隐藏主窗口
|
|
||||||
if not if_quick:
|
|
||||||
|
|
||||||
self.window().hide()
|
|
||||||
self.tray.show()
|
|
||||||
|
|
||||||
def closeEvent(self, event: QCloseEvent):
|
def closeEvent(self, event: QCloseEvent):
|
||||||
"""清理残余进程"""
|
"""清理残余进程"""
|
||||||
@@ -420,11 +491,10 @@ class AUTO_MAA(MSFluentWindow):
|
|||||||
# 清理各功能线程
|
# 清理各功能线程
|
||||||
MainTimer.Timer.stop()
|
MainTimer.Timer.stop()
|
||||||
MainTimer.Timer.deleteLater()
|
MainTimer.Timer.deleteLater()
|
||||||
|
MainTimer.LongTimer.stop()
|
||||||
|
MainTimer.LongTimer.deleteLater()
|
||||||
TaskManager.stop_task("ALL")
|
TaskManager.stop_task("ALL")
|
||||||
|
|
||||||
# 关闭数据库连接
|
|
||||||
Config.close_database()
|
|
||||||
|
|
||||||
# 关闭主题监听
|
# 关闭主题监听
|
||||||
self.themeListener.terminate()
|
self.themeListener.terminate()
|
||||||
self.themeListener.deleteLater()
|
self.themeListener.deleteLater()
|
||||||
|
|||||||
497
app/ui/plan_manager.py
Normal file
@@ -0,0 +1,497 @@
|
|||||||
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
|
# AUTO_MAA is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published
|
||||||
|
# by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
|
||||||
|
# AUTO_MAA is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
|
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
||||||
|
# the GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
|
"""
|
||||||
|
AUTO_MAA
|
||||||
|
AUTO_MAA计划管理界面
|
||||||
|
v4.3
|
||||||
|
作者:DLmaster_361
|
||||||
|
"""
|
||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
from PySide6.QtWidgets import (
|
||||||
|
QWidget,
|
||||||
|
QVBoxLayout,
|
||||||
|
QStackedWidget,
|
||||||
|
QHeaderView,
|
||||||
|
)
|
||||||
|
from qfluentwidgets import (
|
||||||
|
Action,
|
||||||
|
FluentIcon,
|
||||||
|
MessageBox,
|
||||||
|
HeaderCardWidget,
|
||||||
|
CommandBar,
|
||||||
|
TableWidget,
|
||||||
|
)
|
||||||
|
from typing import List, Dict, Union
|
||||||
|
import shutil
|
||||||
|
|
||||||
|
from app.core import Config, MainInfoBar, MaaPlanConfig, SoundPlayer
|
||||||
|
from .Widget import (
|
||||||
|
ComboBoxMessageBox,
|
||||||
|
LineEditSettingCard,
|
||||||
|
ComboBoxSettingCard,
|
||||||
|
SpinBoxSetting,
|
||||||
|
EditableComboBoxSetting,
|
||||||
|
ComboBoxSetting,
|
||||||
|
PivotArea,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class PlanManager(QWidget):
|
||||||
|
"""计划管理父界面"""
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
|
||||||
|
self.setObjectName("计划管理")
|
||||||
|
|
||||||
|
layout = QVBoxLayout(self)
|
||||||
|
|
||||||
|
self.tools = CommandBar()
|
||||||
|
|
||||||
|
self.plan_manager = self.PlanSettingBox(self)
|
||||||
|
|
||||||
|
# 逐个添加动作
|
||||||
|
self.tools.addActions(
|
||||||
|
[
|
||||||
|
Action(FluentIcon.ADD_TO, "新建计划表", triggered=self.add_setting_box),
|
||||||
|
Action(
|
||||||
|
FluentIcon.REMOVE_FROM, "删除计划表", triggered=self.del_setting_box
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
self.tools.addSeparator()
|
||||||
|
self.tools.addActions(
|
||||||
|
[
|
||||||
|
Action(
|
||||||
|
FluentIcon.LEFT_ARROW, "向左移动", triggered=self.left_setting_box
|
||||||
|
),
|
||||||
|
Action(
|
||||||
|
FluentIcon.RIGHT_ARROW, "向右移动", triggered=self.right_setting_box
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
self.tools.addSeparator()
|
||||||
|
|
||||||
|
layout.addWidget(self.tools)
|
||||||
|
layout.addWidget(self.plan_manager)
|
||||||
|
|
||||||
|
def add_setting_box(self):
|
||||||
|
"""添加一个计划表"""
|
||||||
|
|
||||||
|
choice = ComboBoxMessageBox(
|
||||||
|
self.window(),
|
||||||
|
"选择一个计划类型以添加相应计划表",
|
||||||
|
["选择计划类型"],
|
||||||
|
[["MAA"]],
|
||||||
|
)
|
||||||
|
if choice.exec() and choice.input[0].currentIndex() != -1:
|
||||||
|
|
||||||
|
if choice.input[0].currentText() == "MAA":
|
||||||
|
|
||||||
|
index = len(Config.plan_dict) + 1
|
||||||
|
|
||||||
|
maa_plan_config = MaaPlanConfig()
|
||||||
|
maa_plan_config.load(
|
||||||
|
Config.app_path / f"config/MaaPlanConfig/计划_{index}/config.json",
|
||||||
|
maa_plan_config,
|
||||||
|
)
|
||||||
|
maa_plan_config.save()
|
||||||
|
|
||||||
|
Config.plan_dict[f"计划_{index}"] = {
|
||||||
|
"Type": "Maa",
|
||||||
|
"Path": Config.app_path / f"config/MaaPlanConfig/计划_{index}",
|
||||||
|
"Config": maa_plan_config,
|
||||||
|
}
|
||||||
|
|
||||||
|
self.plan_manager.add_MaaPlanSettingBox(index)
|
||||||
|
self.plan_manager.switch_SettingBox(index)
|
||||||
|
|
||||||
|
logger.success(f"计划管理 计划_{index} 添加成功")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"success", "操作成功", f"添加计划表 计划_{index}", 3000
|
||||||
|
)
|
||||||
|
SoundPlayer.play("添加计划表")
|
||||||
|
|
||||||
|
def del_setting_box(self):
|
||||||
|
"""删除一个计划表"""
|
||||||
|
|
||||||
|
name = self.plan_manager.pivot.currentRouteKey()
|
||||||
|
|
||||||
|
if name is None:
|
||||||
|
logger.warning("删除计划表时未选择计划表")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "未选择计划表", "请选择一个计划表", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if len(Config.running_list) > 0:
|
||||||
|
logger.warning("删除计划表时调度队列未停止运行")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "调度中心正在执行任务", "请等待或手动中止任务", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
choice = MessageBox("确认", f"确定要删除 {name} 吗?", self.window())
|
||||||
|
if choice.exec():
|
||||||
|
|
||||||
|
self.plan_manager.clear_SettingBox()
|
||||||
|
|
||||||
|
shutil.rmtree(Config.plan_dict[name]["Path"])
|
||||||
|
Config.change_plan(name, "固定")
|
||||||
|
for i in range(int(name[3:]) + 1, len(Config.plan_dict) + 1):
|
||||||
|
if Config.plan_dict[f"计划_{i}"]["Path"].exists():
|
||||||
|
Config.plan_dict[f"计划_{i}"]["Path"].rename(
|
||||||
|
Config.plan_dict[f"计划_{i}"]["Path"].with_name(f"计划_{i-1}")
|
||||||
|
)
|
||||||
|
Config.change_plan(f"计划_{i}", f"计划_{i-1}")
|
||||||
|
|
||||||
|
self.plan_manager.show_SettingBox(max(int(name[3:]) - 1, 1))
|
||||||
|
|
||||||
|
logger.success(f"计划表 {name} 删除成功")
|
||||||
|
MainInfoBar.push_info_bar("success", "操作成功", f"删除计划表 {name}", 3000)
|
||||||
|
SoundPlayer.play("删除计划表")
|
||||||
|
|
||||||
|
def left_setting_box(self):
|
||||||
|
"""向左移动计划表"""
|
||||||
|
|
||||||
|
name = self.plan_manager.pivot.currentRouteKey()
|
||||||
|
|
||||||
|
if name is None:
|
||||||
|
logger.warning("向左移动计划表时未选择计划表")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "未选择计划表", "请选择一个计划表", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
index = int(name[3:])
|
||||||
|
|
||||||
|
if index == 1:
|
||||||
|
logger.warning("向左移动计划表时已到达最左端")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "已经是第一个计划表", "无法向左移动", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if len(Config.running_list) > 0:
|
||||||
|
logger.warning("向左移动计划表时调度队列未停止运行")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "调度中心正在执行任务", "请等待或手动中止任务", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.plan_manager.clear_SettingBox()
|
||||||
|
|
||||||
|
Config.plan_dict[name]["Path"].rename(
|
||||||
|
Config.plan_dict[name]["Path"].with_name("计划_0")
|
||||||
|
)
|
||||||
|
Config.change_plan(name, "计划_0")
|
||||||
|
Config.plan_dict[f"计划_{index-1}"]["Path"].rename(
|
||||||
|
Config.plan_dict[name]["Path"]
|
||||||
|
)
|
||||||
|
Config.change_plan(f"计划_{index-1}", name)
|
||||||
|
Config.plan_dict[name]["Path"].with_name("计划_0").rename(
|
||||||
|
Config.plan_dict[f"计划_{index-1}"]["Path"]
|
||||||
|
)
|
||||||
|
Config.change_plan("计划_0", f"计划_{index-1}")
|
||||||
|
|
||||||
|
self.plan_manager.show_SettingBox(index - 1)
|
||||||
|
|
||||||
|
logger.success(f"计划表 {name} 左移成功")
|
||||||
|
MainInfoBar.push_info_bar("success", "操作成功", f"左移计划表 {name}", 3000)
|
||||||
|
|
||||||
|
def right_setting_box(self):
|
||||||
|
"""向右移动计划表"""
|
||||||
|
|
||||||
|
name = self.plan_manager.pivot.currentRouteKey()
|
||||||
|
|
||||||
|
if name is None:
|
||||||
|
logger.warning("向右移动计划表时未选择计划表")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "未选择计划表", "请选择一个计划表", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
index = int(name[3:])
|
||||||
|
|
||||||
|
if index == len(Config.plan_dict):
|
||||||
|
logger.warning("向右移动计划表时已到达最右端")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "已经是最后一个计划表", "无法向右移动", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if len(Config.running_list) > 0:
|
||||||
|
logger.warning("向右移动计划表时调度队列未停止运行")
|
||||||
|
MainInfoBar.push_info_bar(
|
||||||
|
"warning", "调度中心正在执行任务", "请等待或手动中止任务", 5000
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.plan_manager.clear_SettingBox()
|
||||||
|
|
||||||
|
Config.plan_dict[name]["Path"].rename(
|
||||||
|
Config.plan_dict[name]["Path"].with_name("计划_0")
|
||||||
|
)
|
||||||
|
Config.change_plan(name, "计划_0")
|
||||||
|
Config.plan_dict[f"计划_{index+1}"]["Path"].rename(
|
||||||
|
Config.plan_dict[name]["Path"]
|
||||||
|
)
|
||||||
|
Config.change_plan(f"计划_{index+1}", name)
|
||||||
|
Config.plan_dict[name]["Path"].with_name("计划_0").rename(
|
||||||
|
Config.plan_dict[f"计划_{index+1}"]["Path"]
|
||||||
|
)
|
||||||
|
Config.change_plan("计划_0", f"计划_{index+1}")
|
||||||
|
|
||||||
|
self.plan_manager.show_SettingBox(index + 1)
|
||||||
|
|
||||||
|
logger.success(f"计划表 {name} 右移成功")
|
||||||
|
MainInfoBar.push_info_bar("success", "操作成功", f"右移计划表 {name}", 3000)
|
||||||
|
|
||||||
|
class PlanSettingBox(QWidget):
|
||||||
|
"""计划管理子页面组"""
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
|
||||||
|
self.setObjectName("计划管理页面组")
|
||||||
|
|
||||||
|
self.pivotArea = PivotArea(self)
|
||||||
|
self.pivot = self.pivotArea.pivot
|
||||||
|
|
||||||
|
self.stackedWidget = QStackedWidget(self)
|
||||||
|
self.stackedWidget.setContentsMargins(0, 0, 0, 0)
|
||||||
|
self.stackedWidget.setStyleSheet("background: transparent; border: none;")
|
||||||
|
|
||||||
|
self.script_list: List[PlanManager.PlanSettingBox.MaaPlanSettingBox] = []
|
||||||
|
|
||||||
|
self.Layout = QVBoxLayout(self)
|
||||||
|
self.Layout.addWidget(self.pivotArea)
|
||||||
|
self.Layout.addWidget(self.stackedWidget)
|
||||||
|
self.Layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
|
||||||
|
self.pivot.currentItemChanged.connect(
|
||||||
|
lambda index: self.switch_SettingBox(
|
||||||
|
int(index[3:]), if_chang_pivot=False
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.show_SettingBox(1)
|
||||||
|
|
||||||
|
def show_SettingBox(self, index) -> None:
|
||||||
|
"""加载所有子界面"""
|
||||||
|
|
||||||
|
Config.search_plan()
|
||||||
|
|
||||||
|
for name, info in Config.plan_dict.items():
|
||||||
|
if info["Type"] == "Maa":
|
||||||
|
self.add_MaaPlanSettingBox(int(name[3:]))
|
||||||
|
|
||||||
|
self.switch_SettingBox(index)
|
||||||
|
|
||||||
|
def switch_SettingBox(self, index: int, if_chang_pivot: bool = True) -> None:
|
||||||
|
"""切换到指定的子界面"""
|
||||||
|
|
||||||
|
if len(Config.plan_dict) == 0:
|
||||||
|
return None
|
||||||
|
|
||||||
|
if index > len(Config.plan_dict):
|
||||||
|
return None
|
||||||
|
|
||||||
|
if if_chang_pivot:
|
||||||
|
self.pivot.setCurrentItem(self.script_list[index - 1].objectName())
|
||||||
|
self.stackedWidget.setCurrentWidget(self.script_list[index - 1])
|
||||||
|
|
||||||
|
def clear_SettingBox(self) -> None:
|
||||||
|
"""清空所有子界面"""
|
||||||
|
|
||||||
|
for sub_interface in self.script_list:
|
||||||
|
Config.gameid_refreshed.disconnect(sub_interface.refresh_gameid)
|
||||||
|
self.stackedWidget.removeWidget(sub_interface)
|
||||||
|
sub_interface.deleteLater()
|
||||||
|
self.script_list.clear()
|
||||||
|
self.pivot.clear()
|
||||||
|
|
||||||
|
def add_MaaPlanSettingBox(self, uid: int) -> None:
|
||||||
|
"""添加一个MAA设置界面"""
|
||||||
|
|
||||||
|
maa_plan_setting_box = self.MaaPlanSettingBox(uid, self)
|
||||||
|
|
||||||
|
self.script_list.append(maa_plan_setting_box)
|
||||||
|
|
||||||
|
self.stackedWidget.addWidget(self.script_list[-1])
|
||||||
|
|
||||||
|
self.pivot.addItem(routeKey=f"计划_{uid}", text=f"计划 {uid}")
|
||||||
|
|
||||||
|
class MaaPlanSettingBox(HeaderCardWidget):
|
||||||
|
"""MAA类计划设置界面"""
|
||||||
|
|
||||||
|
def __init__(self, uid: int, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
|
||||||
|
self.setObjectName(f"计划_{uid}")
|
||||||
|
self.setTitle("MAA计划表")
|
||||||
|
self.config = Config.plan_dict[f"计划_{uid}"]["Config"]
|
||||||
|
|
||||||
|
self.card_Name = LineEditSettingCard(
|
||||||
|
icon=FluentIcon.EDIT,
|
||||||
|
title="计划表名称",
|
||||||
|
content="用于标识计划表的名称",
|
||||||
|
text="请输入计划表名称",
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=self.config.Info_Name,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
self.card_Mode = ComboBoxSettingCard(
|
||||||
|
icon=FluentIcon.DICTIONARY,
|
||||||
|
title="计划模式",
|
||||||
|
content="全局模式下计划内容固定,周计划模式下计划按周一到周日切换",
|
||||||
|
texts=["全局", "周计划"],
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=self.config.Info_Mode,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.table = TableWidget(self)
|
||||||
|
self.table.setColumnCount(8)
|
||||||
|
self.table.setRowCount(6)
|
||||||
|
self.table.setHorizontalHeaderLabels(
|
||||||
|
["全局", "周一", "周二", "周三", "周四", "周五", "周六", "周日"]
|
||||||
|
)
|
||||||
|
self.table.setVerticalHeaderLabels(
|
||||||
|
[
|
||||||
|
"吃理智药",
|
||||||
|
"连战次数",
|
||||||
|
"关卡选择",
|
||||||
|
"备选 - 1",
|
||||||
|
"备选 - 2",
|
||||||
|
"剩余理智",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
self.table.setAlternatingRowColors(False)
|
||||||
|
self.table.setEditTriggers(TableWidget.NoEditTriggers)
|
||||||
|
for col in range(8):
|
||||||
|
self.table.horizontalHeader().setSectionResizeMode(
|
||||||
|
col, QHeaderView.ResizeMode.Stretch
|
||||||
|
)
|
||||||
|
for row in range(6):
|
||||||
|
self.table.verticalHeader().setSectionResizeMode(
|
||||||
|
row, QHeaderView.ResizeMode.ResizeToContents
|
||||||
|
)
|
||||||
|
|
||||||
|
self.item_dict: Dict[
|
||||||
|
str,
|
||||||
|
Dict[
|
||||||
|
str,
|
||||||
|
Union[SpinBoxSetting, ComboBoxSetting, EditableComboBoxSetting],
|
||||||
|
],
|
||||||
|
] = {}
|
||||||
|
|
||||||
|
for col, (group, name_dict) in enumerate(
|
||||||
|
self.config.config_item_dict.items()
|
||||||
|
):
|
||||||
|
|
||||||
|
self.item_dict[group] = {}
|
||||||
|
|
||||||
|
for row, (name, configItem) in enumerate(name_dict.items()):
|
||||||
|
|
||||||
|
if name == "MedicineNumb":
|
||||||
|
self.item_dict[group][name] = SpinBoxSetting(
|
||||||
|
range=(0, 1024),
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=configItem,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
elif name == "SeriesNumb":
|
||||||
|
self.item_dict[group][name] = ComboBoxSetting(
|
||||||
|
texts=["AUTO", "6", "5", "4", "3", "2", "1", "不选择"],
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=configItem,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
elif name == "GameId_Remain":
|
||||||
|
self.item_dict[group][name] = EditableComboBoxSetting(
|
||||||
|
value=Config.gameid_dict[group]["value"],
|
||||||
|
texts=[
|
||||||
|
"不使用" if _ == "当前/上次" else _
|
||||||
|
for _ in Config.gameid_dict[group]["text"]
|
||||||
|
],
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=configItem,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
elif "GameId" in name:
|
||||||
|
self.item_dict[group][name] = EditableComboBoxSetting(
|
||||||
|
value=Config.gameid_dict[group]["value"],
|
||||||
|
texts=Config.gameid_dict[group]["text"],
|
||||||
|
qconfig=self.config,
|
||||||
|
configItem=configItem,
|
||||||
|
parent=self,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.table.setCellWidget(row, col, self.item_dict[group][name])
|
||||||
|
|
||||||
|
Layout = QVBoxLayout()
|
||||||
|
Layout.addWidget(self.card_Name)
|
||||||
|
Layout.addWidget(self.card_Mode)
|
||||||
|
Layout.addWidget(self.table)
|
||||||
|
|
||||||
|
self.viewLayout.addLayout(Layout)
|
||||||
|
self.viewLayout.setSpacing(3)
|
||||||
|
self.viewLayout.setContentsMargins(3, 0, 3, 3)
|
||||||
|
|
||||||
|
self.card_Mode.comboBox.currentIndexChanged.connect(self.switch_mode)
|
||||||
|
Config.gameid_refreshed.connect(self.refresh_gameid)
|
||||||
|
|
||||||
|
self.switch_mode()
|
||||||
|
|
||||||
|
def switch_mode(self) -> None:
|
||||||
|
"""切换计划模式"""
|
||||||
|
|
||||||
|
for group, name_dict in self.item_dict.items():
|
||||||
|
for name, setting_item in name_dict.items():
|
||||||
|
setting_item.setEnabled(
|
||||||
|
(group == "ALL")
|
||||||
|
== (self.config.get(self.config.Info_Mode) == "ALL")
|
||||||
|
)
|
||||||
|
|
||||||
|
def refresh_gameid(self):
|
||||||
|
|
||||||
|
for group, name_dict in self.item_dict.items():
|
||||||
|
|
||||||
|
for name, setting_item in name_dict.items():
|
||||||
|
|
||||||
|
if name == "GameId_Remain":
|
||||||
|
|
||||||
|
setting_item.reLoadOptions(
|
||||||
|
Config.gameid_dict[group]["value"],
|
||||||
|
[
|
||||||
|
"不使用" if _ == "当前/上次" else _
|
||||||
|
for _ in Config.gameid_dict[group]["text"]
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
elif "GameId" in name:
|
||||||
|
|
||||||
|
setting_item.reLoadOptions(
|
||||||
|
Config.gameid_dict[group]["value"],
|
||||||
|
Config.gameid_dict[group]["text"],
|
||||||
|
)
|
||||||
88
app/utils/AUTO_MAA.iss
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
; Script generated by the Inno Setup Script Wizard.
|
||||||
|
; SEE THE DOCUMENTATION FOR DETAILS ON CREATING INNO SETUP SCRIPT FILES!
|
||||||
|
|
||||||
|
#define MyAppName "AUTO_MAA"
|
||||||
|
#define MyAppVersion ""
|
||||||
|
#define MyAppPublisher "AUTO_MAA Team"
|
||||||
|
#define MyAppURL "https://doc.automaa.xyz/"
|
||||||
|
#define MyAppExeName "AUTO_MAA.exe"
|
||||||
|
#define MyAppPath ""
|
||||||
|
#define OutputDir ""
|
||||||
|
|
||||||
|
[Setup]
|
||||||
|
; NOTE: The value of AppId uniquely identifies this application. Do not use the same AppId value in installers for other applications.
|
||||||
|
; (To generate a new GUID, click Tools | Generate GUID inside the IDE.)
|
||||||
|
AppId={{D116A92A-E174-4699-B777-61C5FD837B19}
|
||||||
|
AppName={#MyAppName}
|
||||||
|
AppVersion={#MyAppVersion}
|
||||||
|
AppVerName={#MyAppName}
|
||||||
|
AppPublisher={#MyAppPublisher}
|
||||||
|
AppPublisherURL={#MyAppURL}
|
||||||
|
AppSupportURL={#MyAppURL}
|
||||||
|
AppUpdatesURL={#MyAppURL}
|
||||||
|
DefaultDirName=D:\{#MyAppName}
|
||||||
|
UninstallDisplayIcon={app}\{#MyAppExeName}
|
||||||
|
; "ArchitecturesAllowed=x64compatible" specifies that Setup cannot run
|
||||||
|
; on anything but x64 and Windows 11 on Arm.
|
||||||
|
ArchitecturesAllowed=x64compatible
|
||||||
|
; "ArchitecturesInstallIn64BitMode=x64compatible" requests that the
|
||||||
|
; install be done in "64-bit mode" on x64 or Windows 11 on Arm,
|
||||||
|
; meaning it should use the native 64-bit Program Files directory and
|
||||||
|
; the 64-bit view of the registry.
|
||||||
|
ArchitecturesInstallIn64BitMode=x64compatible
|
||||||
|
DisableProgramGroupPage=yes
|
||||||
|
LicenseFile={#MyAppPath}\LICENSE
|
||||||
|
; Remove the following line to run in administrative install mode (install for all users).
|
||||||
|
PrivilegesRequired=lowest
|
||||||
|
OutputDir={#OutputDir}
|
||||||
|
OutputBaseFilename=AUTO_MAA-Setup
|
||||||
|
SetupIconFile={#MyAppPath}\resources\icons\AUTO_MAA.ico
|
||||||
|
SolidCompression=yes
|
||||||
|
WizardStyle=modern
|
||||||
|
|
||||||
|
[Languages]
|
||||||
|
Name: "Chinese"; MessagesFile: "{#MyAppPath}\resources\docs\ChineseSimplified.isl"
|
||||||
|
Name: "English"; MessagesFile: "compiler:Default.isl"
|
||||||
|
|
||||||
|
[Tasks]
|
||||||
|
Name: "desktopicon"; Description: "{cm:CreateDesktopIcon}"; GroupDescription: "{cm:AdditionalIcons}"; Flags: unchecked
|
||||||
|
|
||||||
|
[Files]
|
||||||
|
Source: "{#MyAppPath}\{#MyAppExeName}"; DestDir: "{app}"; Flags: ignoreversion
|
||||||
|
Source: "{#MyAppPath}\app\*"; DestDir: "{app}\app"; Flags: ignoreversion recursesubdirs createallsubdirs
|
||||||
|
Source: "{#MyAppPath}\resources\*"; DestDir: "{app}\resources"; Flags: ignoreversion recursesubdirs createallsubdirs
|
||||||
|
Source: "{#MyAppPath}\main.py"; DestDir: "{app}"; Flags: ignoreversion
|
||||||
|
Source: "{#MyAppPath}\requirements.txt"; DestDir: "{app}"; Flags: ignoreversion
|
||||||
|
Source: "{#MyAppPath}\README.md"; DestDir: "{app}"; Flags: ignoreversion
|
||||||
|
Source: "{#MyAppPath}\LICENSE"; DestDir: "{app}"; Flags: ignoreversion
|
||||||
|
; NOTE: Don't use "Flags: ignoreversion" on any shared system files
|
||||||
|
|
||||||
|
[Icons]
|
||||||
|
Name: "{autoprograms}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"
|
||||||
|
Name: "{autodesktop}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: desktopicon
|
||||||
|
|
||||||
|
[Run]
|
||||||
|
Filename: "{app}\{#MyAppExeName}"; Description: "{cm:LaunchProgram,{#StringChange(MyAppName, '&', '&&')}}"; Flags: nowait postinstall
|
||||||
|
|
||||||
|
[Code]
|
||||||
|
var
|
||||||
|
DeleteDataQuestion: Boolean;
|
||||||
|
|
||||||
|
function InitializeUninstall: Boolean;
|
||||||
|
begin
|
||||||
|
DeleteDataQuestion := MsgBox('您确认要完全移除 AUTO_MAA 的所有用户数据文件与子组件吗?', mbConfirmation, MB_YESNO) = IDYES;
|
||||||
|
Result := True;
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure CurUninstallStepChanged(CurUninstallStep: TUninstallStep);
|
||||||
|
begin
|
||||||
|
if CurUninstallStep = usPostUninstall then
|
||||||
|
begin
|
||||||
|
DelTree(ExpandConstant('{app}\app'), True, True, True);
|
||||||
|
DelTree(ExpandConstant('{app}\resources'), True, True, True);
|
||||||
|
if DeleteDataQuestion then
|
||||||
|
begin
|
||||||
|
DelTree(ExpandConstant('{app}'), True, True, True);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
67
app/utils/ImageUtils.py
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import base64
|
||||||
|
import hashlib
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
|
class ImageUtils:
|
||||||
|
@staticmethod
|
||||||
|
def get_base64_from_file(image_path):
|
||||||
|
"""从本地文件读取并返回base64编码字符串"""
|
||||||
|
with open(image_path, "rb") as f:
|
||||||
|
return base64.b64encode(f.read()).decode("utf-8")
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def calculate_md5_from_file(image_path):
|
||||||
|
"""从本地文件读取并返回md5值(hex字符串)"""
|
||||||
|
with open(image_path, "rb") as f:
|
||||||
|
return hashlib.md5(f.read()).hexdigest()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def calculate_md5_from_base64(base64_content):
|
||||||
|
"""从base64字符串计算md5"""
|
||||||
|
image_data = base64.b64decode(base64_content)
|
||||||
|
return hashlib.md5(image_data).hexdigest()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def compress_image_if_needed(image_path: Path, max_size_mb=2) -> Path:
|
||||||
|
"""
|
||||||
|
如果图片大于max_size_mb,则压缩并覆盖原文件,返回原始路径(Path对象)
|
||||||
|
"""
|
||||||
|
if hasattr(Image, "Resampling"): # Pillow 9.1.0及以后
|
||||||
|
RESAMPLE = Image.Resampling.LANCZOS
|
||||||
|
else:
|
||||||
|
RESAMPLE = Image.ANTIALIAS
|
||||||
|
|
||||||
|
max_size = max_size_mb * 1024 * 1024
|
||||||
|
if image_path.stat().st_size <= max_size:
|
||||||
|
return image_path
|
||||||
|
|
||||||
|
img = Image.open(image_path)
|
||||||
|
suffix = image_path.suffix.lower()
|
||||||
|
quality = 90 if suffix in [".jpg", ".jpeg"] else None
|
||||||
|
step = 5
|
||||||
|
|
||||||
|
if suffix in [".jpg", ".jpeg"]:
|
||||||
|
while True:
|
||||||
|
img.save(image_path, quality=quality, optimize=True)
|
||||||
|
if image_path.stat().st_size <= max_size or quality <= 10:
|
||||||
|
break
|
||||||
|
quality -= step
|
||||||
|
elif suffix == ".png":
|
||||||
|
width, height = img.size
|
||||||
|
while True:
|
||||||
|
img.save(image_path, optimize=True)
|
||||||
|
if (
|
||||||
|
image_path.stat().st_size <= max_size
|
||||||
|
or width <= 200
|
||||||
|
or height <= 200
|
||||||
|
):
|
||||||
|
break
|
||||||
|
width = int(width * 0.95)
|
||||||
|
height = int(height * 0.95)
|
||||||
|
img = img.resize((width, height), RESAMPLE)
|
||||||
|
else:
|
||||||
|
raise ValueError("仅支持JPG/JPEG和PNG格式图片的压缩。")
|
||||||
|
|
||||||
|
return image_path
|
||||||
@@ -1,490 +0,0 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
|
||||||
# Copyright © <2024> <DLmaster361>
|
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
|
||||||
|
|
||||||
# AUTO_MAA is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published
|
|
||||||
# by the Free Software Foundation, either version 3 of the License,
|
|
||||||
# or (at your option) any later version.
|
|
||||||
|
|
||||||
# AUTO_MAA is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty
|
|
||||||
# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
||||||
# the GNU General Public License for more details.
|
|
||||||
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
|
||||||
|
|
||||||
"""
|
|
||||||
AUTO_MAA
|
|
||||||
AUTO_MAA更新器
|
|
||||||
v1.1
|
|
||||||
作者:DLmaster_361
|
|
||||||
"""
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import json
|
|
||||||
import zipfile
|
|
||||||
import requests
|
|
||||||
import subprocess
|
|
||||||
import time
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from PySide6.QtWidgets import QApplication, QDialog, QVBoxLayout, QHBoxLayout
|
|
||||||
from qfluentwidgets import (
|
|
||||||
ProgressBar,
|
|
||||||
IndeterminateProgressBar,
|
|
||||||
BodyLabel,
|
|
||||||
PushButton,
|
|
||||||
EditableComboBox,
|
|
||||||
)
|
|
||||||
from PySide6.QtGui import QIcon, QCloseEvent
|
|
||||||
from PySide6.QtCore import QThread, Signal, QEventLoop
|
|
||||||
|
|
||||||
|
|
||||||
def version_text(version_numb: list) -> str:
|
|
||||||
"""将版本号列表转为可读的文本信息"""
|
|
||||||
|
|
||||||
if version_numb[3] == 0:
|
|
||||||
version = f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
|
||||||
else:
|
|
||||||
version = (
|
|
||||||
f"v{'.'.join(str(_) for _ in version_numb[0:3])}-beta.{version_numb[3]}"
|
|
||||||
)
|
|
||||||
return version
|
|
||||||
|
|
||||||
|
|
||||||
class UpdateProcess(QThread):
|
|
||||||
|
|
||||||
info = Signal(str)
|
|
||||||
progress = Signal(int, int, int)
|
|
||||||
question = Signal(dict)
|
|
||||||
question_response = Signal(str)
|
|
||||||
accomplish = Signal()
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self, app_path: Path, name: str, main_version: list, updater_version: list
|
|
||||||
) -> None:
|
|
||||||
super(UpdateProcess, self).__init__()
|
|
||||||
|
|
||||||
self.app_path = app_path
|
|
||||||
self.name = name
|
|
||||||
self.main_version = main_version
|
|
||||||
self.updater_version = updater_version
|
|
||||||
self.download_path = app_path / "DOWNLOAD_TEMP.zip" # 临时下载文件的路径
|
|
||||||
self.version_path = app_path / "resources/version.json"
|
|
||||||
self.response = None
|
|
||||||
|
|
||||||
self.question_response.connect(self._capture_response)
|
|
||||||
|
|
||||||
def run(self) -> None:
|
|
||||||
|
|
||||||
# 清理可能存在的临时文件
|
|
||||||
if self.download_path.exists():
|
|
||||||
self.download_path.unlink()
|
|
||||||
|
|
||||||
self.info.emit("正在获取下载链接")
|
|
||||||
url_list = self.get_download_url()
|
|
||||||
url_dict = {}
|
|
||||||
|
|
||||||
# 验证下载地址
|
|
||||||
for i, url in enumerate(url_list):
|
|
||||||
|
|
||||||
if self.isInterruptionRequested():
|
|
||||||
return None
|
|
||||||
|
|
||||||
self.progress.emit(0, len(url_list), i)
|
|
||||||
|
|
||||||
try:
|
|
||||||
self.info.emit(f"正在验证下载地址:{url}")
|
|
||||||
response = requests.get(url, stream=True)
|
|
||||||
if response.status_code != 200:
|
|
||||||
self.info.emit(f"连接失败,错误代码 {response.status_code}")
|
|
||||||
time.sleep(1)
|
|
||||||
continue
|
|
||||||
url_dict[url] = response.elapsed.total_seconds()
|
|
||||||
except requests.RequestException:
|
|
||||||
self.info.emit(f"请求超时")
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
download_url = self.push_question(url_dict)
|
|
||||||
|
|
||||||
# 获取文件大小
|
|
||||||
try:
|
|
||||||
self.info.emit(f"正在连接下载地址:{download_url}")
|
|
||||||
self.progress.emit(0, 0, 0)
|
|
||||||
response = requests.get(download_url, stream=True)
|
|
||||||
if response.status_code != 200:
|
|
||||||
self.info.emit(f"连接失败,错误代码 {response.status_code}")
|
|
||||||
return None
|
|
||||||
file_size = response.headers.get("Content-Length")
|
|
||||||
except requests.RequestException:
|
|
||||||
self.info.emit(f"请求超时")
|
|
||||||
return None
|
|
||||||
|
|
||||||
if file_size is None:
|
|
||||||
file_size = 1
|
|
||||||
else:
|
|
||||||
file_size = int(file_size)
|
|
||||||
|
|
||||||
try:
|
|
||||||
# 下载文件
|
|
||||||
with open(self.download_path, "wb") as f:
|
|
||||||
|
|
||||||
downloaded_size = 0
|
|
||||||
last_download_size = 0
|
|
||||||
speed = 0
|
|
||||||
last_time = time.time()
|
|
||||||
|
|
||||||
for chunk in response.iter_content(chunk_size=8192):
|
|
||||||
|
|
||||||
if self.isInterruptionRequested():
|
|
||||||
break
|
|
||||||
|
|
||||||
# 写入已下载数据
|
|
||||||
f.write(chunk)
|
|
||||||
downloaded_size += len(chunk)
|
|
||||||
|
|
||||||
# 计算下载速度
|
|
||||||
if time.time() - last_time >= 1.0:
|
|
||||||
speed = (
|
|
||||||
(downloaded_size - last_download_size)
|
|
||||||
/ (time.time() - last_time)
|
|
||||||
/ 1024
|
|
||||||
)
|
|
||||||
last_download_size = downloaded_size
|
|
||||||
last_time = time.time()
|
|
||||||
|
|
||||||
# 更新下载进度
|
|
||||||
if speed >= 1024:
|
|
||||||
self.info.emit(
|
|
||||||
f"正在下载:{self.name} 已下载:{downloaded_size / 1048576:.2f}/{file_size / 1048576:.2f} MB ({downloaded_size / file_size * 100:.2f}%) 下载速度:{speed / 1024:.2f} MB/s",
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
self.info.emit(
|
|
||||||
f"正在下载:{self.name} 已下载:{downloaded_size / 1048576:.2f}/{file_size / 1048576:.2f} MB ({downloaded_size / file_size * 100:.2f}%) 下载速度:{speed:.2f} KB/s",
|
|
||||||
)
|
|
||||||
self.progress.emit(0, 100, int(downloaded_size / file_size * 100))
|
|
||||||
|
|
||||||
if self.isInterruptionRequested() and self.download_path.exists():
|
|
||||||
self.download_path.unlink()
|
|
||||||
return None
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
e = str(e)
|
|
||||||
e = "\n".join([e[_ : _ + 75] for _ in range(0, len(e), 75)])
|
|
||||||
self.info.emit(f"下载{self.name}时出错:\n{e}")
|
|
||||||
return None
|
|
||||||
|
|
||||||
# 解压
|
|
||||||
try:
|
|
||||||
|
|
||||||
while True:
|
|
||||||
if self.isInterruptionRequested():
|
|
||||||
self.download_path.unlink()
|
|
||||||
return None
|
|
||||||
try:
|
|
||||||
self.info.emit("正在解压更新文件")
|
|
||||||
self.progress.emit(0, 0, 0)
|
|
||||||
with zipfile.ZipFile(self.download_path, "r") as zip_ref:
|
|
||||||
zip_ref.extractall(self.app_path)
|
|
||||||
break
|
|
||||||
except PermissionError:
|
|
||||||
self.info.emit(f"解压出错:{self.name}正在运行,正在等待其关闭")
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
self.info.emit("正在删除临时文件")
|
|
||||||
self.progress.emit(0, 0, 0)
|
|
||||||
self.download_path.unlink()
|
|
||||||
|
|
||||||
self.info.emit(f"{self.name}更新成功!")
|
|
||||||
self.progress.emit(0, 100, 100)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
|
|
||||||
e = str(e)
|
|
||||||
e = "\n".join([e[_ : _ + 75] for _ in range(0, len(e), 75)])
|
|
||||||
self.info.emit(f"解压更新时出错:\n{e}")
|
|
||||||
return None
|
|
||||||
|
|
||||||
# 更新version文件
|
|
||||||
if not self.isInterruptionRequested and self.name in [
|
|
||||||
"AUTO_MAA主程序",
|
|
||||||
"AUTO_MAA更新器",
|
|
||||||
]:
|
|
||||||
with open(self.version_path, "r", encoding="utf-8") as f:
|
|
||||||
version_info = json.load(f)
|
|
||||||
if self.name == "AUTO_MAA主程序":
|
|
||||||
version_info["main_version"] = ".".join(map(str, self.main_version))
|
|
||||||
elif self.name == "AUTO_MAA更新器":
|
|
||||||
version_info["updater_version"] = ".".join(
|
|
||||||
map(str, self.updater_version)
|
|
||||||
)
|
|
||||||
with open(self.version_path, "w", encoding="utf-8") as f:
|
|
||||||
json.dump(version_info, f, ensure_ascii=False, indent=4)
|
|
||||||
|
|
||||||
# 主程序更新完成后打开AUTO_MAA
|
|
||||||
if not self.isInterruptionRequested and self.name == "AUTO_MAA主程序":
|
|
||||||
subprocess.Popen(
|
|
||||||
str(self.app_path / "AUTO_MAA.exe"),
|
|
||||||
shell=True,
|
|
||||||
creationflags=subprocess.CREATE_NO_WINDOW,
|
|
||||||
)
|
|
||||||
elif not self.isInterruptionRequested and self.name == "MAA":
|
|
||||||
subprocess.Popen(
|
|
||||||
str(self.app_path / "MAA.exe"),
|
|
||||||
shell=True,
|
|
||||||
creationflags=subprocess.CREATE_NO_WINDOW,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.accomplish.emit()
|
|
||||||
|
|
||||||
def get_download_url(self) -> list:
|
|
||||||
"""获取下载链接"""
|
|
||||||
|
|
||||||
try_num = 3
|
|
||||||
for i in range(try_num):
|
|
||||||
try:
|
|
||||||
response = requests.get(
|
|
||||||
"https://gitee.com/DLmaster_361/AUTO_MAA/raw/main/resources/version.json"
|
|
||||||
)
|
|
||||||
if response.status_code != 200:
|
|
||||||
self.info.emit(
|
|
||||||
f"连接失败,错误代码 {response.status_code} ,正在重试({i+1}/{try_num})"
|
|
||||||
)
|
|
||||||
time.sleep(0.1)
|
|
||||||
continue
|
|
||||||
version_remote = response.json()
|
|
||||||
PROXY_list = version_remote["proxy_list"]
|
|
||||||
break
|
|
||||||
except requests.RequestException:
|
|
||||||
self.info.emit(f"请求超时,正在重试({i+1}/{try_num})")
|
|
||||||
time.sleep(0.1)
|
|
||||||
except KeyError:
|
|
||||||
self.info.emit(f"未找到远端代理网址项,正在重试({i+1}/{try_num})")
|
|
||||||
time.sleep(0.1)
|
|
||||||
else:
|
|
||||||
self.info.emit("获取远端代理信息失败,将使用默认代理地址")
|
|
||||||
PROXY_list = [
|
|
||||||
"",
|
|
||||||
"https://gitproxy.click/",
|
|
||||||
"https://cdn.moran233.xyz/",
|
|
||||||
"https://gh.llkk.cc/",
|
|
||||||
"https://github.akams.cn/",
|
|
||||||
"https://www.ghproxy.cn/",
|
|
||||||
"https://ghfast.top/",
|
|
||||||
]
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
url_list = []
|
|
||||||
if self.name == "AUTO_MAA主程序":
|
|
||||||
url_list.append(
|
|
||||||
f"https://gitee.com/DLmaster_361/AUTO_MAA/releases/download/{version_text(self.main_version)}/AUTO_MAA_{version_text(self.main_version)}.zip"
|
|
||||||
)
|
|
||||||
url_list.append(
|
|
||||||
f"https://jp-download.fearr.xyz/AUTO_MAA/AUTO_MAA_{version_text(self.main_version)}.zip"
|
|
||||||
)
|
|
||||||
for i in range(len(PROXY_list)):
|
|
||||||
url_list.append(
|
|
||||||
f"{PROXY_list[i]}https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.main_version)}/AUTO_MAA_{version_text(self.main_version)}.zip"
|
|
||||||
)
|
|
||||||
elif self.name == "AUTO_MAA更新器":
|
|
||||||
url_list.append(
|
|
||||||
f"https://gitee.com/DLmaster_361/AUTO_MAA/releases/download/{version_text(self.main_version)}/Updater_{version_text(self.updater_version)}.zip"
|
|
||||||
)
|
|
||||||
url_list.append(
|
|
||||||
f"https://jp-download.fearr.xyz/AUTO_MAA/Updater_{version_text(self.updater_version)}.zip"
|
|
||||||
)
|
|
||||||
for i in range(len(PROXY_list)):
|
|
||||||
url_list.append(
|
|
||||||
f"{PROXY_list[i]}https://github.com/DLmaster361/AUTO_MAA/releases/download/{version_text(self.main_version)}/Updater_{version_text(self.updater_version)}.zip"
|
|
||||||
)
|
|
||||||
elif self.name == "MAA":
|
|
||||||
url_list.append(
|
|
||||||
f"https://jp-download.fearr.xyz/MAA/MAA-{version_text(self.main_version)}-win-x64.zip"
|
|
||||||
)
|
|
||||||
for i in range(len(PROXY_list)):
|
|
||||||
url_list.append(
|
|
||||||
f"{PROXY_list[i]}https://github.com/MaaAssistantArknights/MaaAssistantArknights/releases/download/{version_text(self.main_version)}/MAA-{version_text(self.main_version)}-win-x64.zip"
|
|
||||||
)
|
|
||||||
return url_list
|
|
||||||
|
|
||||||
def push_question(self, url_dict: dict) -> str:
|
|
||||||
self.question.emit(url_dict)
|
|
||||||
loop = QEventLoop()
|
|
||||||
self.question_response.connect(loop.quit)
|
|
||||||
loop.exec()
|
|
||||||
return self.response
|
|
||||||
|
|
||||||
def _capture_response(self, response: str) -> None:
|
|
||||||
self.response = response
|
|
||||||
|
|
||||||
|
|
||||||
class Updater(QDialog):
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self, app_path: Path, name: str, main_version: list, updater_version: list
|
|
||||||
) -> None:
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
self.setWindowTitle("AUTO_MAA更新器")
|
|
||||||
self.setWindowIcon(
|
|
||||||
QIcon(
|
|
||||||
str(
|
|
||||||
Path(sys.argv[0]).resolve().parent
|
|
||||||
/ "resources/icons/AUTO_MAA_Updater.ico"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
# 创建垂直布局
|
|
||||||
self.Layout = QVBoxLayout(self)
|
|
||||||
|
|
||||||
self.info = BodyLabel("正在初始化", self)
|
|
||||||
self.progress_1 = IndeterminateProgressBar(self)
|
|
||||||
self.progress_2 = ProgressBar(self)
|
|
||||||
self.combo_box = EditableComboBox(self)
|
|
||||||
|
|
||||||
self.button = PushButton("继续", self)
|
|
||||||
self.h_layout = QHBoxLayout()
|
|
||||||
self.h_layout.addStretch(1)
|
|
||||||
self.h_layout.addWidget(self.button)
|
|
||||||
|
|
||||||
self.update_progress(0, 0, 0)
|
|
||||||
|
|
||||||
self.Layout.addWidget(self.info)
|
|
||||||
self.Layout.addStretch(1)
|
|
||||||
self.Layout.addWidget(self.progress_1)
|
|
||||||
self.Layout.addWidget(self.progress_2)
|
|
||||||
self.Layout.addWidget(self.combo_box)
|
|
||||||
self.Layout.addLayout(self.h_layout)
|
|
||||||
self.Layout.addStretch(1)
|
|
||||||
|
|
||||||
self.update_process = UpdateProcess(
|
|
||||||
app_path, name, main_version, updater_version
|
|
||||||
)
|
|
||||||
|
|
||||||
self.update_process.info.connect(self.update_info)
|
|
||||||
self.update_process.progress.connect(self.update_progress)
|
|
||||||
self.update_process.question.connect(self.question)
|
|
||||||
|
|
||||||
self.update_process.start()
|
|
||||||
|
|
||||||
def update_info(self, text: str) -> None:
|
|
||||||
self.info.setText(text)
|
|
||||||
|
|
||||||
def update_progress(
|
|
||||||
self, begin: int, end: int, current: int, if_show_combo_box: bool = False
|
|
||||||
) -> None:
|
|
||||||
|
|
||||||
self.combo_box.setVisible(if_show_combo_box)
|
|
||||||
self.button.setVisible(if_show_combo_box)
|
|
||||||
|
|
||||||
if if_show_combo_box:
|
|
||||||
self.progress_1.setVisible(False)
|
|
||||||
self.progress_2.setVisible(False)
|
|
||||||
self.resize(1000, 90)
|
|
||||||
elif begin == 0 and end == 0:
|
|
||||||
self.progress_2.setVisible(False)
|
|
||||||
self.progress_1.setVisible(True)
|
|
||||||
self.resize(700, 70)
|
|
||||||
else:
|
|
||||||
self.progress_1.setVisible(False)
|
|
||||||
self.progress_2.setVisible(True)
|
|
||||||
self.progress_2.setRange(begin, end)
|
|
||||||
self.progress_2.setValue(current)
|
|
||||||
self.resize(700, 70)
|
|
||||||
|
|
||||||
def question(self, url_dict: dict) -> None:
|
|
||||||
|
|
||||||
self.update_info("测速完成,请选择或自行输入一个合适下载地址:")
|
|
||||||
self.update_progress(0, 0, 0, True)
|
|
||||||
|
|
||||||
url_dict = dict(sorted(url_dict.items(), key=lambda item: item[1]))
|
|
||||||
|
|
||||||
for url, time in url_dict.items():
|
|
||||||
self.combo_box.addItem(f"{url} | 响应时间:{time:.3f}秒")
|
|
||||||
|
|
||||||
self.button.clicked.connect(
|
|
||||||
lambda: self.update_process.question_response.emit(
|
|
||||||
self.combo_box.currentText().split(" | ")[0]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def closeEvent(self, event: QCloseEvent):
|
|
||||||
"""清理残余进程"""
|
|
||||||
|
|
||||||
self.update_process.requestInterruption()
|
|
||||||
self.update_process.quit()
|
|
||||||
self.update_process.wait()
|
|
||||||
|
|
||||||
event.accept()
|
|
||||||
|
|
||||||
|
|
||||||
class AUTO_MAA_Updater(QApplication):
|
|
||||||
def __init__(
|
|
||||||
self, app_path: Path, name: str, main_version: list, updater_version: list
|
|
||||||
) -> None:
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
self.main = Updater(app_path, name, main_version, updater_version)
|
|
||||||
self.main.show()
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
|
|
||||||
# 获取软件自身的路径
|
|
||||||
app_path = Path(sys.argv[0]).resolve().parent
|
|
||||||
|
|
||||||
# 从本地版本信息文件获取当前版本信息
|
|
||||||
if (app_path / "resources/version.json").exists():
|
|
||||||
with (app_path / "resources/version.json").open(
|
|
||||||
mode="r", encoding="utf-8"
|
|
||||||
) as f:
|
|
||||||
version_current = json.load(f)
|
|
||||||
main_version_current = list(
|
|
||||||
map(int, version_current["main_version"].split("."))
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
main_version_current = [0, 0, 0, 0]
|
|
||||||
|
|
||||||
# 从本地配置文件获取更新类型
|
|
||||||
if (app_path / "config/config.json").exists():
|
|
||||||
with (app_path / "config/config.json").open(mode="r", encoding="utf-8") as f:
|
|
||||||
config = json.load(f)
|
|
||||||
if "Update" in config and "UpdateType" in config["Update"]:
|
|
||||||
update_type = config["Update"]["UpdateType"]
|
|
||||||
else:
|
|
||||||
update_type = "main"
|
|
||||||
else:
|
|
||||||
update_type = "main"
|
|
||||||
|
|
||||||
# 从远程服务器获取最新版本信息
|
|
||||||
for _ in range(3):
|
|
||||||
try:
|
|
||||||
response = requests.get(
|
|
||||||
f"https://gitee.com/DLmaster_361/AUTO_MAA/raw/{update_type}/resources/version.json"
|
|
||||||
)
|
|
||||||
version_remote = response.json()
|
|
||||||
main_version_remote = list(
|
|
||||||
map(int, version_remote["main_version"].split("."))
|
|
||||||
)
|
|
||||||
break
|
|
||||||
except Exception as e:
|
|
||||||
err = e
|
|
||||||
time.sleep(0.1)
|
|
||||||
else:
|
|
||||||
sys.exit(f"获取版本信息时出错:\n{err}")
|
|
||||||
|
|
||||||
# 启动更新线程
|
|
||||||
if main_version_remote > main_version_current:
|
|
||||||
app = AUTO_MAA_Updater(
|
|
||||||
app_path,
|
|
||||||
"AUTO_MAA主程序",
|
|
||||||
main_version_remote,
|
|
||||||
[],
|
|
||||||
)
|
|
||||||
sys.exit(app.exec())
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA工具包
|
AUTO_MAA工具包
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -29,6 +29,4 @@ __version__ = "4.2.0"
|
|||||||
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
__author__ = "DLmaster361 <DLmaster_361@163.com>"
|
||||||
__license__ = "GPL-3.0 license"
|
__license__ = "GPL-3.0 license"
|
||||||
|
|
||||||
from .Updater import Updater
|
__all__ = []
|
||||||
|
|
||||||
__all__ = ["Updater"]
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,12 +16,12 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA打包程序
|
AUTO_MAA打包程序
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -35,6 +35,9 @@ from pathlib import Path
|
|||||||
def version_text(version_numb: list) -> str:
|
def version_text(version_numb: list) -> str:
|
||||||
"""将版本号列表转为可读的文本信息"""
|
"""将版本号列表转为可读的文本信息"""
|
||||||
|
|
||||||
|
while len(version_numb) < 4:
|
||||||
|
version_numb.append(0)
|
||||||
|
|
||||||
if version_numb[3] == 0:
|
if version_numb[3] == 0:
|
||||||
version = f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
version = f"v{'.'.join(str(_) for _ in version_numb[0:3])}"
|
||||||
else:
|
else:
|
||||||
@@ -44,6 +47,17 @@ def version_text(version_numb: list) -> str:
|
|||||||
return version
|
return version
|
||||||
|
|
||||||
|
|
||||||
|
def version_info_markdown(info: dict) -> str:
|
||||||
|
"""将版本信息字典转为markdown信息"""
|
||||||
|
|
||||||
|
version_info = ""
|
||||||
|
for key, value in info.items():
|
||||||
|
version_info += f"## {key}\n"
|
||||||
|
for v in value:
|
||||||
|
version_info += f"- {v}\n"
|
||||||
|
return version_info
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|
||||||
root_path = Path(sys.argv[0]).resolve().parent
|
root_path = Path(sys.argv[0]).resolve().parent
|
||||||
@@ -52,58 +66,78 @@ if __name__ == "__main__":
|
|||||||
version = json.load(f)
|
version = json.load(f)
|
||||||
|
|
||||||
main_version_numb = list(map(int, version["main_version"].split(".")))
|
main_version_numb = list(map(int, version["main_version"].split(".")))
|
||||||
updater_version_numb = list(map(int, version["updater_version"].split(".")))
|
|
||||||
|
|
||||||
print("Packaging AUTO_MAA main program ...")
|
print("Packaging AUTO_MAA main program ...")
|
||||||
|
|
||||||
os.system(
|
os.system(
|
||||||
"powershell -Command python -m nuitka --standalone --onefile --mingw64"
|
"powershell -Command python -m nuitka --standalone --onefile --mingw64"
|
||||||
" --enable-plugins=pyside6 --windows-console-mode=disable"
|
" --enable-plugins=pyside6 --windows-console-mode=attach"
|
||||||
" --onefile-tempdir-spec='{TEMP}\\AUTO_MAA'"
|
" --onefile-tempdir-spec='{TEMP}\\AUTO_MAA'"
|
||||||
" --windows-icon-from-ico=resources\\icons\\AUTO_MAA.ico"
|
" --windows-icon-from-ico=resources\\icons\\AUTO_MAA.ico"
|
||||||
" --company-name='AUTO_MAA Team' --product-name=AUTO_MAA"
|
" --company-name='AUTO_MAA Team' --product-name=AUTO_MAA"
|
||||||
f" --file-version={version["main_version"]}"
|
f" --file-version={version['main_version']}"
|
||||||
f" --product-version={version["main_version"]}"
|
f" --product-version={version['main_version']}"
|
||||||
" --file-description='AUTO_MAA Component'"
|
" --file-description='AUTO_MAA Component'"
|
||||||
" --copyright='Copyright © 2024 DLmaster361'"
|
" --copyright='Copyright © 2024-2025 DLmaster361'"
|
||||||
" --assume-yes-for-downloads --output-filename=AUTO_MAA"
|
" --assume-yes-for-downloads --output-filename=AUTO_MAA"
|
||||||
" --remove-output main.py"
|
" --remove-output main.py"
|
||||||
)
|
)
|
||||||
|
|
||||||
print("AUTO_MAA main program packaging completed !")
|
print("AUTO_MAA main program packaging completed !")
|
||||||
|
|
||||||
shutil.copy(root_path / "app/utils/Updater.py", root_path)
|
print("start to create setup program ...")
|
||||||
|
|
||||||
file_content = (root_path / "Updater.py").read_text(encoding="utf-8")
|
(root_path / "AUTO_MAA").mkdir(parents=True, exist_ok=True)
|
||||||
|
shutil.move(root_path / "AUTO_MAA.exe", root_path / "AUTO_MAA/")
|
||||||
|
shutil.copytree(root_path / "app", root_path / "AUTO_MAA/app")
|
||||||
|
shutil.copytree(root_path / "resources", root_path / "AUTO_MAA/resources")
|
||||||
|
shutil.copy(root_path / "main.py", root_path / "AUTO_MAA/")
|
||||||
|
shutil.copy(root_path / "requirements.txt", root_path / "AUTO_MAA/")
|
||||||
|
shutil.copy(root_path / "README.md", root_path / "AUTO_MAA/")
|
||||||
|
shutil.copy(root_path / "LICENSE", root_path / "AUTO_MAA/")
|
||||||
|
|
||||||
(root_path / "Updater.py").write_text(
|
with (root_path / "app/utils/AUTO_MAA.iss").open(mode="r", encoding="utf-8") as f:
|
||||||
file_content.replace(
|
iss = f.read()
|
||||||
"from .version import version_text", "from app import version_text"
|
iss = (
|
||||||
),
|
iss.replace(
|
||||||
encoding="utf-8",
|
'#define MyAppVersion ""',
|
||||||
|
f'#define MyAppVersion "{version["main_version"]}"',
|
||||||
|
)
|
||||||
|
.replace(
|
||||||
|
'#define MyAppPath ""', f'#define MyAppPath "{root_path / "AUTO_MAA"}"'
|
||||||
|
)
|
||||||
|
.replace('#define OutputDir ""', f'#define OutputDir "{root_path}"')
|
||||||
|
)
|
||||||
|
with (root_path / "AUTO_MAA.iss").open(mode="w", encoding="utf-8") as f:
|
||||||
|
f.write(iss)
|
||||||
|
|
||||||
|
os.system(f'ISCC "{root_path / "AUTO_MAA.iss"}"')
|
||||||
|
|
||||||
|
(root_path / "AUTO_MAA_Setup").mkdir(parents=True, exist_ok=True)
|
||||||
|
shutil.move(root_path / "AUTO_MAA-Setup.exe", root_path / "AUTO_MAA_Setup")
|
||||||
|
|
||||||
|
shutil.make_archive(
|
||||||
|
base_name=root_path / f"AUTO_MAA_{version_text(main_version_numb)}",
|
||||||
|
format="zip",
|
||||||
|
root_dir=root_path / "AUTO_MAA_Setup",
|
||||||
|
base_dir=".",
|
||||||
)
|
)
|
||||||
|
|
||||||
print("Packaging AUTO_MAA update program ...")
|
print("setup program created !")
|
||||||
|
|
||||||
os.system(
|
(root_path / "AUTO_MAA.iss").unlink(missing_ok=True)
|
||||||
"powershell -Command python -m nuitka --standalone --onefile --mingw64"
|
shutil.rmtree(root_path / "AUTO_MAA")
|
||||||
" --enable-plugins=pyside6 --windows-console-mode=disable"
|
shutil.rmtree(root_path / "AUTO_MAA_Setup")
|
||||||
" --onefile-tempdir-spec='{TEMP}\\AUTO_MAA_Updater'"
|
|
||||||
" --windows-icon-from-ico=resources\\icons\\AUTO_MAA_Updater.ico"
|
|
||||||
" --company-name='AUTO_MAA Team' --product-name=AUTO_MAA"
|
|
||||||
f" --file-version={version["updater_version"]}"
|
|
||||||
f" --product-version={version["main_version"]}"
|
|
||||||
" --file-description='AUTO_MAA Component'"
|
|
||||||
" --copyright='Copyright © 2024 DLmaster361'"
|
|
||||||
" --assume-yes-for-downloads --output-filename=Updater"
|
|
||||||
" --remove-output Updater.py"
|
|
||||||
)
|
|
||||||
|
|
||||||
print("AUTO_MAA update program packaging completed !")
|
all_version_info = {}
|
||||||
|
for v_i in version["version_info"].values():
|
||||||
(root_path / "Updater.py").unlink()
|
for key, value in v_i.items():
|
||||||
|
if key in all_version_info:
|
||||||
|
all_version_info[key] += value.copy()
|
||||||
|
else:
|
||||||
|
all_version_info[key] = value.copy()
|
||||||
|
|
||||||
(root_path / "version_info.txt").write_text(
|
(root_path / "version_info.txt").write_text(
|
||||||
f"{version_text(main_version_numb)}\n{version_text(updater_version_numb)}{version["announcement"]}",
|
f"{version_text(main_version_numb)}\n\n<!--{json.dumps(version["version_info"], ensure_ascii=False)}-->\n{version_info_markdown(all_version_info)}",
|
||||||
encoding="utf-8",
|
encoding="utf-8",
|
||||||
)
|
)
|
||||||
|
|||||||
31
main.py
@@ -1,5 +1,5 @@
|
|||||||
# <AUTO_MAA:A MAA Multi Account Management and Automation Tool>
|
# AUTO_MAA:A MAA Multi Account Management and Automation Tool
|
||||||
# Copyright © <2024> <DLmaster361>
|
# Copyright © 2024-2025 DLmaster361
|
||||||
|
|
||||||
# This file is part of AUTO_MAA.
|
# This file is part of AUTO_MAA.
|
||||||
|
|
||||||
@@ -16,18 +16,36 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
# along with AUTO_MAA. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# DLmaster_361@163.com
|
# Contact: DLmaster_361@163.com
|
||||||
|
|
||||||
"""
|
"""
|
||||||
AUTO_MAA
|
AUTO_MAA
|
||||||
AUTO_MAA主程序
|
AUTO_MAA主程序
|
||||||
v4.2
|
v4.3
|
||||||
作者:DLmaster_361
|
作者:DLmaster_361
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
# 屏蔽广告
|
||||||
|
import builtins
|
||||||
|
|
||||||
|
original_print = builtins.print
|
||||||
|
|
||||||
|
|
||||||
|
def no_print(*args, **kwargs):
|
||||||
|
if (
|
||||||
|
args
|
||||||
|
and isinstance(args[0], str)
|
||||||
|
and "QFluentWidgets Pro is now released." in args[0]
|
||||||
|
):
|
||||||
|
return
|
||||||
|
return original_print(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
builtins.print = no_print
|
||||||
|
|
||||||
|
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
from PySide6.QtWidgets import QApplication
|
from PySide6.QtWidgets import QApplication
|
||||||
from PySide6.QtCore import Qt
|
|
||||||
from qfluentwidgets import FluentTranslator
|
from qfluentwidgets import FluentTranslator
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
@@ -36,7 +54,6 @@ import sys
|
|||||||
def main():
|
def main():
|
||||||
|
|
||||||
application = QApplication(sys.argv)
|
application = QApplication(sys.argv)
|
||||||
QApplication.setAttribute(Qt.AA_DontCreateNativeWidgetSiblings)
|
|
||||||
|
|
||||||
translator = FluentTranslator()
|
translator = FluentTranslator()
|
||||||
application.installTranslator(translator)
|
application.installTranslator(translator)
|
||||||
@@ -44,7 +61,7 @@ def main():
|
|||||||
from app.ui.main_window import AUTO_MAA
|
from app.ui.main_window import AUTO_MAA
|
||||||
|
|
||||||
window = AUTO_MAA()
|
window = AUTO_MAA()
|
||||||
window.show_ui("显示主窗口")
|
window.show_ui("显示主窗口", if_start=True)
|
||||||
window.start_up_task()
|
window.start_up_task()
|
||||||
sys.exit(application.exec())
|
sys.exit(application.exec())
|
||||||
|
|
||||||
|
|||||||
@@ -3,11 +3,11 @@ plyer
|
|||||||
PySide6
|
PySide6
|
||||||
PySide6-Fluent-Widgets[full]
|
PySide6-Fluent-Widgets[full]
|
||||||
psutil
|
psutil
|
||||||
opencv-python
|
|
||||||
pywin32
|
pywin32
|
||||||
pyautogui
|
keyboard
|
||||||
pycryptodome
|
pycryptodome
|
||||||
requests
|
requests
|
||||||
|
markdown
|
||||||
Jinja2
|
Jinja2
|
||||||
serverchan_sdk
|
nuitka
|
||||||
nuitka==2.6
|
pillow
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
{
|
|
||||||
"main_version": "4.2.0.0",
|
|
||||||
"updater_version": "1.1.0.0",
|
|
||||||
"announcement": "\n# 这是一个中转版本,此版本后更换程序架构方式。\n# 由于更新方法无法通用,您需要在完成本次更新后再次检查更新以获取最新版本。\n",
|
|
||||||
"proxy_list":[
|
|
||||||
"",
|
|
||||||
"https://gitproxy.click/",
|
|
||||||
"https://cdn.moran233.xyz/",
|
|
||||||
"https://gh.llkk.cc/",
|
|
||||||
"https://github.akams.cn/",
|
|
||||||
"https://www.ghproxy.cn/",
|
|
||||||
"https://ghfast.top/"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
403
resources/docs/ChineseSimplified.isl
Normal file
@@ -0,0 +1,403 @@
|
|||||||
|
; *** Inno Setup version 6.4.0+ Chinese Simplified messages ***
|
||||||
|
;
|
||||||
|
; To download user-contributed translations of this file, go to:
|
||||||
|
; https://jrsoftware.org/files/istrans/
|
||||||
|
;
|
||||||
|
; Note: When translating this text, do not add periods (.) to the end of
|
||||||
|
; messages that didn't have them already, because on those messages Inno
|
||||||
|
; Setup adds the periods automatically (appending a period would result in
|
||||||
|
; two periods being displayed).
|
||||||
|
;
|
||||||
|
; Maintained by Zhenghan Yang
|
||||||
|
; Email: 847320916@QQ.com
|
||||||
|
; Translation based on network resource
|
||||||
|
; The latest Translation is on https://github.com/kira-96/Inno-Setup-Chinese-Simplified-Translation
|
||||||
|
;
|
||||||
|
|
||||||
|
[LangOptions]
|
||||||
|
; The following three entries are very important. Be sure to read and
|
||||||
|
; understand the '[LangOptions] section' topic in the help file.
|
||||||
|
LanguageName=简体中文
|
||||||
|
; If Language Name display incorrect, uncomment next line
|
||||||
|
; LanguageName=<7B80><4F53><4E2D><6587>
|
||||||
|
; About LanguageID, to reference link:
|
||||||
|
; https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-lcid/a9eac961-e77d-41a6-90a5-ce1a8b0cdb9c
|
||||||
|
LanguageID=$0804
|
||||||
|
; About CodePage, to reference link:
|
||||||
|
; https://docs.microsoft.com/en-us/windows/win32/intl/code-page-identifiers
|
||||||
|
LanguageCodePage=936
|
||||||
|
; If the language you are translating to requires special font faces or
|
||||||
|
; sizes, uncomment any of the following entries and change them accordingly.
|
||||||
|
;DialogFontName=
|
||||||
|
;DialogFontSize=8
|
||||||
|
;WelcomeFontName=Verdana
|
||||||
|
;WelcomeFontSize=12
|
||||||
|
;TitleFontName=Arial
|
||||||
|
;TitleFontSize=29
|
||||||
|
;CopyrightFontName=Arial
|
||||||
|
;CopyrightFontSize=8
|
||||||
|
|
||||||
|
[Messages]
|
||||||
|
|
||||||
|
; *** 应用程序标题
|
||||||
|
SetupAppTitle=安装
|
||||||
|
SetupWindowTitle=安装 - %1
|
||||||
|
UninstallAppTitle=卸载
|
||||||
|
UninstallAppFullTitle=%1 卸载
|
||||||
|
|
||||||
|
; *** Misc. common
|
||||||
|
InformationTitle=信息
|
||||||
|
ConfirmTitle=确认
|
||||||
|
ErrorTitle=错误
|
||||||
|
|
||||||
|
; *** SetupLdr messages
|
||||||
|
SetupLdrStartupMessage=现在将安装 %1。您想要继续吗?
|
||||||
|
LdrCannotCreateTemp=无法创建临时文件。安装程序已中止
|
||||||
|
LdrCannotExecTemp=无法执行临时目录中的文件。安装程序已中止
|
||||||
|
HelpTextNote=
|
||||||
|
|
||||||
|
; *** 启动错误消息
|
||||||
|
LastErrorMessage=%1。%n%n错误 %2: %3
|
||||||
|
SetupFileMissing=安装目录中缺少文件 %1。请修正这个问题或者获取程序的新副本。
|
||||||
|
SetupFileCorrupt=安装文件已损坏。请获取程序的新副本。
|
||||||
|
SetupFileCorruptOrWrongVer=安装文件已损坏,或是与这个安装程序的版本不兼容。请修正这个问题或获取新的程序副本。
|
||||||
|
InvalidParameter=无效的命令行参数:%n%n%1
|
||||||
|
SetupAlreadyRunning=安装程序正在运行。
|
||||||
|
WindowsVersionNotSupported=此程序不支持当前计算机运行的 Windows 版本。
|
||||||
|
WindowsServicePackRequired=此程序需要 %1 服务包 %2 或更高版本。
|
||||||
|
NotOnThisPlatform=此程序不能在 %1 上运行。
|
||||||
|
OnlyOnThisPlatform=此程序只能在 %1 上运行。
|
||||||
|
OnlyOnTheseArchitectures=此程序只能安装到为下列处理器架构设计的 Windows 版本中:%n%n%1
|
||||||
|
WinVersionTooLowError=此程序需要 %1 版本 %2 或更高。
|
||||||
|
WinVersionTooHighError=此程序不能安装于 %1 版本 %2 或更高。
|
||||||
|
AdminPrivilegesRequired=在安装此程序时您必须以管理员身份登录。
|
||||||
|
PowerUserPrivilegesRequired=在安装此程序时您必须以管理员身份或有权限的用户组身份登录。
|
||||||
|
SetupAppRunningError=安装程序发现 %1 当前正在运行。%n%n请先关闭正在运行的程序,然后点击“确定”继续,或点击“取消”退出。
|
||||||
|
UninstallAppRunningError=卸载程序发现 %1 当前正在运行。%n%n请先关闭正在运行的程序,然后点击“确定”继续,或点击“取消”退出。
|
||||||
|
|
||||||
|
; *** 启动问题
|
||||||
|
PrivilegesRequiredOverrideTitle=选择安装程序模式
|
||||||
|
PrivilegesRequiredOverrideInstruction=选择安装模式
|
||||||
|
PrivilegesRequiredOverrideText1=%1 可以为所有用户安装(需要管理员权限),或仅为您安装。
|
||||||
|
PrivilegesRequiredOverrideText2=%1 只能为您安装,或为所有用户安装(需要管理员权限)。
|
||||||
|
PrivilegesRequiredOverrideAllUsers=为所有用户安装(&A)
|
||||||
|
PrivilegesRequiredOverrideAllUsersRecommended=为所有用户安装(&A) (建议选项)
|
||||||
|
PrivilegesRequiredOverrideCurrentUser=只为我安装(&M)
|
||||||
|
PrivilegesRequiredOverrideCurrentUserRecommended=只为我安装(&M) (建议选项)
|
||||||
|
|
||||||
|
; *** 其他错误
|
||||||
|
ErrorCreatingDir=安装程序无法创建目录“%1”
|
||||||
|
ErrorTooManyFilesInDir=无法在目录“%1”中创建文件,因为里面包含太多文件
|
||||||
|
|
||||||
|
; *** 安装程序公共消息
|
||||||
|
ExitSetupTitle=退出安装程序
|
||||||
|
ExitSetupMessage=安装程序尚未完成。如果现在退出,将不会安装该程序。%n%n您之后可以再次运行安装程序完成安装。%n%n现在退出安装程序吗?
|
||||||
|
AboutSetupMenuItem=关于安装程序(&A)...
|
||||||
|
AboutSetupTitle=关于安装程序
|
||||||
|
AboutSetupMessage=%1 版本 %2%n%3%n%n%1 主页:%n%4
|
||||||
|
AboutSetupNote=
|
||||||
|
TranslatorNote=简体中文翻译由Kira(847320916@qq.com)维护。项目地址:https://github.com/kira-96/Inno-Setup-Chinese-Simplified-Translation
|
||||||
|
|
||||||
|
; *** 按钮
|
||||||
|
ButtonBack=< 上一步(&B)
|
||||||
|
ButtonNext=下一步(&N) >
|
||||||
|
ButtonInstall=安装(&I)
|
||||||
|
ButtonOK=确定
|
||||||
|
ButtonCancel=取消
|
||||||
|
ButtonYes=是(&Y)
|
||||||
|
ButtonYesToAll=全是(&A)
|
||||||
|
ButtonNo=否(&N)
|
||||||
|
ButtonNoToAll=全否(&O)
|
||||||
|
ButtonFinish=完成(&F)
|
||||||
|
ButtonBrowse=浏览(&B)...
|
||||||
|
ButtonWizardBrowse=浏览(&R)...
|
||||||
|
ButtonNewFolder=新建文件夹(&M)
|
||||||
|
|
||||||
|
; *** “选择语言”对话框消息
|
||||||
|
SelectLanguageTitle=选择安装语言
|
||||||
|
SelectLanguageLabel=选择安装时使用的语言。
|
||||||
|
|
||||||
|
; *** 公共向导文字
|
||||||
|
ClickNext=点击“下一步”继续,或点击“取消”退出安装程序。
|
||||||
|
BeveledLabel=
|
||||||
|
BrowseDialogTitle=浏览文件夹
|
||||||
|
BrowseDialogLabel=在下面的列表中选择一个文件夹,然后点击“确定”。
|
||||||
|
NewFolderName=新建文件夹
|
||||||
|
|
||||||
|
; *** “欢迎”向导页
|
||||||
|
WelcomeLabel1=欢迎使用 [name] 安装向导
|
||||||
|
WelcomeLabel2=现在将安装 [name/ver] 到您的电脑中。%n%n建议您在继续安装前关闭所有其他应用程序。
|
||||||
|
|
||||||
|
; *** “密码”向导页
|
||||||
|
WizardPassword=密码
|
||||||
|
PasswordLabel1=这个安装程序有密码保护。
|
||||||
|
PasswordLabel3=请输入密码,然后点击“下一步”继续。密码区分大小写。
|
||||||
|
PasswordEditLabel=密码(&P):
|
||||||
|
IncorrectPassword=您输入的密码不正确,请重新输入。
|
||||||
|
|
||||||
|
; *** “许可协议”向导页
|
||||||
|
WizardLicense=许可协议
|
||||||
|
LicenseLabel=请在继续安装前阅读以下重要信息。
|
||||||
|
LicenseLabel3=请仔细阅读下列许可协议。在继续安装前您必须同意这些协议条款。
|
||||||
|
LicenseAccepted=我同意此协议(&A)
|
||||||
|
LicenseNotAccepted=我不同意此协议(&D)
|
||||||
|
|
||||||
|
; *** “信息”向导页
|
||||||
|
WizardInfoBefore=信息
|
||||||
|
InfoBeforeLabel=请在继续安装前阅读以下重要信息。
|
||||||
|
InfoBeforeClickLabel=准备好继续安装后,点击“下一步”。
|
||||||
|
WizardInfoAfter=信息
|
||||||
|
InfoAfterLabel=请在继续安装前阅读以下重要信息。
|
||||||
|
InfoAfterClickLabel=准备好继续安装后,点击“下一步”。
|
||||||
|
|
||||||
|
; *** “用户信息”向导页
|
||||||
|
WizardUserInfo=用户信息
|
||||||
|
UserInfoDesc=请输入您的信息。
|
||||||
|
UserInfoName=用户名(&U):
|
||||||
|
UserInfoOrg=组织(&O):
|
||||||
|
UserInfoSerial=序列号(&S):
|
||||||
|
UserInfoNameRequired=您必须输入用户名。
|
||||||
|
|
||||||
|
; *** “选择目标目录”向导页
|
||||||
|
WizardSelectDir=选择目标位置
|
||||||
|
SelectDirDesc=您想将 [name] 安装在哪里?
|
||||||
|
SelectDirLabel3=安装程序将安装 [name] 到下面的文件夹中。
|
||||||
|
SelectDirBrowseLabel=点击“下一步”继续。如果您想选择其他文件夹,点击“浏览”。
|
||||||
|
DiskSpaceGBLabel=至少需要有 [gb] GB 的可用磁盘空间。
|
||||||
|
DiskSpaceMBLabel=至少需要有 [mb] MB 的可用磁盘空间。
|
||||||
|
CannotInstallToNetworkDrive=安装程序无法安装到一个网络驱动器。
|
||||||
|
CannotInstallToUNCPath=安装程序无法安装到一个 UNC 路径。
|
||||||
|
InvalidPath=您必须输入一个带驱动器卷标的完整路径,例如:%n%nC:\APP%n%n或UNC路径:%n%n\\server\share
|
||||||
|
InvalidDrive=您选定的驱动器或 UNC 共享不存在或不能访问。请选择其他位置。
|
||||||
|
DiskSpaceWarningTitle=磁盘空间不足
|
||||||
|
DiskSpaceWarning=安装程序至少需要 %1 KB 的可用空间才能安装,但选定驱动器只有 %2 KB 的可用空间。%n%n您一定要继续吗?
|
||||||
|
DirNameTooLong=文件夹名称或路径太长。
|
||||||
|
InvalidDirName=文件夹名称无效。
|
||||||
|
BadDirName32=文件夹名称不能包含下列任何字符:%n%n%1
|
||||||
|
DirExistsTitle=文件夹已存在
|
||||||
|
DirExists=文件夹:%n%n%1%n%n已经存在。您一定要安装到这个文件夹中吗?
|
||||||
|
DirDoesntExistTitle=文件夹不存在
|
||||||
|
DirDoesntExist=文件夹:%n%n%1%n%n不存在。您想要创建此文件夹吗?
|
||||||
|
|
||||||
|
; *** “选择组件”向导页
|
||||||
|
WizardSelectComponents=选择组件
|
||||||
|
SelectComponentsDesc=您想安装哪些程序组件?
|
||||||
|
SelectComponentsLabel2=选中您想安装的组件;取消您不想安装的组件。然后点击“下一步”继续。
|
||||||
|
FullInstallation=完全安装
|
||||||
|
; if possible don't translate 'Compact' as 'Minimal' (I mean 'Minimal' in your language)
|
||||||
|
CompactInstallation=简洁安装
|
||||||
|
CustomInstallation=自定义安装
|
||||||
|
NoUninstallWarningTitle=组件已存在
|
||||||
|
NoUninstallWarning=安装程序检测到下列组件已安装在您的电脑中:%n%n%1%n%n取消选中这些组件不会卸载它们。%n%n确定要继续吗?
|
||||||
|
ComponentSize1=%1 KB
|
||||||
|
ComponentSize2=%1 MB
|
||||||
|
ComponentsDiskSpaceGBLabel=当前选择的组件需要至少 [gb] GB 的磁盘空间。
|
||||||
|
ComponentsDiskSpaceMBLabel=当前选择的组件需要至少 [mb] MB 的磁盘空间。
|
||||||
|
|
||||||
|
; *** “选择附加任务”向导页
|
||||||
|
WizardSelectTasks=选择附加任务
|
||||||
|
SelectTasksDesc=您想要安装程序执行哪些附加任务?
|
||||||
|
SelectTasksLabel2=选择您想要安装程序在安装 [name] 时执行的附加任务,然后点击“下一步”。
|
||||||
|
|
||||||
|
; *** “选择开始菜单文件夹”向导页
|
||||||
|
WizardSelectProgramGroup=选择开始菜单文件夹
|
||||||
|
SelectStartMenuFolderDesc=安装程序应该在哪里放置程序的快捷方式?
|
||||||
|
SelectStartMenuFolderLabel3=安装程序将在下列“开始”菜单文件夹中创建程序的快捷方式。
|
||||||
|
SelectStartMenuFolderBrowseLabel=点击“下一步”继续。如果您想选择其他文件夹,点击“浏览”。
|
||||||
|
MustEnterGroupName=您必须输入一个文件夹名。
|
||||||
|
GroupNameTooLong=文件夹名或路径太长。
|
||||||
|
InvalidGroupName=无效的文件夹名字。
|
||||||
|
BadGroupName=文件夹名不能包含下列任何字符:%n%n%1
|
||||||
|
NoProgramGroupCheck2=不创建开始菜单文件夹(&D)
|
||||||
|
|
||||||
|
; *** “准备安装”向导页
|
||||||
|
WizardReady=准备安装
|
||||||
|
ReadyLabel1=安装程序准备就绪,现在可以开始安装 [name] 到您的电脑。
|
||||||
|
ReadyLabel2a=点击“安装”继续此安装程序。如果您想重新考虑或修改任何设置,点击“上一步”。
|
||||||
|
ReadyLabel2b=点击“安装”继续此安装程序。
|
||||||
|
ReadyMemoUserInfo=用户信息:
|
||||||
|
ReadyMemoDir=目标位置:
|
||||||
|
ReadyMemoType=安装类型:
|
||||||
|
ReadyMemoComponents=已选择组件:
|
||||||
|
ReadyMemoGroup=开始菜单文件夹:
|
||||||
|
ReadyMemoTasks=附加任务:
|
||||||
|
|
||||||
|
; *** TExtractionWizardPage wizard page and Extract7ZipArchive
|
||||||
|
ExtractionLabel=正在提取附加文件...
|
||||||
|
ButtonStopExtraction=停止提取(&S)
|
||||||
|
StopExtraction=您确定要停止提取吗?
|
||||||
|
ErrorExtractionAborted=提取已中止
|
||||||
|
ErrorExtractionFailed=提取失败:%1
|
||||||
|
|
||||||
|
; *** TDownloadWizardPage wizard page and DownloadTemporaryFile
|
||||||
|
DownloadingLabel=正在下载附加文件...
|
||||||
|
ButtonStopDownload=停止下载(&S)
|
||||||
|
StopDownload=您确定要停止下载吗?
|
||||||
|
ErrorDownloadAborted=下载已中止
|
||||||
|
ErrorDownloadFailed=下载失败:%1 %2
|
||||||
|
ErrorDownloadSizeFailed=获取下载大小失败:%1 %2
|
||||||
|
ErrorFileHash1=校验文件哈希失败:%1
|
||||||
|
ErrorFileHash2=无效的文件哈希:预期 %1,实际 %2
|
||||||
|
ErrorProgress=无效的进度:%1 / %2
|
||||||
|
ErrorFileSize=文件大小错误:预期 %1,实际 %2
|
||||||
|
|
||||||
|
; *** “正在准备安装”向导页
|
||||||
|
WizardPreparing=正在准备安装
|
||||||
|
PreparingDesc=安装程序正在准备安装 [name] 到您的电脑。
|
||||||
|
PreviousInstallNotCompleted=先前的程序安装或卸载未完成,您需要重启您的电脑以完成。%n%n在重启电脑后,再次运行安装程序以完成 [name] 的安装。
|
||||||
|
CannotContinue=安装程序不能继续。请点击“取消”退出。
|
||||||
|
ApplicationsFound=以下应用程序正在使用将由安装程序更新的文件。建议您允许安装程序自动关闭这些应用程序。
|
||||||
|
ApplicationsFound2=以下应用程序正在使用将由安装程序更新的文件。建议您允许安装程序自动关闭这些应用程序。安装完成后,安装程序将尝试重新启动这些应用程序。
|
||||||
|
CloseApplications=自动关闭应用程序(&A)
|
||||||
|
DontCloseApplications=不要关闭应用程序(&D)
|
||||||
|
ErrorCloseApplications=安装程序无法自动关闭所有应用程序。建议您在继续之前,关闭所有在使用需要由安装程序更新的文件的应用程序。
|
||||||
|
PrepareToInstallNeedsRestart=安装程序必须重启您的计算机。计算机重启后,请再次运行安装程序以完成 [name] 的安装。%n%n是否立即重新启动?
|
||||||
|
|
||||||
|
; *** “正在安装”向导页
|
||||||
|
WizardInstalling=正在安装
|
||||||
|
InstallingLabel=安装程序正在安装 [name] 到您的电脑,请稍候。
|
||||||
|
|
||||||
|
; *** “安装完成”向导页
|
||||||
|
FinishedHeadingLabel=[name] 安装完成
|
||||||
|
FinishedLabelNoIcons=安装程序已在您的电脑中安装了 [name]。
|
||||||
|
FinishedLabel=安装程序已在您的电脑中安装了 [name]。您可以通过已安装的快捷方式运行此应用程序。
|
||||||
|
ClickFinish=点击“完成”退出安装程序。
|
||||||
|
FinishedRestartLabel=为完成 [name] 的安装,安装程序必须重新启动您的电脑。要立即重启吗?
|
||||||
|
FinishedRestartMessage=为完成 [name] 的安装,安装程序必须重新启动您的电脑。%n%n要立即重启吗?
|
||||||
|
ShowReadmeCheck=是,我想查阅自述文件
|
||||||
|
YesRadio=是,立即重启电脑(&Y)
|
||||||
|
NoRadio=否,稍后重启电脑(&N)
|
||||||
|
; used for example as 'Run MyProg.exe'
|
||||||
|
RunEntryExec=运行 %1
|
||||||
|
; used for example as 'View Readme.txt'
|
||||||
|
RunEntryShellExec=查阅 %1
|
||||||
|
|
||||||
|
; *** “安装程序需要下一张磁盘”提示
|
||||||
|
ChangeDiskTitle=安装程序需要下一张磁盘
|
||||||
|
SelectDiskLabel2=请插入磁盘 %1 并点击“确定”。%n%n如果这个磁盘中的文件可以在下列文件夹之外的文件夹中找到,请输入正确的路径或点击“浏览”。
|
||||||
|
PathLabel=路径(&P):
|
||||||
|
FileNotInDir2=“%2”中找不到文件“%1”。请插入正确的磁盘或选择其他文件夹。
|
||||||
|
SelectDirectoryLabel=请指定下一张磁盘的位置。
|
||||||
|
|
||||||
|
; *** 安装状态消息
|
||||||
|
SetupAborted=安装程序未完成安装。%n%n请修正这个问题并重新运行安装程序。
|
||||||
|
AbortRetryIgnoreSelectAction=选择操作
|
||||||
|
AbortRetryIgnoreRetry=重试(&T)
|
||||||
|
AbortRetryIgnoreIgnore=忽略错误并继续(&I)
|
||||||
|
AbortRetryIgnoreCancel=关闭安装程序
|
||||||
|
|
||||||
|
; *** 安装状态消息
|
||||||
|
StatusClosingApplications=正在关闭应用程序...
|
||||||
|
StatusCreateDirs=正在创建目录...
|
||||||
|
StatusExtractFiles=正在解压缩文件...
|
||||||
|
StatusCreateIcons=正在创建快捷方式...
|
||||||
|
StatusCreateIniEntries=正在创建 INI 条目...
|
||||||
|
StatusCreateRegistryEntries=正在创建注册表条目...
|
||||||
|
StatusRegisterFiles=正在注册文件...
|
||||||
|
StatusSavingUninstall=正在保存卸载信息...
|
||||||
|
StatusRunProgram=正在完成安装...
|
||||||
|
StatusRestartingApplications=正在重启应用程序...
|
||||||
|
StatusRollback=正在撤销更改...
|
||||||
|
|
||||||
|
; *** 其他错误
|
||||||
|
ErrorInternal2=内部错误:%1
|
||||||
|
ErrorFunctionFailedNoCode=%1 失败
|
||||||
|
ErrorFunctionFailed=%1 失败;错误代码 %2
|
||||||
|
ErrorFunctionFailedWithMessage=%1 失败;错误代码 %2.%n%3
|
||||||
|
ErrorExecutingProgram=无法执行文件:%n%1
|
||||||
|
|
||||||
|
; *** 注册表错误
|
||||||
|
ErrorRegOpenKey=打开注册表项时出错:%n%1\%2
|
||||||
|
ErrorRegCreateKey=创建注册表项时出错:%n%1\%2
|
||||||
|
ErrorRegWriteKey=写入注册表项时出错:%n%1\%2
|
||||||
|
|
||||||
|
; *** INI 错误
|
||||||
|
ErrorIniEntry=在文件“%1”中创建 INI 条目时出错。
|
||||||
|
|
||||||
|
; *** 文件复制错误
|
||||||
|
FileAbortRetryIgnoreSkipNotRecommended=跳过此文件(&S) (不推荐)
|
||||||
|
FileAbortRetryIgnoreIgnoreNotRecommended=忽略错误并继续(&I) (不推荐)
|
||||||
|
SourceIsCorrupted=源文件已损坏
|
||||||
|
SourceDoesntExist=源文件“%1”不存在
|
||||||
|
ExistingFileReadOnly2=无法替换现有文件,它是只读的。
|
||||||
|
ExistingFileReadOnlyRetry=移除只读属性并重试(&R)
|
||||||
|
ExistingFileReadOnlyKeepExisting=保留现有文件(&K)
|
||||||
|
ErrorReadingExistingDest=尝试读取现有文件时出错:
|
||||||
|
FileExistsSelectAction=选择操作
|
||||||
|
FileExists2=文件已经存在。
|
||||||
|
FileExistsOverwriteExisting=覆盖已存在的文件(&O)
|
||||||
|
FileExistsKeepExisting=保留现有的文件(&K)
|
||||||
|
FileExistsOverwriteOrKeepAll=为所有冲突文件执行此操作(&D)
|
||||||
|
ExistingFileNewerSelectAction=选择操作
|
||||||
|
ExistingFileNewer2=现有的文件比安装程序将要安装的文件还要新。
|
||||||
|
ExistingFileNewerOverwriteExisting=覆盖已存在的文件(&O)
|
||||||
|
ExistingFileNewerKeepExisting=保留现有的文件(&K) (推荐)
|
||||||
|
ExistingFileNewerOverwriteOrKeepAll=为所有冲突文件执行此操作(&D)
|
||||||
|
ErrorChangingAttr=尝试更改下列现有文件的属性时出错:
|
||||||
|
ErrorCreatingTemp=尝试在目标目录创建文件时出错:
|
||||||
|
ErrorReadingSource=尝试读取下列源文件时出错:
|
||||||
|
ErrorCopying=尝试复制下列文件时出错:
|
||||||
|
ErrorReplacingExistingFile=尝试替换现有文件时出错:
|
||||||
|
ErrorRestartReplace=重启并替换失败:
|
||||||
|
ErrorRenamingTemp=尝试重命名下列目标目录中的一个文件时出错:
|
||||||
|
ErrorRegisterServer=无法注册 DLL/OCX:%1
|
||||||
|
ErrorRegSvr32Failed=RegSvr32 失败;退出代码 %1
|
||||||
|
ErrorRegisterTypeLib=无法注册类库:%1
|
||||||
|
|
||||||
|
; *** 卸载显示名字标记
|
||||||
|
; used for example as 'My Program (32-bit)'
|
||||||
|
UninstallDisplayNameMark=%1 (%2)
|
||||||
|
; used for example as 'My Program (32-bit, All users)'
|
||||||
|
UninstallDisplayNameMarks=%1 (%2, %3)
|
||||||
|
UninstallDisplayNameMark32Bit=32 位
|
||||||
|
UninstallDisplayNameMark64Bit=64 位
|
||||||
|
UninstallDisplayNameMarkAllUsers=所有用户
|
||||||
|
UninstallDisplayNameMarkCurrentUser=当前用户
|
||||||
|
|
||||||
|
; *** 安装后错误
|
||||||
|
ErrorOpeningReadme=尝试打开自述文件时出错。
|
||||||
|
ErrorRestartingComputer=安装程序无法重启电脑,请手动重启。
|
||||||
|
|
||||||
|
; *** 卸载消息
|
||||||
|
UninstallNotFound=文件“%1”不存在。无法卸载。
|
||||||
|
UninstallOpenError=文件“%1”不能被打开。无法卸载。
|
||||||
|
UninstallUnsupportedVer=此版本的卸载程序无法识别卸载日志文件“%1”的格式。无法卸载
|
||||||
|
UninstallUnknownEntry=卸载日志中遇到一个未知条目 (%1)
|
||||||
|
ConfirmUninstall=您确认要完全移除 %1 及其所有组件吗?
|
||||||
|
UninstallOnlyOnWin64=仅允许在 64 位 Windows 中卸载此程序。
|
||||||
|
OnlyAdminCanUninstall=仅使用管理员权限的用户能完成此卸载。
|
||||||
|
UninstallStatusLabel=正在从您的电脑中移除 %1,请稍候。
|
||||||
|
UninstalledAll=已顺利从您的电脑中移除 %1。
|
||||||
|
UninstalledMost=%1 卸载完成。%n%n有部分内容未能被删除,但您可以手动删除它们。
|
||||||
|
UninstalledAndNeedsRestart=为完成 %1 的卸载,需要重启您的电脑。%n%n立即重启电脑吗?
|
||||||
|
UninstallDataCorrupted=文件“%1”已损坏。无法卸载
|
||||||
|
|
||||||
|
; *** 卸载状态消息
|
||||||
|
ConfirmDeleteSharedFileTitle=删除共享的文件吗?
|
||||||
|
ConfirmDeleteSharedFile2=系统表示下列共享的文件已不有其他程序使用。您希望卸载程序删除这些共享的文件吗?%n%n如果删除这些文件,但仍有程序在使用这些文件,则这些程序可能出现异常。如果您不能确定,请选择“否”,在系统中保留这些文件以免引发问题。
|
||||||
|
SharedFileNameLabel=文件名:
|
||||||
|
SharedFileLocationLabel=位置:
|
||||||
|
WizardUninstalling=卸载状态
|
||||||
|
StatusUninstalling=正在卸载 %1...
|
||||||
|
|
||||||
|
; *** Shutdown block reasons
|
||||||
|
ShutdownBlockReasonInstallingApp=正在安装 %1。
|
||||||
|
ShutdownBlockReasonUninstallingApp=正在卸载 %1。
|
||||||
|
|
||||||
|
; The custom messages below aren't used by Setup itself, but if you make
|
||||||
|
; use of them in your scripts, you'll want to translate them.
|
||||||
|
|
||||||
|
[CustomMessages]
|
||||||
|
|
||||||
|
NameAndVersion=%1 版本 %2
|
||||||
|
AdditionalIcons=附加快捷方式:
|
||||||
|
CreateDesktopIcon=创建桌面快捷方式(&D)
|
||||||
|
CreateQuickLaunchIcon=创建快速启动栏快捷方式(&Q)
|
||||||
|
ProgramOnTheWeb=%1 网站
|
||||||
|
UninstallProgram=卸载 %1
|
||||||
|
LaunchProgram=运行 %1
|
||||||
|
AssocFileExtension=将 %2 文件扩展名与 %1 建立关联(&A)
|
||||||
|
AssocingFileExtension=正在将 %2 文件扩展名与 %1 建立关联...
|
||||||
|
AutoStartProgramGroupDescription=启动:
|
||||||
|
AutoStartProgram=自动启动 %1
|
||||||
|
AddonHostProgramNotFound=您选择的文件夹中无法找到 %1。%n%n您要继续吗?
|
||||||
@@ -6,11 +6,21 @@
|
|||||||
"TaskQueue.Recruiting.IsChecked": "True" #自动公招
|
"TaskQueue.Recruiting.IsChecked": "True" #自动公招
|
||||||
"TaskQueue.Base.IsChecked": "True" #基建换班
|
"TaskQueue.Base.IsChecked": "True" #基建换班
|
||||||
"TaskQueue.Combat.IsChecked": "True" #刷理智
|
"TaskQueue.Combat.IsChecked": "True" #刷理智
|
||||||
"TaskQueue.Mission.IsChecked": "True" #领取奖励
|
|
||||||
"TaskQueue.Mall.IsChecked": "True" #获取信用及购物
|
"TaskQueue.Mall.IsChecked": "True" #获取信用及购物
|
||||||
|
"TaskQueue.Mission.IsChecked": "True" #领取奖励
|
||||||
"TaskQueue.AutoRoguelike.IsChecked": "False" #自动肉鸽
|
"TaskQueue.AutoRoguelike.IsChecked": "False" #自动肉鸽
|
||||||
"TaskQueue.Reclamation.IsChecked": "False" #生息演算
|
"TaskQueue.Reclamation.IsChecked": "False" #生息演算
|
||||||
|
"TaskQueue.Order.WakeUp": "0"
|
||||||
|
"TaskQueue.Order.Recruiting": "1"
|
||||||
|
"TaskQueue.Order.Base": "2"
|
||||||
|
"TaskQueue.Order.Combat": "3"
|
||||||
|
"TaskQueue.Order.Mall": "4"
|
||||||
|
"TaskQueue.Order.Mission": "5"
|
||||||
|
"TaskQueue.Order.AutoRoguelike": "6"
|
||||||
|
"TaskQueue.Order.Reclamation": "7"
|
||||||
#刷理智
|
#刷理智
|
||||||
|
"MainFunction.UseMedicine": "True" #吃理智药
|
||||||
|
"MainFunction.UseMedicine.Quantity": "999" #吃理智药数量
|
||||||
"MainFunction.Stage1": "" #主关卡
|
"MainFunction.Stage1": "" #主关卡
|
||||||
"MainFunction.Stage2": "" #备选关卡1
|
"MainFunction.Stage2": "" #备选关卡1
|
||||||
"MainFunction.Stage3": "" #备选关卡2
|
"MainFunction.Stage3": "" #备选关卡2
|
||||||
@@ -28,14 +38,16 @@
|
|||||||
"Penguin.EnablePenguin": "True" #上报企鹅物流
|
"Penguin.EnablePenguin": "True" #上报企鹅物流
|
||||||
"Yituliu.EnableYituliu": "True" #上报一图流
|
"Yituliu.EnableYituliu": "True" #上报一图流
|
||||||
#基建换班
|
#基建换班
|
||||||
"Infrast.CustomInfrastEnabled": "True" #启用自定义基建配置
|
"Infrast.InfrastMode": "Normal"、"Rotation"、"Custom" #基建模式
|
||||||
"Infrast.CustomInfrastPlanIndex": "1" #自定义基建配置索引
|
"Infrast.CustomInfrastPlanIndex": "1" #自定义基建配置索引号
|
||||||
"Infrast.DefaultInfrast": "user_defined" #内置配置
|
"Infrast.DefaultInfrast": "user_defined" #内置配置
|
||||||
"Infrast.IsCustomInfrastFileReadOnly": "False" #自定义基建配置文件只读
|
"Infrast.IsCustomInfrastFileReadOnly": "False" #自定义基建配置文件只读
|
||||||
"Infrast.CustomInfrastFile": "" #自定义基建配置文件地址
|
"Infrast.CustomInfrastFile": "" #自定义基建配置文件地址
|
||||||
#设置
|
#设置
|
||||||
"Start.ClientType": "Bilibili"、 "Official" #服务器
|
"Start.ClientType": "Bilibili"、 "Official" #服务器
|
||||||
G"Timer.Timer1": "False" #时间设置1
|
G"Timer.Timer1": "False" #时间设置1
|
||||||
|
"Connect.AdbPath" #ADB路径
|
||||||
|
"Connect.Address": "127.0.0.1:16448" #连接地址
|
||||||
G"VersionUpdate.ScheduledUpdateCheck": "True" #定时检查更新
|
G"VersionUpdate.ScheduledUpdateCheck": "True" #定时检查更新
|
||||||
G"VersionUpdate.AutoDownloadUpdatePackage": "True" #自动下载更新包
|
G"VersionUpdate.AutoDownloadUpdatePackage": "True" #自动下载更新包
|
||||||
G"VersionUpdate.AutoInstallUpdatePackage": "True" #自动安装更新包
|
G"VersionUpdate.AutoInstallUpdatePackage": "True" #自动安装更新包
|
||||||
@@ -44,4 +56,7 @@ G"Start.MinimizeDirectly": "True" #启动MAA后直接最小化
|
|||||||
"Start.OpenEmulatorAfterLaunch": "True" #启动MAA后自动开启模拟器
|
"Start.OpenEmulatorAfterLaunch": "True" #启动MAA后自动开启模拟器
|
||||||
G"GUI.UseTray": "True" #显示托盘图标
|
G"GUI.UseTray": "True" #显示托盘图标
|
||||||
G"GUI.MinimizeToTray": "False" #最小化时隐藏至托盘
|
G"GUI.MinimizeToTray": "False" #最小化时隐藏至托盘
|
||||||
"Start.EmulatorPath" #模拟器路径
|
"Start.EmulatorPath" #模拟器路径
|
||||||
|
"Start.EmulatorAddCommand": "-v 2" #附加命令
|
||||||
|
"Start.EmulatorWaitSeconds": "10" #等待模拟器启动时间
|
||||||
|
G"VersionUpdate.package": "MirrorChyanAppv5.15.6.zip" #更新包标识
|
||||||
BIN
resources/icons/MirrorChyan.ico
Normal file
|
After Width: | Height: | Size: 17 KiB |
|
Before Width: | Height: | Size: 2.8 MiB |
|
Before Width: | Height: | Size: 2.8 MiB After Width: | Height: | Size: 1.7 MiB |
|
Before Width: | Height: | Size: 61 KiB |
|
Before Width: | Height: | Size: 21 KiB |
BIN
resources/images/notification/six_star.png
Normal file
|
After Width: | Height: | Size: 222 KiB |
BIN
resources/images/notification/test_notify.png
Normal file
|
After Width: | Height: | Size: 32 KiB |
BIN
resources/sounds/both/删除用户.wav
Normal file
BIN
resources/sounds/both/删除脚本实例.wav
Normal file
BIN
resources/sounds/both/删除计划表.wav
Normal file
BIN
resources/sounds/both/删除调度队列.wav
Normal file
BIN
resources/sounds/both/欢迎回来.wav
Normal file
BIN
resources/sounds/both/添加用户.wav
Normal file
BIN
resources/sounds/both/添加脚本实例.wav
Normal file
BIN
resources/sounds/both/添加计划表.wav
Normal file
BIN
resources/sounds/both/添加调度队列.wav
Normal file
BIN
resources/sounds/noisy/ADB失败.wav
Normal file
BIN
resources/sounds/noisy/ADB成功.wav
Normal file
BIN
resources/sounds/noisy/MAA在完成任务前中止.wav
Normal file
BIN
resources/sounds/noisy/MAA在完成任务前退出.wav
Normal file
BIN
resources/sounds/noisy/MAA更新.wav
Normal file
BIN
resources/sounds/noisy/MAA未检测到任何模拟器.wav
Normal file
BIN
resources/sounds/noisy/MAA未能正确登录PRTS.wav
Normal file
BIN
resources/sounds/noisy/MAA的ADB连接异常.wav
Normal file
BIN
resources/sounds/noisy/MAA进程超时.wav
Normal file
BIN
resources/sounds/noisy/MAA部分任务执行失败.wav
Normal file
BIN
resources/sounds/noisy/任务开始.wav
Normal file
BIN
resources/sounds/noisy/任务结束.wav
Normal file
BIN
resources/sounds/noisy/公告展示.wav
Normal file
BIN
resources/sounds/noisy/公告通知.wav
Normal file
BIN
resources/sounds/noisy/六星喜报.wav
Normal file
BIN
resources/sounds/noisy/历史记录查询.wav
Normal file
BIN
resources/sounds/noisy/发生异常.wav
Normal file
BIN
resources/sounds/noisy/发生错误.wav
Normal file
BIN
resources/sounds/noisy/子任务失败.wav
Normal file
BIN
resources/sounds/noisy/排查录入.wav
Normal file
BIN
resources/sounds/noisy/排查重试.wav
Normal file
BIN
resources/sounds/noisy/无新版本.wav
Normal file
BIN
resources/sounds/noisy/有新版本.wav
Normal file
BIN
resources/sounds/noisy/森空岛签到失败.wav
Normal file
BIN
resources/sounds/noisy/森空岛签到成功.wav
Normal file
BIN
resources/sounds/simple/任务开始.wav
Normal file
BIN
resources/sounds/simple/任务结束.wav
Normal file
BIN
resources/sounds/simple/公告展示.wav
Normal file
BIN
resources/sounds/simple/公告通知.wav
Normal file
BIN
resources/sounds/simple/历史记录查询.wav
Normal file
BIN
resources/sounds/simple/发生异常.wav
Normal file
BIN
resources/sounds/simple/发生错误.wav
Normal file
BIN
resources/sounds/simple/无新版本.wav
Normal file
BIN
resources/sounds/simple/有新版本.wav
Normal file
@@ -1,14 +1,43 @@
|
|||||||
{
|
{
|
||||||
"main_version": "4.2.5.1",
|
"main_version": "4.3.11.0",
|
||||||
"updater_version": "1.1.2.1",
|
"version_info": {
|
||||||
"announcement": "\n## 新增功能\n- 暂无\n## 修复BUG\n- 修复统计信息HTML模板公招匹配错误\n## 程序优化\n- 暂无",
|
"4.3.11.0": {
|
||||||
"proxy_list": [
|
"修复BUG": [
|
||||||
"",
|
"临时固定Nuitka打包版本号",
|
||||||
"https://gitproxy.click/",
|
"修复删除计划表引发的错误"
|
||||||
"https://cdn.moran233.xyz/",
|
]
|
||||||
"https://gh.llkk.cc/",
|
},
|
||||||
"https://github.akams.cn/",
|
"4.3.10.0": {
|
||||||
"https://www.ghproxy.cn/",
|
"新增功能": [
|
||||||
"https://ghfast.top/"
|
"更换全新默认主页图",
|
||||||
]
|
"适配 MAA 无`Default`配置情况 #52"
|
||||||
|
],
|
||||||
|
"程序优化": [
|
||||||
|
"静默模式控制时段延长至模拟器完成启动的10s后"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"4.3.10.3": {
|
||||||
|
"程序优化": [
|
||||||
|
"使用 keyboard 模块替代 pyautogui 模块"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"4.3.10.2": {
|
||||||
|
"新增功能": [
|
||||||
|
"公招喜报模板优化",
|
||||||
|
"支持使用命令行调用"
|
||||||
|
],
|
||||||
|
"修复BUG": [
|
||||||
|
"修复更新动作重复执行问题"
|
||||||
|
],
|
||||||
|
"程序优化": [
|
||||||
|
"Mirror 酱链接添加`source`字段,用于标识来源",
|
||||||
|
"优化下载器测速中止条件"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"4.3.10.1": {
|
||||||
|
"新增功能": [
|
||||||
|
"森空岛签到功能上线"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||