71 Commits

Author SHA1 Message Date
wangdage12
d342b37dc0 Update HarvestDirectory path in wixproj file 2025-12-19 21:28:55 +08:00
fanbook-wangdage
179177a77c Merge branch 'main' of https://github.com/wangdage12/Snap.Hutao 2025-12-19 21:20:14 +08:00
fanbook-wangdage
6c68a55d81 解决元数据导致的问题 2025-12-19 21:19:47 +08:00
wangdage12
7bd61c8035 Merge pull request #14 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/upload-artifact-6
Bump actions/upload-artifact from 5 to 6 in /.github/workflows
2025-12-17 20:20:52 +08:00
wangdage12
c19b71e2c4 Merge pull request #15 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/cache-5
Bump actions/cache from 4 to 5 in /.github/workflows
2025-12-17 20:20:38 +08:00
wangdage12
45b7383fc1 Merge pull request #16 from wangdage12/dependabot/github_actions/dot-github/workflows/dessant/lock-threads-6
Bump dessant/lock-threads from 5 to 6 in /.github/workflows
2025-12-17 20:20:23 +08:00
dependabot[bot]
c83a2f3e9d Bump dessant/lock-threads from 5 to 6 in /.github/workflows
Bumps [dessant/lock-threads](https://github.com/dessant/lock-threads) from 5 to 6.
- [Release notes](https://github.com/dessant/lock-threads/releases)
- [Changelog](https://github.com/dessant/lock-threads/blob/main/CHANGELOG.md)
- [Commits](https://github.com/dessant/lock-threads/compare/v5...v6)

---
updated-dependencies:
- dependency-name: dessant/lock-threads
  dependency-version: '6'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-15 02:30:01 +00:00
dependabot[bot]
2bab0baf69 Bump actions/cache from 4 to 5 in /.github/workflows
Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5.
- [Release notes](https://github.com/actions/cache/releases)
- [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md)
- [Commits](https://github.com/actions/cache/compare/v4...v5)

---
updated-dependencies:
- dependency-name: actions/cache
  dependency-version: '5'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-15 02:29:57 +00:00
dependabot[bot]
2726e74731 Bump actions/upload-artifact from 5 to 6 in /.github/workflows
Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 5 to 6.
- [Release notes](https://github.com/actions/upload-artifact/releases)
- [Commits](https://github.com/actions/upload-artifact/compare/v5...v6)

---
updated-dependencies:
- dependency-name: actions/upload-artifact
  dependency-version: '6'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-15 02:29:52 +00:00
fanbook-wangdage
6d08f669e7 更新服务器公钥 2025-12-07 16:35:15 +08:00
fanbook-wangdage
84b9b97059 更新服务器域名、更新Sentry配置 2025-12-05 21:06:59 +08:00
wangdage12
0b846f11b7 Merge pull request #6 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/upload-artifact-5
Bump actions/upload-artifact from 4 to 5 in /.github/workflows
2025-12-03 22:52:23 +08:00
wangdage12
c9adc06210 Merge pull request #7 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/setup-dotnet-5
Bump actions/setup-dotnet from 4 to 5 in /.github/workflows
2025-12-03 22:51:59 +08:00
wangdage12
6c9f50b055 先删除ISSUE模板,之后有必要时再配置 2025-12-03 19:40:24 +08:00
wangdage12
6c515caa88 Update README 2025-12-03 16:54:06 +08:00
dependabot[bot]
b834ae5425 Bump actions/setup-dotnet from 4 to 5 in /.github/workflows
Bumps [actions/setup-dotnet](https://github.com/actions/setup-dotnet) from 4 to 5.
- [Release notes](https://github.com/actions/setup-dotnet/releases)
- [Commits](https://github.com/actions/setup-dotnet/compare/v4...v5)

---
updated-dependencies:
- dependency-name: actions/setup-dotnet
  dependency-version: '5'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-02 15:00:14 +00:00
wangdage12
49ae21e02c Merge pull request #8 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/checkout-6
Bump actions/checkout from 4 to 6 in /.github/workflows
2025-12-02 22:59:09 +08:00
wangdage12
88f81c5582 Update status of new monsters and achievements 2025-12-02 22:56:48 +08:00
wangdage12
274b2766ce Update MSI path in workflow configuration 2025-12-02 22:14:13 +08:00
dependabot[bot]
b9130979c1 Bump actions/upload-artifact from 4 to 5 in /.github/workflows
Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 4 to 5.
- [Release notes](https://github.com/actions/upload-artifact/releases)
- [Commits](https://github.com/actions/upload-artifact/compare/v4...v5)

---
updated-dependencies:
- dependency-name: actions/upload-artifact
  dependency-version: '5'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-02 14:01:28 +00:00
wangdage12
5019987c08 Add Release configuration to MSI installer build 2025-12-02 22:00:31 +08:00
wangdage12
adaf0972ce Update TargetFramework and HarvestDirectory paths 2025-12-02 21:59:30 +08:00
wangdage12
b9169ad41a Fix indentation in MSI build step 2025-12-02 20:35:58 +08:00
wangdage12
00af738520 Add new weapon IDs to WeaponIds.cs 2025-12-02 20:05:44 +08:00
wangdage12
3538665bc6 Add files via upload 2025-12-02 18:21:25 +08:00
wangdage12
c1a1871284 Update README.md 2025-12-02 16:05:37 +08:00
wangdage12
01f965d260 Update completion status in README.md 2025-12-01 18:43:56 +08:00
wangdage12
cadd35a93e Revise README for clarity and updates
Updated README to reflect changes in functionality and metadata progress.
2025-12-01 16:22:52 +08:00
dependabot[bot]
0de6d4b71c Bump actions/checkout from 4 to 6 in /.github/workflows
Bumps [actions/checkout](https://github.com/actions/checkout) from 4 to 6.
- [Release notes](https://github.com/actions/checkout/releases)
- [Changelog](https://github.com/actions/checkout/blob/main/CHANGELOG.md)
- [Commits](https://github.com/actions/checkout/compare/v4...v6)

---
updated-dependencies:
- dependency-name: actions/checkout
  dependency-version: '6'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-12-01 02:55:54 +00:00
wangdage12
a83f119e8a 添加服务器状态 2025-11-27 22:07:07 +08:00
wangdage12
2278c83dd2 Merge pull request #2 from hoshiizumiya/patch-1
Revise README for injection feature and installation updates
2025-11-27 19:12:53 +08:00
wangdage12
867f8b6558 Fix build command in MSI workflow 2025-11-26 20:36:58 +08:00
wangdage12
7e6a6a509c Simplify build command in MSI workflow 2025-11-26 20:20:21 +08:00
wangdage12
7aaa61dcf3 Fix YAML syntax for dotnet publish command 2025-11-26 19:50:20 +08:00
wangdage12
1fe5b4969e Fix path for restoring NuGet packages 2025-11-26 19:46:39 +08:00
wangdage12
e52ed5470e Update NuGet restore command to specify solution file 2025-11-26 19:41:31 +08:00
wangdage12
4434d76e35 Update Wix Toolset version in workflow 2025-11-26 19:36:30 +08:00
wangdage12
4a9c3229a0 Update Wix Toolset version in CI workflow 2025-11-26 19:33:37 +08:00
wangdage12
1090dfa7c6 Fix case sensitivity in Wix Toolset installation command 2025-11-26 19:28:52 +08:00
wangdage12
a4eb3d6f8a Add GitHub Actions workflow for MSI build 2025-11-26 19:25:43 +08:00
wangdage12
d9c43844b7 Merge pull request #1 from hoshiizumiya/main
支持非打包模式直接启动并更改优化设置存储逻辑采用 wix 打包 msi 安装包支持脱离沙箱使用,为注入功能开发奠定基础
2025-11-26 19:24:03 +08:00
wangdage12
2cc2cc80ac 修复KeyPath问题 2025-11-26 19:18:01 +08:00
wangdage12
cc926e4352 指定TargetFramework值 2025-11-26 19:17:01 +08:00
wangdage12
fa4b975f46 Merge branch 'main' into main 2025-11-24 22:03:57 +08:00
wangdage12
299cdee329 Merge pull request #4 from wangdage12/dependabot/github_actions/dot-github/workflows/actions/checkout-6
Bump actions/checkout from 5 to 6 in /.github/workflows
2025-11-24 21:24:48 +08:00
wangdage12
f49f40b5fd Merge pull request #3 from hoshiizumiya/base
完善并启用主题切换功能
2025-11-24 21:22:09 +08:00
dependabot[bot]
ff21a91cc6 Bump actions/checkout from 5 to 6 in /.github/workflows
Bumps [actions/checkout](https://github.com/actions/checkout) from 5 to 6.
- [Release notes](https://github.com/actions/checkout/releases)
- [Changelog](https://github.com/actions/checkout/blob/main/CHANGELOG.md)
- [Commits](https://github.com/actions/checkout/compare/v5...v6)

---
updated-dependencies:
- dependency-name: actions/checkout
  dependency-version: '6'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2025-11-24 02:33:17 +00:00
hoshiizumiya
cc47e050d1 完善并启用主题切换功能
补充部分相关本地化
2025-11-24 00:08:52 +08:00
hoshiizumiya
c2d0156a9d Update README.md 2025-11-23 22:10:08 +08:00
hoshiizumiya
eac314f7d9 Update README with packaging and resource details
Added packaging test instructions and resource notes.
2025-11-23 21:50:21 +08:00
hoshiizumiya
f1132b79f0 Revise README for injection feature and installation updates
Updated README to reflect changes in injection functionality and installation method.
2025-11-23 21:39:08 +08:00
hoshiizumiya
63c4792e00 添加注意 2025-11-23 21:29:19 +08:00
hoshiizumiya
5f196253b3 Enhance failure count handling in RepositoryAffinity
Add lower bounds protection for failure counts to prevent negative values and integer underflow.
2025-11-23 21:24:10 +08:00
hoshiizumiya
a7bb931ea5 Update copyright notice in PackageIdentityDiagnostics.cs 2025-11-23 20:47:40 +08:00
hoshiizumiya
d318dcbfd0 Update copyright information in PackageIdentityAdapter.cs 2025-11-23 20:46:54 +08:00
hoshiizumiya
e045413ac1 Update HarvestDirectory path for project configuration 2025-11-23 20:17:47 +08:00
hoshiizumiya
1039623cbf 完善安装
添加桌面图标,修正注册表安装目标
2025-11-23 20:10:02 +08:00
hoshiizumiya
3f50507490 Delete src/Snap.Hutao/Snap.Hutao.Installer/ExampleComponents.wxs
删除 wix 示例配置代码
2025-11-23 19:37:46 +08:00
hoshiizumiya
e55d6ea1c5 支持非打包模式直接启动并更改优化设置存储逻辑采用wix 打包 msi 安装包支持脱离沙箱使用,为注入功能开发奠定基础
新增 PackageIdentityAdapter 适配打包和非打包模式,支持通过文件系统路径替代 Windows.Storage.ApplicationData。重构 LocalSetting,抽象为 ISettingStorage 接口,支持 JSON 文件存储非打包模式下的设置。改进异常处理和日志记录,增加调试信息和启动诊断日志。隐藏 AppX 打包配置,更新依赖项版本,支持普通管理员模式下的游戏启动和 DLL 注入。补全 SaltConstants 文件,存储数据签名所需的盐值。

已知问题:
注入功能暂不不支持,缺少服务器提供的当前版本注入信息!
没有启用默认管理员权限打开
2025-11-23 16:58:05 +08:00
wangdage12
e96c9f1f49 Revise README for installation and resource details
Updated installation instructions and resource server notes.
2025-11-23 14:58:04 +08:00
wangdage12
f3d47aeffd 修改图片链接 2025-11-23 14:36:15 +08:00
wangdage12
7ee92db156 Update README with metadata repository information
Added metadata repository and mirror links to README.
2025-11-22 16:09:19 +08:00
wangdage12
7c25951950 Revise README for project resources and documentation
Updated the README to clarify project requirements and documentation links.
2025-11-21 21:53:55 +08:00
wangdage12
2ad17f60bf Update README with GetImage.py usage instructions
Added instructions for using GetImage.py to restore image resources from logs.
2025-11-21 15:16:02 +08:00
wangdage12
4fdea542b0 添加从请求日志中恢复旧图片资源的工具
添加从请求日志中恢复旧图片资源的工具
2025-11-21 14:59:30 +08:00
wangdage12
4d5e5342f2 Add README for development tools 2025-11-21 14:56:46 +08:00
wangdage12
fde960b7cc Update README with compilation instructions
Added a note about using Visual Studio 2026 for compilation.
2025-11-21 14:44:56 +08:00
wangdage12
cfc7ba6274 Delete .github/FUNDING.yml 2025-11-21 14:41:21 +08:00
wangdage12
c463f1809c Update README.md 2025-11-21 14:38:50 +08:00
wangdage12
3d352c1c12 Update README.md 2025-11-20 21:37:30 +08:00
wangdage12
fbf3c0c695 Update README.md 2025-11-20 21:15:25 +08:00
58 changed files with 1646 additions and 1080 deletions

13
.github/FUNDING.yml vendored
View File

@@ -1,13 +0,0 @@
# These are supported funding model platforms
github: [DGP-Studio]
patreon: # Replace with a single Patreon username
open_collective: snaphutao
ko_fi: # Replace with a single Ko-fi username
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
liberapay: # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username
otechie: # Replace with a single Otechie username
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
custom: https://afdian.com/a/DismissedLight

View File

@@ -1,93 +0,0 @@
name: 问题反馈
description: 通过这个议题向开发团队反馈你发现的程序中的问题
title: "[Bug]: 在这里填写一个合适的标题"
type: "Bug"
labels: ["priority:none"]
body:
- type: markdown
attributes:
value: |
> **请在上方以一句话简短地概括你的问题作为标题**
> 请按下方的要求填写完整的问题表单,以便我们更快的定位问题。
- type: input
id: winver
attributes:
label: Windows 版本
description: |
`Win+R` 输入 `winver` 回车后在打开的窗口第二行可以找到
placeholder: 22000.556
validations:
required: true
- type: input
id: shver
attributes:
label: Snap Hutao 版本
description: 在应用标题,应用程序的反馈中心界面中可以找到
placeholder: 1.9.9.0
validations:
required: true
- type: input
id: deviceid
attributes:
label: 设备 ID
description: |
> 在胡桃工具箱的反馈中心界面,你可以找到并复制你的设备 ID
> 如果你的问题涉及程序崩溃,请填写该项,这将有助于我们定位问题
> 如果你的程序已经无法启动,请下载并运行[诊断工具](https://github.com/DGP-Automation/ISSUE_TEMPLATES/releases/download/diagnosis_tools/Snap.Hutao.Diagnostic.Tooling.exe),它将显示你的设备 ID
validations:
required: true
- type: dropdown
id: user-set-category
attributes:
label: 问题分类
description: 请设置一个你认为合适的分类,这将帮助我们快速定位问题
options:
- 安装和环境
- 游戏启动器
- 祈愿记录
- 成就管理
- 我的角色
- 实时便笺
- 养成计算
- 深境螺旋/胡桃数据库
- Wiki
- 米游社账号面板
- 每日签到奖励
- 胡桃通行证/胡桃云
- 用户界面
- 文件缓存
- 公告
- 其它
validations:
required: true
- type: textarea
id: what-happened
attributes:
label: 发生了什么?
description: |
详细的描述问题发生前后的行为,以便我们解决问题。**如果你的问题涉及程序崩溃,你应当检查 Windows 事件查看器,并将相关的 `.Net 错误`详情附上**
如果你无法找到该日志,请下载并运行[诊断工具](https://github.com/DGP-Automation/ISSUE_TEMPLATES/releases/download/diagnosis_tools/Snap.Hutao.Diagnostic.Tooling.exe),它将转储问题日志至工具运行目录中的 `Snap.Hutao Error Log.txt`
validations:
required: true
- type: textarea
id: what-expected
attributes:
label: 你期望发生的行为?
description: 详细的描述你期望发生的行为,突出与目前(可能不正确的)行为的不同
validations:
required: false
- type: checkboxes
id: checklist-final
attributes:
label: 最后一步
description: 回顾你的回答
options:
- label: 我认为上述的描述已经足以详细,以允许开发人员能复现该问题
required: true

View File

@@ -1,26 +0,0 @@
name: 功能请求
description: 通过这个议题来向开发团队分享你的想法
title: "[Feat]: 在这里填写一个合适的标题"
type: "Feature"
labels: ["needs-triage", "priority:none"]
body:
- type: markdown
attributes:
value: |
请按下方的要求填写完整的问题表单。
- type: textarea
id: back
attributes:
label: 背景与动机
description: 添加此功能的理由,如果你想要实现多个功能,请分别发起多个单独的议题
validations:
required: true
- type: textarea
id: req
attributes:
label: 想要实现或优化的功能
description: 详细的描述一下你想要的功能,描述的越具体,采纳的可能性越高
validations:
required: true

View File

@@ -1,84 +0,0 @@
name: 网络问题
description: 通过这个议题来反馈网络问题
title: "[Network]: 在这里填写一个合适的标题"
type: "Bug"
labels: ["area-Network"]
assignees:
- Lightczx
- Masterain98
body:
- type: markdown
attributes:
value: |
**请先在上方为工单设置一个合适的标题**
**请按下方的要求填写完整的问题表单,以便我们更快的定位问题。**
- type: textarea
id: network-diagnosis-report
attributes:
label: 提交你的网络诊断报告
description: |
停下!
**在填写下面的问题之前请先使用我们的网络诊断工具**
**这个工具将会生成一份报告并加密压缩,请将这份报告拖入下面的框中,让其与你的工单一起被上传提交**
- 你可以点击下面的链接以下载网络诊断工具:
- [GitHub](https://github.com/Masterain98/network-diagnosis-tool/releases/latest/download/SH-Network-Diagnosis.exe)
validations:
required: true
- type: input
id: user-geo-location
attributes:
label: 你的地理位置
description: |
中国用户请精确到省级行政区
海外用户请精确到国家
placeholder: 北京
validations:
required: true
- type: dropdown
id: user-isp
attributes:
label: 你的运营商
description: 海外用户请选其它
options:
- 中国电信
- 中国联通
- 中国移动
- 中国广电
- 其它
validations:
required: true
- type: dropdown
id: user-issue-category
attributes:
label: 你的问题
description: 选择一个问题类别
options:
- 完全无法连接服务器
- 连接速度慢
- 获取到了不正确的页面或数据
- 客户端图片下载错误
- 客户端图片预下载错误
- 其它
validations:
required: true
- type: textarea
id: what-happened
attributes:
label: 你的问题(补充)
description: 如果你在上一项中选择了`其它`或者你有更多信息需要提供,请在这里写下来
validations:
required: false
- type: checkboxes
id: checklist-final
attributes:
label: 最后一步
description: 检查你提交的议题
options:
- label: 我已经在该议题中上传了包含网络诊断报告的加密压缩包
required: true

View File

@@ -1,93 +0,0 @@
name: BUG Report [English Form]
description: Tell us what issue you get
title: "[ENG][Bug]: Place your Issue Title Here"
type: "Bug"
labels: ["priority:none"]
body:
- type: markdown
attributes:
value: |
> **Please use one sentence to briefly describe your issue as title above**
> Please follow the instruction below to fill the form, so we can locate the issue quickly
- type: input
id: winver
attributes:
label: Windows Version
description: |
Use `Win+R` and input `winver`, Windows build version is usually at the second line
placeholder: e.g. 22000.556
validations:
required: true
- type: input
id: shver
attributes:
label: Snap Hutao Version
description: You can find the version in application's title bar
placeholder: e.g. 1.9.9.0
validations:
required: true
- type: input
id: deviceid
attributes:
label: Device ID
description: |
> In Snap Hutao's Feedback Center, you can find and copy your device ID
> If your issue is about program crash, please fill this so we can dump the log and locate the source easier
> If your program cannot startup, please download and run [Diagnostic Tooling](https://github.com/DGP-Automation/ISSUE_TEMPLATES/releases/download/diagnosis_tools/Snap.Hutao.Diagnostic.Tooling.exe), it will shows your device ID.
validations:
required: true
- type: dropdown
id: user-set-category
attributes:
label: Issue Category
description: Please select the most associated category of your issue
options:
- Installation and Environment
- Game Launcher
- Wish Export
- Achievement
- My Character
- Realtime Note
- Develop Plan
- Spiral Abyss
- Wiki
- MiHoYo Account Panel
- Daily Checkin Reward
- Hutao Passport/Hutao Cloud
- User Interface
- File Cache
- Announcement
- Other
validations:
required: true
- type: textarea
id: what-happened
attributes:
label: What Happened?
description: |
Describe your issue in detail to help us identify the issue. **If your issue is about program crash, you should check Windows Event Viewer, and attach associated `.Net Error` details here**If your program cannot startup, please download and run [this PowerShell script](https://github.com/DGP-Studio/ISSUE_TEMPLATES/releases/download/get_device_id/GetHutaoDeviceId.ps1), it will shows your device ID.
If you cannot find it, please download and run [Diagnosis Tool](https://github.com/DGP-Automation/ISSUE_TEMPLATES/releases/download/diagnosis_tools/Snap.Hutao.Diagnostic.Tooling.exe), it will dump the error log to `Snap.Hutao Error Log.txt` in the working directory of the tool.
validations:
required: true
- type: textarea
id: what-expected
attributes:
label: What is expected?
description: Describe expected outcome, highlight the difference with current outcome
validations:
required: false
- type: checkboxes
id: checklist-final
attributes:
label: Last Step
description: Review your Form
options:
- label: I believe the description above is detail enough to allow developers to reproduce the issue
required: true

View File

@@ -1,26 +0,0 @@
name: Feature Request [English Form]
description: Tell us about your thought
title: "[Feat]: Place your title here"
type: "Feature"
labels: ["needs-triage", "priority:none"]
body:
- type: markdown
attributes:
value: |
Please fill the form below
- type: textarea
id: back
attributes:
label: Background & Motivation
description: Reason why this feature is needed. If multiple features is requested, please open multiple issues for each of them.
validations:
required: true
- type: textarea
id: req
attributes:
label: Detail of the Feature
description: Descripbe the feaure in detail. The more detailed and convincing the desciprtion the more likyly feature will be accepted.
validations:
required: true

View File

@@ -1,79 +0,0 @@
name: Network Issue [English Form]
description: Submit this issue form when network issue affect your client experience
title: "[Network]: Place your title here"
type: "Bug"
labels: ["area-Network"]
assignees:
- Lightczx
- Masterain98
body:
- type: markdown
attributes:
value: |
**Please use one sentence to briefly describe your issue as title above**
**Please follow the instruction below to fill the form, so we can locate the issue quickly**
- type: textarea
id: network-diagnosis-report
attributes:
label: Submit Your Network Diagnosis Report
description: |
STOP HERE!
**Please run our network diagnosis tool before filling this form**
**The diagnosis tool will generate a report and add it into a password-protected archive. Drag the `.zip` archive to the box below so it can be uploaded.**
- Use the following link to download the Network Diagnosis Tool:
- [GitHub](https://github.com/Masterain98/network-diagnosis-tool/releases/latest/download/SH-Network-Diagnosis.exe)
validations:
required: true
- type: input
id: user-geo-location
attributes:
label: Your Geographical Location
description: |
Description accurate to country
placeholder: USA
validations:
required: true
- type: input
id: user-isp
attributes:
label: Your ISP Name
description: |
Name of your Internet service provider
placeholder: AT&T
validations:
required: true
- type: dropdown
id: user-issue-category
attributes:
label: Issue Category
description: Select an issue category
options:
- Cannot connect to server completely
- Slow spped
- Fetched wrong page or data
- Image download error in the client
- Image set pre-download error (client welcome wizard process)
- Other
validations:
required: true
- type: textarea
id: what-happened
attributes:
label: Your Issue (cont.)
description: If you selected `Other` in previous dropdown, please explain your issue in detail here.
validations:
required: false
- type: checkboxes
id: checklist-final
attributes:
label: One Last Step
description: Check your issue form
options:
- label: I confirm I have attached the network diagnosis report archive in the issue
required: true

View File

@@ -1,31 +0,0 @@
name: Publish Process
description: FOR ADMIN USE ONLY. WILL CAUSE A BAN IF NO PERMISSION.
title: "[Publish]: Version 1.9.98"
labels: ["Publish"]
assignees:
- Lightczx
body:
- type: textarea
id: main-body
attributes:
label: Publish Process
value: |
## 创建版本
- [ ] 同步一次 [Crowdin](https://crowdin.com/project/snap-hutao) 翻译
- [ ] 发布 RC 版本Optional
- [ ] 合并入主分支
- [ ] 整理更新内容,等待翻译
- [ ] 在 [Snap.Hutao.Docs@next-patch](https://github.com/DGP-Studio/Snap.Hutao.Docs/tree/next-patch) 分支更新文档并直接开 PR
- [ ] 更新日志
- [ ] 功能文档更新
- type: checkboxes
id: checklist-final
attributes:
label: Final Check
description: Understand what you are doing
options:
- label: I understand that I will get banned from repository if I don't have permission to use this template
required: true

View File

@@ -1,14 +0,0 @@
blank_issues_enabled: false
contact_links:
- name: Snap Hutao 官方文档 / Snap Hutao Document
url: https://hut.ao
about: 请在提出问题前阅读文档 / Read the document before submit the issue
- name: 常见问题 / FAQ
url: https://hut.ao/advanced/FAQ.html
about: 常见的用户提出的问题 / Common questions asked by users
- name: 常见程序异常 / Common Program Exceptions
url: https://hut.ao/advanced/exceptions.html
about: 用户通常能自行解决这些问题 / Users may solve these problems by themselves

View File

@@ -1,13 +0,0 @@
name: 内部任务
description: 此Issue模板仅用于创建内部任务非 DGP Studio 成员请勿使用
title: "[Task]: 在这里填写一个合适的标题"
type: "Task"
labels: ["priority:none"]
body:
- type: textarea
id: content
attributes:
label: 背景与动机
description: 添加相关的说明
validations:
required: true

View File

@@ -73,7 +73,7 @@ jobs:
runs-on: ${{ needs.select-runner.outputs.runner }}
steps:
- name: Checkout
uses: actions/checkout@v5
uses: actions/checkout@v6
- name: Setup .NET (self-hosted)
if: ${{ needs.select-runner.outputs.runner == 'sjc1' }}
@@ -95,7 +95,7 @@ jobs:
- name: Cache NuGet packages
if: ${{ needs.select-runner.outputs.runner == 'windows-latest' }}
uses: actions/cache@v4
uses: actions/cache@v5
with:
path: ~/.nuget/packages
key: ${{ runner.os }}-nuget-${{ hashFiles('**/Snap.Hutao.csproj') }}
@@ -113,7 +113,7 @@ jobs:
- name: Upload signed msix
if: success() && github.event_name != 'pull_request'
uses: actions/upload-artifact@v5
uses: actions/upload-artifact@v6
with:
name: Snap.Hutao.Alpha-${{ steps.cake.outputs.version }}
path: ${{ github.workspace }}/src/output/Snap.Hutao.Alpha-${{ steps.cake.outputs.version }}.msix

View File

@@ -21,7 +21,7 @@ jobs:
runs-on: windows-latest
steps:
- name: Checkout
uses: actions/checkout@v5
uses: actions/checkout@v6
with:
ref: develop
fetch-depth: 0
@@ -59,7 +59,7 @@ jobs:
- name: Cache NuGet packages
if: ${{ steps.merge.outputs.continue == 'true' }}
uses: actions/cache@v4
uses: actions/cache@v5
with:
path: ~/.nuget/packages
key: ${{ runner.os }}-nuget-${{ hashFiles('**/Snap.Hutao.csproj') }}
@@ -77,7 +77,7 @@ jobs:
- name: Upload signed msix
if: ${{ success() && steps.merge.outputs.continue == 'true' }}
uses: actions/upload-artifact@v5
uses: actions/upload-artifact@v6
with:
name: Snap.Hutao.Canary-${{ steps.cake.outputs.version }}
path: ${{ github.workspace }}/src/output/Snap.Hutao.Canary-${{ steps.cake.outputs.version }}.msix

View File

@@ -17,7 +17,7 @@ jobs:
action:
runs-on: ubuntu-latest
steps:
- uses: dessant/lock-threads@v5
- uses: dessant/lock-threads@v6
with:
issue-inactive-days: '30'
issue-comment: 'This issue has been automatically locked since there has not been any recent activity after it was closed. Please open a new issue for related topic.'

38
.github/workflows/msi-build.yml vendored Normal file
View File

@@ -0,0 +1,38 @@
name: Build MSI Installer
on:
push:
branches: [ main ]
pull_request:
jobs:
build:
runs-on: windows-latest
steps:
- name: Checkout source
uses: actions/checkout@v6
- name: Setup .NET SDK
uses: actions/setup-dotnet@v5
with:
dotnet-version: 10.0.x
- name: Install Wix Toolset 4
run: dotnet tool install --global wix --version 4.0.1
- name: Restore NuGet packages
run: dotnet restore src/Snap.Hutao/Snap.Hutao.slnx
- name: Build WinUI 3 project (self-contained)
run: dotnet build src/Snap.Hutao/Snap.Hutao/Snap.Hutao.csproj -c Release
- name: Build MSI installer
run: dotnet build src/Snap.Hutao/Snap.Hutao.Installer/Snap.Hutao.Installer.wixproj -c Release
- name: Upload MSI Artifact
uses: actions/upload-artifact@v6
with:
name: Snap.Hutao-MSI
path: |
src/Snap.Hutao/Snap.Hutao.Installer/bin/x64/Release/en-US/*.msi

5
.gitignore vendored
View File

@@ -1,4 +1,4 @@
desktop.ini
desktop.ini
*.csproj.user
*.DotSettings.user
@@ -25,3 +25,6 @@ src/Snap.Hutao/Snap.Hutao/Generated Files/
tools/
src/Snap.Hutao/Snap.Hutao/AppPackages
/src/Snap.Hutao/Snap.Hutao.Installer/obj
/src/Snap.Hutao/Snap.Hutao.Installer/bin/x64/Release/en-US
/src/Snap.Hutao/Snap.Hutao.Installer/bin

198
README.md
View File

@@ -1,54 +1,11 @@
<p align="center">
<img src="https://github.com/user-attachments/assets/976e057c-f01e-486b-9fa0-04744ae96f99" alt="Snap Hutao Banner" width="600"/>
</p>
<h1 align="center">Snap Hutao</h1>
<p align="center">
🎮 开源的原神工具箱,专为 Windows 平台设计,改善桌面端玩家的游戏体验
<br/>
🎮 An open-source Genshin Impact toolkit for Windows, designed to improve the desktop gaming experience
</p>
<div align="center">
<table>
<tr>
<td align="center" style="padding:0 10px;">
<b>Latest CI/CD Build</sub>
</td>
<td align="center" style="padding:0 10px;">
<b>Latest Release</sub>
</td>
<td align="center" style="padding:0 10px;">
<b>Downloads</sub>
</td>
</tr>
<tr>
<td align="center" style="padding:0 10px;">
<a href="https://ci.appveyor.com/project/DGP-Studio/snap-hutao">
<img src="https://ci.appveyor.com/api/projects/status/n4s40t9llru4si9y?svg=true" alt="Build Status"/>
</a>
</td>
<td align="center" style="padding:0 10px;">
<a href="https://github.com/DGP-Studio/Snap.Hutao/releases/latest">
<img src="https://img.shields.io/github/release/DGP-Studio/Snap.Hutao?style=flat" alt="Release"/>
</a>
</td>
<td align="center" style="padding:0 10px;">
<img src="https://img.shields.io/github/downloads/DGP-Studio/Snap.Hutao/total.svg?style=flat" alt="Downloads"/>
</td>
</tr>
</table>
</div>
---
## 📖 简介 / Introduction
**中文**
胡桃工具箱是一款以 MIT 协议开源的原神工具箱,专为现代化 Windows 平台设计,旨在改善桌面端玩家的游戏体验。
该版本注入功能暂不可用,并且由于缺失资源和开发能力,不建议长期使用
**English**
Snap Hutao is an open-source Genshin Impact toolkit under MIT license, designed for modern Windows platform to improve the gaming experience for desktop players.
@@ -56,89 +13,98 @@ Snap Hutao is an open-source Genshin Impact toolkit under MIT license, designed
## 🚀 安装 / Installation
**中文**
你可以按照 [快速开始](https://hut.ao/zh/quick-start.html) 文档中提供的流程安装并设置 Snap Hutao。
> 如果你的设备不支持ipv6请下载末尾带有`ipv4`的压缩包,正常情况下请尽量下载普通包(服务器速度快)
**English**
You can follow the instructions in the [Quick Start](https://hut.ao/en/quick-start.html) document to install and set up Snap Hutao.
目前 Sanp.Hutao.Rev 更新了打包方式,并采用了标准现代的 msi 安装,方便程序获取管理员权限和更多的功能设置,不再需要原 Depolyment
只有`.msi`安装包安装的可以和之前的版本共存,如果通过`.msix`安装包安装则可能出现`0x80073CF3`,备份旧版本数据文件夹后卸载旧版本即可继续安装,将旧版本数据文件夹里面的文件复制到该版本的数据文件夹中即可恢复数据
---
## 🌍 本地化翻译 / Localization
## 开发
项目启动位置已升级为 VS2026 的 slnx 格式 Snap.Hutao\src\Snap.Hutao\Snap.Hutao.slnx
> [!WARNING]
> 要使该项目可以长期运行,我们需要以下资源
> 1. `src/Snap.Hutao/Snap.Hutao/Web/Hoyolab/DataSigning/SaltConstants.cs`中的新签名值
> 2. 元数据的编写
> 3. 图片资源
Snap Hutao 使用 [Crowdin](https://translate.hut.ao/) 作为客户端文本翻译平台,在该平台上你可以为你熟悉的语言提交翻译文本。我们感谢每一个为 Snap Hutao 做出贡献的社区成员,并且欢迎更多的朋友能参与到这个项目中。
V6.2的元数据已在编写中
测试仓库位置http://server.wdg.cloudns.ch:3000/wdg1122/Snap.Metadata.Test
**目前元数据的编写进度:**
Snap Hutao uses [Crowdin](https://translate.hut.ao/) as a client text translation platform where you can submit translated text for languages you are familiar with. We are grateful to every community member who has contributed to Snap Hutao and welcome more friends to participate in this project.
| 项目V6.2 | 是否完成 |
| ----------- | ----------- |
| 新角色的基本数据 | ✔️ |
| 新版本角色/怪物基础数值 | ❔ |
| 新角色的详细资料、名片等 | ❌ |
| 新武器 | ✔️ |
| 新材料 | ❇️ |
| 新怪物 | ❇️ |
| 新圣遗物 | / |
| 新卡池 | ❇️ |
| 新成就 | ✔️ |
| 深境螺旋 | 💠 |
| 幻想真境剧诗 | 💠 |
| 幽境危战 | ✔️ |
| Language | Status |
|----------|--------|
| zh-TW | [![zh-TW](https://img.shields.io/badge/dynamic/json?color=blue&label=zh-TW&style=flat&logo=crowdin&query=%24.zh-TW&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| en | [![en](https://img.shields.io/badge/dynamic/json?color=blue&label=en&style=flat&logo=crowdin&query=%24.en&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| fr | [![fr](https://img.shields.io/badge/dynamic/json?color=blue&label=fr&style=flat&logo=crowdin&query=%24.fr&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| id | [![id](https://img.shields.io/badge/dynamic/json?color=blue&label=id&style=flat&logo=crowdin&query=%24.id&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| ja | [![ja](https://img.shields.io/badge/dynamic/json?color=blue&label=ja&style=flat&logo=crowdin&query=%24.ja&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| ko | [![ko](https://img.shields.io/badge/dynamic/json?color=blue&label=ko&style=flat&logo=crowdin&query=%24.ko&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| pt-PT | [![pt-PT](https://img.shields.io/badge/dynamic/json?color=blue&label=pt-PT&style=flat&logo=crowdin&query=%24.pt-PT&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| ru | [![ru](https://img.shields.io/badge/dynamic/json?color=blue&label=ru&style=flat&logo=crowdin&query=%24.ru&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
| vi | [![vi](https://img.shields.io/badge/dynamic/json?color=blue&label=vi&style=flat&logo=crowdin&query=%24.vi&url=https%3A%2F%2Fawesome-crowdin-proxy.qhy040404.workers.dev%2Fstats-15670597-565845.json)](https://crowdin.com/project/snap-hutao) |
✔️:已完成
❌:未编写
❇️:编写中
❔:数据暂时无法得到
/ :似乎不需要变动
💠:低优先级,以后编写
**若需编译项目,请使用[Visual Studio 2026](https://visualstudio.microsoft.com/zh-hans/)**
调试选项请选择unpackaged不打包
**原开发文档现在还可使用其中的AI功能很好用以下是开发文档链接**
https://deepwiki.com/DGP-Studio/Snap.Hutao
https://deepwiki.com/DGP-Studio/Snap.Hutao.Server
## 打包测试
由于采用了 wix 进行打包程序VS 需要安装 **HeatWave for VS2022**2026兼容。需要 msi 安装包时,右键选中 Snap.Hutao.Installer 生成后即可在目标目录找到。默认目录Snap.Hutao.Installer\bin\x64\Release\en-US\Snap.Hutao.Installer.msi
### 资源
> 注意普通包的资源服务器只能使用ipv6连接也就是说你的电脑必须有ipv6并且建议你手动配置DNS为`223.5.5.5`
> 如果你的设备不支持ipv6请下载末尾带有`ipv4`的压缩包
> 由于数据文件夹中有元数据的仓库和图片缓存,才得以恢复资源文件
> 如果你发现之前版本可以显示的图片不能显示了,请查找旧数据文件夹
> `C:\Users\<用户名>\AppData\Local\Packages\xxxDGPStudio.SnapHutao_xxx\LocalCache\ImageCache`
> 并将`ImageCache`文件夹提供给我,我会尽力恢复资源
[服务器状态页面](http://serverjp.wdg.cloudns.ch:3001/status/hts)
**元数据仓库:**
https://github.com/wangdage12/Snap.Metadata
镜像:
![http://serverjp.wdg.cloudns.ch:3001/api/badge/6/status?style=flat-square](http://serverjp.wdg.cloudns.ch:3001/api/badge/6/status?style=flat-square)
http://server.wdg.cloudns.ch:3000/wdg1122/Snap.Metadata
![http://serverjp.wdg.cloudns.ch:3001/api/badge/7/status?style=flat-square](http://serverjp.wdg.cloudns.ch:3001/api/badge/7/status?style=flat-square)
http://serverjp.wdg.cloudns.ch:3000/wdg1122/Snap.Metadata
---
## 🛠️ 贡献 / Contribute
**临时API**
- [向我们提交 PR / Make Pull Requests](https://hut.ao/development/contribute.html)
- [为我们更新文档 / Enhance our Document](https://github.com/DGP-Studio/Snap.Hutao.Docs)
- [通过 DeepWiKi 了解项目结构 / Understand Project Structure with DeepWiKi](https://deepwiki.com/DGP-Studio/Snap.Hutao)
- [![Ask DeepWiki](https://deepwiki.com/badge.svg)](https://deepwiki.com/DGP-Studio/Snap.Hutao)
![http://serverjp.wdg.cloudns.ch:3001/api/badge/8/status?style=flat-square](http://serverjp.wdg.cloudns.ch:3001/api/badge/8/status?style=flat-square)
http://server.wdg.cloudns.ch:5222/
![http://serverjp.wdg.cloudns.ch:3001/api/badge/9/status?style=flat-square](http://serverjp.wdg.cloudns.ch:3001/api/badge/9/status?style=flat-square)
http://serverjp.wdg.cloudns.ch:5222/
---
## 🙏 特别感谢 / Special Thanks
**临时资源站:**
http://server.wdg.cloudns.ch:8007/
- [HolographicHat](https://github.com/HolographicHat)
- [UIGF organization](https://uigf.org)
**特定的原神项目 / Specific Genshin-related Projects**
- [Scighost/Starward](https://github.com/Scighost/Starward)
---
## ⚙️ 使用的技术栈 / Tech Stack
- [CommunityToolkit/dotnet](https://github.com/CommunityToolkit/dotnet)
- [CommunityToolkit/Labs-Windows](https://github.com/CommunityToolkit/Labs-Windows)
- [CommunityToolkit/Windows](https://github.com/CommunityToolkit/Windows)
- [dotnet/efcore](https://github.com/dotnet/efcore)
- [dotnet/runtime](https://github.com/dotnet/runtime)
- [microsoft/vs-validation](https://github.com/microsoft/vs-validation)
- [microsoft/WindowsAppSDK](https://github.com/microsoft/WindowsAppSDK)
- [microsoft/microsoft-ui-xaml](https://github.com/microsoft/microsoft-ui-xaml)
- [quartznet/quartznet](https://github.com/quartznet/quartznet)
---
## ❤️ 赞助商 / Sponsorship
Snap Hutao is currently using sponsored software from the following service providers.
<img src="./res/assets/readmeSponsors.svg" alt="Readme Sponsors" />
- 🏠 [Netlify](https://www.netlify.com/) provides document and home page hosting service for Snap Hutao
- 🌍 [Crowdin](https://crowdin.com/) provides its SaaS platform to help Snap Hutao's localization
- 🗄️ [Navicat](https://navicat.com/) provides Snap Hutao with advanced database management tools
- 🔒 Free code signing provided by [SignPath.io](https://signpath.io/), certificate by [SignPath Foundation](https://signpath.org/)
- 🔑 [1Password](https://1password.com/) provides Snap Hutao development team with their amazing password management software
- 🐳 [DigitalOcean](https://www.digitalocean.com) provides reliable cloud database and container service for Snap Hutao database backup
- 📊 [Ducalis.io](https://hi.ducalis.io/) provides Snap Hutao project with a complete decision-making toolkit for project management
- ☁️ [Cloudflare](https://www.cloudflare.com/) sponsors Snap Hutao with their Business Plan, ensuring secure, fast, and reliable worldwide connection to our infrastructure
- 🔐 [Termius](https://termius.com) provides a secure, reliable, and collaborative SSH client
---
## 📈 开发 / Development
![Snap.Hutao](https://repobeats.axiom.co/api/embed/f029553fbe0c60689b1710476ec8512452163fc9.svg)
[![Star History Chart](https://api.star-history.com/svg?repos=DGP-Studio/Snap.Hutao&type=Date)](https://star-history.com/#DGP-Studio/Snap.Hutao&Date)
[![](https://opengraph.snapgenshin.cn/gitcode?repo=DGP-Studio/Snap.Hutao)](https://github.com/DGP-Studio/Snap.Hutao)
http://serverjp.wdg.cloudns.ch:8001/

View File

@@ -0,0 +1,3 @@
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
</Wix>

View File

@@ -0,0 +1,8 @@
<!--
This file contains the declaration of all the localizable strings.
-->
<WixLocalization xmlns="http://wixtoolset.org/schemas/v4/wxl" Culture="en-US">
<String Id="DowngradeError" Value="A newer version of [ProductName] is already installed." />
</WixLocalization>

View File

@@ -0,0 +1,79 @@
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
<Package
Name="Snap.Hutao"
Manufacturer="Millennium Science Technology R-D Inst"
Version="1.17.4.0"
UpgradeCode="121203be-60cb-408f-92cc-7080f6598e68"
Scope="perMachine">
<MajorUpgrade DowngradeErrorMessage="A newer version of [ProductName] is already installed." />
<MediaTemplate EmbedCab="yes" />
<Feature Id="ProductFeature" Title="Snap.Hutao" Level="1">
<ComponentGroupRef Id="MainAppComponents" />
<!-- 快捷方式组件 -->
<ComponentRef Id="ApplicationShortcut" />
<ComponentRef Id="DesktopShortcut" />
</Feature>
</Package>
<!-- 安装目录 -->
<Fragment>
<StandardDirectory Id="ProgramFiles64Folder">
<Directory Id="INSTALLFOLDER" Name="Snap.Hutao" />
</StandardDirectory>
<StandardDirectory Id="ProgramMenuFolder">
<Directory Id="ApplicationProgramsFolder" Name="Snap Hutao" />
</StandardDirectory>
<StandardDirectory Id="DesktopFolder" />
</Fragment>
<!-- 桌面快捷方式 -->
<Fragment>
<Component Id="DesktopShortcut" Directory="DesktopFolder" Guid="*">
<Shortcut
Id="DesktopShortcut_Normal"
Name="Snap Hutao"
Description="Snap Hutao Client"
Target="[INSTALLFOLDER]Snap.Hutao.exe"
WorkingDirectory="INSTALLFOLDER" />
<!-- KeyPath 必须是 HKCU因为快捷方式安装到用户目录 -->
<RegistryValue
Root="HKCU"
Key="Software\Snap.Hutao"
Name="DesktopShortcutInstalled"
Type="integer"
Value="1"
KeyPath="yes" />
</Component>
</Fragment>
<!-- 开始菜单快捷方式 -->
<Fragment>
<Component Id="ApplicationShortcut" Directory="ApplicationProgramsFolder" Guid="*">
<Shortcut
Id="ApplicationStartMenuShortcut"
Name="Snap Hutao"
Description="Snap Hutao Client"
Target="[INSTALLFOLDER]Snap.Hutao.exe"
WorkingDirectory="INSTALLFOLDER" />
<RemoveFolder Id="CleanUpShortCut" Directory="ApplicationProgramsFolder" On="uninstall" />
<!-- KeyPath 依然必须改为 HKCU -->
<RegistryValue
Root="HKCU"
Key="Software\Snap.Hutao"
Name="StartMenuShortcutInstalled"
Type="integer"
Value="1"
KeyPath="yes" />
</Component>
</Fragment>
</Wix>

View File

@@ -0,0 +1,24 @@
<Project Sdk="WixToolset.Sdk/6.0.2">
<PropertyGroup>
<SuppressIces>ICE03;ICE60</SuppressIces>
<Platform>x64</Platform>
<TargetFramework>net10.0-windows10.0.26100.0</TargetFramework>
<Configuration>Release</Configuration>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\Snap.Hutao\Snap.Hutao.csproj" />
</ItemGroup>
<ItemGroup>
<HarvestDirectory Include="..\Snap.Hutao\bin\Release\net10.0-windows10.0.26100.0\win-x64">
<ComponentGroupName>MainAppComponents</ComponentGroupName>
<DirectoryRefId>INSTALLFOLDER</DirectoryRefId>
<SuppressCom>true</SuppressCom>
<SuppressRegistry>true</SuppressRegistry>
<SuppressRootDirectory>true</SuppressRootDirectory>
</HarvestDirectory>
<PackageReference Include="WixToolset.Heat" Version="4.0.1" />
</ItemGroup>
</Project>

View File

@@ -1,4 +1,8 @@
# Snap.Hutao.SourceGeneration
> 生成器包的备份目前还可以从nuget上获取所以暂时不需要使用该目录
> https://www.nuget.org/packages/Snap.Hutao.SourceGeneration/1.3.14
Source Code Generator for Snap.Hutao
# Development Guideline

View File

@@ -1,24 +1,24 @@
<Solution>
<Configurations>
<Platform Name="Any CPU" />
<Platform Name="ARM64" />
<Platform Name="x64" />
<Platform Name="x86" />
</Configurations>
<Folder Name="/Solution Items/">
<File Path=".editorconfig" />
<File Path=".vsconfig" />
</Folder>
<Project Path="Snap.Hutao.Test\Snap.Hutao.Test.csproj" />
<Project Path="Snap.Hutao\Snap.Hutao.csproj">
<!-- For Rider -->
<Configuration Solution="Debug|Any CPU" Project="Debug|x64|Deploy" />
<Configuration Solution="Debug|x64" Project="Debug|x64|Deploy" />
<Configuration Solution="Release|Any CPU" Project="Release|x64|Deploy" />
<Configuration Solution="Release|x64" Project="Release|x64|Deploy" />
<!-- For Visual Studio -->
<Project Path="Snap.Hutao.Installer/Snap.Hutao.Installer.wixproj" Type="b7dd6f7e-def8-4e67-b5b7-07ef123db6f0" Id="91a04cd0-28cc-4562-92e1-202bc163edd7">
<Platform Solution="*|Any CPU" Project="x64" />
<Platform Solution="*|arm64" Project="arm64" />
<Platform Solution="*|x64" Project="x64" />
<Platform Solution="*|x86" Project="x86" />
<Deploy />
</Project>
<Project Path="Snap.Hutao.Test\Snap.Hutao.Test.csproj">
<Build Solution="*|ARM64" Project="false" />
<Build Solution="*|x86" Project="false" />
</Project>
<Project Path="Snap.Hutao\Snap.Hutao.csproj">
<Platform Project="x64" />
<Deploy Solution="*|Any CPU" />
<Deploy Solution="*|x64" />
</Project>
</Solution>

View File

@@ -13,6 +13,7 @@ using Snap.Hutao.Service;
using Snap.Hutao.UI.Xaml;
using Snap.Hutao.UI.Xaml.Control.Theme;
using System.Diagnostics;
using System.IO;
namespace Snap.Hutao;
@@ -64,6 +65,11 @@ public sealed partial class App : Application
protected override void OnLaunched(LaunchActivatedEventArgs args)
{
// ⚠️ 添加启动诊断
#if DEBUG
Core.ApplicationModel.PackageIdentityDiagnostics.LogDiagnostics();
#endif
DebugPatchXamlDiagnosticsRemoveRootObjectFromLVT();
try
@@ -71,18 +77,41 @@ public sealed partial class App : Application
// Important: You must call AppNotificationManager::Default().Register
// before calling AppInstance.GetCurrent.GetActivatedEventArgs.
AppNotificationManager.Default.NotificationInvoked += activation.NotificationInvoked;
try
{
AppNotificationManager.Default.Register();
}
catch
{
// In unpackaged mode, this might fail - continue anyway
}
// E_INVALIDARG E_OUTOFMEMORY
AppActivationArguments activatedEventArgs = AppInstance.GetCurrent().GetActivatedEventArgs();
AppActivationArguments? activatedEventArgs = null;
PrivateNamedPipeClient? namedPipeClient = null;
if (serviceProvider.GetRequiredService<PrivateNamedPipeClient>().TryRedirectActivationTo(activatedEventArgs))
try
{
activatedEventArgs = AppInstance.GetCurrent().GetActivatedEventArgs();
namedPipeClient = serviceProvider.GetRequiredService<PrivateNamedPipeClient>();
}
catch
{
// In unpackaged mode, AppInstance might not work
// Create a default activation argument for launch
}
if (activatedEventArgs is not null && namedPipeClient is not null)
{
if (namedPipeClient.TryRedirectActivationTo(activatedEventArgs))
{
SentrySdk.AddBreadcrumb(BreadcrumbFactory.CreateInfo("Application exiting on RedirectActivationTo", "Hutao"));
XamlApplicationLifetime.ActivationAndInitializationCompleted = true;
Exit();
return;
}
}
logger.LogInformation($"{ConsoleBanner}");
@@ -90,10 +119,30 @@ public sealed partial class App : Application
// Manually invoke
SentrySdk.AddBreadcrumb(BreadcrumbFactory.CreateInfo("Activate and Initialize", "Application"));
activation.ActivateAndInitialize(HutaoActivationArguments.FromAppActivationArguments(activatedEventArgs));
HutaoActivationArguments hutaoArgs = activatedEventArgs is not null
? HutaoActivationArguments.FromAppActivationArguments(activatedEventArgs)
: HutaoActivationArguments.CreateDefaultLaunchArguments();
activation.ActivateAndInitialize(hutaoArgs);
}
catch (Exception ex)
{
// ⚠️ 添加更详细的异常日志
try
{
string errorPath = Path.Combine(
Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
"Hutao",
"startup_error.txt");
Directory.CreateDirectory(Path.GetDirectoryName(errorPath)!);
File.WriteAllText(errorPath, $"[{DateTime.Now:yyyy-MM-dd HH:mm:ss}] Startup Error:\n{ex}");
}
catch
{
// Ignore
}
SentrySdk.CaptureException(ex);
SentrySdk.Flush();

View File

@@ -31,8 +31,15 @@ public static partial class Bootstrap
[STAThread]
private static void Main(string[] args)
{
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Starting...");
#endif
if (Mutex.TryOpenExisting(LockName, out _))
{
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Another instance is running");
#endif
return;
}
@@ -42,9 +49,16 @@ public static partial class Bootstrap
mutexSecurity.AddAccessRule(new(SecurityIdentifiers.Everyone, MutexRights.FullControl, AccessControlType.Allow));
mutex = MutexAcl.Create(true, LockName, out bool created, mutexSecurity);
Debug.Assert(created);
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Mutex created");
#endif
}
catch (WaitHandleCannotBeOpenedException)
{
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] WaitHandleCannotBeOpenedException");
#endif
return;
}
@@ -54,34 +68,70 @@ public static partial class Bootstrap
{
if (!OSPlatformSupported())
{
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] OS not supported");
#endif
return;
}
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Setting environment variables");
#endif
Environment.SetEnvironmentVariable("WEBVIEW2_DEFAULT_BACKGROUND_COLOR", "00000000");
Environment.SetEnvironmentVariable("DOTNET_SYSTEM_BUFFERS_SHAREDARRAYPOOL_MAXARRAYSPERPARTITION", "128");
AppContext.SetData("MVVMTOOLKIT_ENABLE_INOTIFYPROPERTYCHANGING_SUPPORT", false);
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Initializing COM wrappers");
#endif
ComWrappersSupport.InitializeComWrappers();
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Initializing DI container");
#endif
// By adding the using statement, we can dispose the injected services when closing
using (ServiceProvider serviceProvider = DependencyInjection.Initialize())
{
Thread.CurrentThread.Name = "Snap Hutao Application Main Thread";
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Calling Application.Start()");
#endif
// If you hit a COMException REGDB_E_CLASSNOTREG (0x80040154) during debugging
// You can delete bin and obj folder and then rebuild.
// In a Desktop app this runs a message pump internally,
// and does not return until the application shuts down.
Application.Start(AppInitializationCallback);
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Application.Start() returned");
#endif
XamlApplicationLifetime.Exited = true;
}
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Flushing Sentry");
#endif
SentrySdk.Flush();
}
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.Main] Exiting");
#endif
}
private static void InitializeApp(ApplicationInitializationCallbackParams param)
{
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.InitializeApp] Callback invoked");
#endif
Gen2GcCallback.Register(() =>
{
SentrySdk.AddBreadcrumb(BreadcrumbFactory.CreateDebug("Gen2 GC triggered.", "Runtime"));
@@ -90,8 +140,17 @@ public static partial class Bootstrap
IServiceProvider serviceProvider = Ioc.Default;
_ = serviceProvider.GetRequiredService<ITaskContext>();
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.InitializeApp] Creating App instance");
#endif
// ⚠️ 只创建 App
// TaskContext 将在第一次被需要时自动创建(延迟初始化)
_ = serviceProvider.GetRequiredService<App>();
#if DEBUG
System.Diagnostics.Debug.WriteLine("[Bootstrap.InitializeApp] Initialization complete (TaskContext will be lazily created)");
#endif
}
private static bool OSPlatformSupported()

View File

@@ -4,14 +4,13 @@
using System.Collections.Frozen;
using System.Security.Cryptography;
using System.Text;
using Windows.ApplicationModel;
namespace Snap.Hutao.Core.ApplicationModel;
internal static class LimitedAccessFeatures
{
private static readonly string PackagePublisherId = Package.Current.Id.PublisherId;
private static readonly string PackageFamilyName = Package.Current.Id.FamilyName;
private static readonly string PackagePublisherId = PackageIdentityAdapter.PublisherId;
private static readonly string PackageFamilyName = PackageIdentityAdapter.FamilyName;
private static readonly FrozenDictionary<string, string> Features = WinRTAdaptive.ToFrozenDictionary(
[
@@ -67,8 +66,15 @@ internal static class LimitedAccessFeatures
KeyValuePair.Create("com.microsoft.windows.windowdecorations", "425261a8-7f73-4319-8a53-fc13f87e1717")
]);
public static LimitedAccessFeatureRequestResult TryUnlockFeature(string featureId)
public static Windows.ApplicationModel.LimitedAccessFeatureRequestResult TryUnlockFeature(string featureId)
{
if (!PackageIdentityAdapter.HasPackageIdentity)
{
// In unpackaged mode, we can't unlock limited access features
// Create a dummy result - actual implementation will handle the failure
return default;
}
return Windows.ApplicationModel.LimitedAccessFeatures.TryUnlockFeature(featureId, GetToken(featureId), GetAttestation(featureId));
}

View File

@@ -0,0 +1,109 @@
// Copyright (c) Millennium-Science-Technology-R-D-Inst. All rights reserved.
// Licensed under the MIT license.
using System.Diagnostics;
using System.IO;
using System.Reflection;
namespace Snap.Hutao.Core.ApplicationModel;
/// <summary>
/// Adapter to handle both packaged and unpackaged app scenarios
/// </summary>
internal static class PackageIdentityAdapter
{
private static readonly Lazy<bool> LazyHasPackageIdentity = new(CheckPackageIdentity);
private static readonly Lazy<string> LazyAppDirectory = new(GetAppDirectoryPath);
private static readonly Lazy<Version> LazyAppVersion = new(GetAppVersionInternal);
private static readonly Lazy<string> LazyFamilyName = new(GetFamilyNameInternal);
private static readonly Lazy<string> LazyPublisherId = new(GetPublisherIdInternal);
/// <summary>
/// Check if the app has package identity
/// </summary>
public static bool HasPackageIdentity => LazyHasPackageIdentity.Value;
/// <summary>
/// Get application installation directory
/// </summary>
public static string AppDirectory => LazyAppDirectory.Value;
/// <summary>
/// Get application version
/// </summary>
public static Version AppVersion => LazyAppVersion.Value;
/// <summary>
/// Get package family name (or fallback for unpackaged)
/// </summary>
public static string FamilyName => LazyFamilyName.Value;
/// <summary>
/// Get publisher ID (or fallback for unpackaged)
/// </summary>
public static string PublisherId => LazyPublisherId.Value;
private static bool CheckPackageIdentity()
{
try
{
// Try to access Package.Current - if it throws, we don't have package identity
_ = Windows.ApplicationModel.Package.Current.Id;
return true;
}
catch
{
return false;
}
}
private static string GetAppDirectoryPath()
{
if (HasPackageIdentity)
{
return Windows.ApplicationModel.Package.Current.InstalledLocation.Path;
}
// Unpackaged: use the exe directory
string? exePath = Process.GetCurrentProcess().MainModule?.FileName;
ArgumentException.ThrowIfNullOrEmpty(exePath);
string? directory = Path.GetDirectoryName(exePath);
ArgumentException.ThrowIfNullOrEmpty(directory);
return directory;
}
private static Version GetAppVersionInternal()
{
if (HasPackageIdentity)
{
return Windows.ApplicationModel.Package.Current.Id.Version.ToVersion();
}
// Unpackaged: use assembly version
Assembly assembly = Assembly.GetExecutingAssembly();
Version? version = assembly.GetName().Version;
return version ?? new Version(1, 0, 0, 0);
}
private static string GetFamilyNameInternal()
{
if (HasPackageIdentity)
{
return Windows.ApplicationModel.Package.Current.Id.FamilyName;
}
// Unpackaged: use a deterministic fallback
return "Snap.Hutao.Unpackaged";
}
private static string GetPublisherIdInternal()
{
if (HasPackageIdentity)
{
return Windows.ApplicationModel.Package.Current.Id.PublisherId;
}
// Unpackaged: use a fallback
return "CN=Millennium-Science-Technology-R-D-Inst";
}
}

View File

@@ -0,0 +1,43 @@
// Copyright (c) Millennium-Science-Technology-R-D-Inst. All rights reserved.
// Licensed under the MIT license.
using System.Diagnostics;
using System.IO;
namespace Snap.Hutao.Core.ApplicationModel;
/// <summary>
/// Diagnostic helper for PackageIdentityAdapter
/// </summary>
internal static class PackageIdentityDiagnostics
{
public static void LogDiagnostics()
{
try
{
string logPath = Path.Combine(
Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
"Hutao",
"startup_diagnostics.txt");
Directory.CreateDirectory(Path.GetDirectoryName(logPath)!);
using (StreamWriter writer = File.CreateText(logPath))
{
writer.WriteLine($"[{DateTime.Now:yyyy-MM-dd HH:mm:ss}] Startup Diagnostics");
writer.WriteLine($"HasPackageIdentity: {PackageIdentityAdapter.HasPackageIdentity}");
writer.WriteLine($"AppVersion: {PackageIdentityAdapter.AppVersion}");
writer.WriteLine($"AppDirectory: {PackageIdentityAdapter.AppDirectory}");
writer.WriteLine($"FamilyName: {PackageIdentityAdapter.FamilyName}");
writer.WriteLine($"PublisherId: {PackageIdentityAdapter.PublisherId}");
writer.WriteLine("---");
}
Debug.WriteLine($"Diagnostics written to: {logPath}");
}
catch (Exception ex)
{
Debug.WriteLine($"Failed to write diagnostics: {ex.Message}");
}
}
}

View File

@@ -2,6 +2,7 @@
// Licensed under the MIT license.
using Microsoft.EntityFrameworkCore;
using Snap.Hutao.Core.ApplicationModel;
using Snap.Hutao.Model.Entity.Database;
using System.Security.Cryptography;
using System.Text;
@@ -17,7 +18,20 @@ internal sealed partial class HutaoDiagnostics : IHutaoDiagnostics
[GeneratedConstructor]
public partial HutaoDiagnostics(IServiceProvider serviceProvider);
public ApplicationDataContainer LocalSettings { get => ApplicationData.Current.LocalSettings; }
public ApplicationDataContainer? LocalSettings
{
get
{
if (PackageIdentityAdapter.HasPackageIdentity)
{
return ApplicationData.Current.LocalSettings;
}
// In unpackaged mode, ApplicationDataContainer is not available
// Return null - scripting/diagnostics code should handle this gracefully
return null;
}
}
public async ValueTask<int> ExecuteSqlAsync(string sql)
{

View File

@@ -8,7 +8,7 @@ namespace Snap.Hutao.Core.Diagnostics;
[SuppressMessage("", "SH001", Justification = "IHutaoDiagnostics must be public in order to be exposed to the scripting environment")]
public interface IHutaoDiagnostics
{
ApplicationDataContainer LocalSettings { get; }
ApplicationDataContainer? LocalSettings { get; }
ValueTask<int> ExecuteSqlAsync(string sql);

View File

@@ -4,6 +4,7 @@
using Microsoft.Web.WebView2.Core;
using Microsoft.Win32;
using Microsoft.Windows.AppNotifications;
using Snap.Hutao.Core.ApplicationModel;
using Snap.Hutao.Core.ExceptionService;
using Snap.Hutao.Core.IO;
using Snap.Hutao.Core.IO.Hashing;
@@ -13,31 +14,32 @@ using System.Diagnostics;
using System.IO;
using System.Security.Cryptography;
using System.Text;
using Windows.ApplicationModel;
using Windows.Storage;
namespace Snap.Hutao.Core;
internal static class HutaoRuntime
{
public static Version Version { get; } = Package.Current.Id.Version.ToVersion();
public static Version Version { get; } = PackageIdentityAdapter.AppVersion;
public static string UserAgent { get; } = $"Snap Hutao/{Version}";
public static string DataDirectory { get; } = InitializeDataDirectory();
public static string LocalCacheDirectory { get; } = ApplicationData.Current.LocalCacheFolder.Path;
public static string LocalCacheDirectory { get; } = InitializeLocalCacheDirectory();
public static string FamilyName { get; } = Package.Current.Id.FamilyName;
public static string FamilyName { get; } = PackageIdentityAdapter.FamilyName;
public static string DeviceId { get; } = InitializeDeviceId();
public static WebView2Version WebView2Version { get; } = InitializeWebView2();
public static bool IsProcessElevated { get; } = LocalSetting.Get(SettingKeys.OverrideElevationRequirement, false) || Environment.IsPrivilegedProcess;
// ⚠️ 延迟初始化以避免循环依赖
private static readonly Lazy<bool> LazyIsProcessElevated = new(GetIsProcessElevated);
public static bool IsProcessElevated => LazyIsProcessElevated.Value;
// Requires main thread
public static bool IsAppNotificationEnabled { get; } = AppNotificationManager.Default.Setting is AppNotificationSetting.Enabled;
public static bool IsAppNotificationEnabled { get; } = CheckAppNotificationEnabled();
public static string? GetDisplayName()
{
@@ -106,32 +108,57 @@ internal static class HutaoRuntime
return string.Intern(directory);
}
private static string InitializeDataDirectory()
private static bool GetIsProcessElevated()
{
// Delete the previous data folder if it exists
// ⚠️ 这里调用 LocalSetting 时,确保 DataDirectory 已经初始化完成
try
{
string previousDirectory = LocalSetting.Get(SettingKeys.PreviousDataDirectoryToDelete, string.Empty);
if (!string.IsNullOrEmpty(previousDirectory) && Directory.Exists(previousDirectory))
{
Directory.SetReadOnly(previousDirectory, false);
Directory.Delete(previousDirectory, true);
return LocalSetting.Get(SettingKeys.OverrideElevationRequirement, false) || Environment.IsPrivilegedProcess;
}
}
finally
catch
{
LocalSetting.Set(SettingKeys.PreviousDataDirectoryToDelete, string.Empty);
// 如果读取失败,使用默认值
return Environment.IsPrivilegedProcess;
}
}
// Check if the preferred path is set
string currentDirectory = LocalSetting.Get(SettingKeys.DataDirectory, string.Empty);
if (!string.IsNullOrEmpty(currentDirectory))
private static string InitializeLocalCacheDirectory()
{
Directory.CreateDirectory(currentDirectory);
return currentDirectory;
if (PackageIdentityAdapter.HasPackageIdentity)
{
return Windows.Storage.ApplicationData.Current.LocalCacheFolder.Path;
}
// Unpackaged: use %LOCALAPPDATA%\Snap.Hutao\Cache
string localAppData = Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData);
const string FolderName
#if IS_ALPHA_BUILD
= "HutaoAlpha";
#elif IS_CANARY_BUILD
= "HutaoCanary";
#else
= "Hutao";
#endif
string cacheDir = Path.Combine(localAppData, FolderName, "Cache");
Directory.CreateDirectory(cacheDir);
return cacheDir;
}
private static bool CheckAppNotificationEnabled()
{
try
{
return AppNotificationManager.Default.Setting is AppNotificationSetting.Enabled;
}
catch
{
// In unpackaged mode, this might fail - return false
return false;
}
}
private static string InitializeDataDirectory()
{
const string FolderName
#if IS_ALPHA_BUILD
= "HutaoAlpha";
@@ -141,30 +168,43 @@ internal static class HutaoRuntime
= "Hutao";
#endif
// ⚠️ 不要在这里调用 LocalSetting - 会导致循环依赖
// 先确定默认的数据目录位置
// Check if the old documents path exists
string myDocumentsHutaoDirectory = Path.GetFullPath(Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.MyDocuments), FolderName));
if (Directory.Exists(myDocumentsHutaoDirectory))
{
LocalSetting.Set(SettingKeys.DataDirectory, myDocumentsHutaoDirectory);
return myDocumentsHutaoDirectory;
}
// Prefer LocalApplicationData
string localApplicationData = ApplicationData.Current.LocalFolder.Path;
string path = Path.GetFullPath(Path.Combine(localApplicationData, FolderName));
// Use LocalApplicationData
string localApplicationData;
if (PackageIdentityAdapter.HasPackageIdentity)
{
localApplicationData = Windows.Storage.ApplicationData.Current.LocalFolder.Path;
}
else
{
// Unpackaged: use %LOCALAPPDATA%
localApplicationData = Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData);
}
string defaultPath = Path.GetFullPath(Path.Combine(localApplicationData, FolderName));
// ⚠️ 延迟处理:在第一次使用 LocalSetting 后再检查是否有自定义路径
// 这里返回默认路径,后续通过 LocalSetting 可能会更新
try
{
Directory.CreateDirectory(path);
Directory.CreateDirectory(defaultPath);
}
catch (Exception ex)
{
// FileNotFoundException | UnauthorizedAccessException
// We don't have enough permission
HutaoException.InvalidOperation($"Failed to create data folder: {path}", ex);
HutaoException.InvalidOperation($"Failed to create data folder: {defaultPath}", ex);
}
LocalSetting.Set(SettingKeys.DataDirectory, path);
return path;
return defaultPath;
}
private static string InitializeDeviceId()

View File

@@ -1,11 +1,10 @@
// Copyright (c) DGP Studio. All rights reserved.
// Licensed under the MIT license.
using Snap.Hutao.Core.ApplicationModel;
using System.IO;
using System.Security.AccessControl;
using System.Security.Principal;
using Windows.ApplicationModel;
using Windows.Storage;
namespace Snap.Hutao.Core;
@@ -13,7 +12,7 @@ internal static class InstalledLocation
{
public static string GetAbsolutePath(string relativePath)
{
return Path.Combine(Package.Current.InstalledLocation.Path, relativePath);
return Path.Combine(PackageIdentityAdapter.AppDirectory, relativePath);
}
public static void CopyFileFromApplicationUri(string url, string path)
@@ -23,8 +22,26 @@ internal static class InstalledLocation
static async Task CopyApplicationUriFileCoreAsync(string url, string path)
{
await Task.CompletedTask.ConfigureAwait(ConfigureAwaitOptions.ForceYielding);
StorageFile file = await StorageFile.GetFileFromApplicationUriAsync(url.ToUri());
using (Stream outputStream = (await file.OpenReadAsync()).AsStreamForRead())
Uri uri = url.ToUri();
Stream outputStream;
if (PackageIdentityAdapter.HasPackageIdentity)
{
// Packaged: use StorageFile
Windows.Storage.StorageFile file = await Windows.Storage.StorageFile.GetFileFromApplicationUriAsync(uri);
outputStream = (await file.OpenReadAsync()).AsStreamForRead();
}
else
{
// Unpackaged: read from file system directly
// Assume ms-appx:/// points to the app directory
string localPath = uri.LocalPath.TrimStart('/');
string fullPath = Path.Combine(PackageIdentityAdapter.AppDirectory, localPath);
outputStream = File.OpenRead(fullPath);
}
using (outputStream)
{
if (File.Exists(path))
{

View File

@@ -74,8 +74,15 @@ internal sealed partial class AppActivation : IAppActivation, IAppActivationActi
public void ActivateAndInitialize(HutaoActivationArguments args)
{
#if DEBUG
Debug.WriteLine("[AppActivation] ActivateAndInitialize called");
#endif
if (Volatile.Read(ref isActivating) is 1)
{
#if DEBUG
Debug.WriteLine("[AppActivation] Already activating, returning");
#endif
return;
}
@@ -85,21 +92,52 @@ internal sealed partial class AppActivation : IAppActivation, IAppActivationActi
{
try
{
#if DEBUG
Debug.WriteLine("[AppActivation] Starting activation process");
#endif
using (await activateLock.LockAsync().ConfigureAwait(false))
{
if (Interlocked.CompareExchange(ref isActivating, 1, 0) is not 0)
{
#if DEBUG
Debug.WriteLine("[AppActivation] Race condition detected, returning");
#endif
return;
}
#if DEBUG
Debug.WriteLine("[AppActivation] Calling UnsynchronizedHandleActivationAsync");
#endif
await UnsynchronizedHandleActivationAsync(args).ConfigureAwait(false);
#if DEBUG
Debug.WriteLine("[AppActivation] Calling UnsynchronizedHandleInitializationAsync");
#endif
await UnsynchronizedHandleInitializationAsync().ConfigureAwait(false);
#if DEBUG
Debug.WriteLine("[AppActivation] Initialization completed successfully");
#endif
}
}
catch (Exception ex)
{
#if DEBUG
Debug.WriteLine($"[AppActivation] Exception during activation: {ex}");
#endif
throw;
}
finally
{
XamlApplicationLifetime.ActivationAndInitializationCompleted = true;
Interlocked.Exchange(ref isActivating, 0);
#if DEBUG
Debug.WriteLine("[AppActivation] ActivationAndInitializationCompleted set to true");
#endif
}
}
}
@@ -313,16 +351,36 @@ internal sealed partial class AppActivation : IAppActivation, IAppActivationActi
private async ValueTask<Window?> WaitWindowAsync<TWindow>()
where TWindow : Window
{
#if DEBUG
Debug.WriteLine($"[AppActivation.WaitWindowAsync] Waiting for window type: {typeof(TWindow).Name}");
#endif
await taskContext.SwitchToMainThreadAsync();
#if DEBUG
Debug.WriteLine("[AppActivation.WaitWindowAsync] Switched to main thread");
#endif
if (currentXamlWindowReference.Window is not { } window)
{
#if DEBUG
Debug.WriteLine("[AppActivation.WaitWindowAsync] Creating new window instance");
#endif
try
{
window = serviceProvider.GetRequiredService<TWindow>();
#if DEBUG
Debug.WriteLine($"[AppActivation.WaitWindowAsync] Window created successfully: {window.GetType().Name}");
#endif
}
catch (COMException)
catch (COMException ex)
{
#if DEBUG
Debug.WriteLine($"[AppActivation.WaitWindowAsync] COMException: {ex}");
#endif
if (XamlApplicationLifetime.Exiting)
{
return default;
@@ -330,11 +388,33 @@ internal sealed partial class AppActivation : IAppActivation, IAppActivationActi
throw;
}
catch (Exception ex)
{
#if DEBUG
Debug.WriteLine($"[AppActivation.WaitWindowAsync] Exception creating window: {ex}");
#endif
throw;
}
currentXamlWindowReference.Window = window;
}
else
{
#if DEBUG
Debug.WriteLine($"[AppActivation.WaitWindowAsync] Using existing window: {window.GetType().Name}");
#endif
}
#if DEBUG
Debug.WriteLine("[AppActivation.WaitWindowAsync] Calling window.SwitchTo()");
#endif
window.SwitchTo();
#if DEBUG
Debug.WriteLine("[AppActivation.WaitWindowAsync] Window activated");
#endif
window.AppWindow?.MoveInZOrderAtTop();
return window;
}

View File

@@ -66,4 +66,14 @@ internal sealed class HutaoActivationArguments
return result;
}
public static HutaoActivationArguments CreateDefaultLaunchArguments()
{
return new HutaoActivationArguments
{
IsRedirectTo = false,
Kind = HutaoActivationKind.Launch,
LaunchActivatedArguments = string.Empty
};
}
}

View File

@@ -20,9 +20,9 @@ internal static class LoggerFactoryExtension
#if DEBUG || IS_ALPHA_BUILD || IS_CANARY_BUILD
// Alpha and Canary produces noisy events
options.Dsn = "https://ec3799184191c344ca06c592cb97a464@sentry.snapgenshin.com/4";
options.Dsn = "https://2d3047ff2d451986bc7ef395d1f1fe63@o4507525750521856.ingest.us.sentry.io/4510413123682304";
#else
options.Dsn = "https://1a1151ce5ac4e7f1536edf085bd483ec@sentry.snapgenshin.com/2";
options.Dsn = "https://2d3047ff2d451986bc7ef395d1f1fe63@o4507525750521856.ingest.us.sentry.io/4510413123682304";
#endif
#if DEBUG
@@ -36,8 +36,8 @@ internal static class LoggerFactoryExtension
options.Environment = GetBuildEnvironment();
// Suppress logs to generate events and breadcrumbs
options.MinimumBreadcrumbLevel = LogLevel.None;
options.MinimumEventLevel = LogLevel.None;
options.MinimumBreadcrumbLevel = LogLevel.Information;
options.MinimumEventLevel = LogLevel.Error;
options.ProfilesSampleRate = 1.0D;
options.TracesSampleRate = 1.0D;

View File

@@ -1,12 +1,16 @@
// Copyright (c) DGP Studio. All rights reserved.
// Licensed under the MIT license.
using Snap.Hutao.Core.ApplicationModel;
using Snap.Hutao.Core.ExceptionService;
using Snap.Hutao.Factory.Process;
using Snap.Hutao.Win32;
using Snap.Hutao.Win32.Foundation;
using System.Collections.Concurrent;
using System.Collections.Frozen;
using System.Diagnostics;
using System.IO;
using System.Text.Json;
using Windows.Storage;
namespace Snap.Hutao.Core.Setting;
@@ -36,40 +40,20 @@ internal static class LocalSetting
typeof(ApplicationDataCompositeValue)
];
private static readonly ApplicationDataContainer Container = ApplicationData.Current.LocalSettings;
private static readonly Lazy<ISettingStorage> LazyStorage = new(CreateStorage);
private static ISettingStorage Storage => LazyStorage.Value;
public static T Get<T>(string key, T defaultValue = default!)
{
Debug.Assert(SupportedTypes.Contains(typeof(T)));
if (Container.Values.TryGetValue(key, out object? value))
{
// unbox the value
return value is null ? defaultValue : (T)value;
}
Set(key, defaultValue);
return defaultValue;
return Storage.Get(key, defaultValue);
}
public static void Set<T>(string key, T value)
{
Debug.Assert(SupportedTypes.Contains(typeof(T)));
try
{
Container.Values[key] = value;
}
catch (Exception ex)
{
// 状态管理器无法写入设置
if (HutaoNative.IsWin32(ex.HResult, WIN32_ERROR.ERROR_STATE_WRITE_SETTING_FAILED))
{
HutaoNative.Instance.ShowErrorMessage(ex.Message, ExceptionFormat.Format(ex));
ProcessFactory.KillCurrent();
}
throw;
}
Storage.Set(key, value);
}
public static void SetIf<T>(bool condition, string key, T value)
@@ -103,4 +87,299 @@ internal static class LocalSetting
Set(key, newValue);
return oldValue;
}
private static ISettingStorage CreateStorage()
{
if (PackageIdentityAdapter.HasPackageIdentity)
{
return new PackagedSettingStorage();
}
return new UnpackagedSettingStorage();
}
private interface ISettingStorage
{
T Get<T>(string key, T defaultValue);
void Set<T>(string key, T value);
}
private sealed class PackagedSettingStorage : ISettingStorage
{
private readonly ApplicationDataContainer container = ApplicationData.Current.LocalSettings;
public T Get<T>(string key, T defaultValue)
{
if (container.Values.TryGetValue(key, out object? value))
{
// unbox the value
return value is null ? defaultValue : (T)value;
}
Set(key, defaultValue);
return defaultValue;
}
public void Set<T>(string key, T value)
{
try
{
container.Values[key] = value;
}
catch (Exception ex)
{
// 状态管理器无法写入设置
if (HutaoNative.IsWin32(ex.HResult, WIN32_ERROR.ERROR_STATE_WRITE_SETTING_FAILED))
{
HutaoNative.Instance.ShowErrorMessage(ex.Message, ExceptionFormat.Format(ex));
ProcessFactory.KillCurrent();
}
throw;
}
}
}
private sealed class UnpackagedSettingStorage : ISettingStorage
{
private readonly string settingsFilePath;
private readonly ConcurrentDictionary<string, object?> cache = new();
private readonly object fileLock = new();
private readonly JsonSerializerOptions jsonOptions = new()
{
WriteIndented = true,
Converters =
{
new ApplicationDataCompositeValueJsonConverter(),
}
};
public UnpackagedSettingStorage()
{
string localAppData = Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData);
const string FolderName
#if IS_ALPHA_BUILD
= "HutaoAlpha";
#elif IS_CANARY_BUILD
= "HutaoCanary";
#else
= "Hutao";
#endif
string settingsDir = Path.Combine(localAppData, FolderName, "Settings");
Directory.CreateDirectory(settingsDir);
settingsFilePath = Path.Combine(settingsDir, "LocalSettings.json");
LoadFromFile();
}
public T Get<T>(string key, T defaultValue)
{
if (cache.TryGetValue(key, out object? value))
{
if (value is null)
{
return defaultValue;
}
// Handle JSON deserialization for complex types
if (value is JsonElement jsonElement)
{
try
{
// ⚠️ 特殊处理JSON 数字类型转换
Type targetType = typeof(T);
if (jsonElement.ValueKind == JsonValueKind.Number)
{
if (targetType == typeof(int))
{
return (T)(object)jsonElement.GetInt32();
}
if (targetType == typeof(long))
{
return (T)(object)jsonElement.GetInt64();
}
if (targetType == typeof(short))
{
return (T)(object)jsonElement.GetInt16();
}
if (targetType == typeof(byte))
{
return (T)(object)jsonElement.GetByte();
}
if (targetType == typeof(uint))
{
return (T)(object)jsonElement.GetUInt32();
}
if (targetType == typeof(ulong))
{
return (T)(object)jsonElement.GetUInt64();
}
if (targetType == typeof(ushort))
{
return (T)(object)jsonElement.GetUInt16();
}
if (targetType == typeof(float))
{
return (T)(object)jsonElement.GetSingle();
}
if (targetType == typeof(double))
{
return (T)(object)jsonElement.GetDouble();
}
}
// 其他类型使用标准反序列化
return jsonElement.Deserialize<T>(jsonOptions) ?? defaultValue;
}
catch
{
return defaultValue;
}
}
// ⚠️ 如果是直接从 cache 读取的值,也可能需要类型转换
// 例如double -> int
if (value is double doubleValue)
{
Type targetType = typeof(T);
if (targetType == typeof(int))
{
return (T)(object)(int)doubleValue;
}
if (targetType == typeof(long))
{
return (T)(object)(long)doubleValue;
}
if (targetType == typeof(short))
{
return (T)(object)(short)doubleValue;
}
if (targetType == typeof(byte))
{
return (T)(object)(byte)doubleValue;
}
}
return (T)value;
}
Set(key, defaultValue);
return defaultValue;
}
public void Set<T>(string key, T value)
{
cache[key] = value;
SaveToFile();
}
private void LoadFromFile()
{
lock (fileLock)
{
if (!File.Exists(settingsFilePath))
{
return;
}
try
{
string json = File.ReadAllText(settingsFilePath);
Dictionary<string, JsonElement>? data = JsonSerializer.Deserialize<Dictionary<string, JsonElement>>(json, jsonOptions);
if (data is not null)
{
foreach ((string key, JsonElement value) in data)
{
cache[key] = value;
}
}
}
catch
{
// If file is corrupted, start fresh
}
}
}
private void SaveToFile()
{
lock (fileLock)
{
try
{
// Convert cache to serializable dictionary
Dictionary<string, object?> serializableData = new(cache);
string json = JsonSerializer.Serialize(serializableData, jsonOptions);
File.WriteAllText(settingsFilePath, json);
}
catch (Exception ex)
{
Debug.WriteLine($"Failed to save settings: {ex.Message}");
}
}
}
}
// Converter for ApplicationDataCompositeValue
private sealed class ApplicationDataCompositeValueJsonConverter : System.Text.Json.Serialization.JsonConverter<ApplicationDataCompositeValue>
{
public override ApplicationDataCompositeValue? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
if (reader.TokenType != JsonTokenType.StartObject)
{
return null;
}
ApplicationDataCompositeValue composite = new();
while (reader.Read())
{
if (reader.TokenType == JsonTokenType.EndObject)
{
return composite;
}
if (reader.TokenType != JsonTokenType.PropertyName)
{
continue;
}
string? key = reader.GetString();
reader.Read();
if (key is not null)
{
composite[key] = reader.TokenType switch
{
JsonTokenType.String => reader.GetString(),
JsonTokenType.Number => reader.TryGetInt64(out long l) ? l : reader.GetDouble(),
JsonTokenType.True => true,
JsonTokenType.False => false,
_ => null
};
}
}
return composite;
}
public override void Write(Utf8JsonWriter writer, ApplicationDataCompositeValue value, JsonSerializerOptions options)
{
writer.WriteStartObject();
foreach ((string key, object? val) in value)
{
writer.WritePropertyName(key);
if (val is null)
{
writer.WriteNullValue();
}
else
{
JsonSerializer.Serialize(writer, val, val.GetType(), options);
}
}
writer.WriteEndObject();
}
}
}

View File

@@ -147,6 +147,27 @@ internal sealed class ProcessFactory
{
string repoDirectory = HutaoRuntime.GetDataRepositoryDirectory();
string fullTrustFilePath = Path.Combine(repoDirectory, "Snap.ContentDelivery", "Snap.Hutao.FullTrust.exe");
// Check if FullTrust executable exists - if not, fallback to normal admin mode
if (!File.Exists(fullTrustFilePath))
{
string errorMessage = $"""
Island 功能需要的 FullTrust 进程文件不存在,将使用普通管理员模式启动游戏。
预期路径:{fullTrustFilePath}
原因ContentDelivery 仓库尚未下载或初始化失败(常见于非打包模式首次运行)
Island 功能将不可用,但游戏可以正常启动。
等待仓库下载完成后可重新尝试使用 Island 功能。
""";
// Capture as breadcrumb instead of exception
SentrySdk.AddBreadcrumb(errorMessage, category: "process.fulltrust", level: Sentry.BreadcrumbLevel.Warning);
// Fallback to normal admin mode - Island features will not work but game can launch
return CreateUsingShellExecuteRunAs(arguments, fileName, workingDirectory);
}
StartUsingShellExecuteRunAs(fullTrustFilePath);
FullTrustProcessStartInfoRequest request = new()

View File

@@ -124,6 +124,8 @@ internal static class AvatarIds
public static readonly AvatarId Flins = 10000120;
public static readonly AvatarId Aino = 10000121;
public static readonly AvatarId Nefer = 10000122;
public static readonly AvatarId Durin = 10000123;
public static readonly AvatarId Jahoda = 10000124;
private static readonly FrozenSet<AvatarId> StandardWishIds =
[

View File

@@ -23,7 +23,8 @@ internal static class WeaponIds
12401U, 12402U, 12403U, 12405U,
13401U, 13407U,
14401U, 14402U, 14403U, 14409U,
15401U, 15402U, 15403U, 15405U
15401U, 15402U, 15403U, 15405U,
15434U
];
public static readonly FrozenSet<WeaponId> OrangeStandardWishIds =
@@ -33,6 +34,7 @@ internal static class WeaponIds
13502U, 13505U,
14501U, 14502U,
15501U, 15502U,
15515U, 15518U
];
public static bool IsOrangeStandardWish(in WeaponId weaponId)

View File

@@ -13,7 +13,7 @@
<Identity
Name="60568DGPStudio.SnapHutao"
Publisher="CN=35C8E923-85DF-49A7-9172-B39DC6312C52"
Version="1.17.1.0" />
Version="1.17.4.0" />
<Properties>
<DisplayName>Snap Hutao</DisplayName>

View File

@@ -1181,6 +1181,9 @@ Space Available: {2}</value>
<data name="ServiceYaeWaitForGameResponseMessage" xml:space="preserve">
<value>Waiting for game data</value>
</data>
<data name="UIViewMainTitleBarBackgroundActivityAction" xml:space="preserve">
<value>Background task</value>
</data>
<data name="UIViewPageAvatarPropertyRecommendedAppendProperties" xml:space="preserve">
<value>Additional Property Recommendation</value>
</data>
@@ -3989,4 +3992,7 @@ Space Available: {2}</value>
<data name="WindowIdentifyMonitorHeader" xml:space="preserve">
<value>Monitor ID</value>
</data>
<data name="UIViewMainTitleBarInvertTheme" xml:space="preserve">
<value>Invert Theme</value>
</data>
</root>

View File

@@ -4020,4 +4020,7 @@
<data name="WindowIdentifyMonitorHeader" xml:space="preserve">
<value>显示器编号</value>
</data>
<data name="UIViewMainTitleBarInvertTheme" xml:space="preserve">
<value>主题切换</value>
</data>
</root>

View File

@@ -24,8 +24,10 @@ internal static class AvatarViewBuilderExtension
{
if (detailedCharacter.Costumes is [{ Id: { } id }, ..])
{
MetadataCostume costume = avatar.Costumes.Single(c => c.Id == id);
MetadataCostume? costume = avatar.Costumes.SingleOrDefault(c => c.Id == id);
if (costume != null)
{
ArgumentNullException.ThrowIfNull(costume.FrontIcon);
ArgumentNullException.ThrowIfNull(costume.SideIcon);
@@ -34,6 +36,13 @@ internal static class AvatarViewBuilderExtension
builder.View.SideIcon = AvatarIconConverter.IconNameToUri(costume.SideIcon);
}
else
{
// Costume not found in metadata, fallback to default avatar icon
builder.View.Icon = AvatarIconConverter.IconNameToUri(avatar.Icon);
builder.View.SideIcon = AvatarIconConverter.IconNameToUri(avatar.SideIcon);
}
}
else
{
builder.View.Icon = AvatarIconConverter.IconNameToUri(avatar.Icon);
builder.View.SideIcon = AvatarIconConverter.IconNameToUri(avatar.SideIcon);

View File

@@ -77,22 +77,49 @@ internal sealed class GameIslandInterop : IGameIslandInterop
{
nint handle = accessor.SafeMemoryMappedViewHandle.DangerousGetHandle();
InitializeIslandEnvironment(handle, context.LaunchOptions, context.IsOversea);
if (!resume)
{
if (context.Process is not FullTrustProcess fullTrustProcess)
{
throw HutaoException.InvalidOperation("Process is not full trust");
}
ArgumentException.ThrowIfNullOrEmpty(islandPath);
if (!File.Exists(islandPath))
{
throw HutaoException.InvalidOperation(SH.ServiceGameIslandTargetVersionFileNotExists);
}
// Support both FullTrust and normal admin mode
if (context.Process is FullTrustProcess fullTrustProcess)
{
// Use FullTrust process for injection (suspended process)
fullTrustProcess.LoadLibrary(FullTrustLoadLibraryRequest.Create("Island", islandPath));
fullTrustProcess.ResumeMainThread();
}
else
{
// Use native injection for normal admin mode
// The process was already started by CreateUsingShellExecuteRunAs
// Just inject the DLL into the running process
try
{
// Wait a bit for process to initialize
// await Task.Delay(500, token).ConfigureAwait(false);
// ⚠️此处需要更多调查
// Inject using RemoteThread
DllInjectionUtilities.InjectUsingRemoteThread(islandPath, context.Process.Id);
}
catch (Exception ex)
{
// Log the injection failure but don't crash - game can still run
SentrySdk.AddBreadcrumb(
$"Island DLL injection failed: {ex.Message}",
category: "island.injection",
level: Sentry.BreadcrumbLevel.Error);
// Re-throw to let the caller handle it
throw HutaoException.Throw($"Island DLL 注入失败: {ex.Message}", ex);
}
}
}
await PeriodicUpdateIslandEnvironmentAsync(context, handle, token).ConfigureAwait(false);
}

View File

@@ -45,6 +45,7 @@ internal sealed class GameProcessFactory
string gameFilePath = context.FileSystem.GameFilePath;
string gameDirectory = context.FileSystem.GameDirectory;
// ProcessFactory.CreateUsingFullTrustSuspended will automatically fallback to normal mode if FullTrust.exe is missing
return launchOptions.IsIslandEnabled.Value
? ProcessFactory.CreateUsingFullTrustSuspended(commandLine, gameFilePath, gameDirectory)
: ProcessFactory.CreateUsingShellExecuteRunAs(commandLine, gameFilePath, gameDirectory);

View File

@@ -2,17 +2,17 @@
// Licensed under the MIT license.
using Snap.Hutao.Core.IO.Hashing;
using Snap.Hutao.Core.Setting;
using Snap.Hutao.Web.Hutao;
using System.Collections.Immutable;
using System.Runtime.InteropServices;
using System.Security.Cryptography;
using Windows.Storage;
namespace Snap.Hutao.Service.Git;
internal static class RepositoryAffinity
{
private static readonly ApplicationDataContainer RepositoryContainer = ApplicationData.Current.LocalSettings.CreateContainer("RepositoryAffinity", ApplicationDataCreateDisposition.Always);
private const string RepositoryAffinityPrefix = "RepositoryAffinity::";
private static readonly Lock SyncRoot = new();
public static ImmutableArray<GitRepository> Sort(ImmutableArray<GitRepository> repositories)
@@ -23,9 +23,11 @@ internal static class RepositoryAffinity
for (int i = 0; i < repositories.Length; i++)
{
GitRepository repository = repositories[i];
ApplicationDataContainer container = RepositoryContainer.CreateContainer(repository.Name, ApplicationDataCreateDisposition.Always);
string key = Hash.ToHexString(HashAlgorithmName.SHA256, repository.HttpsUrl.OriginalString.ToUpperInvariant());
counts[i] = container.Values[key] is int c ? c : 0;
string key = GetSettingKey(repository.Name, repository.HttpsUrl.OriginalString);
// 对读取值做下限保护,确保排序使用的是非负失败计数
int raw = LocalSetting.Get(key, 0);
counts[i] = Math.Max(0, raw);
}
Array.Sort(counts, ImmutableCollectionsMarshal.AsArray(repositories));
@@ -42,10 +44,14 @@ internal static class RepositoryAffinity
{
lock (SyncRoot)
{
ApplicationDataContainer container = RepositoryContainer.CreateContainer(name, ApplicationDataCreateDisposition.Always);
string key = Hash.ToHexString(HashAlgorithmName.SHA256, url.ToUpperInvariant());
object box = container.Values[key];
container.Values[key] = box is int count ? unchecked(count + 1) : 1;
string key = GetSettingKey(name, url);
int currentCount = LocalSetting.Get(key, 0);
// 防止整数上溢:当已到达 int.MaxValue 时不再自增
if (currentCount < int.MaxValue)
{
LocalSetting.Set(key, currentCount + 1);
}
}
}
@@ -58,10 +64,20 @@ internal static class RepositoryAffinity
{
lock (SyncRoot)
{
ApplicationDataContainer container = RepositoryContainer.CreateContainer(name, ApplicationDataCreateDisposition.Always);
string key = Hash.ToHexString(HashAlgorithmName.SHA256, url.ToUpperInvariant());
object box = container.Values[key];
container.Values[key] = box is int count ? unchecked(count - 1) : 0;
string key = GetSettingKey(name, url);
int currentCount = LocalSetting.Get(key, 0);
// 失败次数不允许小于 0避免出现负数或整型下溢
if (currentCount > 0)
{
LocalSetting.Set(key, currentCount - 1);
}
}
}
private static string GetSettingKey(string name, string url)
{
string urlHash = Hash.ToHexString(HashAlgorithmName.SHA256, url.ToUpperInvariant());
return $"{RepositoryAffinityPrefix}{name}::{urlHash}";
}
}

View File

@@ -1,4 +1,4 @@
<Project Sdk="Microsoft.NET.Sdk">
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>WinExe</OutputType>
<TargetFramework>net10.0-windows10.0.26100.0</TargetFramework>
@@ -10,6 +10,9 @@
<RuntimeIdentifier>win-x64</RuntimeIdentifier>
<UseWinUI>true</UseWinUI>
<UseWPF>False</UseWPF>
<!-- 配置版本号 -->
<Version>1.17.4.0</Version>
<UseWindowsForms>False</UseWindowsForms>
<ImplicitUsings>False</ImplicitUsings>
<Nullable>enable</Nullable>
@@ -47,12 +50,18 @@
<AppxPackageSigningEnabled>True</AppxPackageSigningEnabled>
<PackageCertificateKeyFile>Snap.Hutao_TemporaryKey.pfx</PackageCertificateKeyFile>
<AppxPackageSigningTimestampDigestAlgorithm>SHA256</AppxPackageSigningTimestampDigestAlgorithm>
<!-- 关闭AppX打包 -->
<AppxPackage>false</AppxPackage>
<!-- 不设置打包类型 -->
<WindowsPackageType>None</WindowsPackageType>
</PropertyGroup>
<ItemGroup>
<None Remove="Package.appxmanifest" />
<None Remove="Package.development.appxmanifest" />
</ItemGroup>
<ItemGroup>
<Manifest Include="$(ApplicationManifest)" />
<AppxManifest Include="Package.appxmanifest" Condition="'$(ConfigurationName)'!='Debug'" />
<AppxManifest Include="Package.development.appxmanifest" Condition="'$(ConfigurationName)'=='Debug'" />
</ItemGroup>
<Target Name="DeleteLibsAfterBuild" AfterTargets="Build">
@@ -241,15 +250,15 @@
<PackageReference Include="CommunityToolkit.WinUI.Controls.SettingsControls" Version="8.2.251105-build.1544" />
<PackageReference Include="CommunityToolkit.WinUI.Converters" Version="8.2.251105-build.1544" />
<PackageReference Include="CommunityToolkit.WinUI.Media" Version="8.2.251105-build.1544" />
<PackageReference Include="Google.Protobuf" Version="3.33.0" />
<PackageReference Include="Google.Protobuf" Version="3.33.1" />
<PackageReference Include="JetBrains.Annotations" Version="2025.2.2" />
<PackageReference Include="Microsoft.CodeAnalysis.BannedApiAnalyzers" Version="4.14.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Scripting" Version="4.14.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="4.14.0" />
<PackageReference Include="Microsoft.CodeAnalysis.Workspaces.MSBuild" Version="4.14.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Scripting" Version="5.0.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="5.0.0" />
<PackageReference Include="Microsoft.CodeAnalysis.Workspaces.MSBuild" Version="5.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="10.0.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
@@ -265,7 +274,7 @@
<PackageReference Include="Microsoft.VisualStudio.Validation" Version="17.13.22" />
<PackageReference Include="Microsoft.Web.WebView2" Version="1.0.3595.46" />
<PackageReference Include="Microsoft.Windows.CsWinRT" Version="2.3.0-prerelease.251015.2" />
<PackageReference Include="Microsoft.Windows.SDK.BuildTools" Version="10.0.26100.6901" />
<PackageReference Include="Microsoft.Windows.SDK.BuildTools" Version="10.0.26100.7175" />
<PackageReference Include="Microsoft.WindowsAppSDK" Version="1.8.251106002" />
<PackageReference Include="QRCoder" Version="1.7.0" />
<PackageReference Include="Quartz.Extensions.DependencyInjection" Version="3.15.1" />
@@ -274,10 +283,6 @@
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Snap.Hutao.Deployment.Runtime" Version="2.5.12">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Snap.Hutao.Elevated.Launcher.Runtime" Version="1.3.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets>
@@ -286,7 +291,7 @@
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Snap.Hutao.SourceGeneration" Version="1.3.14">
<PackageReference Include="Snap.Hutao.SourceGeneration" Version="1.3.15">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>

View File

@@ -13,33 +13,32 @@
<x:String x:Key="IconGame">https://launcher-webstatic.mihoyo.com/launcher-public/2024/04/15/9ebf1bc5af2d83ca5fca21adb49cf341_2571779162329842818.png</x:String>
<!-- AvatarCard -->
<x:String x:Key="UI_AvatarIcon_Costume_Card">https://api.snapgenshin.com/static/raw/AvatarCard/UI_AvatarIcon_Costume_Card.png</x:String>
<x:String x:Key="UI_AvatarIcon_Costume_Card">https://htserver.wdg.cloudns.ch/static/raw/AvatarCard/UI_AvatarIcon_Costume_Card.png</x:String>
<!-- Bg -->
<x:String x:Key="UI_ItemIcon_None">https://api.snapgenshin.com/static/raw/Bg/UI_ItemIcon_None.png</x:String>
<x:String x:Key="UI_ItemIcon_None">https://htserver.wdg.cloudns.ch/static/raw/Bg/UI_ItemIcon_None.png</x:String>
<!-- Mark -->
<x:String x:Key="UI_MarkQuest_Events_Proce">https://api.snapgenshin.com/static/raw/Mark/UI_MarkQuest_Events_Proce.png</x:String>
<x:String x:Key="UI_MarkQuest_Events_Start">https://api.snapgenshin.com/static/raw/Mark/UI_MarkQuest_Events_Start.png</x:String>
<x:String x:Key="UI_MarkQuest_Main_Proce">https://api.snapgenshin.com/static/raw/Mark/UI_MarkQuest_Main_Proce.png</x:String>
<x:String x:Key="UI_MarkQuest_Main_Start">https://api.snapgenshin.com/static/raw/Mark/UI_MarkQuest_Main_Start.png</x:String>
<x:String x:Key="UI_MarkTower">https://api.snapgenshin.com/static/raw/Mark/UI_MarkTower.png</x:String>
<x:String x:Key="UI_MarkQuest_Events_Proce">https://htserver.wdg.cloudns.ch/static/raw/Mark/UI_MarkQuest_Events_Proce.png</x:String>
<x:String x:Key="UI_MarkQuest_Events_Start">https://htserver.wdg.cloudns.ch/static/raw/Mark/UI_MarkQuest_Events_Start.png</x:String>
<x:String x:Key="UI_MarkQuest_Main_Proce">https://htserver.wdg.cloudns.ch/static/raw/Mark/UI_MarkQuest_Main_Proce.png</x:String>
<x:String x:Key="UI_MarkQuest_Main_Start">https://htserver.wdg.cloudns.ch/static/raw/Mark/UI_MarkQuest_Main_Start.png</x:String>
<x:String x:Key="UI_MarkTower">https://htserver.wdg.cloudns.ch/static/raw/Mark/UI_MarkTower.png</x:String>
<!-- ItemIcon -->
<x:String x:Key="UI_ItemIcon_106">https://api.snapgenshin.com/static/raw/ItemIcon/UI_ItemIcon_106.png</x:String>
<x:String x:Key="UI_ItemIcon_204">https://api.snapgenshin.com/static/raw/ItemIcon/UI_ItemIcon_204.png</x:String>
<x:String x:Key="UI_ItemIcon_210">https://api.snapgenshin.com/static/raw/ItemIcon/UI_ItemIcon_210.png</x:String>
<x:String x:Key="UI_ItemIcon_220021">https://api.snapgenshin.com/static/raw/ItemIcon/UI_ItemIcon_220021.png</x:String>
<x:String x:Key="UI_ItemIcon_106">https://htserver.wdg.cloudns.ch/static/raw/ItemIcon/UI_ItemIcon_106.png</x:String>
<x:String x:Key="UI_ItemIcon_204">https://htserver.wdg.cloudns.ch/static/raw/ItemIcon/UI_ItemIcon_204.png</x:String>
<x:String x:Key="UI_ItemIcon_210">https://htserver.wdg.cloudns.ch/static/raw/ItemIcon/UI_ItemIcon_210.png</x:String>
<x:String x:Key="UI_ItemIcon_220021">https://htserver.wdg.cloudns.ch/static/raw/ItemIcon/UI_ItemIcon_220021.png</x:String>
<!-- EmotionIcon -->
<x:String x:Key="UI_EmotionIcon52">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon52.png</x:String>
<x:String x:Key="UI_EmotionIcon71">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon71.png</x:String>
<x:String x:Key="UI_EmotionIcon89">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon89.png</x:String>
<x:String x:Key="UI_EmotionIcon250">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon250.png</x:String>
<x:String x:Key="UI_EmotionIcon271">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon271.png</x:String>
<x:String x:Key="UI_EmotionIcon272">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon272.png</x:String>
<x:String x:Key="UI_EmotionIcon293">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon293.png</x:String>
<x:String x:Key="UI_EmotionIcon433">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon433.png</x:String>
<x:String x:Key="UI_EmotionIcon445">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon445.png</x:String>
<x:String x:Key="UI_EmotionIcon585">https://api.snapgenshin.com/static/raw/EmotionIcon/UI_EmotionIcon585.png</x:String>
<x:String x:Key="UI_EmotionIcon52">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon52.png</x:String>
<x:String x:Key="UI_EmotionIcon71">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon71.png</x:String>
<x:String x:Key="UI_EmotionIcon89">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon89.png</x:String>
<x:String x:Key="UI_EmotionIcon250">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon250.png</x:String>
<x:String x:Key="UI_EmotionIcon271">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon271.png</x:String>
<x:String x:Key="UI_EmotionIcon272">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon272.png</x:String>
<x:String x:Key="UI_EmotionIcon293">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon293.png</x:String>
<x:String x:Key="UI_EmotionIcon433">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon433.png</x:String>
<x:String x:Key="UI_EmotionIcon445">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon445.png</x:String>
<x:String x:Key="UI_EmotionIcon585">https://htserver.wdg.cloudns.ch/static/raw/EmotionIcon/UI_EmotionIcon585.png</x:String>
</ResourceDictionary>

View File

@@ -79,9 +79,8 @@
<Button
Padding="6"
Command="{Binding InvertAppThemeCommand}"
Content="[Dev] Invert Theme"
Style="{ThemeResource SettingButtonStyle}"
Visibility="{Binding IsDebug}"/>
Content="{shuxm:ResourceString Name=UIViewMainTitleBarInvertTheme}"
Style="{ThemeResource SettingButtonStyle}"/>
<Button
Padding="0"

View File

@@ -553,7 +553,7 @@
</ScrollViewer>
</PivotItem>
<PivotItem Header="{shuxm:ResourceString Name=ViewPageLaunchGameIslandOptionsHeader}" IsEnabled="False">
<PivotItem Header="{shuxm:ResourceString Name=ViewPageLaunchGameIslandOptionsHeader}">
<Border
Margin="16"
Padding="0"

View File

@@ -105,7 +105,7 @@ internal static class StaticResource
foreach ((string key, object value) in map)
{
if ((int)value < (int)LatestResourceVersionMap[key])
if (Convert.ToInt32(value) < Convert.ToInt32(LatestResourceVersionMap[key]))
{
return true;
}
@@ -120,7 +120,7 @@ internal static class StaticResource
ApplicationDataCompositeValue map = LocalSetting.Get(ContractMap, DefaultResourceVersionMap);
foreach ((string key, object value) in LatestResourceVersionMap)
{
if (!map.TryGetValue(key, out object current) || (int)value > (int)current)
if (!map.TryGetValue(key, out object current) || Convert.ToInt32(value) > Convert.ToInt32(current))
{
result.Add(key);
}

View File

@@ -3,6 +3,7 @@
using Microsoft.UI.Xaml.Controls;
using Snap.Hutao.Core;
using Snap.Hutao.Core.ApplicationModel;
using Snap.Hutao.Core.Setting;
using Snap.Hutao.Factory.ContentDialog;
using Snap.Hutao.Factory.Picker;
@@ -71,9 +72,19 @@ internal sealed class SettingStorageSetDataFolderOperation
try
{
Directory.SetReadOnly(oldFolderPath, false);
if (PackageIdentityAdapter.HasPackageIdentity)
{
// Packaged: use StorageFolder API
StorageFolder oldFolder = await StorageFolder.GetFolderFromPathAsync(oldFolderPath);
await oldFolder.CopyAsync(newFolderPath).ConfigureAwait(false);
}
else
{
// Unpackaged: use standard file I/O
await CopyDirectoryAsync(oldFolderPath, newFolderPath).ConfigureAwait(false);
}
}
catch (Exception ex)
{
Messenger.Send(InfoBarMessage.Error(ex));
@@ -84,4 +95,22 @@ internal sealed class SettingStorageSetDataFolderOperation
LocalSetting.Set(SettingKeys.DataDirectory, newFolderPath);
return true;
}
private static async ValueTask CopyDirectoryAsync(string sourceDir, string destDir)
{
await Task.Run(() =>
{
// Create all directories
foreach (string dirPath in Directory.GetDirectories(sourceDir, "*", SearchOption.AllDirectories))
{
Directory.CreateDirectory(dirPath.Replace(sourceDir, destDir));
}
// Copy all files
foreach (string filePath in Directory.GetFiles(sourceDir, "*", SearchOption.AllDirectories))
{
File.Copy(filePath, filePath.Replace(sourceDir, destDir), true);
}
}).ConfigureAwait(false);
}
}

View File

@@ -5,6 +5,7 @@ using CommunityToolkit.Mvvm.ComponentModel;
using Microsoft.UI.Xaml.Controls;
using Microsoft.Windows.AppLifecycle;
using Snap.Hutao.Core;
using Snap.Hutao.Core.ApplicationModel;
using Snap.Hutao.Core.Caching;
using Snap.Hutao.Core.Logging;
using Snap.Hutao.Core.Setting;
@@ -134,9 +135,20 @@ internal sealed partial class SettingStorageViewModel : Abstraction.ViewModel
// TODO: prompt user that restart will be non-elevated
try
{
if (PackageIdentityAdapter.HasPackageIdentity)
{
AppInstance.Restart(string.Empty);
}
else
{
// Unpackaged: manually restart the process
string exePath = System.Diagnostics.Process.GetCurrentProcess().MainModule?.FileName
?? throw new InvalidOperationException("Cannot get process path");
System.Diagnostics.Process.Start(exePath);
Snap.Hutao.Factory.Process.ProcessFactory.KillCurrent();
}
}
catch (COMException ex)
{
if (HutaoNative.IsWin32(ex.HResult, WIN32_ERROR.ERROR_PACKAGE_UPDATING))

View File

@@ -6,9 +6,9 @@ namespace Snap.Hutao.Web.Endpoint.Hutao;
[Service(ServiceLifetime.Singleton, typeof(IHutaoEndpoints), Key = HutaoEndpointsKind.Release)]
internal sealed class HutaoEndpointsForRelease : IHutaoEndpoints
{
string IHomaRootAccess.Root { get => "http://server.wdg.cloudns.ch:5222"; }
string IHomaRootAccess.Root { get => "https://htserver.wdg.cloudns.ch/api"; }
string IInfrastructureRootAccess.Root { get => "http://server.wdg.cloudns.ch:5222"; }
string IInfrastructureRootAccess.Root { get => "https://htserver.wdg.cloudns.ch/api"; }
string IInfrastructureRawRootAccess.RawRoot { get => "http://server.wdg.cloudns.ch:5222"; }
string IInfrastructureRawRootAccess.RawRoot { get => "https://htserver.wdg.cloudns.ch/api"; }
}

View File

@@ -5,7 +5,7 @@ namespace Snap.Hutao.Web.Endpoint.Hutao;
internal static class StaticResourcesEndpoints
{
public static string Root { get => "http://server.wdg.cloudns.ch:8007"; }
public static string Root { get => "https://htserver.wdg.cloudns.ch"; }
public static Uri UIIconNone { get; } = StaticRaw("Bg", "UI_Icon_None.png").ToUri();

View File

@@ -1,11 +0,0 @@
namespace Snap.Hutao.Web.Hoyolab.DataSigning;
internal static class SaltConstants
{
public const string CNVersion = "2.95.1";
public const string OSVersion = "2.54.0";
public const string CNK2 = "sfYPEgpxkOe1I3XVMLdwp1Lyt9ORgZsq";
public const string CNLK2 = "sidQFEglajEz7FA0Aj7HQPV88zpf17SO";
public const string OSK2 = "599uqkwc0dlqu3h6epzjzfhgyyrd44ae";
public const string OSLK2 = "rk4xg2hakoi26nljpr099fv9fck1ah10";
}

View File

@@ -17,13 +17,13 @@ internal sealed partial class HutaoPassportClient
{
private const string PublicKey = """
-----BEGIN PUBLIC KEY-----
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA5W2SEyZSlP2zBI1Sn8Gd
TwbZoXlUGNKyoVrY8SVYu9GMefdGZCrUQNkCG/Np8pWPmSSEFGd5oeug/oIMtCZQ
NOn0drlR+pul/XZ1KQhKmj/arWjN1XNok2qXF7uxhqD0JyNT/Fxy6QvzqIpBsM9S
7ajm8/BOGlPG1SInDPaqTdTRTT30AuN+IhWEEFwT3Ctv1SmDupHs2Oan5qM7Y3uw
b6K1rbnk5YokiV2FzHajGUymmSKXqtG1USZzwPqImpYb4Z0M/StPFWdsKqexBqMM
mkXckI5O98GdlszEmQ0Ejv5Fx9fR2rXRwM76S4iZTfabYpiMbb4bM42mHMauupj6
9QIDAQAB
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAy4i0acb1/rTjwNt4Wsi/
AgRLztRRGhludiYOWskqS6o0wTy6+DkGNZew/8qy93Tmv/mhYMoBhDJACD7Dpzu6
2cdiPl6MW8wuAE+H86Mh7ghWxnUAvdK6Cp3qbk7MaJF2zI/2yYesGYhcf7HZmZmC
RldyQnH9SP30FRhmbSAqGAjVSObwfa3W9islkbYB2SkcXguK+hONZmtqISoiUK1/
d+ZEpL01MNWI06iUxin+iT3yk68o6reLOk/Yoqjj12pONIwbu7Up4noLhhhmBdR3
7Xy9csCounngKoBw+7tEmmzJzeqYm/zeHp/Jpy/996dIxAiq6jVvNpWaT9Es6s08
cwIDAQAB
-----END PUBLIC KEY-----
""";

68
tools/GetImages.py Normal file
View File

@@ -0,0 +1,68 @@
import hashlib
import os
import shutil
import re
def sha1_name(filename):
"""对输入文件名进行 SHA1 哈希,返回大写 40 位字符串"""
sha = hashlib.sha1(filename.encode('utf-8')).hexdigest().upper()
return sha
def search_and_copy(original_name, search_dir, output_dir):
sha_name = sha1_name(original_name)
print(f"SHA1 计算结果: {sha_name}")
# 查找匹配文件
matched_path = None
for root, dirs, files in os.walk(search_dir):
for file in files:
if file.upper() == sha_name:
matched_path = os.path.join(root, file)
break
if not matched_path:
print("❌ 未找到匹配文件!")
return
# 创建输出目录
os.makedirs(output_dir, exist_ok=True)
# 复制并重命名
# 文件名是url取最后一部分作为原文件名
original_filename = os.path.basename(original_name)
# 创建url中最后一个文件夹作为输出目录
last_folder = original_name.split('/')[-2]
output_subdir = os.path.join(output_dir, last_folder)
os.makedirs(output_subdir, exist_ok=True)
new_path = os.path.join(output_subdir, original_filename)
shutil.copy(matched_path, new_path)
print(f"✔ 已复制并重命名文件: {new_path}")
def extract_urls_from_log(log_file):
"""
从日志中提取原始文件 URL并打印到终端
"""
base_url = "https://api.snapgenshin.com/static/raw"
url_pattern = re.compile(r'GET\s+/static/raw/([^/]+)/([^ ]+)\s+HTTP')
urls = []
with open(log_file, 'r', encoding='utf-8') as f:
for line in f:
match = url_pattern.search(line)
if match:
category, filename = match.groups()
full_url = f"{base_url}/{category}/{filename}"
urls.append(full_url)
return urls
if __name__ == "__main__":
logfile="1.txt" # 日志文件路径
original_file = extract_urls_from_log(logfile)
search_directory = "C:\\Users\\username\\AppData\\Local\\Packages\\60568DGPStudio.SnapHutao_wbnnev551gwxy1\\LocalCache\\ImageCache" # 搜索目录
output_directory = "C:\\Users\\username\\AppData\\Local\\Packages\\60568DGPStudio.SnapHutao_wbnnev551gwxy1\\LocalCache\\ImageCache\\output" # 输出目录
for url in original_file:
search_and_copy(url, search_directory, output_directory)

19
tools/README.md Normal file
View File

@@ -0,0 +1,19 @@
# 对开发有用的工具
对开发有用的工具,不参与编译
## GetImage.py
由于没有备份的图片资源并且缓存目录中的图片名全部经过SHA1处理过该工具用于通过请求日志在旧的资源文件夹中还原图片资源结构
### 用法
**先获取资源服务器上的404日志如下所示**
<img width="1161" height="57" alt="image" src="https://github.com/user-attachments/assets/54270165-f077-4c79-ae87-da16592e77fe" />
复制404日志粘贴到文本中
**然后修改脚本,确保路径正确**
`GetImage.py:62`开始修改,确保路径正确,搜索路径一定要为旧资源文件夹,`username`是你的用户名,`60568DGPStudio.SnapHutao_wbnnev551gwxy1`改为你实际的旧数据文件夹
运行后将自动搜索缺失的文件,然后会在输出路径中生成原始文件,可以直接上传至服务器来补全图片资源