Compare commits

...

334 commits

Author SHA1 Message Date
Nikolay Kuznetsov f905428238
Merge pull request #23 from DubyaDude/master
Move to api/v1 + additional QOL changes
2022-06-16 20:37:22 +02:00
Usman Shafiq b74c421bbb Added IsRetired (ApprovalStatus 3) 2022-06-15 18:15:32 -04:00
Usman Shafiq 71efc3990f Fixed download file name 2022-06-12 02:35:56 -04:00
Usman Shafiq d57e730baa Version bump 2022-06-12 00:57:50 -04:00
Usman Shafiq 2adb976b21 On Agree Removed Windows Notif + Move To Mods. Search bar enabled by default + made bigger 2022-06-12 00:22:23 -04:00
Usman Shafiq 59cd085130 Moved to api/v1 + Moved Broken to bottom 2022-06-12 00:04:03 -04:00
Nikolay Kuznetsov 992480facc Handle missing directories for deleting all mods 2021-07-16 01:59:08 +02:00
Nikolay Kuznetsov 8160521d10 Add a bit of exception handling to headpats/hugs buttons 2021-07-16 01:59:08 +02:00
Nikolay Kuznetsov a6321a9ecc Let SemVer handle four-digit version numbers 2021-07-16 01:59:08 +02:00
Nikolay Kuznetsov 88c987d714 Bump version again 2021-07-16 01:59:08 +02:00
Nikolay Kuznetsov 610098dcbf
Fix formatting 2021-07-15 22:45:05 +02:00
Nikolay Kuznetsov ec8cd210f7 Bump version 2021-07-15 19:43:24 +02:00
Nikolay Kuznetsov 66d2b41de8 Update default mods 2021-07-15 19:42:36 +02:00
Nikolay Kuznetsov ed27510ccc Update hardcoded categories for new mods 2021-07-15 19:42:15 +02:00
Nikolay Kuznetsov 0f60e68c02 Use latest ML installer instead of 0.3.0 2021-07-15 19:41:57 +02:00
Nikolay Kuznetsov f250bb893c Fix hyperlinks in mod info page 2021-05-29 22:02:39 +02:00
Nikolay Kuznetsov b1e2b3510b Don't ILRepack the debug build 2021-05-26 22:46:17 +02:00
Nikolay Kuznetsov 29ebdd82f9 Add mod author column to the mods list, add a few hardcoded author names 2021-05-26 22:46:04 +02:00
Nikolay Kuznetsov 6cc5510c5d Fix update check 2021-05-26 00:59:35 +02:00
Nikolay Kuznetsov e1b7320449 Use correct nuget restore 2021-05-26 00:45:04 +02:00
Nikolay Kuznetsov 8f93c766ac Bump version 2021-05-26 00:40:15 +02:00
Nikolay Kuznetsov 75a3cc8392 Make mods list a bit faster 2021-05-26 00:33:55 +02:00
Nikolay Kuznetsov ea59c178e6 Add more hardcoded categories for new mods 2021-05-26 00:33:28 +02:00
Nikolay Kuznetsov 9d453e7419 Fix mod list failing to load if mods directory has non-mod dlls 2021-05-17 23:16:09 +02:00
Nikolay Kuznetsov 3a9b80afdb Properly download stuff to memory 2021-05-17 23:15:55 +02:00
Nikolay Kuznetsov 4b69295fda Handle failing tasks that are not awaited 2021-05-17 23:15:38 +02:00
Nikolay Kuznetsov f755fc8b4a Add hardcoded categories for a few pending mods 2021-05-17 22:52:07 +02:00
Nikolay Kuznetsov c9b6c594cc Fix NRE on startup 2021-05-17 22:51:25 +02:00
Nikolay Kuznetsov 3cebca05b6 Add issue reporting warning to intro page 2021-05-17 22:51:16 +02:00
Nikolay Kuznetsov 5b4a6a9c1b Fork initial: VRCMelonManager based on ModAssistant 1.1.18 2021-05-17 22:27:27 +02:00
Vyolex 7788c2573c
Fix rate limit issues for playlist downloading (#326)
* Fix inconsistent use of UTC and local time, changed display to local time.
2021-04-14 23:09:55 +01:00
Jack Baron ab775ea496
update readme logo 2021-03-27 14:40:54 +00:00
Jack Baron 2d2477d19d
Updated version to 1.1.18 2021-03-27 11:39:01 +00:00
Jack Baron 37375b1026
Merge pull request #293 from Stevoisiak/oneclick-registry-description
Add description for OneClick registry keys
2021-03-27 11:33:11 +00:00
wgzeyu ee06f36ab9
Update zh.xaml (#311)
* Update zh.xaml

* Update zh.xaml
2021-03-27 11:26:10 +00:00
Robert Reichel 84c879c16f
add missing German translations (#312) 2021-03-27 11:26:01 +00:00
Jack Baron 2c32da0ed5
run CI in PRs 2021-03-27 11:24:05 +00:00
Jack Baron b9e58c0b8c
decode uri components in playlist filenames
closes #317
2021-03-22 06:50:15 +00:00
Assistant 6b4450156e
Merge pull request #314 from ErisApps/feature/name_consistency
Fixed inconsistent naming ModAssistant vs Mod Assistant
2021-03-16 07:37:36 -06:00
Eris 6d7e02aaa4 Fixed inconsistent naming ModAssistant vs Mod Assistant
Melopod ƸӜƷ — Today at 22:42
@Assistant hey, would you prefer it being ModAssistant or Mod Assistant
Space or no space

Assistant — Today at 22:43
Mod Assistant
2021-03-15 23:05:39 +01:00
Jack Baron 7a824f8230
Updated version to 1.1.17 2021-03-13 22:39:12 +00:00
Jack Baron 260441282a
cleanup redundant comments 2021-03-13 22:36:00 +00:00
Uncouth 5198383b2f
Update sv.xaml (#308) 2021-03-13 22:34:55 +00:00
larynai bc001f5852
Update ru.xaml (#303)
Minor updates of localization
2021-03-13 22:34:29 +00:00
Migush 3bf63e3122
Updated nl.xaml (#304)
* Update nl.xaml
2021-03-13 22:29:18 +00:00
Assistant 8fe1e5494e
Merge pull request #294 from Stevoisiak/read-only-language-var
Make availableLanguageCodes readonly
2021-01-24 14:17:09 -07:00
Stevoisiak 177820b954 Shift arguments by 3
Addresses https://github.com/Assistant/ModAssistant/pull/293#discussion_r560598751
2021-01-19 19:55:59 -05:00
Stevoisiak 02c30d5158 Make availableLanguageCodes readonly
Resolves IDE0044 style rule
2021-01-19 14:08:06 -05:00
Stevoisiak 140dde19fd Add description for OneClick registry keys
Adds a description to protocol handler registry keys when setting up OneClick install.
2021-01-19 12:34:54 -05:00
Jack Baron 3b9dc222d0
Merge pull request #289 from SieR-VR/master
Update ko.xaml
2021-01-07 07:04:16 +00:00
SieR c33baacc9e
Update ko.xaml
I'm stupid
2021-01-06 16:46:47 +09:00
Jack Baron 2c7018a9d1
Merge pull request #272 from simeng/feature/nb-translation
Norwegian (bokmål) translation
2020-12-23 04:17:59 +00:00
Kim Jaejun a298d226ec
Added missing translations for Korean (#281) 2020-12-23 04:17:23 +00:00
Jack Baron df1cd91358
Merge pull request #278 from ErisApps/feature/Added_missing_translations_for_Dutch_and_French
Added missing translations for Dutch and French
2020-11-26 09:24:08 +00:00
Eris d5a2884390 Added missing translations for Dutch and French 2020-11-26 00:02:10 +01:00
Simen Graaten 118352b5f3
Fixed some translation strings 2020-11-16 14:31:46 +00:00
Simen Graaten ed8170c5e0
Update ModAssistant.csproj 2020-11-16 14:31:46 +00:00
Simen Graaten c592e0ccbc
Add a norwegian (bokmål) translation. 2020-11-16 14:31:46 +00:00
Simen Graaten 71819281a5
Update Languages.cs 2020-11-16 14:31:25 +00:00
Jack Baron 89bdd100bb
format all files 2020-11-16 14:19:59 +00:00
Jack Baron 688857aea9
ignore unreachable code warnings on autoupdater 2020-11-16 14:11:01 +00:00
Jack Baron 5901ffb5de
use pattern matching and interpolation 2020-11-16 14:10:11 +00:00
Jack Baron d3ad20d082
remove seemingly unused param 2020-11-16 14:09:03 +00:00
Jack Baron 1884b7f9ec
disable naming warnings for JSON types 2020-11-16 14:08:18 +00:00
Jack Baron c10f9b88f6
use simple types 2020-11-16 14:06:24 +00:00
Jack Baron cbdc019b1c
simplify ratelimit init 2020-11-16 14:04:32 +00:00
Jack Baron 1c0f6dcaba
simplify object initialisers 2020-11-16 14:03:46 +00:00
Jack Baron d4753910f4
add readonly modifiers where appropriate 2020-11-16 14:03:01 +00:00
Jack Baron cfe66c84e1
simplify all 2020-11-16 14:02:28 +00:00
Jack Baron 2f0375bb86
fix unlocalised dialog title 2020-11-16 13:57:05 +00:00
Jack Baron 57822f3721
fixed incorrect language code for dutch 2020-11-16 13:37:59 +00:00
Jack Baron ac027753ad
add missing swedish localisations
thanks RangeValley#0237
2020-11-16 13:37:16 +00:00
Jack Baron 47602291a0
remove and sort usings 2020-11-16 13:22:30 +00:00
Jack Baron b5223f36a9
Merge pull request #266 from DomDom3333/master
Added check and Warning for Uninstalling BSIPA
2020-11-05 16:35:40 +00:00
Jack Baron 0ff1c5164c
add missing translations 2020-11-05 16:31:35 +00:00
Jack Baron be06ee7a99
localise bsipa uninstall fail box 2020-11-05 16:31:24 +00:00
Jack Baron 8659cbb7f7
improve readability 2020-11-05 16:25:56 +00:00
DomDom3333 7ba79ecd41 Did a stupid and forgot some brackets.
Just forgot some brackets and blindly commited before checking. Fixed now.
2020-10-27 21:58:13 +01:00
DomDom3333 5f814ab58e Added check and Warning for BSIPA
Simple check if IPA.exe and IPA Directory exists before trying to delete it.
2020-10-27 16:34:14 +01:00
Jack Baron 1c1332f1b5
Merge pull request #264 from ErisApps/feature/Renamed_#support_to_#pc-help
Renamed the #support channelname to #pc-help
2020-10-26 21:01:19 +00:00
Eris 0b702e8840 Renamed the #support channelname to #pc-help 2020-10-26 20:05:19 +01:00
Jack Baron 503aad4af2
Updated version to 1.1.16 2020-10-25 01:13:42 +00:00
Jack Baron 1c14aba0d6
Merge pull request #262 from wgzeyu/patch-23
Update Chinese localization
2020-10-25 01:12:52 +00:00
wgzeyu e4a6fe0335
Update Chinese localization 2020-10-25 08:55:14 +08:00
Jack Baron f41b7a7b23
Merge pull request #235 from wgzeyu/patch-22
Add chinese guide
2020-10-25 01:29:28 +01:00
Jack Baron b268d472bf
don't check for updates when running debug 2020-10-25 01:15:21 +01:00
Jack Baron ad62d5ecbd
Merge pull request #240 from TheJmJ/master
Issue #239: Quick hotfix for error that comes if not up-to-date
2020-10-25 01:13:01 +01:00
Jack Baron 223da16d7f
minor cleanup to alias resolution 2020-10-25 01:11:52 +01:00
Jack Baron e3726cbc63
Merge pull request #257 from Assistant/fix/spam-pats-hugs
[Fix] Spamming pat/hug buttons causes crashes
2020-10-22 19:14:53 +01:00
Jack Baron d9d9c9e5d0
only reenable pat/hug buttons when popups are open
closes #253
2020-10-22 19:12:35 +01:00
Jack Baron 69f3f0556a
Merge pull request #256 from ErisApps/feature/Added_revised_Swedish_translations
Added/revised Swedish translations
2020-10-22 16:53:30 +01:00
Eris bb5729ef9b Added/revised Swedish translations
All credits for those translations go to Davy#0013 ( as I don't speak nor understand a single word of what's written here 😅 )
2020-10-22 17:45:23 +02:00
Jack Baron 24918e6079
Merge pull request #209 from Parapass/patch-5
Check if Playlists folder exists before downloading
2020-10-22 15:36:58 +01:00
Jack Baron 8a05d0e4a0
only create playlists dir when needed 2020-10-22 15:34:34 +01:00
Jack Baron 3039f744a5
rewrite create function
CreateDirectory does nothing if the dir exists
2020-10-22 15:32:09 +01:00
Jack Baron 60893fd40c
Merge pull request #255 from Awagi/patch-3
🇫🇷 Update translation
2020-10-22 15:23:05 +01:00
Awagi ea5b271954
🇫🇷 Update translation 2020-10-22 12:18:50 +02:00
Jack Baron 271f649cb5
Merge pull request #251 from paulvandenburg/feature/nl-translation-fixes
Add missing NL translations, fix typos and suboptimal NL translations
2020-10-21 11:24:47 +01:00
Jack Baron 9eda65c5bc
Merge pull request #250 from mhombach/feature/issue-249-missing-german-translation
feat(249): added missing german translation
2020-10-21 11:24:39 +01:00
Paul van den Burg 9e92464ceb Add missing NL translations, fix typos and suboptimal NL translations 2020-10-21 00:41:23 +02:00
Jack Baron 4832649002
one day I'll get CI right 2020-10-20 21:29:07 +01:00
mhombach 3f9cf7c2cd feat(249): german translation
- added all missing translation
- changed "About:List:Header" because it sounded awkward in german (it's still not 100% but I did't want to change the whole structure of the sentence)
2020-10-20 22:28:00 +02:00
Jack Baron f105e5336f
Updated version to 1.1.15 2020-10-20 21:25:46 +01:00
Jack Baron 84382c48b2
properly resolve version for release 2020-10-20 21:24:53 +01:00
Jack Baron 7f94b2bf33
i left debug code in oops 2020-10-20 21:22:29 +01:00
Jack Baron 1b1c209df4
Updated version to 1.1.14 2020-10-20 20:52:53 +01:00
Jack Baron 5294455fdd
unfuck actions yml 2020-10-20 20:50:00 +01:00
Jack Baron 18c35bb70f
Merge pull request #245 from Assistant/feat/translation-stubs
Translation Stubs Tool
2020-10-20 20:46:15 +01:00
Jack Baron b0f9e8b6db
Merge pull request #237 from RedBrumbler/patch-6
Improved some phrases
2020-10-19 14:54:17 +01:00
Jack Baron db9c241d26
"Tik op hoofd" sounds like :owoHit:
Co-authored-by: Eris <erisapps@outlook.be>
2020-10-19 14:53:04 +01:00
Jack Baron 98181a50df
Merge pull request #246 from ErisApps/feature/Added_Swedish_translation_for_NoMods_key
Added Swedish translation for NoMods loc key
2020-10-19 14:41:46 +01:00
Eris f905e0d172 Added Swedish translation for NoMods loc key 2020-10-19 15:40:28 +02:00
Jack Baron a3ded0c9bf
Merge pull request #244 from ErisApps/feature/Added_translations_for_NoMods_key
Feature/added translations for no mods key
2020-10-19 14:29:38 +01:00
Jack Baron 4878bac01e
generate translation stubs 2020-10-19 14:28:20 +01:00
Eris def19229f5 Added Dutch translation for NoMods loc key 2020-10-19 15:27:17 +02:00
Eris e852ad3516 Added French translation for NoMods loc key 2020-10-19 15:26:57 +02:00
Jack Baron a0314af766
add translation stubs tool 2020-10-19 14:23:49 +01:00
Jack Baron 6651df67ee
oops forgot this 2020-10-19 14:19:02 +01:00
Jack Baron addf3e858f
optimise CI 2020-10-19 14:12:43 +01:00
Jack Baron ca87635473
Merge pull request #241 from PulseLane/master
Add informative text for versions with no updated mods
2020-10-19 14:05:47 +01:00
Jack Baron 272251f79b
move warning to center of mods panel 2020-10-19 14:01:57 +01:00
Jack Baron 5a36a2f4db
tweak wording and add stub translations 2020-10-19 13:35:41 +01:00
PulseLane b850b928db Add informative text for versions with no updated mods 2020-10-15 23:43:31 -07:00
Jere Tuohino 1c8ae4e604 Issue #239: Quick hotfix for error that comes if older version of BS is installed 2020-10-14 02:14:33 +03:00
RedBrumbler 5245c7838c
Improved some phrases 2020-09-19 00:02:38 +02:00
wgzeyu 10c90ef9ec
Add chinese guide 2020-09-13 18:00:54 +08:00
Assistant 215bc0f081
Merge pull request #220 from wgzeyu/patch-20
Update Chinese localization
2020-08-19 20:42:06 -06:00
Assistant 068647f8f5
Merge pull request #226 from DeadlyKitten/bugfix/desktop.ini
Add exclusion for desktop.ini to anti-piracy
2020-08-19 20:41:19 -06:00
Steven ff357cbc03 Add exclusion for desktop.ini to anti-piracy 2020-08-18 22:12:50 -07:00
wgzeyu 9f013d3af6
Update Chinese localization 2020-07-30 23:08:35 +08:00
Megalon 58de249f1d Create Playlists folder if it doesn't exist 2020-06-29 10:56:01 -07:00
Parapass c170d91d26
Check if Playlists folder exists before downloading
strings 3head
2020-06-29 08:29:59 -04:00
Assistant 2d53be9009 Updated version to 1.1.13 2020-06-24 00:48:22 -06:00
Assistant e10cd23a91
Merge pull request #205 from wgzeyu/patch-17
Update Chinese localization
2020-06-24 00:46:57 -06:00
Assistant 29322a4807 Removed delay from OCI close 2020-06-24 00:32:14 -06:00
wgzeyu d9594ff6d8
Update Chinese localization 2020-06-22 14:41:11 +08:00
Assistant c92da4c012 Improved OCI UX 2020-06-22 00:17:30 -06:00
Assistant 7ff93ec823 Added options for OCI window 2020-06-21 22:05:53 -06:00
Assistant eead2780e9
Merge pull request #194 from Rychard/try-catch-clipboard
Suppress exceptions when setting clipboard data
2020-06-21 18:58:08 -06:00
Assistant 57496ff873
Merge pull request #195 from RedBrumbler/patch-5
Update nl.xaml
2020-06-21 18:57:35 -06:00
Assistant e01cb6b968
Merge pull request #198 from larynai/patch-1
Update ru.xaml
2020-06-21 18:57:21 -06:00
Assistant 8ecba1e076
Merge pull request #201 from DrTexx/master
README.md typo fix
2020-06-21 18:57:09 -06:00
Assistant bc3b5b6748
Merge pull request #203 from wgzeyu/patch-15
Update Chinese localization
2020-06-21 18:56:42 -06:00
Assistant 9fefe75735 Fix property instancing order issue 2020-06-21 18:52:55 -06:00
wgzeyu 3a459a262a
Update Chinese localization 2020-06-21 02:04:56 +08:00
Denver P 05890fec66
README.md typo fix 2020-06-18 18:55:18 +10:00
larynai 7e17946202
Update ru.xaml 2020-06-16 16:05:46 +03:00
RedBrumbler 44e87bf3ec
Update nl.xaml 2020-06-12 12:41:05 +02:00
megalon adaca10ddb
Merge pull request #145 from UncouthMedia/patch-1
Create sv.xaml
2020-06-09 16:32:27 -07:00
Megalon 86977c68eb Add "sv" to available language codes 2020-06-09 16:27:13 -07:00
megalon 4fc3417360
Merge branch 'master' into patch-1 2020-06-09 16:23:22 -07:00
Joshua Shearer 8d8bf90088 Suppress exceptions when setting clipboard data 2020-06-09 09:29:49 -04:00
Assistant 6c2894e5d4 Updated version to 1.1.12 2020-06-08 21:59:07 -06:00
Assistant 2301eb3303
Merge pull request #192 from wgzeyu/patch-13
Add description for 围城 event
2020-06-08 21:55:57 -06:00
wgzeyu 98d1549fcb
Add description for 围城 event
“我们发现 围城 的网站未经作者同意擅自打包贩卖歌曲、模型,并抹黑社区中的成员。
请各位玩家引以为鉴,不要攻击其他玩家或支持这种行为,谢谢配合。”
2020-06-09 11:49:17 +08:00
Assistant 4cbf2c244f Updated version to 1.1.11 2020-06-08 21:00:03 -06:00
Assistant de39c64ddd
Merge pull request #179 from wgzeyu/patch-11
Update chinese localisation
2020-06-08 20:46:23 -06:00
Assistant 436593a509
Merge pull request #187 from RedBrumbler/patch-4
Update nl.xaml
2020-06-08 20:46:06 -06:00
Assistant 474c51d165
Merge pull request #185 from Awagi/patch-2
FR localisation update
2020-06-08 20:43:56 -06:00
megalon 3072ded757
Merge pull request #190 from megalon/feature/oneclick-menu
Move OneClick options into a menu
2020-06-08 19:43:15 -07:00
Megalon c54165ec72 why this 2020-06-08 19:42:20 -07:00
Megalon f78e26ac2f Add sv.xaml to project 2020-06-08 19:37:27 -07:00
Assistant 1fd55d9df6
Merge pull request #189 from Rychard/verify-resource-before-access
Don't attempt to access non-existent manifest resources
2020-06-08 20:29:45 -06:00
megalon 869802f59d
Merge pull request #182 from megalon/fix/text
Fix text not fitting inside buttons
2020-06-08 19:25:05 -07:00
Megalon 359d77a96b Remove "Close window" option 2020-06-08 19:16:57 -07:00
Megalon eb246cf7d0 Update margins 2020-06-08 19:15:02 -07:00
Megalon 8aa828739c Better styling for OneClick menu 2020-06-08 19:04:45 -07:00
Megalon 4e70f55bf8 Move OneClick buttons to menu 2020-06-08 18:38:13 -07:00
Joshua Shearer affa4aa6ff Don't attempt to access non-existent manifest resources 2020-06-05 10:20:17 -04:00
RedBrumbler cbc84472a7
Update nl.xaml 2020-06-05 12:54:50 +02:00
Awagi e91fc54a90
Update fr.xaml 2020-06-04 21:26:35 +02:00
Megalon fa9dec967c Fix hugs / headpats buttons 2020-06-02 16:41:04 -07:00
Megalon 2c7d6fc349 Scale side button text 2020-06-02 16:41:04 -07:00
Megalon e785ccea3e Fix size of Mod Info and Install buttons
- Removed extra stackpanel
- Fixed columns not rezising
- Added padding 
- Added min width
2020-06-02 16:41:04 -07:00
Megalon 2c2a44b3c3 Fix Intro button padding 2020-06-02 16:41:04 -07:00
Assistant b9319f9990 -e 2020-06-02 16:48:30 -06:00
Assistant 6327c197fb Fix'd localizations not loading correctly 2020-06-02 16:47:59 -06:00
wgzeyu a120caf001
Update zh.xaml 2020-06-02 11:51:50 +08:00
Assistant fe2fcfb7f9 Updated version to 1.1.10 2020-06-01 21:15:14 -06:00
megalon d1203ac255
Merge pull request #175 from megalon/feature/language-select
Language select
2020-06-01 19:52:01 -07:00
megalon 53dd2115ef
Merge pull request #168 from wgzeyu/patch-10
Update chinese localisation
2020-06-01 19:37:22 -07:00
megalon d96cb8bb95
Merge pull request #170 from larynai/master
russian localization small update
2020-06-01 19:36:57 -07:00
Megalon 11664afdf4 Label for language combobox 2020-06-01 19:28:50 -07:00
Megalon 3dc29753f1 Save selected language and load on startup 2020-06-01 19:28:50 -07:00
Megalon 97ef2c1285 Add LanguageCode setting 2020-06-01 19:28:50 -07:00
Megalon e3f8ab9696 Move language combobox 2020-06-01 19:28:50 -07:00
Megalon 79936c79f3 Show NativeName instead of EnglishName 2020-06-01 19:28:50 -07:00
Megalon ff327ecffc Language select and refactor 2020-06-01 19:28:50 -07:00
Megalon 3fe99bb5e2 Move LoadLanguage to Utils 2020-06-01 19:28:50 -07:00
Megalon 83760c6a0a Temp combobox 2020-06-01 19:28:50 -07:00
megalon 6495cab513
Merge pull request #178 from megalon/fix/not-closing
Force shutdown app when main window closed
2020-06-01 19:28:29 -07:00
Megalon ef197d1611 Shutdown app when main window closed 2020-06-01 19:25:35 -07:00
larynai a6b6c79515
Update ru.xaml 2020-05-30 18:32:42 +03:00
larynai c16568dfd2
Update ru.xaml 2020-05-30 18:29:13 +03:00
Uncouth 9319d0dbc0
Update sv.xaml
Basically finished? Need someone to take a look and see if I've mistaken any translatable parts for code. Unsure if to translate lines 11-13, might need some context.
2020-05-28 22:20:08 +02:00
Uncouth 28e1624b64
Update sv.xaml
Translated some more. Forgot about this to be honest.  Finishing it tomorrow.
2020-05-28 00:49:13 +02:00
Assistant 425d96f83d Remove accidental debug message 2020-05-24 17:51:00 -06:00
larynai b81e264709
russian localization update 2020-05-23 16:44:40 +03:00
wgzeyu 626881b730
Update zh.xaml 2020-05-21 01:26:06 +08:00
wgzeyu cf4d0c4c51
Update chinese localisation 2020-05-20 21:15:45 +08:00
Assistant 56d21f1432 Placeholder localizations 2020-05-19 19:17:00 -06:00
Assistant 145c350c14 improved filetype detection 2020-05-19 19:09:03 -06:00
Assistant 9622cef810 Fixed log uploading 2020-05-19 18:38:08 -06:00
Assistant 0576d293b8 Remove debugging message 2020-05-19 17:19:32 -06:00
Assistant c9bceb8a21 Fixes #165 2020-05-19 17:13:55 -06:00
Assistant ec03c8127b Added option to have OCI window close after its finished 2020-05-19 16:48:35 -06:00
Assistant dba0a7d951 Merge branch 'master' of https://github.com/Assistant/ModAssistant 2020-05-19 15:55:45 -06:00
Assistant d880ae4024 Added missing placeholders 2020-05-19 15:54:18 -06:00
Assistant 3a146cb8f4
Merge pull request #164 from mischiminator/master
Fixed typos
2020-05-19 15:41:12 -06:00
Kariko 932dd2f0f3
Fixed typos 2020-05-19 23:37:44 +02:00
Assistant 23135b0725
Merge pull request #163 from mischiminator/master
Updated German Translation
2020-05-19 15:28:57 -06:00
Kariko 86818d44b4
Update de.xaml
Added new translations
2020-05-19 23:18:25 +02:00
Kariko cf8cec9a65
Merge pull request #1 from Assistant/master
fork update
2020-05-19 23:14:20 +02:00
Assistant c59105b91c Updated version to 1.1.9 2020-05-19 12:13:45 -06:00
Assistant 16dd4ba72f
Merge pull request #162 from Assistant/OCI-overhaul
OCI overhaul
2020-05-19 11:59:00 -06:00
Assistant 05dc6b502a
Merge pull request #161 from Parapass/patch-4
Add the OneClick UI for Playlists
2020-05-19 11:57:05 -06:00
Assistant a2dd0f71b1 why did this not commit what the fuck 2020-05-19 11:48:25 -06:00
Assistant d353ce7f4b Big commits will make me hate myself eventually
but I'll hate myself anyways so lets go
2020-05-19 11:47:45 -06:00
Parapass ff2a63429c
do localization thing - part 2 2020-05-19 07:49:37 -04:00
Parapass 62abdfa0b5
do localization thing - part 1 2020-05-19 07:49:06 -04:00
Parapass 0ed8aafd99
Copy+paste mo-- I mean add the OneClick to Options CS File 2020-05-19 07:46:55 -04:00
Parapass 1d65e3ad13
Add the OneClick UI 2020-05-19 07:44:09 -04:00
Assistant 7d60f3f4a4 bug fixes 2020-05-19 02:10:44 -06:00
Assistant 80a9339227 made rate limits nullable 2020-05-18 19:10:54 -06:00
Assistant 9009ea4a87 added Russian localization 2020-05-18 19:08:59 -06:00
Assistant 1a55d70928 Some error checking 2020-05-18 19:08:21 -06:00
Assistant 4a71c5cfe1
Merge pull request #157 from larynai/master
Russian Localization | Update
2020-05-18 18:38:08 -06:00
Assistant 3bb30e57f3
Merge pull request #158 from claudiodemarzo/master
Italian localisation | updated
2020-05-18 18:36:51 -06:00
Assistant a556b56b75
Merge branch 'master' into master 2020-05-18 18:36:40 -06:00
Assistant e40a791beb
Merge pull request #156 from RedBrumbler/patch-2
fixed some typos
2020-05-18 18:35:04 -06:00
Assistant 2a8f2c3a47
Merge pull request #155 from Awagi/patch-1
French localisation update
2020-05-18 18:34:45 -06:00
Assistant 57c358b0c8
Merge pull request #153 from wgzeyu/patch-9
Update Chinese translation
2020-05-18 18:34:23 -06:00
Claudio De Marzo 290a6072e3
Update it.xaml
updated with new strings
2020-05-18 22:59:43 +02:00
larynai 0eae0730a1
Russian Localization | Update 2020-05-18 22:48:10 +03:00
RedBrumbler ed2249cd7a
fixed some typos 2020-05-18 20:52:42 +02:00
Awagi 4084f7b803
French localisation update
Translated 6 new strings
2020-05-18 20:01:46 +02:00
wgzeyu 9a6e0aa08c
Update zh.xaml 2020-05-19 00:34:10 +08:00
Assistant ff3b45e031 Updated version to 1.1.8 2020-05-18 10:25:32 -06:00
Assistant 70e8116611 Fixed playlist not being copied 2020-05-18 10:24:58 -06:00
Assistant a9559d8cb2 Updated version to 1.1.7 2020-05-18 09:38:16 -06:00
Assistant 0b677b75fb Filled in localizations 2020-05-18 09:31:58 -06:00
Assistant eb1cd8c497
Merge pull request #151 from Tonitrua0512/kor
Korean Localization
2020-05-18 09:24:37 -06:00
Assistant 7a6770c8bb
Merge pull request #148 from claudiodemarzo/master
italian localisation
2020-05-18 09:22:34 -06:00
Assistant cca63ad28e
Merge pull request #147 from mischiminator/master
German localisation
2020-05-18 09:22:17 -06:00
Assistant 7f570b8aa0
Merge pull request #146 from Awagi/master
French localisation
2020-05-18 09:21:52 -06:00
Assistant ffba3ca42a
Merge pull request #150 from Assistant/search
Search bar
2020-05-18 09:12:38 -06:00
Assistant 4e185333ad
Merge pull request #152 from Assistant/playlists
Playlists
2020-05-18 09:08:00 -06:00
Assistant e7dbe58025 Playlist downloading 2020-05-18 09:05:42 -06:00
Assistant 07c2d90b09 Added Install Playlist button. 2020-05-18 04:56:16 -06:00
Tonitrua0512 7d5c42862f kor translate 2020-05-18 14:14:02 +09:00
Assistant b4583a3530 Merge branch 'master' into playlists 2020-05-17 19:14:34 -06:00
Assistant 82deee8bc6 Add ProgressBar support to playlist downloading 2020-05-17 19:14:20 -06:00
Assistant 1f2640bd78 Improved options layout 2020-05-17 19:07:32 -06:00
Megalon 72f73cac5b Fix hardcoded text resource 2020-05-16 14:56:41 -07:00
Assistant 39843e706c Search bar 2020-05-16 05:31:38 -06:00
claudiodemarzo 5b5a68bd6b
italian localisation 2020-05-15 12:37:32 +02:00
Kariko 2fb34aac5d Created de.xaml
German translation for Mod Assisstant
2020-05-15 08:55:23 +02:00
Awagi 6010314835
Fixed typos and reviewed 2020-05-14 11:55:36 +02:00
Awagi 11551bfcd5
French localisation
Translated terms in French
Also Wiki links redirect to French pages
2020-05-14 01:23:30 +02:00
UncouthMedia caeb2557ea
Create sv.xaml
Work in progress, about half-way done.
2020-05-13 16:07:33 +02:00
Assistant 51f99aebbf
Merge pull request #144 from wgzeyu/patch-8
Update zh.xaml
2020-05-12 20:39:10 -06:00
wgzeyu c345cea27c
Update zh.xaml 2020-05-13 10:16:51 +08:00
Assistant 3213942190 Improved arguments 2020-05-12 19:57:05 -06:00
Assistant a1b9176570 Fix'd localizations 2020-05-12 18:58:20 -06:00
Assistant 3a5c2061b1
Merge pull request #143 from RedBrumbler/patch-1
Created dutch localistation file
2020-05-12 18:39:37 -06:00
Assistant 35830234d8
Merge pull request #131 from Parapass/patch-3
Add 'if' and 'else' to Open AppData Button
2020-05-12 18:33:59 -06:00
Assistant a6c5db47b7
Merge pull request #139 from megalon/feature/reinstall-selected
Separate Install and Reinstall buttons
2020-05-12 18:32:21 -06:00
Assistant 4d8f67e293
Merge pull request #135 from wgzeyu/patch-7
Change the name of the Chinese tutorial website
2020-05-12 18:21:05 -06:00
RedBrumbler 94f09dd2e7
Created dutch localistation file
Did my best to translate everything as good as I could

Should probably have been sleeping tbh, it's 2 am and I am here making a localisation file....
2020-05-13 01:55:17 +02:00
Megalon 2dea0440e0 Add ReinstallInstalled option to UI 2020-05-09 14:40:45 -07:00
Megalon 2799bdd4aa ReinstallInstalled setting 2020-05-09 14:40:22 -07:00
Megalon 1d906810b9 Remove advanced install button 2020-05-09 14:39:13 -07:00
Assistant 3379379a45 Playlist logic 2020-05-09 10:28:39 -06:00
Megalon 27ff6576e9 Fix offset button text 2020-05-09 01:53:41 -07:00
Megalon 14ac339270 More styling 2020-05-09 01:41:00 -07:00
Megalon 139790ec06 Fix typo in BSMG theme 2020-05-09 01:10:20 -07:00
Megalon e43c8c8ef9 Styling for menu 2020-05-09 01:10:10 -07:00
Megalon 49e8c04441 Make reinstall functional 2020-05-08 23:23:04 -07:00
Megalon e69ab66062 Simple advanced install button 2020-05-08 22:59:32 -07:00
wgzeyu a51f8016d3
Change the name of the Chinese tutorial website 2020-05-06 07:26:16 +08:00
Assistant c9ac2b7b9c
Merge pull request #134 from Assistant/OneClickInstall-refactor
Refactored OneClick Install code
2020-05-05 13:44:54 -06:00
Assistant 00561c98b7 Refactored OneClick Install code 2020-05-05 13:43:50 -06:00
Para d29aa406a7
HahaFix 2020-05-04 12:11:54 -04:00
Assistant aeceb27226
Merge pull request #133 from megalon/fix/invalid-semver
Handle invalid semver from beatmods
2020-05-04 04:30:42 -06:00
Megalon 05715487da Handle invalid semver from beatmods 2020-05-04 01:40:01 -07:00
Assistant c8459e5292 Updated version to 1.1.6 2020-05-03 22:38:49 -06:00
Assistant 2367876df1
Merge pull request #130 from megalon/fix/SecurityProtocol
Enable TLS 1.2 security protocol
2020-05-03 09:05:56 -06:00
Para a39938ccea
assistant's request 2020-05-03 10:57:33 -04:00
Para 6eac4d7258
make it less passive-aggressive peepoBruh 2020-05-03 10:27:07 -04:00
Para 4a9fd20ff3
add Options:AppDataNotFound 2020-05-03 10:26:24 -04:00
Para 8c53243031
Add 'if' and 'else' to OpenAppDataButton
This would solve one of the issues I DM'd you about if the folder doesn't exist.
2020-05-03 10:24:30 -04:00
Megalon 05e62191ac Enable TLS 1.2 security protocol 2020-05-02 14:08:47 -07:00
Assistant 9b878a43c3
Merge pull request #128 from megalon/fix/version-check
Fix mod version checker
Closes #121
2020-05-02 13:38:15 -06:00
Megalon bd28a2e8b6 Clarify variable name 2020-05-02 11:42:11 -07:00
Megalon 5ca687fe81 Only skip installed mods that are newer
- Also removed ModsAlreadyUpToDate message
2020-05-02 11:38:55 -07:00
Megalon 715a867653 Fix message always showing mods up to date 2020-05-02 11:22:47 -07:00
Assistant 183f469d85
Update FUNDING.yml 2020-05-02 11:53:10 -06:00
Megalon a452ae3d79 Message if all mods already up to date 2020-05-01 19:57:25 -07:00
Megalon 42d3bef0dc Fix BSIPA ignoring version check 2020-05-01 19:45:59 -07:00
Megalon 6ad4d0f984 Organize libs and give credit 2020-05-01 19:38:48 -07:00
Megalon 8d9efc1577 Fix crash with semver parser 2020-05-01 19:38:33 -07:00
Megalon e1490bcb1c Check if mod is outdated before installing 2020-05-01 19:21:26 -07:00
Assistant b13605d700
Merge pull request #127 from wgzeyu/patch-6
Adjust some translations
2020-04-30 04:38:36 -06:00
wgzeyu 0002c9e058
Update zh.xaml 2020-04-24 17:31:20 +08:00
Megalon f8507924c4 Update version display 2020-04-21 17:46:28 -07:00
Megalon efd01ff9c3 Add semver
https://github.com/maxhauser/semver
2020-04-21 17:26:46 -07:00
Assistant 90537aae3a
Merge pull request #125 from megalon/fix/saving
Quick fix for incorrect mods showing as "installed"
2020-04-21 17:57:53 -06:00
Megalon 5dbc50c4b9 Clarify option in settings 2020-04-21 15:50:24 -07:00
Megalon abace2afc0 Check mod status when checking hash 2020-04-21 15:42:03 -07:00
Assistant 925de391e2 Update version to 1.1.5 2020-04-11 16:12:45 -06:00
Assistant 51bdb82a15 Closes #117 2020-04-09 09:48:51 -06:00
Assistant b0d685d356
Merge pull request #115 from wgzeyu/patch-5
Update zh.xaml
2020-04-05 09:51:13 -06:00
wgzeyu aa606835c3
Update zh.xaml 2020-04-05 23:08:53 +08:00
Assistant 8f79fba624 Error handling for overwriting files in use. 2020-04-05 07:04:51 -06:00
Assistant 58c1550fd9
Merge pull request #108 from Parapass/patch-1
Fix 'Delete All Mods' button
2020-03-27 20:28:31 -06:00
Parapass f9e05641fb
fix Delete All Mods button 2020-03-27 12:57:30 -04:00
Assistant a1af3d7413 Updated version to 1.1.4 2020-03-26 09:39:18 -06:00
Assistant 8040af8954 2020-03-26 09:38:38 -06:00
Assistant cb4c475d46 Updated version to 1.1.3 2020-03-18 08:39:22 -06:00
Assistant 1f1fccd953 Fixed hyperlink formatting 2020-03-18 08:37:23 -06:00
Assistant 8227be4ec2 Fixes #98 2020-03-15 08:27:44 -06:00
Assistant c04f8403aa
Merge pull request #99 from Parapass/yeet-modsaber
Remove modsaber link from Options C# file
2020-03-14 08:45:41 -06:00
Parapass 759888e53b
remove modsaber from options cs file 2020-03-14 10:43:16 -04:00
Assistant 800abf1663 Updated version to 1.1.2 2020-03-05 16:27:29 -07:00
Assistant b13b922c5f Fixed #95 with HttpClient timeout to 240 seconds 2020-03-05 16:26:50 -07:00
Assistant b7bc73a1ff Fixes #94 2020-03-03 14:17:56 -07:00
Assistant b7831df8e8
Merge pull request #93 from Parapass/patch-2
Add support for custom bloqs from ModelSaber
2020-02-29 10:33:05 -07:00
Parapass c840f76172
Add support for custom bloqs from ModelSaber 2020-02-28 19:13:07 -05:00
Assistant d5885f1a60 Updated version to 1.1.1 2020-02-28 08:38:49 -07:00
Assistant 2eae95e13a Fixed #91 v2 2020-02-28 05:24:51 -07:00
Assistant 4cb40f7906 Fixed wiki link 2020-02-28 01:36:34 -07:00
Assistant ba58d0a859 Fixed #91 2020-02-28 01:28:02 -07:00
wgzeyu cc5a6b1cb2
Correct Chinese language files (#90) 2020-02-28 00:03:39 -07:00
wgzeyu c34934bca8
Add Chinese language file (#88) 2020-02-27 21:14:48 -07:00
82 changed files with 3456 additions and 2856 deletions

View file

@ -14,3 +14,6 @@ dotnet_separate_import_directive_groups = false
[*.md]
trim_trailing_whitespace = false
[*.yml]
indent_size = 2

5
.github/FUNDING.yml vendored
View file

@ -1,4 +1 @@
patreon: AssistantMoe
ko_fi: N4N8JX7B
liberapay: Assistant
custom: ['https://paypal.me/AssistantMoe', 'https://bs.assistant.moe/Donate/']
custom: ['https://github.com/Assistant/ModAssistant']

View file

@ -1,52 +1,44 @@
name: .NET Build
on: [push]
on:
push:
pull_request:
branches:
- master
jobs:
build:
runs-on: windows-latest
steps:
- uses: actions/checkout@v1
# Set Path workaround for https://github.com/actions/virtual-environments/issues/263
- name: "Temp step to Set Path for Windows"
run: |
echo "::add-path::C:\Program Files\Git\mingw64\bin"
echo "::add-path::C:\Program Files\Git\usr\bin"
echo "::add-path::C:\Program Files\Git\bin"
- name: Setup MSBuild
uses: warrenbuckley/Setup-MSBuild@v1
- name: Install dependencies
run: msbuild -t:restore
- name: Checkout
uses: actions/checkout@v2
- name: Setup msbuild
uses: microsoft/setup-msbuild@v1.0.2
- name: Restore NuGet packages
working-directory: ${{env.GITHUB_WORKSPACE}}
run: nuget restore VRCMelonAssistant.sln
- name: Build project
run: msbuild ModAssistant/ModAssistant.csproj /t:Build /p:Configuration=Release
run: msbuild VRCMelonAssistant/VRCMelonAssistant.csproj /t:Build /p:Configuration=Release
- name: Cleanup release
shell: bash
run: |
find "ModAssistant/bin/Release" -type f ! -name "ModAssistant.exe" -delete
cp "LICENSE" "ModAssistant/bin/Release/LICENSE.ModAssistant.txt"
find "VRCMelonAssistant/bin/Release" -type f ! -name "VRCMelonAssistant.exe" -delete
cp "LICENSE" "VRCMelonAssistant/bin/Release/LICENSE.VRCMelonAssistant.txt"
- name: Upload Build
if: contains(github.ref, 'refs/tags/') == false
uses: actions/upload-artifact@v1
if: startsWith(github.ref, 'refs/tags/') == false
uses: actions/upload-artifact@v2
with:
name: ModAssistant-${{ github.sha }}
path: ./ModAssistant/bin/Release
- name: Create Release
if: contains(github.ref, 'refs/tags/') == true
id: create_release
uses: actions/create-release@v1.0.0
name: VRCMelonAssistant-${{ github.sha }}
path: ./VRCMelonAssistant/bin/Release/
- name: Extract Release Version
if: startsWith(github.ref, 'refs/tags/')
id: get_version
shell: bash
run: echo ::set-output name=version::${GITHUB_REF#refs/tags/v}
- name: Release
if: startsWith(github.ref, 'refs/tags/')
uses: softprops/action-gh-release@v1
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
with:
tag_name: ${{ github.ref }}
release_name: Mod Assistant ${{ github.ref }}
draft: false
prerelease: false
- name: Upload Release Asset
if: contains(github.ref, 'refs/tags/') == true
uses: actions/upload-release-asset@v1.0.1
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
with:
upload_url: ${{ steps.create_release.outputs.upload_url }}
asset_path: ./ModAssistant/bin/Release/ModAssistant.exe
asset_name: ModAssistant.exe
asset_content_type: application/vnd.microsoft.portable-executable
name: VRChat Melon Assistant v${{ steps.get_version.outputs.version }}
files: ./VRCMelonAssistant/bin/Release/VRCMelonAssistant.exe

View file

@ -1,60 +0,0 @@
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<configSections>
<sectionGroup name="userSettings" type="System.Configuration.UserSettingsGroup, System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" >
<section name="ModAssistant.Properties.Settings" type="System.Configuration.ClientSettingsSection, System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" allowExeDefinition="MachineToLocalUser" requirePermission="false" />
<section name="ModAssistant.Settings1" type="System.Configuration.ClientSettingsSection, System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" allowExeDefinition="MachineToLocalUser" requirePermission="false" />
</sectionGroup>
</configSections>
<startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.6.1" />
</startup>
<userSettings>
<ModAssistant.Properties.Settings>
<setting name="InstallFolder" serializeAs="String">
<value />
</setting>
<setting name="StoreType" serializeAs="String">
<value />
</setting>
<setting name="SaveSelected" serializeAs="String">
<value>False</value>
</setting>
<setting name="CheckInstalled" serializeAs="String">
<value>True</value>
</setting>
<setting name="SavedMods" serializeAs="String">
<value>False</value>
</setting>
<setting name="Agreed" serializeAs="String">
<value>False</value>
</setting>
<setting name="SelectInstalled" serializeAs="String">
<value>False</value>
</setting>
<setting name="GameVersion" serializeAs="String">
<value />
</setting>
<setting name="AllGameVersions" serializeAs="String">
<value />
</setting>
<setting name="UpgradeRequired" serializeAs="String">
<value>True</value>
</setting>
<setting name="LastTab" serializeAs="String">
<value />
</setting>
<setting name="SelectedTheme" serializeAs="String">
<value />
</setting>
</ModAssistant.Properties.Settings>
<ModAssistant.Settings1>
<setting name="InstallFolder" serializeAs="String">
<value />
</setting>
<setting name="StoreVariant" serializeAs="String">
<value />
</setting>
</ModAssistant.Settings1>
</userSettings>
</configuration>

View file

@ -1,162 +0,0 @@
using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Threading.Tasks;
using System.Windows;
namespace ModAssistant
{
/// <summary>
/// Interaction logic for App.xaml
/// </summary>
public partial class App : Application
{
public static string BeatSaberInstallDirectory;
public static string BeatSaberInstallType;
public static bool SaveModSelection;
public static bool CheckInstalledMods;
public static bool SelectInstalledMods;
public static string Version = Assembly.GetExecutingAssembly().GetName().Version.ToString();
public static List<string> SavedMods = ModAssistant.Properties.Settings.Default.SavedMods.Split(',').ToList();
private async void Application_Startup(object sender, StartupEventArgs e)
{
// Load localisation languages
LoadLanguage(CultureInfo.CurrentCulture.Name);
// Uncomment the next line to debug localisation
// LoadLanguage("en-DEBUG");
if (ModAssistant.Properties.Settings.Default.UpgradeRequired)
{
ModAssistant.Properties.Settings.Default.Upgrade();
ModAssistant.Properties.Settings.Default.UpgradeRequired = false;
ModAssistant.Properties.Settings.Default.Save();
}
Version = Version.Substring(0, Version.Length - 2);
BeatSaberInstallDirectory = Utils.GetInstallDir();
while (string.IsNullOrEmpty(App.BeatSaberInstallDirectory))
{
string title = (string)Current.FindResource("App:InstallDirDialog:Title");
string body = (string)Current.FindResource("App:InstallDirDialog:OkCancel");
if (System.Windows.Forms.MessageBox.Show(body, title, System.Windows.Forms.MessageBoxButtons.OKCancel) == System.Windows.Forms.DialogResult.OK)
{
App.BeatSaberInstallDirectory = Utils.GetManualDir();
}
else
{
Environment.Exit(0);
}
}
BeatSaberInstallType = ModAssistant.Properties.Settings.Default.StoreType;
SaveModSelection = ModAssistant.Properties.Settings.Default.SaveSelected;
CheckInstalledMods = ModAssistant.Properties.Settings.Default.CheckInstalled;
SelectInstalledMods = ModAssistant.Properties.Settings.Default.SelectInstalled;
if (e.Args.Length == 0)
{
await Task.Run(async () => await Updater.Run());
MainWindow window = new MainWindow();
window.Show();
}
else
{
ArgumentHandler(e.Args);
}
}
private void ArgumentHandler(string[] args)
{
switch (args[0])
{
case "--install":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--install"));
}
else
{
OneClickInstaller.InstallAsset(args[1]);
}
break;
case "--no-update":
MainWindow window = new MainWindow();
window.Show();
break;
case "--register":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--register"));
}
else
{
OneClickInstaller.Register(args[1], true);
}
break;
case "--unregister":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--unregister"));
}
else
{
OneClickInstaller.Unregister(args[1], true);
}
break;
default:
Utils.SendNotify((string)Current.FindResource("App:UnrecognizedArgument"));
break;
}
Current.Shutdown();
}
private void Application_DispatcherUnhandledException(object sender, System.Windows.Threading.DispatcherUnhandledExceptionEventArgs e)
{
string title = (string)Current.FindResource("App:Exception");
string body = (string)Current.FindResource("App:UnhandledException");
MessageBox.Show($"{body}: {e.Exception}", "Exception", MessageBoxButton.OK, MessageBoxImage.Warning);
e.Handled = true;
Application.Current.Shutdown();
}
private ResourceDictionary LanguagesDict
{
get
{
return Resources.MergedDictionaries[1];
}
}
private void LoadLanguage(string culture)
{
try
{
LanguagesDict.Source = new Uri($"Localisation/{culture}.xaml", UriKind.Relative);
}
catch (IOException)
{
if (culture.Contains("-"))
{
LoadLanguage(culture.Split('-').First());
}
// Can't load language file
}
}
}
}

View file

@ -1,53 +0,0 @@
using System;
using System.Collections.Generic;
using ModAssistant.Pages;
namespace ModAssistant
{
public class Mod
{
public string name;
public string version;
public string gameVersion;
public string _id;
public string authorId;
public string uploadedDate;
public string updatedDate;
public Author author;
public string description;
public string link;
public string category;
public DownloadLink[] downloads;
public bool required;
public Dependency[] dependencies;
public List<Mod> Dependents = new List<Mod>();
public Mods.ModListItem ListItem;
public class Author
{
public string _id;
public string username;
public string lastLogin;
}
public class DownloadLink
{
public string type;
public string url;
public FileHashes[] hashMd5;
}
public class FileHashes
{
public string hash;
public string file;
}
public class Dependency
{
public string name;
public string _id;
public Mod Mod;
}
}
}

View file

@ -1,321 +0,0 @@
using Microsoft.Win32;
using System;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Net;
using System.Threading.Tasks;
using System.Windows;
using static ModAssistant.Http;
namespace ModAssistant
{
class OneClickInstaller
{
private const string ModelSaberURLPrefix = "https://modelsaber.com/files/";
private const string BeatSaverURLPrefix = "https://beatsaver.com";
private static readonly string BeatSaberPath = App.BeatSaberInstallDirectory;
private const string CustomAvatarsFolder = "CustomAvatars";
private const string CustomSabersFolder = "CustomSabers";
private const string CustomPlatformsFolder = "CustomPlatforms";
private static readonly string CustomSongsFolder = Path.Combine("Beat Saber_Data", "CustomLevels");
private static readonly string[] Protocols = new[] { "modelsaber", "beatsaver" };
private const bool BypassDownloadCounter = false;
public static async void InstallAsset(string link)
{
Uri uri = new Uri(link);
if (!Protocols.Contains(uri.Scheme)) return;
switch (uri.Scheme)
{
case "modelsaber":
await ModelSaber(uri);
break;
case "beatsaver":
await BeatSaver(uri);
break;
}
}
private static async Task BeatSaver(Uri uri)
{
string Key = uri.Host;
BeatSaverApiResponse Response;
try
{
var resp = await HttpClient.GetAsync(BeatSaverURLPrefix + "/api/maps/detail/" + Key);
var body = await resp.Content.ReadAsStringAsync();
Response = JsonSerializer.Deserialize<BeatSaverApiResponse>(body);
}
catch (Exception e)
{
MessageBox.Show($"{Application.Current.FindResource("OneClick:MapDownloadFailed")}\n\n" + e);
return;
}
string zip = Path.Combine(BeatSaberPath, CustomSongsFolder, Response.hash) + ".zip";
string directory = Path.Combine(
BeatSaberPath,
CustomSongsFolder,
string.Concat(
(Response.key + " (" + Response.metadata.songName + " - " + Response.metadata.levelAuthorName + ")")
.Split(Utils.Constants.IllegalCharacters)
)
);
if (BypassDownloadCounter)
{
await DownloadAsset(BeatSaverURLPrefix + Response.directDownload, CustomSongsFolder, Response.hash + ".zip");
}
else
{
await DownloadAsset(BeatSaverURLPrefix + Response.downloadURL, CustomSongsFolder, Response.hash + ".zip");
}
if (File.Exists(zip))
{
using (FileStream stream = new FileStream(zip, FileMode.Open))
using (ZipArchive archive = new ZipArchive(stream))
{
foreach (ZipArchiveEntry file in archive.Entries)
{
string fileDirectory = Path.GetDirectoryName(Path.Combine(directory, file.FullName));
if (!Directory.Exists(fileDirectory))
{
Directory.CreateDirectory(fileDirectory);
}
if (!string.IsNullOrEmpty(file.Name))
{
file.ExtractToFile(Path.Combine(directory, file.FullName), true);
}
}
}
File.Delete(zip);
}
else
{
string line1 = (string)Application.Current.FindResource("OneClick:SongDownload:Failed");
string line2 = (string)Application.Current.FindResource("OneClick:SongDownload:NetworkIssues");
string title = (string)Application.Current.FindResource("OneClick:SongDownload:FailedTitle");
MessageBox.Show($"{line1}\n{line2}", title);
}
}
private static async Task ModelSaber(Uri uri)
{
switch (uri.Host)
{
case "avatar":
await DownloadAsset(ModelSaberURLPrefix + uri.Host + uri.AbsolutePath, CustomAvatarsFolder);
break;
case "saber":
await DownloadAsset(ModelSaberURLPrefix + uri.Host + uri.AbsolutePath, CustomSabersFolder);
break;
case "platform":
await DownloadAsset(ModelSaberURLPrefix + uri.Host + uri.AbsolutePath, CustomPlatformsFolder);
break;
}
}
private static async Task DownloadAsset(string link, string folder, string fileName = null)
{
if (string.IsNullOrEmpty(BeatSaberPath))
{
Utils.SendNotify((string)Application.Current.FindResource("OneClick:InstallDirNotFound"));
}
try
{
Directory.CreateDirectory(Path.Combine(BeatSaberPath, folder));
if (string.IsNullOrEmpty(fileName))
{
fileName = WebUtility.UrlDecode(Path.Combine(BeatSaberPath, folder, new Uri(link).Segments.Last()));
}
else
{
fileName = WebUtility.UrlDecode(Path.Combine(BeatSaberPath, folder, fileName));
}
await Utils.Download(link, fileName);
Utils.SendNotify(string.Format((string)Application.Current.FindResource("OneClick:InstalledAsset"), Path.GetFileNameWithoutExtension(fileName)));
}
catch
{
Utils.SendNotify((string)Application.Current.FindResource("OneClick:AssetInstallFailed"));
}
}
public static void Register(string Protocol, bool Background = false)
{
if (IsRegistered(Protocol) == true)
return;
try
{
if (Utils.IsAdmin)
{
RegistryKey ProtocolKey = Registry.ClassesRoot.OpenSubKey(Protocol, true);
if (ProtocolKey == null)
ProtocolKey = Registry.ClassesRoot.CreateSubKey(Protocol, true);
RegistryKey CommandKey = ProtocolKey.CreateSubKey(@"shell\open\command", true);
if (CommandKey == null)
CommandKey = Registry.ClassesRoot.CreateSubKey(@"shell\open\command", true);
if (ProtocolKey.GetValue("OneClick-Provider", "").ToString() != "ModAssistant")
{
ProtocolKey.SetValue("URL Protocol", "", RegistryValueKind.String);
ProtocolKey.SetValue("OneClick-Provider", "ModAssistant", RegistryValueKind.String);
CommandKey.SetValue("", $"\"{Utils.ExePath}\" \"--install\" \"%1\"");
}
Utils.SendNotify(string.Format((string)Application.Current.FindResource("OneClick:ProtocolHandler:Registered"), Protocol));
}
else
{
Utils.StartAsAdmin($"\"--register\" \"{Protocol}\"");
}
}
catch (Exception e)
{
MessageBox.Show(e.ToString());
}
if (Background)
Application.Current.Shutdown();
else
Pages.Options.Instance.UpdateHandlerStatus();
}
public static void Unregister(string Protocol, bool Background = false)
{
if (IsRegistered(Protocol) == false)
return;
try
{
if (Utils.IsAdmin)
{
using (RegistryKey ProtocolKey = Registry.ClassesRoot.OpenSubKey(Protocol, true))
{
if (ProtocolKey != null
&& ProtocolKey.GetValue("OneClick-Provider", "").ToString() == "ModAssistant")
{
Registry.ClassesRoot.DeleteSubKeyTree(Protocol);
}
}
Utils.SendNotify(string.Format((string)Application.Current.FindResource("OneClick:ProtocolHandler:Unregistered"), Protocol));
}
else
{
Utils.StartAsAdmin($"\"--unregister\" \"{Protocol}\"");
}
}
catch (Exception e)
{
MessageBox.Show(e.ToString());
}
if (Background)
Application.Current.Shutdown();
else
Pages.Options.Instance.UpdateHandlerStatus();
}
public static bool IsRegistered(string Protocol)
{
RegistryKey ProtocolKey = Registry.ClassesRoot.OpenSubKey(Protocol);
if (ProtocolKey != null
&& ProtocolKey.GetValue("OneClick-Provider", "").ToString() == "ModAssistant")
return true;
else
return false;
}
}
#pragma warning disable IDE1006 // Naming Styles
class BeatSaverApiResponse
{
public Metadata metadata { get; set; }
public Stats stats { get; set; }
public string description { get; set; }
public DateTime? deletedAt { get; set; }
public string _id { get; set; }
public string key { get; set; }
public string name { get; set; }
public Uploader uploader { get; set; }
public DateTime uploaded { get; set; }
public string hash { get; set; }
public string directDownload { get; set; }
public string downloadURL { get; set; }
public string coverURL { get; set; }
public class Difficulties
{
public bool easy { get; set; }
public bool normal { get; set; }
public bool hard { get; set; }
public bool expert { get; set; }
public bool expertPlus { get; set; }
}
public class Metadata
{
public Difficulties difficulties { get; set; }
public Characteristic[] characteristics { get; set; }
public string songName { get; set; }
public string songSubName { get; set; }
public string songAuthorName { get; set; }
public string levelAuthorName { get; set; }
public double bpm { get; set; }
}
public class Characteristic
{
public string name { get; set; }
public CharacteristicDifficulties difficulties { get; set; }
}
public class CharacteristicDifficulties
{
public Difficulty easy { get; set; }
public Difficulty normal { get; set; }
public Difficulty hard { get; set; }
public Difficulty expert { get; set; }
public Difficulty expertPlus { get; set; }
}
public class Difficulty
{
public double? duration { get; set; }
public double? length { get; set; }
public double bombs { get; set; }
public double notes { get; set; }
public double obstacles { get; set; }
public double njs { get; set; }
}
public class Stats
{
public int downloads { get; set; }
public int plays { get; set; }
public int downVotes { get; set; }
public int upVotes { get; set; }
public double heat { get; set; }
public double rating { get; set; }
}
public class Uploader
{
public string _id { get; set; }
public string username { get; set; }
}
}
}
#pragma warning restore IDE1006 // Naming Styles

View file

@ -1,323 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;
using ModAssistant.Pages;
using static ModAssistant.Http;
namespace ModAssistant
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public static MainWindow Instance;
public static bool ModsOpened = false;
public static bool ModsLoading = false;
public static string GameVersion;
public static string GameVersionOverride;
public TaskCompletionSource<bool> VersionLoadStatus = new TaskCompletionSource<bool>();
public string MainText
{
get
{
return MainTextBlock.Text;
}
set
{
Dispatcher.Invoke(new Action(() => { MainWindow.Instance.MainTextBlock.Text = value; }));
}
}
public MainWindow()
{
InitializeComponent();
Instance = this;
const int ContentWidth = 1280;
const int ContentHeight = 720;
double ChromeWidth = SystemParameters.WindowNonClientFrameThickness.Left + SystemParameters.WindowNonClientFrameThickness.Right;
double ChromeHeight = SystemParameters.WindowNonClientFrameThickness.Top + SystemParameters.WindowNonClientFrameThickness.Bottom;
double ResizeBorder = SystemParameters.ResizeFrameVerticalBorderWidth;
Width = ChromeWidth + ContentWidth + 2 * ResizeBorder;
Height = ChromeHeight + ContentHeight + 2 * ResizeBorder;
VersionText.Text = App.Version;
if (Utils.IsVoid())
{
Main.Content = Invalid.Instance;
MainWindow.Instance.ModsButton.IsEnabled = false;
MainWindow.Instance.OptionsButton.IsEnabled = false;
MainWindow.Instance.IntroButton.IsEnabled = false;
MainWindow.Instance.AboutButton.IsEnabled = false;
MainWindow.Instance.GameVersionsBox.IsEnabled = false;
return;
}
Themes.LoadThemes();
Themes.FirstLoad(Properties.Settings.Default.SelectedTheme);
Task.Run(() => LoadVersionsAsync());
if (!Properties.Settings.Default.Agreed || string.IsNullOrEmpty(Properties.Settings.Default.LastTab))
{
Main.Content = Intro.Instance;
}
else
{
switch (Properties.Settings.Default.LastTab)
{
case "Intro":
Main.Content = Intro.Instance;
break;
case "Mods":
_ = ShowModsPage();
break;
case "About":
Main.Content = About.Instance;
break;
case "Options":
Main.Content = Options.Instance;
Themes.LoadThemes();
break;
default:
Main.Content = Intro.Instance;
break;
}
}
}
private async void LoadVersionsAsync()
{
try
{
var resp = await HttpClient.GetAsync(Utils.Constants.BeatModsAPIUrl + "version");
var body = await resp.Content.ReadAsStringAsync();
List<string> versions = JsonSerializer.Deserialize<string[]>(body).ToList();
resp = await HttpClient.GetAsync(Utils.Constants.BeatModsAlias);
body = await resp.Content.ReadAsStringAsync();
object jsonObject = JsonSerializer.DeserializeObject(body);
Dispatcher.Invoke(() =>
{
GameVersion = GetGameVersion(versions, jsonObject);
GameVersionsBox.ItemsSource = versions;
GameVersionsBox.SelectedValue = GameVersion;
if (!string.IsNullOrEmpty(GameVersionOverride))
{
GameVersionsBox.Visibility = Visibility.Collapsed;
GameVersionsBoxOverride.Visibility = Visibility.Visible;
GameVersionsBoxOverride.Text = GameVersionOverride;
GameVersionsBoxOverride.IsEnabled = false;
}
if (!string.IsNullOrEmpty(GameVersion) && Properties.Settings.Default.Agreed)
{
MainWindow.Instance.ModsButton.IsEnabled = true;
}
});
VersionLoadStatus.SetResult(true);
}
catch (Exception e)
{
Dispatcher.Invoke(() =>
{
GameVersionsBox.IsEnabled = false;
MessageBox.Show($"{Application.Current.FindResource("MainWindow:GameVersionLoadFailed")}\n{e}");
});
VersionLoadStatus.SetResult(false);
}
}
private string GetGameVersion(List<string> versions, object aliases)
{
string version = Utils.GetVersion();
if (!string.IsNullOrEmpty(version) && versions.Contains(version))
{
return version;
}
string aliasOf = CheckAliases(versions, aliases, version);
if (!string.IsNullOrEmpty(aliasOf))
{
return aliasOf;
}
string versionsString = String.Join(",", versions.ToArray());
if (Properties.Settings.Default.AllGameVersions != versionsString)
{
Properties.Settings.Default.AllGameVersions = versionsString;
Properties.Settings.Default.Save();
string title = (string)Application.Current.FindResource("MainWindow:GameUpdateDialog:Title");
string line1 = (string)Application.Current.FindResource("MainWindow:GameUpdateDialog:Line1");
string line2 = (string)Application.Current.FindResource("MainWindow:GameUpdateDialog:Line2");
Utils.ShowMessageBoxAsync($"{line1}\n\n{line2}", title);
return versions[0];
}
if (!string.IsNullOrEmpty(Properties.Settings.Default.GameVersion) && versions.Contains(Properties.Settings.Default.GameVersion))
return Properties.Settings.Default.GameVersion;
return versions[0];
}
private string CheckAliases(List<string> versions, object aliases, string detectedVersion)
{
Dictionary<string, object> Objects = (Dictionary<string, object>)aliases;
foreach (string version in versions)
{
object[] aliasArray = (object[])Objects[version];
foreach (object alias in aliasArray)
{
if (alias.ToString() == detectedVersion)
{
GameVersionOverride = detectedVersion;
return version;
}
}
}
return string.Empty;
}
private async Task ShowModsPage()
{
void OpenModsPage()
{
Main.Content = Mods.Instance;
Properties.Settings.Default.LastTab = "Mods";
Properties.Settings.Default.Save();
Mods.Instance.RefreshColumns();
}
if (ModsOpened == true && Mods.Instance.PendingChanges == false)
{
OpenModsPage();
return;
}
Main.Content = Loading.Instance;
if (ModsLoading) return;
ModsLoading = true;
await Mods.Instance.LoadMods();
ModsLoading = false;
if (ModsOpened == false) ModsOpened = true;
if (Mods.Instance.PendingChanges == true) Mods.Instance.PendingChanges = false;
if (Main.Content == Loading.Instance)
{
OpenModsPage();
}
}
private void ModsButton_Click(object sender, RoutedEventArgs e)
{
_ = ShowModsPage();
}
private void IntroButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = Intro.Instance;
Properties.Settings.Default.LastTab = "Intro";
Properties.Settings.Default.Save();
}
private void AboutButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = About.Instance;
Properties.Settings.Default.LastTab = "About";
Properties.Settings.Default.Save();
}
private void OptionsButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = Options.Instance;
Themes.LoadThemes();
Properties.Settings.Default.LastTab = "Options";
Properties.Settings.Default.Save();
}
private void InstallButton_Click(object sender, RoutedEventArgs e)
{
Mods.Instance.InstallMods();
}
private void InfoButton_Click(object sender, RoutedEventArgs e)
{
if ((Mods.ModListItem)Mods.Instance.ModsListView.SelectedItem == null)
{
MessageBox.Show((string)Application.Current.FindResource("MainWindow:NoModSelected"));
return;
}
Mods.ModListItem mod = ((Mods.ModListItem)Mods.Instance.ModsListView.SelectedItem);
string infoUrl = mod.ModInfo.link;
if (string.IsNullOrEmpty(infoUrl))
{
MessageBox.Show(string.Format((string)Application.Current.FindResource("MainWindow:NoModInfoPage"), mod.ModName));
}
else
{
System.Diagnostics.Process.Start(infoUrl);
}
}
private async void GameVersionsBox_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
string oldGameVersion = GameVersion;
GameVersion = (sender as ComboBox).SelectedItem.ToString();
if (string.IsNullOrEmpty(oldGameVersion)) return;
Properties.Settings.Default.GameVersion = GameVersion;
Properties.Settings.Default.Save();
if (ModsOpened)
{
var prevPage = Main.Content;
Mods.Instance.PendingChanges = true;
await ShowModsPage();
Main.Content = prevPage;
}
}
private void Window_PreviewMouseDown(object sender, MouseButtonEventArgs e)
{
About.Instance.PatUp.IsOpen = false;
About.Instance.PatButton.IsEnabled = true;
About.Instance.HugUp.IsOpen = false;
About.Instance.HugButton.IsEnabled = true;
}
private void Window_SizeChanged(object sender, SizeChangedEventArgs e)
{
if (Main.Content == Mods.Instance)
{
Mods.Instance.RefreshColumns();
}
}
private void BackgroundVideo_MediaEnded(object sender, RoutedEventArgs e)
{
BackgroundVideo.Position = TimeSpan.Zero;
BackgroundVideo.Play();
}
}
}

View file

@ -1,101 +0,0 @@
<Page
x:Class="ModAssistant.Pages.Mods"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="{DynamicResource Mods:Title}"
d:DesignHeight="629"
d:DesignWidth="1182"
Loaded="Page_Loaded"
mc:Ignorable="d">
<Grid>
<ListView
Name="ModsListView"
Grid.Column="1"
SelectionChanged="ModsListView_SelectionChanged"
SelectionMode="Single">
<ListView.View>
<GridView>
<GridView.Columns>
<GridViewColumn Width="30">
<GridViewColumn.CellTemplate>
<DataTemplate>
<CheckBox
Name="ModCheckBox"
Checked="ModCheckBox_Checked"
IsChecked="{Binding Path=IsSelected, Mode=TwoWay}"
IsEnabled="{Binding IsEnabled}"
Tag="{Binding ModInfo}"
Unchecked="ModCheckBox_Unchecked" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn DisplayMemberBinding="{Binding ModName}" Header="{DynamicResource Mods:Header:Name}" />
<GridViewColumn x:Name="InstalledColumn" Header="{DynamicResource Mods:Header:Installed}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBlock
Foreground="{Binding GetVersionColor}"
Text="{Binding InstalledVersion}"
TextDecorations="{Binding GetVersionDecoration}" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn DisplayMemberBinding="{Binding ModVersion}" Header="{DynamicResource Mods:Header:Latest}" />
<GridViewColumn x:Name="DescriptionColumn" Header="{DynamicResource Mods:Header:Description}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Margin="{Binding PromotionMargin}">
<Hyperlink NavigateUri="{Binding PromotionLink, TargetNullValue=about:blank}" RequestNavigate="Hyperlink_RequestNavigate">
<TextBlock Text="{Binding PromotionText}" />
</Hyperlink>
</TextBlock>
<TextBlock Text="{Binding ModDescription}" />
</StackPanel>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn
x:Name="UninstallColumn"
Width="70"
Header="{DynamicResource Mods:Header:Uninstall}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<Button
Name="Uninstall"
Click="Uninstall_Click"
Content="{DynamicResource Mods:UninstallButton}"
Foreground="Red"
IsEnabled="{Binding CanDelete}"
Tag="{Binding ModInfo}"
Visibility="{Binding CanSeeDelete}" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
</GridView.Columns>
</GridView>
</ListView.View>
<ListView.GroupStyle>
<GroupStyle>
<GroupStyle.HeaderTemplate>
<DataTemplate>
<TextBlock
Padding="6,0,0,0"
FontSize="14"
FontWeight="Bold"
Text="{Binding Name}" />
</DataTemplate>
</GroupStyle.HeaderTemplate>
</GroupStyle>
</ListView.GroupStyle>
</ListView>
</Grid>
</Page>

View file

@ -1,670 +0,0 @@
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Threading;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Forms;
using System.Windows.Navigation;
using static ModAssistant.Http;
namespace ModAssistant.Pages
{
/// <summary>
/// Interaction logic for Mods.xaml
/// </summary>
public sealed partial class Mods : Page
{
public static Mods Instance = new Mods();
public List<string> DefaultMods = new List<string>() { "SongCore", "ScoreSaber", "BeatSaverDownloader", "BeatSaverVoting", "PlaylistCore", "Survey" };
public Mod[] ModsList;
public Mod[] AllModsList;
public static List<Mod> InstalledMods = new List<Mod>();
public List<string> CategoryNames = new List<string>();
public CollectionView view;
public bool PendingChanges;
private readonly SemaphoreSlim _modsLoadSem = new SemaphoreSlim(1, 1);
public List<ModListItem> ModList { get; set; }
public Mods()
{
InitializeComponent();
}
private void RefreshModsList()
{
if (view != null)
{
view.Refresh();
}
}
public void RefreshColumns()
{
if (MainWindow.Instance.Main.Content != Mods.Instance) return;
double viewWidth = ModsListView.ActualWidth;
double totalSize = 0;
GridViewColumn description = null;
GridView grid = ModsListView.View as GridView;
if (grid != null)
{
foreach (var column in grid.Columns)
{
if (column.Header?.ToString() == FindResource("Mods:Header:Description").ToString())
{
description = column;
}
else
{
totalSize += column.ActualWidth;
}
if (double.IsNaN(column.Width))
{
column.Width = column.ActualWidth;
column.Width = double.NaN;
}
}
double descriptionNewWidth = viewWidth - totalSize - 35;
description.Width = descriptionNewWidth > 200 ? descriptionNewWidth : 200;
}
}
public async Task LoadMods()
{
var versionLoadSuccess = await MainWindow.Instance.VersionLoadStatus.Task;
if (versionLoadSuccess == false) return;
await _modsLoadSem.WaitAsync();
try
{
MainWindow.Instance.InstallButton.IsEnabled = false;
MainWindow.Instance.GameVersionsBox.IsEnabled = false;
MainWindow.Instance.InfoButton.IsEnabled = false;
if (ModsList != null)
{
Array.Clear(ModsList, 0, ModsList.Length);
}
if (AllModsList != null)
{
Array.Clear(AllModsList, 0, AllModsList.Length);
}
InstalledMods = new List<Mod>();
CategoryNames = new List<string>();
ModList = new List<ModListItem>();
ModsListView.Visibility = Visibility.Hidden;
if (App.CheckInstalledMods)
{
MainWindow.Instance.MainText = $"{FindResource("Mods:CheckingInstalledMods")}...";
await Task.Run(async () => await CheckInstalledMods());
InstalledColumn.Width = double.NaN;
UninstallColumn.Width = 70;
DescriptionColumn.Width = 750;
}
else
{
InstalledColumn.Width = 0;
UninstallColumn.Width = 0;
DescriptionColumn.Width = 800;
}
MainWindow.Instance.MainText = $"{FindResource("Mods:LoadingMods")}...";
await Task.Run(async () => await PopulateModsList());
ModsListView.ItemsSource = ModList;
view = (CollectionView)CollectionViewSource.GetDefaultView(ModsListView.ItemsSource);
PropertyGroupDescription groupDescription = new PropertyGroupDescription("Category");
view.GroupDescriptions.Add(groupDescription);
this.DataContext = this;
RefreshModsList();
ModsListView.Visibility = Visibility.Visible;
MainWindow.Instance.MainText = $"{FindResource("Mods:FinishedLoadingMods")}.";
MainWindow.Instance.InstallButton.IsEnabled = true;
MainWindow.Instance.GameVersionsBox.IsEnabled = true;
}
finally
{
_modsLoadSem.Release();
}
}
public async Task CheckInstalledMods()
{
await GetAllMods();
List<string> empty = new List<string>();
GetBSIPAVersion();
CheckInstallDir("IPA/Pending/Plugins", empty);
CheckInstallDir("IPA/Pending/Libs", empty);
CheckInstallDir("Plugins", empty);
CheckInstallDir("Libs", empty);
}
public async Task GetAllMods()
{
var resp = await HttpClient.GetAsync(Utils.Constants.BeatModsAPIUrl + "mod");
var body = await resp.Content.ReadAsStringAsync();
AllModsList = JsonSerializer.Deserialize<Mod[]>(body);
}
private void CheckInstallDir(string directory, List<string> blacklist)
{
if (!Directory.Exists(Path.Combine(App.BeatSaberInstallDirectory, directory)))
{
return;
}
foreach (string file in Directory.GetFileSystemEntries(Path.Combine(App.BeatSaberInstallDirectory, directory)))
{
if (File.Exists(file) && Path.GetExtension(file) == ".dll" || Path.GetExtension(file) == ".manifest")
{
Mod mod = GetModFromHash(Utils.CalculateMD5(file));
if (mod != null)
{
AddDetectedMod(mod);
}
}
}
}
public void GetBSIPAVersion()
{
string InjectorPath = Path.Combine(App.BeatSaberInstallDirectory, "Beat Saber_Data", "Managed", "IPA.Injector.dll");
if (!File.Exists(InjectorPath)) return;
string InjectorHash = Utils.CalculateMD5(InjectorPath);
foreach (Mod mod in AllModsList)
{
if (mod.name.ToLower() == "bsipa")
{
foreach (Mod.DownloadLink download in mod.downloads)
{
foreach (Mod.FileHashes fileHash in download.hashMd5)
{
if (fileHash.hash == InjectorHash)
{
AddDetectedMod(mod);
}
}
}
}
}
}
private void AddDetectedMod(Mod mod)
{
if (!InstalledMods.Contains(mod))
{
InstalledMods.Add(mod);
if (App.SelectInstalledMods && !DefaultMods.Contains(mod.name))
{
DefaultMods.Add(mod.name);
}
}
}
private Mod GetModFromHash(string hash)
{
foreach (Mod mod in AllModsList)
{
if (mod.name.ToLower() != "bsipa")
{
foreach (Mod.DownloadLink download in mod.downloads)
{
foreach (Mod.FileHashes fileHash in download.hashMd5)
{
if (fileHash.hash == hash)
return mod;
}
}
}
}
return null;
}
public async Task PopulateModsList()
{
try
{
var resp = await HttpClient.GetAsync(Utils.Constants.BeatModsAPIUrl + Utils.Constants.BeatModsModsOptions + "&gameVersion=" + MainWindow.GameVersion);
var body = await resp.Content.ReadAsStringAsync();
ModsList = JsonSerializer.Deserialize<Mod[]>(body);
}
catch (Exception e)
{
System.Windows.MessageBox.Show($"{FindResource("Mods:LoadFailed")}.\n\n" + e);
return;
}
foreach (Mod mod in ModsList)
{
bool preSelected = mod.required;
if (DefaultMods.Contains(mod.name) || (App.SaveModSelection && App.SavedMods.Contains(mod.name)))
{
preSelected = true;
if (!App.SavedMods.Contains(mod.name))
{
App.SavedMods.Add(mod.name);
}
}
RegisterDependencies(mod);
ModListItem ListItem = new ModListItem()
{
IsSelected = preSelected,
IsEnabled = !mod.required,
ModName = mod.name,
ModVersion = mod.version,
ModDescription = mod.description.Replace("\r\n", " ").Replace("\n", " "),
ModInfo = mod,
Category = mod.category
};
foreach (Promotion promo in Promotions.ActivePromotions)
{
if (mod.name == promo.ModName)
{
ListItem.PromotionText = promo.Text;
ListItem.PromotionLink = promo.Link;
}
}
foreach (Mod installedMod in InstalledMods)
{
if (mod.name == installedMod.name)
{
ListItem.InstalledModInfo = installedMod;
ListItem.IsInstalled = true;
ListItem.InstalledVersion = installedMod.version;
break;
}
}
mod.ListItem = ListItem;
ModList.Add(ListItem);
}
foreach (Mod mod in ModsList)
{
ResolveDependencies(mod);
}
}
public async void InstallMods()
{
MainWindow.Instance.InstallButton.IsEnabled = false;
string installDirectory = App.BeatSaberInstallDirectory;
foreach (Mod mod in ModsList)
{
if (mod.name.ToLower() == "bsipa")
{
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstallingMod"), mod.name)}...";
await Task.Run(async () => await InstallMod(mod, installDirectory));
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstalledMod"), mod.name)}.";
if (!File.Exists(Path.Combine(installDirectory, "winhttp.dll")))
{
await Task.Run(() =>
Process.Start(new ProcessStartInfo
{
FileName = Path.Combine(installDirectory, "IPA.exe"),
WorkingDirectory = installDirectory,
Arguments = "-n"
}).WaitForExit()
);
}
Pages.Options.Instance.YeetBSIPA.IsEnabled = true;
}
else if (mod.ListItem.IsSelected)
{
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstallingMod"), mod.name)}...";
await Task.Run(async () => await InstallMod(mod, Path.Combine(installDirectory, @"IPA\Pending")));
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstalledMod"), mod.name)}.";
}
}
MainWindow.Instance.MainText = $"{FindResource("Mods:FinishedInstallingMods")}.";
MainWindow.Instance.InstallButton.IsEnabled = true;
RefreshModsList();
}
private async Task InstallMod(Mod mod, string directory)
{
string downloadLink = null;
foreach (Mod.DownloadLink link in mod.downloads)
{
if (link.type == "universal")
{
downloadLink = link.url;
break;
}
else if (link.type.ToLower() == App.BeatSaberInstallType.ToLower())
{
downloadLink = link.url;
break;
}
}
if (string.IsNullOrEmpty(downloadLink))
{
System.Windows.MessageBox.Show(string.Format((string)FindResource("Mods:ModDownloadLinkMissing"), mod.name));
return;
}
using (Stream stream = await DownloadMod(Utils.Constants.BeatModsURL + downloadLink))
using (ZipArchive archive = new ZipArchive(stream))
{
foreach (ZipArchiveEntry file in archive.Entries)
{
string fileDirectory = Path.GetDirectoryName(Path.Combine(directory, file.FullName));
if (!Directory.Exists(fileDirectory))
{
Directory.CreateDirectory(fileDirectory);
}
if (!string.IsNullOrEmpty(file.Name))
{
file.ExtractToFile(Path.Combine(directory, file.FullName), true);
}
}
}
if (App.CheckInstalledMods)
{
mod.ListItem.IsInstalled = true;
mod.ListItem.InstalledVersion = mod.version;
mod.ListItem.InstalledModInfo = mod;
}
}
private async Task<Stream> DownloadMod(string link)
{
var resp = await HttpClient.GetAsync(link);
return await resp.Content.ReadAsStreamAsync();
}
private void RegisterDependencies(Mod dependent)
{
if (dependent.dependencies.Length == 0)
return;
foreach (Mod mod in ModsList)
{
foreach (Mod.Dependency dep in dependent.dependencies)
{
if (dep.name == mod.name)
{
dep.Mod = mod;
mod.Dependents.Add(dependent);
}
}
}
}
private void ResolveDependencies(Mod dependent)
{
if (dependent.ListItem.IsSelected && dependent.dependencies.Length > 0)
{
foreach (Mod.Dependency dependency in dependent.dependencies)
{
if (dependency.Mod.ListItem.IsEnabled)
{
dependency.Mod.ListItem.PreviousState = dependency.Mod.ListItem.IsSelected;
dependency.Mod.ListItem.IsSelected = true;
dependency.Mod.ListItem.IsEnabled = false;
ResolveDependencies(dependency.Mod);
}
}
}
}
private void UnresolveDependencies(Mod dependent)
{
if (!dependent.ListItem.IsSelected && dependent.dependencies.Length > 0)
{
foreach (Mod.Dependency dependency in dependent.dependencies)
{
if (!dependency.Mod.ListItem.IsEnabled)
{
bool needed = false;
foreach (Mod dep in dependency.Mod.Dependents)
{
if (dep.ListItem.IsSelected)
{
needed = true;
break;
}
}
if (!needed && !dependency.Mod.required)
{
dependency.Mod.ListItem.IsSelected = dependency.Mod.ListItem.PreviousState;
dependency.Mod.ListItem.IsEnabled = true;
UnresolveDependencies(dependency.Mod);
}
}
}
}
}
private void ModCheckBox_Checked(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.CheckBox).Tag as Mod);
mod.ListItem.IsSelected = true;
ResolveDependencies(mod);
App.SavedMods.Add(mod.name);
Properties.Settings.Default.SavedMods = string.Join(",", App.SavedMods.ToArray());
Properties.Settings.Default.Save();
RefreshModsList();
}
private void ModCheckBox_Unchecked(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.CheckBox).Tag as Mod);
mod.ListItem.IsSelected = false;
UnresolveDependencies(mod);
App.SavedMods.Remove(mod.name);
Properties.Settings.Default.SavedMods = string.Join(",", App.SavedMods.ToArray());
Properties.Settings.Default.Save();
RefreshModsList();
}
public class Category
{
public string CategoryName { get; set; }
public List<ModListItem> Mods = new List<ModListItem>();
}
public class ModListItem
{
public string ModName { get; set; }
public string ModVersion { get; set; }
public string ModDescription { get; set; }
public bool PreviousState { get; set; }
public bool IsEnabled { get; set; }
public bool IsSelected { get; set; }
public Mod ModInfo { get; set; }
public string Category { get; set; }
public Mod InstalledModInfo { get; set; }
public bool IsInstalled { get; set; }
private string _installedVersion { get; set; }
public string InstalledVersion
{
get
{
return (string.IsNullOrEmpty(_installedVersion) || !IsInstalled) ? "-" : _installedVersion;
}
set
{
_installedVersion = value;
}
}
public string GetVersionColor
{
get
{
if (!IsInstalled) return "Black";
return InstalledVersion == ModVersion ? "Green" : "Red";
}
}
public string GetVersionDecoration
{
get
{
if (!IsInstalled) return "None";
return InstalledVersion == ModVersion ? "None" : "Strikethrough";
}
}
public bool CanDelete
{
get
{
return (!ModInfo.required && IsInstalled);
}
}
public string CanSeeDelete
{
get
{
if (!ModInfo.required && IsInstalled)
return "Visible";
else
return "Hidden";
}
}
public string PromotionText { get; set; }
public string PromotionLink { get; set; }
public string PromotionMargin
{
get
{
if (string.IsNullOrEmpty(PromotionText)) return "0";
return "0,0,5,0";
}
}
}
private void ModsListView_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
if ((Mods.ModListItem)Mods.Instance.ModsListView.SelectedItem == null)
{
MainWindow.Instance.InfoButton.IsEnabled = false;
}
else
{
MainWindow.Instance.InfoButton.IsEnabled = true;
}
}
public void UninstallBSIPA(Mod.DownloadLink links)
{
Process.Start(new ProcessStartInfo
{
FileName = Path.Combine(App.BeatSaberInstallDirectory, "IPA.exe"),
WorkingDirectory = App.BeatSaberInstallDirectory,
Arguments = "--revert -n"
}).WaitForExit();
foreach (Mod.FileHashes files in links.hashMd5)
{
string file = files.file.Replace("IPA/", "").Replace("Data", "Beat Saber_Data");
if (File.Exists(Path.Combine(App.BeatSaberInstallDirectory, file)))
File.Delete(Path.Combine(App.BeatSaberInstallDirectory, file));
}
Pages.Options.Instance.YeetBSIPA.IsEnabled = false;
}
private void Uninstall_Click(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.Button).Tag as Mod);
string title = string.Format((string)FindResource("Mods:UninstallBox:Title"), mod.name);
string body1 = string.Format((string)FindResource("Mods:UninstallBox:Body1"), mod.name);
string body2 = string.Format((string)FindResource("Mods:UninstallBox:Body2"), mod.name);
var result = System.Windows.Forms.MessageBox.Show($"{body1}\n{body2}", title, MessageBoxButtons.YesNo);
if (result == DialogResult.Yes)
{
UninstallModFromList(mod);
}
}
private void UninstallModFromList(Mod mod)
{
UninstallMod(mod.ListItem.InstalledModInfo);
mod.ListItem.IsInstalled = false;
mod.ListItem.InstalledVersion = null;
if (App.SelectInstalledMods)
{
mod.ListItem.IsSelected = false;
UnresolveDependencies(mod);
App.SavedMods.Remove(mod.name);
Properties.Settings.Default.SavedMods = string.Join(",", App.SavedMods.ToArray());
Properties.Settings.Default.Save();
RefreshModsList();
}
view.Refresh();
}
public void UninstallMod(Mod mod)
{
Mod.DownloadLink links = null;
foreach (Mod.DownloadLink link in mod.downloads)
{
if (link.type.ToLower() == "universal" || link.type.ToLower() == App.BeatSaberInstallType.ToLower())
{
links = link;
break;
}
}
if (mod.name.ToLower() == "bsipa")
UninstallBSIPA(links);
foreach (Mod.FileHashes files in links.hashMd5)
{
if (File.Exists(Path.Combine(App.BeatSaberInstallDirectory, files.file)))
File.Delete(Path.Combine(App.BeatSaberInstallDirectory, files.file));
if (File.Exists(Path.Combine(App.BeatSaberInstallDirectory, "IPA", "Pending", files.file)))
File.Delete(Path.Combine(App.BeatSaberInstallDirectory, "IPA", "Pending", files.file));
}
}
private void Hyperlink_RequestNavigate(object sender, RequestNavigateEventArgs e)
{
Process.Start(new ProcessStartInfo(e.Uri.AbsoluteUri));
e.Handled = true;
}
private void Page_Loaded(object sender, RoutedEventArgs e)
{
RefreshColumns();
}
}
}

View file

@ -1,283 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using Path = System.IO.Path;
namespace ModAssistant.Pages
{
/// <summary>
/// Interaction logic for Options.xaml
/// </summary>
public partial class Options : Page
{
public static Options Instance = new Options();
public string InstallDirectory { get; set; }
public string InstallType { get; set; }
public bool SaveSelection { get; set; }
public bool CheckInstalledMods { get; set; }
public bool SelectInstalledMods { get; set; }
public bool ModelSaberProtocolHandlerEnabled { get; set; }
public bool BeatSaverProtocolHandlerEnabled { get; set; }
public bool ModSaberProtocolHandlerEnabled { get; set; }
public string LogURL { get; private set; }
public Options()
{
InitializeComponent();
InstallDirectory = App.BeatSaberInstallDirectory;
InstallType = App.BeatSaberInstallType;
SaveSelection = App.SaveModSelection;
CheckInstalledMods = App.CheckInstalledMods;
SelectInstalledMods = App.SelectInstalledMods;
if (!CheckInstalledMods)
SelectInstalled.IsEnabled = false;
UpdateHandlerStatus();
this.DataContext = this;
}
public void UpdateHandlerStatus()
{
ModelSaberProtocolHandlerEnabled = OneClickInstaller.IsRegistered("modelsaber");
BeatSaverProtocolHandlerEnabled = OneClickInstaller.IsRegistered("beatsaver");
ModSaberProtocolHandlerEnabled = OneClickInstaller.IsRegistered("modsaber");
}
private void SelectDirButton_Click(object sender, RoutedEventArgs e)
{
Utils.GetManualDir();
DirectoryTextBlock.Text = InstallDirectory;
GameTypeTextBlock.Text = InstallType;
}
private void OpenDirButton_Click(object sender, RoutedEventArgs e)
{
Utils.OpenFolder(InstallDirectory);
}
private void Test_Click(object sender, RoutedEventArgs e)
{
MessageBox.Show(Utils.GetSteamDir());
}
private void SaveSelected_Checked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.SaveSelected = true;
App.SaveModSelection = true;
Properties.Settings.Default.Save();
}
private void SaveSelected_Unchecked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.SaveSelected = false;
App.SaveModSelection = false;
Properties.Settings.Default.Save();
}
private void CheckInstalled_Checked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.CheckInstalled = true;
App.CheckInstalledMods = true;
CheckInstalledMods = true;
Properties.Settings.Default.Save();
SelectInstalled.IsEnabled = true;
if (MainWindow.ModsOpened)
{
Mods.Instance.PendingChanges = true;
}
}
private void CheckInstalled_Unchecked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.CheckInstalled = false;
App.CheckInstalledMods = false;
CheckInstalledMods = false;
Properties.Settings.Default.Save();
SelectInstalled.IsEnabled = false;
if (MainWindow.ModsOpened)
{
Mods.Instance.PendingChanges = true;
}
}
public void ModelSaberProtocolHandler_Checked(object sender, RoutedEventArgs e)
{
OneClickInstaller.Register("modelsaber");
}
public void ModelSaberProtocolHandler_Unchecked(object sender, RoutedEventArgs e)
{
OneClickInstaller.Unregister("modelsaber");
}
public void BeatSaverProtocolHandler_Checked(object sender, RoutedEventArgs e)
{
OneClickInstaller.Register("beatsaver");
}
public void BeatSaverProtocolHandler_Unchecked(object sender, RoutedEventArgs e)
{
OneClickInstaller.Unregister("beatsaver");
}
private void SelectInstalled_Checked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.SelectInstalled = true;
App.SelectInstalledMods = true;
SelectInstalledMods = true;
Properties.Settings.Default.Save();
}
private void SelectInstalled_Unchecked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.SelectInstalled = false;
App.SelectInstalledMods = false;
SelectInstalledMods = false;
Properties.Settings.Default.Save();
}
private async void OpenLogsDirButton_Click(object sender, RoutedEventArgs e)
{
try
{
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:UploadingLog")}...";
await Task.Run(async () => await UploadLog());
System.Diagnostics.Process.Start(LogURL);
Clipboard.SetText(LogURL);
MainWindow.Instance.MainText = (string)Application.Current.FindResource("Options:LogUrlCopied");
}
catch (Exception exception)
{
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:LogUploadFailed")}.";
string title = (string)Application.Current.FindResource("Options:LogUploadFailed:Title");
string body = (string)Application.Current.FindResource("Options:LogUploadFailed:Body");
MessageBox.Show($"{body}\n ================= \n" + exception, title);
Utils.OpenFolder(Path.Combine(InstallDirectory, "Logs"));
}
}
private async Task UploadLog()
{
const string DateFormat = "yyyy-mm-dd HH:mm:ss";
DateTime now = DateTime.Now;
var nvc = new List<KeyValuePair<string, string>>()
{
new KeyValuePair<string, string>("title", $"_latest.log ({now.ToString(DateFormat)})"),
new KeyValuePair<string, string>("expireUnit", "hour"),
new KeyValuePair<string, string>("expireLength", "5"),
new KeyValuePair<string, string>("code", File.ReadAllText(Path.Combine(InstallDirectory, "Logs", "_latest.log"))),
};
var req = new HttpRequestMessage(HttpMethod.Post, Utils.Constants.TeknikAPIUrl + "Paste")
{
Content = new FormUrlEncodedContent(nvc),
};
var resp = await Http.HttpClient.SendAsync(req);
var body = await resp.Content.ReadAsStringAsync();
var TeknikResponse = Http.JsonSerializer.Deserialize<Utils.TeknikPasteResponse>(body);
LogURL = TeknikResponse.result.url;
}
private void OpenAppDataButton_Click(object sender, RoutedEventArgs e)
{
string location = Path.Combine(
Environment.GetFolderPath(Environment.SpecialFolder.UserProfile),
"AppData", "LocalLow", "Hyperbolic Magnetism");
Utils.OpenFolder(location);
}
private async void YeetBSIPAButton_Click(object sender, RoutedEventArgs e)
{
if (Mods.Instance.AllModsList == null)
{
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:GettingModList")}...";
await Task.Run(async () => await Mods.Instance.GetAllMods());
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:FindingBSIPAVersion")}...";
await Task.Run(() => Mods.Instance.GetBSIPAVersion());
}
foreach (Mod mod in Mods.InstalledMods)
{
if (mod.name.ToLower() == "bsipa")
{
Mods.Instance.UninstallMod(mod);
break;
}
}
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:BSIPAUninstalled")}...";
}
private async void YeetModsButton_Click(object sender, RoutedEventArgs e)
{
string title = (string)Application.Current.FindResource("Options:YeetModsBox:Title");
string line1 = (string)Application.Current.FindResource("Options:YeetModsBox:RemoveAllMods");
string line2 = (string)Application.Current.FindResource("Options:YeetModsBox:CannotBeUndone");
var resp = System.Windows.Forms.MessageBox.Show($"{line1}\n{line2}", title, System.Windows.Forms.MessageBoxButtons.YesNo);
if (resp == System.Windows.Forms.DialogResult.Yes)
{
if (Mods.Instance.AllModsList == null)
{
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options: GettingModList")}...";
await Task.Run(async () => await Mods.Instance.CheckInstalledMods());
}
foreach (Mod mod in Mods.InstalledMods)
{
Mods.Instance.UninstallMod(mod);
}
if (Directory.Exists(Path.Combine(App.BeatSaberInstallDirectory, "Plugins")))
Directory.Delete(Path.Combine(App.BeatSaberInstallDirectory, "Plugins"), true);
if (Directory.Exists(Path.Combine(App.BeatSaberInstallDirectory, "Libs")))
Directory.Delete(Path.Combine(App.BeatSaberInstallDirectory, "Libs"), true);
if (Directory.Exists(Path.Combine(App.BeatSaberInstallDirectory, "IPA")))
Directory.Delete(Path.Combine(App.BeatSaberInstallDirectory, "IPA"), true);
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:AllModsUninstalled")}...";
}
}
private void ApplicationThemeComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
if ((sender as ComboBox).SelectedItem == null)
{
Themes.ApplyWindowsTheme();
MainWindow.Instance.MainText = (string)Application.Current.FindResource("Options:CurrentThemeRemoved");
}
else
{
Themes.ApplyTheme((sender as ComboBox).SelectedItem.ToString());
}
}
private void ApplicationThemeExportTemplate_Click(object sender, RoutedEventArgs e)
{
Themes.WriteThemeToDisk("Ugly Kulu-Ya-Ku");
Themes.LoadThemes();
}
private void ApplicationThemeOpenThemesFolder_Click(object sender, RoutedEventArgs e)
{
if (Directory.Exists(Themes.ThemeDirectory))
{
Utils.OpenFolder(Themes.ThemeDirectory);
}
else
{
MessageBox.Show((string)Application.Current.FindResource("Options:ThemeFolderNotFound"));
}
}
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 121 KiB

View file

@ -1,103 +0,0 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<!--
ModAssistant BSMG theem by Megalon2D
https://twitter.com/megalon2D
-->
<!-- Standard Styles -->
<Color x:Key="StandardContent">#DCDDDE</Color>
<Color x:Key="StandardPrimary">#36393F</Color>
<Color x:Key="StandardSecondary">#2F3136</Color>
<Color x:Key="StandardBackground">#202225</Color>
<Color x:Key="StandardBorder">#202225</Color>
<Color x:Key="StandardHighlight">#7289DA</Color>
<Color x:Key="StandardActive">#44347C</Color>
<Color x:Key="StandardIcon">#DCDDDE</Color>
<Color x:Key="DiscordTextBox">#40444B</Color>
<Color x:Key="DiscordHighlight">#32353B</Color>
<!-- Default text -->
<SolidColorBrush x:Key="TextColor" Color="{StaticResource ResourceKey=StandardContent}" />
<SolidColorBrush x:Key="TextHighlighted" Color="Black" />
<!-- Buttons (Info/Mods/About/Options as well as More Info and Install/Update) -->
<SolidColorBrush x:Key="ButtonBackground" Color="#303237" />
<SolidColorBrush x:Key="ButtonOutline" Color="#202225" />
<SolidColorBrush x:Key="ButtonHighlightedBackground" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="ButtonClickedBackground" Color="{DynamicResource ResourceKey=StandardActive}" />
<SolidColorBrush x:Key="ButtonDisabledBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="ButtonDangerBackground" Color="{DynamicResource ResourceKey=StandardActive}" />
<!-- Page Buttons (Side of Main Page) -->
<SolidColorBrush x:Key="PageButtonBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="PageButtonOutline" Color="{DynamicResource ResourceKey=StandardBorder}" />
<SolidColorBrush x:Key="PageButtonHighlightedBackground" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="PageButtonClickedBackground" Color="{DynamicResource ResourceKey=StandardActive}" />
<SolidColorBrush x:Key="PageButtonDisabledBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<!-- Mod List -->
<SolidColorBrush x:Key="ModColumnBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ModColumnBorderBrush" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ModColumnHeaderHighlighted" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="ModListBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ModListItemHighlighted" Color="{DynamicResource ResourceKey=DiscordHighlight}" />
<SolidColorBrush x:Key="ModListItemHighlightedOutline" Color="Transparent" />
<SolidColorBrush x:Key="ModListItemSelected" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="ModListItemSelectedOutline" Color="Transparent" />
<SolidColorBrush x:Key="ModListBorderBrush" Color="Transparent" />
<!-- Combo Box (Version select) -->
<SolidColorBrush x:Key="ComboBoxBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ComboBoxOutline" Color="{DynamicResource ResourceKey=StandardBorder}" />
<SolidColorBrush x:Key="ComboBoxHighlighted" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="ComboBoxSelected" Color="{DynamicResource ResourceKey=StandardActive}" />
<SolidColorBrush x:Key="ComboBoxClicked" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ComboBoxArrow" Color="{DynamicResource ResourceKey=StandardBorder}" />
<!-- Checkboxes (Mod List and Options) -->
<SolidColorBrush x:Key="CheckboxDefaultBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="CheckboxDefaultOutlineColor" Color="{DynamicResource ResourceKey=StandardBorder}" />
<SolidColorBrush x:Key="CheckboxDisabledBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="CheckboxDisabledOutlineColor" Color="{DynamicResource ResourceKey=StandardBorder}" />
<SolidColorBrush x:Key="CheckboxDisabledTickColor" Color="{DynamicResource ResourceKey=StandardBorder}" />
<SolidColorBrush x:Key="CheckboxHoveredBackground" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="CheckboxHoveredTickColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="CheckboxTickColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="CheckboxPressedBackground" Color="{DynamicResource ResourceKey=StandardActive}" />
<!-- Scroll Bars -->
<SolidColorBrush x:Key="ScrollBarBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="ScrollBarBorder" Color="{DynamicResource ResourceKey=Transparent}" />
<SolidColorBrush x:Key="ScrollBarTextColor" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ScrollBarDisabled" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="ScrollBarArrowColor" Color="{DynamicResource ResourceKey=StandardActive}" />
<SolidColorBrush x:Key="ScrollBarArrowClicked" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="ScrollBarArrowHovered" Color="{DynamicResource ResourceKey=StandardHighlight}" />
<SolidColorBrush x:Key="ScrollBarHandle" Color="{DynamicResource ResourceKey=StandardBackground}" />
<SolidColorBrush x:Key="ScrollBarHandleHovered" Color="{DynamicResource ResourceKey=StandardBackground}" />
<SolidColorBrush x:Key="ScrollBarHandleClick" Color="{DynamicResource ResourceKey=StandardBackground}" />
<GridLength x:Key="ScrollBarButtonHeight">0</GridLength>
<!-- Various important elements that need to be controlled independently -->
<SolidColorBrush x:Key="ModAssistantBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="FrameBackgroundColor" Color="#00000000" />
<SolidColorBrush x:Key="BottomStatusBarBackground" Color="{DynamicResource ResourceKey=DiscordTextBox}" />
<SolidColorBrush x:Key="BottomStatusBarOutline" Color="{DynamicResource ResourceKey=DiscordTextBox}" />
<SolidColorBrush x:Key="DirectoryBackground" Color="{DynamicResource ResourceKey=DiscordTextBox}" />
<SolidColorBrush x:Key="DirectoryOutline" Color="{DynamicResource ResourceKey=DiscordTextBox}" />
<!-- Colors for the corresponding icons. -->
<SolidColorBrush x:Key="InfoIconColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="ModsIconColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="AboutIconColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="OptionsIconColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="LoadingIconColor" Color="{DynamicResource ResourceKey=StandardIcon}" />
<!-- Background and Side image settings. -->
<!-- Fill, None, Uniform, UniformToFill -->
<Stretch x:Key="BackgroundImageStretch">UniformToFill</Stretch>
<!-- Bottom, Center, Top -->
<VerticalAlignment x:Key="SideImageYPosition">Top</VerticalAlignment>
</ResourceDictionary>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6 KiB

137
README.md
View file

@ -1,138 +1,51 @@
[![ModAssistant](https://cdn.assistant.moe/images/ModAssistant/Icons/Banner.svg)](https://github.com/Assistant/ModAssistant/releases/latest)
[![Download here!](https://cdn.assistant.moe/images/ModAssistant/Icons/Download.svg)](https://github.com/Assistant/ModAssistant/releases/latest)
# [Download here!](https://github.com/knah/VRCMelonAssistant/releases/latest)
ModAssistant is a PC mod installer for Beat Saber. It uses mods from [BeatMods](https://beatmods.com/).
VRChat Melon Assistant is a PC mod installer for VRChat. It uses mods published in [VRChat Modding Group Discord](https://discord.gg/rCqKSvR).
It's a (very stripped down) fork of [Assistant's Mod Assistant](https://github.com/Assistant/ModAssistant), a mod manager for Beat Saber.
**Modifying the VRChat client is not allowed by VRChat Terms of Service and can lead to your account being banned.** Mods available via this installer are manually checked to minimize the chance of that happening, but the risk is always there.
VRChat Melon Assistant is not affiliated with and/or endorsed by VRChat Inc.
* [Features](#Features)
* [Usage](#Usage)
* [Themes](#Themes)
* [Custom Themes](#Custom-Themes)
* [Built In](#Built-In)
* [Packaged `.mat` Files](#Packaged-mat-Files)
* [Loose Folder Themes](#Loose-Folder-Themes)
* [Overriding Themes](#Overriding-Themes)
* [Common Issues](#Common-Issues)
## Features
ModAssistant boasts a rich feature set, some of which include:
* Dependency resolution
VRChat Melon Assistant boasts a rich feature set, some of which include:
* Installed mod detection
* Mod uninstallation
* OneClick&trade; Install support
* Broken mod move-aside (temporarily uninstalls them until a fix is available)
* Complex theming engine
* Localization support
* Headpats and Hugs
## Usage
Download the newest installer from the release section and run it. This application auto-updates when launched, there is no need to download a new release each time.
1. **Run the game at least once before trying to mod the game!** This applies to reinstalling your game too. All mods are moved into an `Old X.X.X Plugins` folder on first launch to avoid version mismatches, so make sure to do this before installing mods on a fresh version.
2. Once that's done, simply check off the mods that you wish to install and click the <kbd>Install or Update</kbd> button. Likewise, click the <kbd>Uninstall</kbd> button to remove any mods.
3. Mods are installed to `IPA/Pending` until the game is run. Boot the game to complete mod installation.
Download the newest build from the release section and run it. This application auto-updates when launched, there is no need to download a new release each time.
Then, simply check off the mods that you wish to install or update and click the <kbd>Install or Update</kbd> button. Likewise, click the <kbd>Uninstall</kbd> button to remove any mods.
## Themes
<details>
<summary><b>Light</b></summary>
<div>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light/Intro.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light/Mods.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light/About.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light/Options.png" /></p>
</div>
</details>
<details>
<summary><b>Dark</b></summary>
<div>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Dark/Intro.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Dark/Mods.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Dark/About.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Dark/Options.png" /></p>
</div>
</details>
<details>
<summary><b>BSMG</b></summary>
<div>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/BSMG/Intro.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/BSMG/Mods.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/BSMG/About.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/BSMG/Options.png" /></p>
</div>
</details>
<details>
<summary><b>Light Pink</b></summary>
<div>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light Pink/Intro.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light Pink/Mods.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light Pink/About.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Light Pink/Options.png" /></p>
</div>
</details>
<details>
<summary><b>Your own!</b></summary>
<div>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Custom/Intro.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Custom/Mods.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Custom/About.png" /></p>
<p><img src="https://cdn.assistant.moe/images/ModAssistant/Themes/Custom/Options.png" /></p>
</div>
</details>
### Custom Themes
Custom themes are located in a folder called `Themes` in the same folder as `ModAssistant.exe`. ModAssistant can load themes from one of three sources.
### Built In
These come with the program and you can't change them, however you can overwrite them by creating one of the other two theme types with the same name.
If you have a particularly popular theme, you can submit a [Pull Request](https://github.com/Assistant/ModAssistant/pulls) to add your theme as a built-in theme.
### Packaged `.mat` Files
These are pre-packaged theme files. Under the hood they are renamed`.zip` files, and the file structure is the same as that of `Folders` themes. These will be overwritten by `Folders` themes with the same name.
To create one follow the instructions on `Folders` themes, and zip the files up into a zip archive, and rename it to `<themeName>.mat`.
### Loose Folder Themes
These will overwrite all other themes, and are loaded from a folder named after the theme. There are 4 types of files you can include:
* `Theme.xaml`
* This file determines the colors and styling of the theme.
* The filename isn't important, but the `.xaml` file extension is.
* To see an example of this file press the <kbd>Export Template</kbd> button in the `Options` page. It will create a folder in `Themes` called `Ugly Kulu-Ya-Ku`. You can open that file to use as a template for your own themes, or just use it.
* `Waifu.png`
* This will be loaded as the background image.
* It will be centered, and you can pick how to stretch it in the associated `.xaml` file.
* The filename isn't important, but the `.png` file extension is.
* `Waifu.side.png`
* This will be loaded as the left side image.<br />It will be left aligned, and you can pick its vertical alignment in the associated `.xaml` file.
* The filename isn't important, but the `.side.png` file extension is.
* `Video.{mp4, webm, mkv, avi, m2ts}`
* This will be loaded as a background video, with sound.
* The filename isn't important, but the file extension must be supported (`.mp4`, `.webm`, `.mkv`, `.avi`, `.m2ts`)
* Whether the file works or not will depend on what codecs the file has, and whether those are available on your machine.
### Overriding Themes
You can mix and match parts from different themes by giving them the same name.
The priority in which they will be used is `Loose Folder Themes` > `Packaged .mat files` > `Built in`. Overriding themes will only change the files that are included.
Examples:
* Adding `/Themes/Dark.mat` which includes `.png` and `.xaml` files will override both those aspects of the `Dark` theme.
* Adding `/Themes/Dark/image.png` will use that image as the background for the `Dark` theme, overriding both the built in theme and `Dark.mat` if it exists.
VRChat Melon Assistant should support themes for Mod Assistant. Check [its README](https://github.com/Assistant/ModAssistant#themes) for more info on theming.
However, this is not a supported feature. If something doesn't work, don't complain to either Assistant or knah.
## Common Issues
**I hit install but I don't see anything in game!**
Double check that you followed the [Usage](#usage) instructions correctly.
Make sure you're looking in the right place. Sometimes mod menus move as modding libraries/practices change.
Double check that you followed the [Usage](#usage) instructions correctly.
Make sure you're looking in the right place. Sometimes mod menus move as modding libraries/practices change.
Additionally, make sure the proper VRChat installation directory is selected in option tab.
**I don't see a certain mod in the mods list!**
Mod Assistant uses mods from [BeatMods](https://beatmods.com/) and shows the whatever is available there. If you need to install a mod manually, please refer to the [Beat Saber Modding Group Wiki](https://bsmg.wiki/pc-modding.html#manual-installation).
VRChat Melon Assistant uses mods from VRChat Modding Group and shows whatever is available for download. It's recommended to avoid non-VRCMG mods due to rampant spread of malware disguised as mods.
**I hit install but now my game won't launch, I can't click any buttons, I only see a black screen, etc**
Please visit the [Beat Saber Modding Group](https://discord.gg/beatsabermods) `#pc-help` channels. Check the pinned messages or ask for help and see if you can work out things out.
Please visit the [VRChat Modding Group](https://discord.gg/rCqKSvR) `#help-and-support` channel. Check the pinned messages or ask for help and see if you can work out things out.
## Credits
Lemon icon from Twitter Emoji
https://github.com/twitter/twemoji
Original Mod Assistant is made by Assistant and used under the terms of MIT License.
https://github.com/Assistant/ModAssistant
semver by Max Hauser
https://github.com/maxhauser/semver

View file

@ -3,7 +3,14 @@ Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio 15
VisualStudioVersion = 15.0.27130.2027
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ModAssistant", "ModAssistant\ModAssistant.csproj", "{6A224B82-40DA-40B3-94DC-EFBEC2BDDA39}"
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "VRCMelonAssistant", "VRCMelonAssistant\VRCMelonAssistant.csproj", "{6A224B82-40DA-40B3-94DC-EFBEC2BDDA39}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{8DBEE4E3-1033-433D-B1F3-F7D789808141}"
ProjectSection(SolutionItems) = preProject
README.md = README.md
.github\workflows\dotnet.yml = .github\workflows\dotnet.yml
.github\FUNDING.yml = .github\FUNDING.yml
EndProjectSection
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution

View file

@ -0,0 +1,39 @@
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<configSections>
<sectionGroup name="userSettings" type="System.Configuration.UserSettingsGroup, System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
<section name="VRCMelonAssistant.Properties.Settings" type="System.Configuration.ClientSettingsSection, System, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" allowExeDefinition="MachineToLocalUser" requirePermission="false" />
</sectionGroup>
</configSections>
<startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.6.1" />
</startup>
<userSettings>
<VRCMelonAssistant.Properties.Settings>
<setting name="InstallFolder" serializeAs="String">
<value />
</setting>
<setting name="StoreType" serializeAs="String">
<value />
</setting>
<setting name="Agreed" serializeAs="String">
<value>False</value>
</setting>
<setting name="UpgradeRequired" serializeAs="String">
<value>True</value>
</setting>
<setting name="LastTab" serializeAs="String">
<value />
</setting>
<setting name="SelectedTheme" serializeAs="String">
<value />
</setting>
<setting name="CloseWindowOnFinish" serializeAs="String">
<value>False</value>
</setting>
<setting name="LanguageCode" serializeAs="String">
<value />
</setting>
</VRCMelonAssistant.Properties.Settings>
</userSettings>
</configuration>

View file

@ -1,5 +1,5 @@
<Application
x:Class="ModAssistant.App"
x:Class="VRCMelonAssistant.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
DispatcherUnhandledException="Application_DispatcherUnhandledException"
@ -21,6 +21,8 @@
<!-- Load SVG icons, courtesy of lolPants. -->
<ResourceDictionary x:Name="Icons" Source="Resources/Icons.xaml" />
<ResourceDictionary x:Name="AppResources" Source="Resources/AppResources.xaml" />
<!-- Load Styles needed for the Theme engine to work. -->
<ResourceDictionary x:Name="Button_Style" Source="Styles/Button.xaml" />
<ResourceDictionary x:Name="Label_Style" Source="Styles/Label.xaml" />
@ -28,6 +30,8 @@
<ResourceDictionary x:Name="GridViewColumnHeader_Style" Source="Styles/GridViewColumnHeader.xaml" />
<ResourceDictionary x:Name="ListView_Style" Source="Styles/ListView.xaml" />
<ResourceDictionary x:Name="ListViewItem_Style" Source="Styles/ListViewItem.xaml" />
<ResourceDictionary x:Name="Menu_Style" Source="Styles/Menu.xaml" />
<ResourceDictionary x:Name="MenuItem_Style" Source="Styles/MenuItem.xaml" />
<ResourceDictionary x:Name="CheckBox_Style" Source="Styles/CheckBox.xaml" />
<ResourceDictionary x:Name="ComboBoxItem_Style" Source="Styles/ComboBoxItem.xaml" />
<ResourceDictionary x:Name="ComboBox_Style" Source="Styles/ComboBox.xaml" />

View file

@ -0,0 +1,198 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Reflection;
using System.Threading.Tasks;
using System.Windows;
namespace VRCMelonAssistant
{
/// <summary>
/// Interaction logic for App.xaml
/// </summary>
public partial class App : Application
{
public static string VRChatInstallDirectory;
public static string VRChatInstallType;
public static bool CloseWindowOnFinish;
public static string Version = Assembly.GetExecutingAssembly().GetName().Version.ToString();
public static MainWindow window;
public static string Arguments;
public static bool Update = true;
public static bool GUI = true;
private async void Application_Startup(object sender, StartupEventArgs e)
{
// Set SecurityProtocol to prevent crash with TLS
ServicePointManager.SecurityProtocol |= SecurityProtocolType.Tls12;
if (VRCMelonAssistant.Properties.Settings.Default.UpgradeRequired)
{
VRCMelonAssistant.Properties.Settings.Default.Upgrade();
VRCMelonAssistant.Properties.Settings.Default.UpgradeRequired = false;
VRCMelonAssistant.Properties.Settings.Default.Save();
}
Pages.Options options = Pages.Options.Instance;
options.InstallDirectory =
VRChatInstallDirectory = Utils.GetInstallDir();
Languages.LoadLanguages();
while (string.IsNullOrEmpty(VRChatInstallDirectory))
{
string title = (string)Current.FindResource("App:InstallDirDialog:Title");
string body = (string)Current.FindResource("App:InstallDirDialog:OkCancel");
if (System.Windows.Forms.MessageBox.Show(body, title, System.Windows.Forms.MessageBoxButtons.OKCancel) == System.Windows.Forms.DialogResult.OK)
{
VRChatInstallDirectory = Utils.GetManualDir();
}
else
{
Environment.Exit(0);
}
}
options.InstallType =
VRChatInstallType = VRCMelonAssistant.Properties.Settings.Default.StoreType;
options.CloseWindowOnFinish =
CloseWindowOnFinish = VRCMelonAssistant.Properties.Settings.Default.CloseWindowOnFinish;
await ArgumentHandler(e.Args);
await Init();
}
private async Task Init()
{
if (Update)
{
try
{
await Task.Run(async () => await Updater.Run());
}
catch (UnauthorizedAccessException)
{
Utils.StartAsAdmin(Arguments, true);
}
}
if (GUI)
{
window = new MainWindow();
window.Show();
}
else
{
//Application.Current.Shutdown();
}
}
private async Task ArgumentHandler(string[] args)
{
Arguments = string.Join(" ", args);
while (args.Length > 0)
{
switch (args[0])
{
case "--install":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--install"));
}
if (CloseWindowOnFinish)
{
await Task.Delay(5 * 1000);
Current.Shutdown();
}
Update = false;
GUI = false;
args = Shift(args, 2);
break;
case "--no-update":
Update = false;
args = Shift(args);
break;
case "--language":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--language"));
}
else
{
if (Languages.LoadLanguage(args[1]))
{
VRCMelonAssistant.Properties.Settings.Default.LanguageCode = args[1];
VRCMelonAssistant.Properties.Settings.Default.Save();
Languages.UpdateUI(args[1]);
}
}
args = Shift(args, 2);
break;
case "--register":
if (args.Length < 3 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--register"));
}
Update = false;
GUI = false;
args = Shift(args, 3);
break;
case "--unregister":
if (args.Length < 2 || string.IsNullOrEmpty(args[1]))
{
Utils.SendNotify(string.Format((string)Current.FindResource("App:InvalidArgument"), "--unregister"));
}
Update = false;
GUI = false;
args = Shift(args, 2);
break;
case "--runforever":
while (true)
{
}
default:
Utils.SendNotify((string)Current.FindResource("App:UnrecognizedArgument"));
args = Shift(args);
break;
}
}
}
private static string[] Shift(string[] array, int places = 1)
{
if (places >= array.Length) return Array.Empty<string>();
string[] newArray = new string[array.Length - places];
for (int i = places; i < array.Length; i++)
{
newArray[i - places] = array[i];
}
return newArray;
}
private void Application_DispatcherUnhandledException(object sender, System.Windows.Threading.DispatcherUnhandledExceptionEventArgs e)
{
string title = (string)Current.FindResource("App:Exception");
string body = (string)Current.FindResource("App:UnhandledException");
MessageBox.Show($"{body}: {e.Exception}", title, MessageBoxButton.OK, MessageBoxImage.Warning);
e.Handled = true;
Current.Shutdown();
}
}
}

View file

@ -1,9 +1,8 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Windows;
namespace ModAssistant
namespace VRCMelonAssistant
{
class Diagnostics
{
@ -37,11 +36,11 @@ namespace ModAssistant
{
MessageBox.Show("! " + file);
}
}
if (entries.Count > 0)
{
entries[entries.Count - 1] = entries[entries.Count - 1].Replace("├", "└");
}
return entries.ToArray();
}

View file

@ -0,0 +1,125 @@
using System.Collections.Generic;
namespace VRCMelonAssistant
{
public static class HardcodedCategories
{
private static readonly Dictionary<string, List<string>> CategoryContents = new()
{
{"Safety & Security", new() {"Advanced Safety", "Finitizer", "True Shader Anticrash", "Safety-Presets", "Final IK Sanity"}},
{"Core mods and libraries", new() {"UI Expansion Kit", "ActionMenuApi", "VRCModUpdater.Loader", "VRChatUtilityKit"}},
{"All-in-one mods", new() {"emmVRCLoader"}},
{"Camera mods", new() {
"CameraMinus", "DesktopCamera", "BetterSteadycam", "ITR's Melon Cameras", "CameraResChanger",
"LocalCameraMod", "Lag Free Screenshots"
}},
{"Performance & Fidelity", new() {
"Core Limiter", "MirrorResolutionUnlimiter", "AvatarHider", "Runtime Graphics Settings",
"GamePriority", "FrameFocus", "ClearVRAM", "NoPerformanceStats", "Turbones"
}},
{"Utilities & Tweaks", new() {
"ReloadAvatars", "KeyboardPaste", "No Outlines", "UnmuteSound", "SparkleBeGone",
"BTKSAImmersiveHud", "OGTrustRanks", "ToggleMicIcon", "Friends+ home",
"MicSensitivity", "CloningBeGone", "ToggleFullScreen", "View Point Tweaker",
"SettingsRestart", "SmallUserVolume", "TeleportCameraToYou",
"BetterPortalPlacement", "BetterDirections", "ChairExitController",
"Panic Button Rework", "SelectYourself", "Trust Color Changer", "Voice Falloff Override"
}},
{"Hardware support", new() {"LeapMotionExtension", "ThumbParams", "VRCFaceTracking", "VRCPimaxEyeTracker", "VRCBhapticsIntegration"}},
{"Dynamic bones", new() {
"ImmersiveTouch", "Dynamic Bones Safety", "MultiplayerDynamicBonesMod", "Multiplayer Dynamic Bones",
}},
{"World tweaks", new() {
"PostProcessing", "NearClipPlaneAdj", "RemoveChairs", "ComponentToggle", "No Grabby Hands", "AOOverride"
}},
{"Fixes", new() {"Invite+ fix", "CursorLockFix", "DownloadFix", "ProneUiFix"}},
{"New features & Overhauls", new() {
"IKTweaks", "JoinNotifier", "FBT Saver", "BTKSANameplateMod", "AdvancedInvites", "VRCVideoLibrary",
"BTKSASelfPortrait", "OldMate", "BetterLoadingScreen", "Loading Screen Pictures", "FavCat",
"ActionMenuUtils", "WorldPredownload", "AskToPortal", "Headlight", "ITR's Player Tracer",
"InstanceHistory", "PortableMirrorMod", "VRCBonesController", "CalibrationLinesVisualizer",
"ITR's Collider Mod", "RememberMe", "TriggerESP"
}},
{"UI mods", new() {
"Particle and DynBone limiter settings UI", "CalibrateConfirm", "Emoji Page Buttons",
"UserInfoExtensions", "MLConsoleViewer", "OwO Mod", "ActiveBackground", "PlayerList", "ComfyVRMenu",
"DiscordMute", "MicToggle", "VRCPlusPet", "AMMusic", "Friend Notes", "GestureIndicator",
"NameplateStats", "PreviewScroller", "ProPlates", "QuickMenuVolume", "ToggleUIStickers"
}},
{"Movement", new() {
"TeleporterVR", "ImmobilizePlayerMod", "TrackingRotator", "OculusPlayspaceMover",
"ITR's Gravity Controller", "QMFreeze", "Double-Tap Runner", "Player Rotater",
"HeadTurn"
}},
{"Very Niche Mods", new() {"HWIDPatch", "No Steam. At all.", "Vertex Animation Remover", "LocalPlayerPrefs", "BTKSAGestureMod",}}
};
private static readonly Dictionary<string, string> CategoryDescriptions = new()
{
{"Safety & Security", "Crash less, block annoyances"},
{"Core mods and libraries", "Other mods might require these"},
{"All-in-one mods", "It does a lot of stuff"},
{"Camera mods", "For all your screenshot or streaming needs"},
{"Performance & Fidelity", "Improve performance or make the game look better"},
{"Utilities & Tweaks", "Small mods that address specific issues"},
{"Hardware support", "For all exotic hardware out there"},
{"Dynamic bones", "Mods that affect jiggly bits"},
{"World tweaks", "Change aspects of the world you're in"},
{"Fixes", "It's not a bug, it's a feature"},
{"New features & Overhauls", "Mods that introduce new features or significantly change existing ones"},
{"UI mods", "Modify the user interface or introduce new functionality to it"},
{"Movement", "Move in new exciting ways"},
{"Very Niche Mods", "Only use these if you're really sure you need them"}
};
private static readonly Dictionary<string, string> ModNameToCategory = new();
static HardcodedCategories()
{
foreach (var keyValuePair in CategoryContents)
foreach (var s in keyValuePair.Value)
ModNameToCategory.Add(s.ToLowerInvariant(), keyValuePair.Key);
}
public static string GetCategoryFor(Mod mod)
{
foreach (var alias in mod.aliases)
{
if (ModNameToCategory.TryGetValue(alias.ToLowerInvariant(), out var result)) return result;
}
foreach (var version in mod.versions)
{
if (ModNameToCategory.TryGetValue(version.name.ToLowerInvariant(), out var result)) return result;
}
return null;
}
public static string GetCategoryDescription(string category)
{
return CategoryDescriptions.TryGetValue(category, out var result) ? result : "";
}
private static List<(string Original, string Replace)> ourAuthorReplaces =
new()
{
("<@!170953680718266369>", "ImTiara"),
("<@!286669951987613706>", "Rafa"),
("<@!168795588366696450>", "Grummus"),
("<@!167335587488071682>", "KortyBoi/Lily"),
("<@!127978642981650432>", "tetra"),
("<@!155396491853168640>", "Dawn/arion")
};
public static string FixupAuthor(string authorName)
{
if (string.IsNullOrEmpty(authorName) || !authorName.Contains("@")) return authorName;
foreach (var authorReplace in ourAuthorReplaces)
authorName = authorName.Replace(authorReplace.Original, authorReplace.Replace);
return authorName;
}
}
}

View file

@ -3,7 +3,7 @@ using System.Net;
using System.Net.Http;
using System.Web.Script.Serialization;
namespace ModAssistant
namespace VRCMelonAssistant
{
static class Http
{
@ -22,11 +22,11 @@ namespace ModAssistant
_client = new HttpClient(handler)
{
Timeout = TimeSpan.FromSeconds(30),
Timeout = TimeSpan.FromSeconds(240),
};
ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls12;
_client.DefaultRequestHeaders.Add("User-Agent", "ModAssistant/" + App.Version);
_client.DefaultRequestHeaders.Add("User-Agent", "VRCMelonAssistant/" + App.Version);
return _client;
}

View file

@ -1,9 +1,9 @@
using System;
using System.Diagnostics;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Documents;
namespace ModAssistant
namespace VRCMelonAssistant
{
public static class HyperlinkExtensions
{
@ -29,10 +29,21 @@ namespace ModAssistant
hyperlink.RequestNavigate -= Hyperlink_RequestNavigate;
}
private static void Hyperlink_RequestNavigate(object sender, System.Windows.Navigation.RequestNavigateEventArgs e)
public static void Hyperlink_RequestNavigate(object sender, System.Windows.Navigation.RequestNavigateEventArgs e)
{
Process.Start(new ProcessStartInfo(e.Uri.AbsoluteUri));
e.Handled = true;
}
public static void NoAwait(this Task task)
{
task.ContinueWith(t =>
{
if (t.IsFaulted)
{
Utils.ShowErrorMessageBox("Exception in free-floating task", t.Exception);
}
});
}
}
}

View file

@ -0,0 +1,118 @@
using System;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Threading.Tasks;
using System.Windows;
using VRCMelonAssistant.Pages;
namespace VRCMelonAssistant
{
public static class InstallHandlers
{
public static bool IsMelonLoaderInstalled()
{
return File.Exists(Path.Combine(App.VRChatInstallDirectory, "version.dll")) && File.Exists(Path.Combine(App.VRChatInstallDirectory, "MelonLoader", "Dependencies", "Bootstrap.dll"));
}
public static bool RemoveMelonLoader()
{
MainWindow.Instance.MainText = $"{(string)App.Current.FindResource("Mods:UnInstallingMelonLoader")}...";
try
{
var versionDllPath = Path.Combine(App.VRChatInstallDirectory, "version.dll");
var melonLoaderDirPath = Path.Combine(App.VRChatInstallDirectory, "MelonLoader");
if (File.Exists(versionDllPath))
File.Delete(versionDllPath);
if (Directory.Exists(melonLoaderDirPath))
Directory.Delete(melonLoaderDirPath, true);
}
catch (Exception ex)
{
MessageBox.Show($"{App.Current.FindResource("Mods:UninstallMLFailed")}.\n\n" + ex);
return false;
}
return true;
}
public static async Task InstallMelonLoader()
{
if (!RemoveMelonLoader()) return;
try
{
MainWindow.Instance.MainText = $"{(string) App.Current.FindResource("Mods:DownloadingMelonLoader")}...";
using var installerZip = await DownloadFileToMemory("https://github.com/LavaGang/MelonLoader/releases/latest/download/MelonLoader.x64.zip");
using var zipReader = new ZipArchive(installerZip, ZipArchiveMode.Read);
MainWindow.Instance.MainText = $"{(string) App.Current.FindResource("Mods:UnpackingMelonLoader")}...";
foreach (var zipArchiveEntry in zipReader.Entries)
{
var targetFileName = Path.Combine(App.VRChatInstallDirectory, zipArchiveEntry.FullName);
Directory.CreateDirectory(Path.GetDirectoryName(targetFileName));
using var targetFile = File.OpenWrite(targetFileName);
using var entryStream = zipArchiveEntry.Open();
await entryStream.CopyToAsync(targetFile);
}
Directory.CreateDirectory(Path.Combine(App.VRChatInstallDirectory, "Mods"));
Directory.CreateDirectory(Path.Combine(App.VRChatInstallDirectory, "Plugins"));
}
catch (Exception ex)
{
MessageBox.Show($"{App.Current.FindResource("Mods:InstallMLFailed")}.\n\n" + ex);
}
}
internal static async Task<Stream> DownloadFileToMemory(string link)
{
using var resp = await Http.HttpClient.GetAsync(link);
var newStream = new MemoryStream();
await resp.Content.CopyToAsync(newStream);
newStream.Position = 0;
return newStream;
}
public static async Task InstallMod(Mod mod)
{
string downloadLink = mod.versions[0].downloadLink;
if (string.IsNullOrEmpty(downloadLink))
{
MessageBox.Show(string.Format((string)App.Current.FindResource("Mods:ModDownloadLinkMissing"), mod.versions[0].name));
return;
}
if (mod.installedFilePath != null)
File.Delete(mod.installedFilePath);
string targetFilePath = "";
using (var resp = await Http.HttpClient.GetAsync(downloadLink))
{
var stream = new MemoryStream();
await resp.Content.CopyToAsync(stream);
stream.Position = 0;
targetFilePath = Path.Combine(App.VRChatInstallDirectory, mod.versions[0].IsPlugin ? "Plugins" : "Mods",
mod.versions[0].IsBroken ? "Broken" : (mod.versions[0].IsRetired ? "Retired" : ""), resp.RequestMessage.RequestUri.Segments.Last());
Directory.CreateDirectory(Path.GetDirectoryName(targetFilePath));
using var targetFile = File.OpenWrite(targetFilePath);
await stream.CopyToAsync(targetFile);
}
mod.ListItem.IsInstalled = true;
mod.installedFilePath = targetFilePath;
mod.ListItem.InstalledVersion = mod.versions[0].modVersion;
mod.ListItem.InstalledModInfo = mod;
}
}
}

View file

@ -0,0 +1,74 @@
using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Linq;
using System.Windows;
using VRCMelonAssistant.Pages;
namespace VRCMelonAssistant
{
class Languages
{
public static string LoadedLanguage { get; private set; }
public static List<CultureInfo> LoadedLanguages { get => availableCultures.ToList(); }
public static bool FirstRun = true;
private static readonly string[] availableLanguageCodes = { "en" };
private static IEnumerable<CultureInfo> availableCultures;
public static void LoadLanguages()
{
var allCultures = CultureInfo.GetCultures(CultureTypes.AllCultures);
// Get CultureInfo for any of the available translations
availableCultures = allCultures.Where(cultureInfo => availableLanguageCodes.Any(code => code.Equals(cultureInfo.Name)));
string savedLanguageCode = Properties.Settings.Default.LanguageCode;
if (!LoadLanguage(savedLanguageCode))
{
// If no language code was saved, load system language
if (!LoadLanguage(CultureInfo.CurrentUICulture.Name))
{
LoadLanguage("en");
}
}
UpdateUI(LoadedLanguage);
}
public static void UpdateUI(string languageCode)
{
if (Options.Instance != null && Options.Instance.LanguageSelectComboBox != null)
{
Options.Instance.LanguageSelectComboBox.ItemsSource = availableCultures.Select(cultureInfo => cultureInfo.NativeName).ToList();
Options.Instance.LanguageSelectComboBox.SelectedIndex = LoadedLanguages.FindIndex(cultureInfo => cultureInfo.Name.Equals(languageCode));
}
}
public static ResourceDictionary LanguagesDict
{
get
{
return Application.Current.Resources.MergedDictionaries[1];
}
}
public static bool LoadLanguage(string languageCode)
{
if (string.IsNullOrEmpty(languageCode)) return false;
try
{
LanguagesDict.Source = new Uri($"Localisation/{languageCode}.xaml", UriKind.Relative);
LoadedLanguage = languageCode;
return true;
}
catch (IOException)
{
if (languageCode.Contains("-"))
{
return LoadLanguage(languageCode.Split('-').First());
}
return false;
}
}
}
}

View file

@ -0,0 +1,41 @@
using System;
using System.Collections.Generic;
using VRCMelonAssistant.Pages;
namespace VRCMelonAssistant
{
public class Mod
{
public int _id;
public string uploadDate;
public string category;
public string[] aliases;
public ModVersion[] versions;
public Mods.ModListItem ListItem;
public string installedFilePath;
public string installedVersion;
public bool installedInBrokenDir;
public bool installedInRetiredDir;
public class ModVersion
{
public int _version;
public string name;
public string modVersion;
public string modType;
public string author;
public string description;
public string downloadLink;
public string sourceLink;
public string hash;
public string updateDate;
public string vrchatVersion;
public string loaderVersion;
public int approvalStatus;
public bool IsBroken => approvalStatus == 2;
public bool IsRetired => approvalStatus == 3;
public bool IsPlugin => modType.Equals("plugin", StringComparison.InvariantCultureIgnoreCase);
}
}
}

View file

@ -1,6 +1,4 @@
using System;
namespace ModAssistant
namespace VRCMelonAssistant
{
class Promotions
{
@ -8,9 +6,9 @@ namespace ModAssistant
{
new Promotion
{
ModName = "YUR Fit Calorie Tracker",
ModName = "emmVRCLoader",
Text = "Join our Discord!",
Link = "https://yur.chat"
Link = "https://discord.gg/emmvrc"
}
};
}

View file

@ -1,17 +1,17 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Windows;
using System.IO;
using System.Windows.Media;
using ModAssistant.Pages;
using System.Reflection;
using Microsoft.Win32;
using System.Windows.Media.Imaging;
using System.IO.Compression;
using System.Linq;
using System.Reflection;
using System.Windows;
using System.Windows.Markup;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using Microsoft.Win32;
using VRCMelonAssistant.Pages;
namespace ModAssistant
namespace VRCMelonAssistant
{
public class Themes
{
@ -22,15 +22,15 @@ namespace ModAssistant
/// <summary>
/// Local dictionary of Resource Dictionaries mapped by their names.
/// </summary>
private static Dictionary<string, Theme> loadedThemes = new Dictionary<string, Theme>();
private static List<string> preInstalledThemes = new List<string> { "Light", "Dark", "BSMG", "Light Pink" };
private static readonly Dictionary<string, Theme> loadedThemes = new Dictionary<string, Theme>();
private static readonly List<string> preInstalledThemes = new List<string> { "Light", "Dark", "Light Pink" };
/// <summary>
/// Index of "LoadedTheme" in App.xaml
/// </summary>
private static readonly int LOADED_THEME_INDEX = 3;
private static List<string> supportedVideoExtensions = new List<string>() { ".mp4", ".webm", ".mkv", ".avi", ".m2ts" };
private static readonly List<string> supportedVideoExtensions = new List<string>() { ".mp4", ".webm", ".mkv", ".avi", ".m2ts" };
/// <summary>
/// Load all themes from local Themes subfolder and from embedded resources.
@ -49,20 +49,27 @@ namespace ModAssistant
string location = $"Themes/{localTheme}.xaml";
Uri local = new Uri(location, UriKind.Relative);
ResourceDictionary localDictionary = new ResourceDictionary();
localDictionary.Source = local;
ResourceDictionary localDictionary = new ResourceDictionary
{
Source = local
};
/*
* Load any Waifus that come with these built-in themes, too.
* The format must be: Background.png and Sidebar.png as a subfolder with the same name as the theme name.
* For example: "Themes/Dark/Background.png", or "Themes/Ugly Kulu-Ya-Ku/Sidebar.png"
*/
Waifus waifus = new Waifus();
waifus.Background = GetImageFromEmbeddedResources(localTheme, "Background");
waifus.Sidebar = GetImageFromEmbeddedResources(localTheme, "Sidebar");
Waifus waifus = new Waifus
{
Background = GetImageFromEmbeddedResources(localTheme, "Background"),
Sidebar = GetImageFromEmbeddedResources(localTheme, "Sidebar")
};
Theme theme = new Theme(localTheme, localDictionary)
{
Waifus = waifus
};
Theme theme = new Theme(localTheme, localDictionary);
theme.Waifus = waifus;
loadedThemes.Add(localTheme, theme);
}
@ -110,23 +117,30 @@ namespace ModAssistant
{
if (string.IsNullOrEmpty(savedTheme))
{
Themes.ApplyWindowsTheme();
try
{
ApplyWindowsTheme();
}
catch
{
ApplyTheme("Light", false);
}
return;
}
try
{
Themes.ApplyTheme(savedTheme, false);
ApplyTheme(savedTheme, false);
}
catch (ArgumentException)
{
Themes.ApplyWindowsTheme();
ApplyWindowsTheme();
MainWindow.Instance.MainText = (string)Application.Current.FindResource("Themes:ThemeNotFound");
}
}
/// <summary>
/// Applies a loaded theme to ModAssistant.
/// Applies a loaded theme to VRCMelonAssistant.
/// </summary>
/// <param name="theme">Name of the theme.</param>
/// <param name="sendMessage">Send message to MainText (default: true).</param>
@ -195,7 +209,7 @@ namespace ModAssistant
* Writing it as is instead of using XAMLWriter keeps the source as is with comments, spacing, and organization.
* Using XAMLWriter would compress it into an unorganized mess.
*/
using (Stream s = Assembly.GetExecutingAssembly().GetManifestResourceStream($"ModAssistant.Themes.{themeName}.xaml"))
using (Stream s = Assembly.GetExecutingAssembly().GetManifestResourceStream($"VRCMelonAssistant.Themes.{themeName}.xaml"))
using (FileStream writer = new FileStream($@"{ThemeDirectory}\\{themeName}\\{themeName}.xaml", FileMode.Create))
{
byte[] buffer = new byte[s.Length];
@ -243,8 +257,10 @@ namespace ModAssistant
/// <returns></returns>
private static Theme LoadTheme(string directory, string name)
{
Theme theme = new Theme(name, null);
theme.Waifus = new Waifus();
Theme theme = new Theme(name, null)
{
Waifus = new Waifus()
};
foreach (string file in Directory.EnumerateFiles(directory).OrderBy(x => x))
{
@ -268,8 +284,11 @@ namespace ModAssistant
try
{
Uri resourceSource = new Uri(info.FullName);
ResourceDictionary dictionary = new ResourceDictionary();
dictionary.Source = resourceSource;
ResourceDictionary dictionary = new ResourceDictionary
{
Source = resourceSource
};
theme.ThemeDictionary = dictionary;
}
catch (Exception ex)
@ -373,7 +392,7 @@ namespace ModAssistant
/*
* Check to see if the lengths of each file are different. If they are, overwrite what currently exists.
* The reason we are also checking LoadedTheme against the name variable is to prevent overwriting a file that's
* already being used by ModAssistant and causing a System.IO.IOException.
* already being used by VRCMelonAssistant and causing a System.IO.IOException.
*/
FileInfo existingInfo = new FileInfo(videoName);
if (existingInfo.Length != file.Length && LoadedTheme != name)
@ -398,8 +417,10 @@ namespace ModAssistant
}
}
Theme theme = new Theme(name, dictionary);
theme.Waifus = waifus;
Theme theme = new Theme(name, dictionary)
{
Waifus = waifus
};
return theme;
}
@ -426,9 +447,19 @@ namespace ModAssistant
private static BitmapImage GetImageFromEmbeddedResources(string themeName, string imageName)
{
var assembly = Assembly.GetExecutingAssembly();
var resourceNames = assembly.GetManifestResourceNames();
var desiredResourceName = $"VRCMelonAssistant.Themes.{themeName}.{imageName}.png";
// Don't attempt to access non-existent manifest resources
if (!resourceNames.Contains(desiredResourceName))
{
return null;
}
try
{
using (Stream stream = Assembly.GetExecutingAssembly().GetManifestResourceStream($"ModAssistant.Themes.{themeName}.{imageName}.png"))
using (Stream stream = assembly.GetManifestResourceStream(desiredResourceName))
{
byte[] imageBytes = new byte[stream.Length];
stream.Read(imageBytes, 0, (int)stream.Length);
@ -477,8 +508,6 @@ namespace ModAssistant
ChangeColor(icons, "InfoIconColor", "info_circleDrawingGroup");
ChangeColor(icons, "OptionsIconColor", "cogDrawingGroup");
ChangeColor(icons, "ModsIconColor", "microchipDrawingGroup");
ChangeColor(icons, "LoadingIconColor", "loadingInnerDrawingGroup");
ChangeColor(icons, "LoadingIconColor", "loadingMiddleDrawingGroup");
ChangeColor(icons, "LoadingIconColor", "loadingOuterDrawingGroup");
}

View file

@ -3,22 +3,28 @@ using System.Diagnostics;
using System.IO;
using System.Threading.Tasks;
using System.Windows;
using static ModAssistant.Http;
using static VRCMelonAssistant.Http;
namespace ModAssistant
namespace VRCMelonAssistant
{
class Updater
{
private static readonly string APILatestURL = "https://api.github.com/repos/Assistant/ModAssistant/releases/latest";
private static readonly string APILatestURL = "https://api.github.com/repos/knah/VRCMelonAssistant/releases/latest";
private static Update LatestUpdate;
private static Version CurrentVersion;
private static Version LatestVersion;
private static bool NeedsUpdate = false;
private static string NewExe = Path.Combine(Path.GetDirectoryName(Utils.ExePath), "ModAssistant.exe");
private static readonly string NewExe = Path.Combine(Path.GetDirectoryName(Utils.ExePath), "VRCMelonAssistant.exe");
private static readonly string Arguments = App.Arguments;
#pragma warning disable CS0162 // Unreachable code detected
public static async Task<bool> CheckForUpdate()
{
#if DEBUG
return false;
#endif
var resp = await HttpClient.GetAsync(APILatestURL);
var body = await resp.Content.ReadAsStringAsync();
LatestUpdate = JsonSerializer.Deserialize<Update>(body);
@ -28,10 +34,11 @@ namespace ModAssistant
return (LatestVersion > CurrentVersion);
}
#pragma warning restore CS0162 // Unreachable code detected
public static async Task Run()
{
if (Path.GetFileName(Utils.ExePath).Equals("ModAssistant.old.exe")) RunNew();
if (Path.GetFileName(Utils.ExePath).Equals("VRCMelonAssistant.old.exe")) RunNew();
try
{
NeedsUpdate = await CheckForUpdate();
@ -46,12 +53,12 @@ namespace ModAssistant
public static async Task StartUpdate()
{
string OldExe = Path.Combine(Path.GetDirectoryName(Utils.ExePath), "ModAssistant.old.exe");
string OldExe = Path.Combine(Path.GetDirectoryName(Utils.ExePath), "VRCMelonAssistant.old.exe");
string DownloadLink = null;
foreach (Update.Asset asset in LatestUpdate.assets)
{
if (asset.name == "ModAssistant.exe")
if (asset.name == "VRCMelonAssistant.exe")
{
DownloadLink = asset.browser_download_url;
}
@ -77,7 +84,7 @@ namespace ModAssistant
private static void RunNew()
{
Process.Start(NewExe);
Process.Start(NewExe, Arguments);
Application.Current.Dispatcher.Invoke(() => { Application.Current.Shutdown(); });
}
}

View file

@ -1,6 +1,6 @@
using Microsoft.Win32;
using System;
using System.Collections.Generic;
using System.Configuration;
using System.Diagnostics;
using System.IO;
using System.Linq;
@ -11,9 +11,11 @@ using System.Text;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using System.Windows;
using static ModAssistant.Http;
using Microsoft.Win32;
using VRCMelonAssistant.Pages;
using static VRCMelonAssistant.Http;
namespace ModAssistant
namespace VRCMelonAssistant
{
public class Utils
{
@ -22,13 +24,9 @@ namespace ModAssistant
public class Constants
{
public const string BeatSaberAPPID = "620980";
public const string BeatModsAPIUrl = "https://beatmods.com/api/v1/";
public const string TeknikAPIUrl = "https://api.teknik.io/v1/";
public const string BeatModsURL = "https://beatmods.com";
public const string BeatModsAlias = "https://alias.beatmods.com/aliases.json";
public const string VRChatAppId = "438100";
public const string VRCMGModsJson = "https://api.vrcmg.com/v1/mods";
public const string WeebCDNAPIURL = "https://pat.assistant.moe/api/v1.0/";
public const string BeatModsModsOptions = "mod?status=approved";
public const string MD5Spacer = " ";
public static readonly char[] IllegalCharacters = new char[]
{
@ -40,20 +38,6 @@ namespace ModAssistant
};
}
public class TeknikPasteResponse
{
public Result result;
public class Result
{
public string id;
public string url;
public string title;
public string syntax;
public DateTime? expiration;
public string password;
}
}
public class WeebCDNRandomResponse
{
public int index;
@ -123,8 +107,8 @@ namespace ModAssistant
if (!string.IsNullOrEmpty(InstallDir)
&& Directory.Exists(InstallDir)
&& Directory.Exists(Path.Combine(InstallDir, "Beat Saber_Data", "Plugins"))
&& File.Exists(Path.Combine(InstallDir, "Beat Saber.exe")))
&& Directory.Exists(Path.Combine(InstallDir, "VRChat_Data", "Plugins"))
&& File.Exists(Path.Combine(InstallDir, "VRChat.exe")))
{
return InstallDir;
}
@ -162,13 +146,14 @@ namespace ModAssistant
public static string SetDir(string directory, string store)
{
App.BeatSaberInstallDirectory = directory;
App.BeatSaberInstallType = store;
App.VRChatInstallDirectory = directory;
App.VRChatInstallType = store;
Pages.Options.Instance.InstallDirectory = directory;
Pages.Options.Instance.InstallType = store;
Properties.Settings.Default.InstallFolder = directory;
Properties.Settings.Default.StoreType = store;
Properties.Settings.Default.Save();
MainWindow.Instance?.MarkModsPageForRefresh();
return directory;
}
@ -208,9 +193,9 @@ namespace ModAssistant
regex = new Regex("\\s\"installdir\"\\s+\"(.+)\"");
foreach (string path in SteamPaths)
{
if (File.Exists(Path.Combine(@path, @"appmanifest_" + Constants.BeatSaberAPPID + ".acf")))
if (File.Exists(Path.Combine(@path, @"appmanifest_" + Constants.VRChatAppId + ".acf")))
{
using (StreamReader reader = new StreamReader(Path.Combine(@path, @"appmanifest_" + Constants.BeatSaberAPPID + ".acf")))
using (StreamReader reader = new StreamReader(Path.Combine(@path, @"appmanifest_" + Constants.VRChatAppId + ".acf")))
{
string line;
while ((line = reader.ReadLine()) != null)
@ -218,7 +203,7 @@ namespace ModAssistant
Match match = regex.Match(line);
if (match.Success)
{
if (File.Exists(Path.Combine(@path, @"common", match.Groups[1].Value, "Beat Saber.exe")))
if (File.Exists(Path.Combine(@path, @"common", match.Groups[1].Value, "VRChat.exe")))
{
return SetDir(Path.Combine(@path, @"common", match.Groups[1].Value), "Steam");
}
@ -232,18 +217,18 @@ namespace ModAssistant
public static string GetVersion()
{
string filename = Path.Combine(App.BeatSaberInstallDirectory, "Beat Saber_Data", "globalgamemanagers");
string filename = Path.Combine(App.VRChatInstallDirectory, "VRChat_Data", "globalgamemanagers");
using (FileStream fs = new FileStream(filename, FileMode.Open, FileAccess.Read))
{
byte[] file = File.ReadAllBytes(filename);
byte[] bytes = new byte[16];
byte[] bytes = new byte[32];
fs.Read(file, 0, Convert.ToInt32(fs.Length));
fs.Close();
int index = Encoding.Default.GetString(file).IndexOf("public.app-category.games") + 136;
int index = Encoding.UTF8.GetString(file).IndexOf("public.app-category.games") + 136;
Array.Copy(file, index, bytes, 0, 16);
string version = Encoding.Default.GetString(bytes).Trim(Utils.Constants.IllegalCharacters);
Array.Copy(file, index, bytes, 0, 32);
string version = Encoding.UTF8.GetString(bytes).Trim(Constants.IllegalCharacters);
return version;
}
@ -256,9 +241,9 @@ namespace ModAssistant
if (!string.IsNullOrEmpty(OculusInstall))
{
if (File.Exists(Path.Combine(OculusInstall, "Software", "hyperbolic-magnetism-beat-saber", "Beat Saber.exe")))
if (File.Exists(Path.Combine(OculusInstall, "Software", "vrchat-vrchat", "VRChat.exe")))
{
return SetDir(Path.Combine(OculusInstall, "Software", "hyperbolic-magnetism-beat-saber"), "Oculus");
return SetDir(Path.Combine(OculusInstall, "Software", "vrchat-vrchat"), "Oculus");
}
}
@ -292,8 +277,8 @@ namespace ModAssistant
string GUIDLetter = guidLetterVolumes.FirstOrDefault(x => libraryPath.Contains(x.Key)).Value;
if (!string.IsNullOrEmpty(GUIDLetter))
{
string finalPath = Path.Combine(GUIDLetter, libraryPath.Substring(49), @"Software\hyperbolic-magnetism-beat-saber");
if (File.Exists(Path.Combine(finalPath, "Beat Saber.exe")))
string finalPath = Path.Combine(GUIDLetter, libraryPath.Substring(49), @"Software\vrchat-vrchat");
if (File.Exists(Path.Combine(finalPath, "VRChat.exe")))
{
return SetDir(finalPath, "Oculus");
}
@ -308,7 +293,7 @@ namespace ModAssistant
public static string GetManualDir()
{
var dialog = new Microsoft.Win32.SaveFileDialog()
var dialog = new SaveFileDialog()
{
Title = (string)Application.Current.FindResource("Utils:InstallDir:DialogTitle"),
Filter = "Directory|*.this.directory",
@ -321,10 +306,10 @@ namespace ModAssistant
path = path.Replace("\\select.this.directory", "");
path = path.Replace(".this.directory", "");
path = path.Replace("\\select.directory", "");
if (File.Exists(Path.Combine(path, "Beat Saber.exe")))
if (File.Exists(Path.Combine(path, "VRChat.exe")))
{
string store;
if (File.Exists(Path.Combine(path, "Beat Saber_Data", "Plugins", "steam_api64.dll")))
if (File.Exists(Path.Combine(path, "VRChat_Data", "Plugins", "steam_api64.dll")))
{
store = "Steam";
}
@ -338,18 +323,20 @@ namespace ModAssistant
return null;
}
public static bool IsVoid()
public static string GetManualFile(string filter = "", string title = "Open File")
{
string directory = App.BeatSaberInstallDirectory;
var dialog = new OpenFileDialog()
{
Title = title,
Filter = filter,
Multiselect = false,
};
if (File.Exists(Path.Combine(directory, "IGG-GAMES.COM.url")) ||
File.Exists(Path.Combine(directory, "SmartSteamEmu.ini")) ||
File.Exists(Path.Combine(directory, "GAMESTORRENT.CO.url")) ||
File.Exists(Path.Combine(directory, "Beat Saber_Data", "Plugins", "BSteam crack.dll")) ||
File.Exists(Path.Combine(directory, "Beat Saber_Data", "Plugins", "HUHUVR_steam_api64.dll")) ||
Directory.GetFiles(Path.Combine(directory, "Beat Saber_Data", "Plugins"), "*.ini", SearchOption.TopDirectoryOnly).Length > 0)
return true;
return false;
if (dialog.ShowDialog() == true)
{
return dialog.FileName;
}
return null;
}
public static byte[] StreamToArray(Stream input)
@ -373,7 +360,7 @@ namespace ModAssistant
{
try
{
System.Diagnostics.Process.Start(new System.Diagnostics.ProcessStartInfo()
Process.Start(new System.Diagnostics.ProcessStartInfo()
{
FileName = location,
UseShellExecute = true,
@ -386,6 +373,13 @@ namespace ModAssistant
MessageBox.Show($"{string.Format((string)Application.Current.FindResource("Utils:CannotOpenFolder"), location)}.");
}
public static void Log(string message, string severity = "LOG")
{
string path = Path.GetDirectoryName(ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.PerUserRoamingAndLocal).FilePath);
string logFile = $"{path}{Path.DirectorySeparatorChar}log.log";
File.AppendAllText(logFile, $"[{DateTime.UtcNow:yyyy-mm-dd HH:mm:ss.ffffff}][{severity.ToUpper()}] {message}\n");
}
public static async Task Download(string link, string output)
{
var resp = await HttpClient.GetAsync(link);
@ -414,5 +408,36 @@ namespace ModAssistant
ShowMessageBoxDelegate caller = new ShowMessageBoxDelegate(ShowMessageBox);
caller.BeginInvoke(Message, null, null, null);
}
public static void ShowErrorMessageBox(string title, Exception ex)
{
MessageBox.Show(MainWindow.Instance, $"{title}\n{ex}", "Error",
MessageBoxButton.OK, MessageBoxImage.Error);
}
/// <summary>
/// Attempts to write the specified string to the <see cref="System.Windows.Clipboard"/>.
/// </summary>
/// <param name="text">The string to be written</param>
public static void SetClipboard(string text)
{
bool success = false;
try
{
Clipboard.SetText(text);
success = true;
}
catch (Exception)
{
// Swallow exceptions relating to writing data to clipboard.
}
// This could be placed in the try/catch block but we don't
// want to suppress exceptions for non-clipboard operations
if (success)
{
SendNotify($"Copied text to clipboard");
}
}
}
}

View file

@ -0,0 +1,55 @@
/*
Copyright (c) 2013 Max Hauser
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
using System.Text;
namespace VRCMelonAssistant.Libs
{
internal static class IntExtensions
{
/// <summary>
/// The number of digits in a non-negative number. Returns 1 for all
/// negative numbers. That is ok because we are using it to calculate
/// string length for a <see cref="StringBuilder"/> for numbers that
/// aren't supposed to be negative, but when they are it is just a little
/// slower.
/// </summary>
/// <remarks>
/// This approach is based on https://stackoverflow.com/a/51099524/268898
/// where the poster offers performance benchmarks showing this is the
/// fastest way to get a number of digits.
/// </remarks>
public static int Digits(this int n)
{
if (n < 10) return 1;
if (n < 100) return 2;
if (n < 1_000) return 3;
if (n < 10_000) return 4;
if (n < 100_000) return 5;
if (n < 1_000_000) return 6;
if (n < 10_000_000) return 7;
if (n < 100_000_000) return 8;
if (n < 1_000_000_000) return 9;
return 10;
}
}
}

View file

@ -0,0 +1,630 @@
/*
Copyright (c) 2013 Max Hauser
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
using System;
using System.Globalization;
using System.Text;
#if !NETSTANDARD
using System.Runtime.Serialization;
using System.Security.Permissions;
#endif
using System.Text.RegularExpressions;
namespace VRCMelonAssistant.Libs
{
/// <summary>
/// A semantic version implementation.
/// Conforms with v2.0.0 of http://semver.org
/// </summary>
#if NETSTANDARD
public sealed class SemVersion : IComparable<SemVersion>, IComparable
#else
[Serializable]
public sealed class SemVersion : IComparable<SemVersion>, IComparable, ISerializable
#endif
{
private static readonly Regex ParseEx =
new Regex(@"^(?<major>\d+)" +
@"(?>\.(?<minor>\d+))?" +
@"(?>\.(?<patch>\d+))?" +
@"(?>\.(?<extra>\d+))?" +
@"(?>\-(?<pre>[0-9A-Za-z\-\.]+))?" +
@"(?>\+(?<build>[0-9A-Za-z\-\.]+))?$",
#if NETSTANDARD
RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture,
#else
RegexOptions.CultureInvariant | RegexOptions.Compiled | RegexOptions.ExplicitCapture,
#endif
TimeSpan.FromSeconds(0.5));
#if !NETSTANDARD
#pragma warning disable CA1801 // Parameter unused
/// <summary>
/// Deserialize a <see cref="SemVersion"/>.
/// </summary>
/// <exception cref="ArgumentNullException">The <paramref name="info"/> parameter is null.</exception>
private SemVersion(SerializationInfo info, StreamingContext context)
#pragma warning restore CA1801 // Parameter unused
{
if (info == null) throw new ArgumentNullException(nameof(info));
var semVersion = Parse(info.GetString("SemVersion"));
Major = semVersion.Major;
Minor = semVersion.Minor;
Patch = semVersion.Patch;
Prerelease = semVersion.Prerelease;
Build = semVersion.Build;
}
#endif
/// <summary>
/// Constructs a new instance of the <see cref="SemVersion" /> class.
/// </summary>
/// <param name="major">The major version.</param>
/// <param name="minor">The minor version.</param>
/// <param name="patch">The patch version.</param>
/// <param name="prerelease">The prerelease version (e.g. "alpha").</param>
/// <param name="build">The build metadata (e.g. "nightly.232").</param>
public SemVersion(int major, int minor = 0, int patch = 0, int extra = 0, string prerelease = "", string build = "")
{
Major = major;
Minor = minor;
Patch = patch;
Extra = extra;
Prerelease = prerelease ?? "";
Build = build ?? "";
}
/// <summary>
/// Constructs a new instance of the <see cref="SemVersion"/> class from
/// a <see cref="System.Version"/>.
/// </summary>
/// <param name="version">The <see cref="Version"/> that is used to initialize
/// the Major, Minor, Patch and Build.</param>
/// <returns>A <see cref="SemVersion"/> with the same Major and Minor version.
/// The Patch version will be the fourth part of the version number. The
/// build meta data will contain the third part of the version number if
/// it is greater than zero.</returns>
public SemVersion(Version version)
{
if (version == null)
throw new ArgumentNullException(nameof(version));
Major = version.Major;
Minor = version.Minor;
if (version.Revision >= 0)
Patch = version.Revision;
Prerelease = "";
Build = version.Build > 0 ? version.Build.ToString(CultureInfo.InvariantCulture) : "";
}
/// <summary>
/// Converts the string representation of a semantic version to its <see cref="SemVersion"/> equivalent.
/// </summary>
/// <param name="version">The version string.</param>
/// <param name="strict">If set to <see langword="true"/> minor and patch version are required,
/// otherwise they are optional.</param>
/// <returns>The <see cref="SemVersion"/> object.</returns>
/// <exception cref="ArgumentNullException">The <paramref name="version"/> is <see langword="null"/>.</exception>
/// <exception cref="ArgumentException">The <paramref name="version"/> has an invalid format.</exception>
/// <exception cref="InvalidOperationException">The <paramref name="version"/> is missing Minor or Patch versions and <paramref name="strict"/> is <see langword="true"/>.</exception>
/// <exception cref="OverflowException">The Major, Minor, or Patch versions are larger than <code>int.MaxValue</code>.</exception>
public static SemVersion Parse(string version, bool strict = false)
{
var match = ParseEx.Match(version);
if (!match.Success)
throw new ArgumentException($"Invalid version '{version}'.", nameof(version));
var major = int.Parse(match.Groups["major"].Value, CultureInfo.InvariantCulture);
var minorMatch = match.Groups["minor"];
int minor = 0;
if (minorMatch.Success)
minor = int.Parse(minorMatch.Value, CultureInfo.InvariantCulture);
else if (strict)
throw new InvalidOperationException("Invalid version (no minor version given in strict mode)");
var patchMatch = match.Groups["patch"];
int patch = 0;
if (patchMatch.Success)
patch = int.Parse(patchMatch.Value, CultureInfo.InvariantCulture);
else if (strict)
throw new InvalidOperationException("Invalid version (no patch version given in strict mode)");
var extraMatch = match.Groups["extra"];
int extra = 0;
if (extraMatch.Success)
extra = int.Parse(extraMatch.Value, CultureInfo.InvariantCulture);
var prerelease = match.Groups["pre"].Value;
var build = match.Groups["build"].Value;
return new SemVersion(major, minor, patch, extra, prerelease, build);
}
/// <summary>
/// Converts the string representation of a semantic version to its <see cref="SemVersion"/>
/// equivalent and returns a value that indicates whether the conversion succeeded.
/// </summary>
/// <param name="version">The version string.</param>
/// <param name="semver">When the method returns, contains a <see cref="SemVersion"/> instance equivalent
/// to the version string passed in, if the version string was valid, or <see langword="null"/> if the
/// version string was not valid.</param>
/// <param name="strict">If set to <see langword="true"/> minor and patch version are required,
/// otherwise they are optional.</param>
/// <returns><see langword="false"/> when a invalid version string is passed, otherwise <see langword="true"/>.</returns>
public static bool TryParse(string version, out SemVersion semver, bool strict = false)
{
semver = null;
if (version is null) return false;
var match = ParseEx.Match(version);
if (!match.Success) return false;
if (!int.TryParse(match.Groups["major"].Value, NumberStyles.Integer, CultureInfo.InvariantCulture, out var major))
return false;
var minorMatch = match.Groups["minor"];
int minor = 0;
if (minorMatch.Success)
{
if (!int.TryParse(minorMatch.Value, NumberStyles.Integer, CultureInfo.InvariantCulture, out minor))
return false;
}
else if (strict) return false;
var patchMatch = match.Groups["patch"];
int patch = 0;
if (patchMatch.Success)
{
if (!int.TryParse(patchMatch.Value, NumberStyles.Integer, CultureInfo.InvariantCulture, out patch))
return false;
}
else if (strict) return false;
var extraMatch = match.Groups["extra"];
int extra = 0;
if (extraMatch.Success)
{
if (!int.TryParse(extraMatch.Value, NumberStyles.Integer, CultureInfo.InvariantCulture, out extra))
return false;
}
var prerelease = match.Groups["pre"].Value;
var build = match.Groups["build"].Value;
semver = new SemVersion(major, minor, patch, extra, prerelease, build);
return true;
}
/// <summary>
/// Checks whether two semantic versions are equal.
/// </summary>
/// <param name="versionA">The first version to compare.</param>
/// <param name="versionB">The second version to compare.</param>
/// <returns><see langword="true"/> if the two values are equal, otherwise <see langword="false"/>.</returns>
public static bool Equals(SemVersion versionA, SemVersion versionB)
{
if (ReferenceEquals(versionA, versionB)) return true;
if (versionA is null || versionB is null) return false;
return versionA.Equals(versionB);
}
/// <summary>
/// Compares the specified versions.
/// </summary>
/// <param name="versionA">The first version to compare.</param>
/// <param name="versionB">The second version to compare.</param>
/// <returns>A signed number indicating the relative values of <paramref name="versionA"/> and <paramref name="versionB"/>.</returns>
public static int Compare(SemVersion versionA, SemVersion versionB)
{
if (ReferenceEquals(versionA, versionB)) return 0;
if (versionA is null) return -1;
if (versionB is null) return 1;
return versionA.CompareTo(versionB);
}
/// <summary>
/// Make a copy of the current instance with changed properties.
/// </summary>
/// <param name="major">The value to replace the major version or <see langword="null"/> to leave it unchanged.</param>
/// <param name="minor">The value to replace the minor version or <see langword="null"/> to leave it unchanged.</param>
/// <param name="patch">The value to replace the patch version or <see langword="null"/> to leave it unchanged.</param>
/// <param name="prerelease">The value to replace the prerelease version or <see langword="null"/> to leave it unchanged.</param>
/// <param name="build">The value to replace the build metadata or <see langword="null"/> to leave it unchanged.</param>
/// <returns>The new version object.</returns>
/// <remarks>
/// The change method is intended to be called using named argument syntax, passing only
/// those fields to be changed.
/// </remarks>
/// <example>
/// To change only the patch version:
/// <code>version.Change(patch: 4)</code>
/// </example>
public SemVersion Change(int? major = null, int? minor = null, int? patch = null, int? extra = null,
string prerelease = null, string build = null)
{
return new SemVersion(
major ?? Major,
minor ?? Minor,
patch ?? Patch,
extra ?? Extra,
prerelease ?? Prerelease,
build ?? Build);
}
/// <summary>
/// Gets the major version.
/// </summary>
/// <value>
/// The major version.
/// </value>
public int Major { get; }
/// <summary>
/// Gets the minor version.
/// </summary>
/// <value>
/// The minor version.
/// </value>
public int Minor { get; }
/// <summary>
/// Gets the patch version.
/// </summary>
/// <value>
/// The patch version.
/// </value>
public int Patch { get; }
/// <summary>
/// Handles the fourth number present in assembly versions
/// </summary>
public int Extra { get; }
/// <summary>
/// Gets the prerelease version.
/// </summary>
/// <value>
/// The prerelease version. Empty string if this is a release version.
/// </value>
public string Prerelease { get; }
/// <summary>
/// Gets the build metadata.
/// </summary>
/// <value>
/// The build metadata. Empty string if there is no build metadata.
/// </value>
public string Build { get; }
/// <summary>
/// Returns the <see cref="string" /> equivalent of this version.
/// </summary>
/// <returns>
/// The <see cref="string" /> equivalent of this version.
/// </returns>
public override string ToString()
{
// Assume all separators ("..-+"), at most 2 extra chars
var estimatedLength = 4 + Major.Digits() + Minor.Digits() + Patch.Digits()
+ Prerelease.Length + Build.Length;
var version = new StringBuilder(estimatedLength);
version.Append(Major);
version.Append('.');
version.Append(Minor);
version.Append('.');
version.Append(Patch);
if (Extra != 0)
{
version.Append('.');
version.Append(Extra);
}
if (Prerelease.Length > 0)
{
version.Append('-');
version.Append(Prerelease);
}
if (Build.Length > 0)
{
version.Append('+');
version.Append(Build);
}
return version.ToString();
}
/// <summary>
/// Compares the current instance with another object of the same type and returns an integer that indicates
/// whether the current instance precedes, follows, or occurs in the same position in the sort order as the
/// other object.
/// </summary>
/// <param name="obj">An object to compare with this instance.</param>
/// <returns>
/// A value that indicates the relative order of the objects being compared.
/// The return value has these meanings:
/// Less than zero: This instance precedes <paramref name="obj" /> in the sort order.
/// Zero: This instance occurs in the same position in the sort order as <paramref name="obj" />.
/// Greater than zero: This instance follows <paramref name="obj" /> in the sort order.
/// </returns>
/// <exception cref="InvalidCastException">The <paramref name="obj"/> is not a <see cref="SemVersion"/>.</exception>
public int CompareTo(object obj)
{
return CompareTo((SemVersion)obj);
}
/// <summary>
/// Compares the current instance with another object of the same type and returns an integer that indicates
/// whether the current instance precedes, follows, or occurs in the same position in the sort order as the
/// other object.
/// </summary>
/// <param name="other">An object to compare with this instance.</param>
/// <returns>
/// A value that indicates the relative order of the objects being compared.
/// The return value has these meanings:
/// Less than zero: This instance precedes <paramref name="other" /> in the sort order.
/// Zero: This instance occurs in the same position in the sort order as <paramref name="other" />.
/// Greater than zero: This instance follows <paramref name="other" /> in the sort order.
/// </returns>
public int CompareTo(SemVersion other)
{
var r = CompareByPrecedence(other);
if (r != 0) return r;
#pragma warning disable CA1062 // Validate arguments of public methods
// If other is null, CompareByPrecedence() returns 1
return CompareComponent(Build, other.Build);
#pragma warning restore CA1062 // Validate arguments of public methods
}
/// <summary>
/// Returns whether two semantic versions have the same precedence. Versions
/// that differ only by build metadata have the same precedence.
/// </summary>
/// <param name="other">The semantic version to compare to.</param>
/// <returns><see langword="true"/> if the version precedences are equal.</returns>
public bool PrecedenceMatches(SemVersion other)
{
return CompareByPrecedence(other) == 0;
}
/// <summary>
/// Compares two semantic versions by precedence as defined in the SemVer spec. Versions
/// that differ only by build metadata have the same precedence.
/// </summary>
/// <param name="other">The semantic version.</param>
/// <returns>
/// A value that indicates the relative order of the objects being compared.
/// The return value has these meanings:
/// Less than zero: This instance precedes <paramref name="other" /> in the sort order.
/// Zero: This instance occurs in the same position in the sort order as <paramref name="other" />.
/// Greater than zero: This instance follows <paramref name="other" /> in the sort order.
/// </returns>
public int CompareByPrecedence(SemVersion other)
{
if (other is null)
return 1;
var r = Major.CompareTo(other.Major);
if (r != 0) return r;
r = Minor.CompareTo(other.Minor);
if (r != 0) return r;
r = Patch.CompareTo(other.Patch);
if (r != 0) return r;
r = Extra.CompareTo(other.Extra);
if (r != 0) return r;
return CompareComponent(Prerelease, other.Prerelease, true);
}
private static int CompareComponent(string a, string b, bool nonemptyIsLower = false)
{
var aEmpty = string.IsNullOrEmpty(a);
var bEmpty = string.IsNullOrEmpty(b);
if (aEmpty && bEmpty)
return 0;
if (aEmpty)
return nonemptyIsLower ? 1 : -1;
if (bEmpty)
return nonemptyIsLower ? -1 : 1;
var aComps = a.Split('.');
var bComps = b.Split('.');
var minLen = Math.Min(aComps.Length, bComps.Length);
for (int i = 0; i < minLen; i++)
{
var ac = aComps[i];
var bc = bComps[i];
var aIsNum = int.TryParse(ac, out var aNum);
var bIsNum = int.TryParse(bc, out var bNum);
int r;
if (aIsNum && bIsNum)
{
r = aNum.CompareTo(bNum);
if (r != 0) return r;
}
else
{
if (aIsNum)
return -1;
if (bIsNum)
return 1;
r = string.CompareOrdinal(ac, bc);
if (r != 0)
return r;
}
}
return aComps.Length.CompareTo(bComps.Length);
}
/// <summary>
/// Determines whether the specified <see cref="object" /> is equal to this instance.
/// </summary>
/// <param name="obj">The <see cref="object" /> to compare with this instance.</param>
/// <returns>
/// <see langword="true"/> if the specified <see cref="object" /> is equal to this instance, otherwise <see langword="false"/>.
/// </returns>
/// <exception cref="InvalidCastException">The <paramref name="obj"/> is not a <see cref="SemVersion"/>.</exception>
public override bool Equals(object obj)
{
if (obj is null)
return false;
if (ReferenceEquals(this, obj))
return true;
var other = (SemVersion)obj;
return Major == other.Major
&& Minor == other.Minor
&& Patch == other.Patch
&& string.Equals(Prerelease, other.Prerelease, StringComparison.Ordinal)
&& string.Equals(Build, other.Build, StringComparison.Ordinal);
}
/// <summary>
/// Returns a hash code for this instance.
/// </summary>
/// <returns>
/// A hash code for this instance, suitable for use in hashing algorithms and data structures like a hash table.
/// </returns>
public override int GetHashCode()
{
unchecked
{
// TODO verify this. Some versions start result = 17. Some use 37 instead of 31
int result = Major.GetHashCode();
result = result * 31 + Minor.GetHashCode();
result = result * 31 + Patch.GetHashCode();
result = result * 31 + Prerelease.GetHashCode();
result = result * 31 + Build.GetHashCode();
return result;
}
}
#if !NETSTANDARD
/// <summary>
/// Populates a <see cref="SerializationInfo"/> with the data needed to serialize the target object.
/// </summary>
/// <param name="info">The <see cref="SerializationInfo"/> to populate with data.</param>
/// <param name="context">The destination (see <see cref="SerializationInfo"/>) for this serialization.</param>
[SecurityPermission(SecurityAction.Demand, SerializationFormatter = true)]
public void GetObjectData(SerializationInfo info, StreamingContext context)
{
if (info == null) throw new ArgumentNullException(nameof(info));
info.AddValue("SemVersion", ToString());
}
#endif
#pragma warning disable CA2225 // Operator overloads have named alternates
/// <summary>
/// Implicit conversion from <see cref="string"/> to <see cref="SemVersion"/>.
/// </summary>
/// <param name="version">The semantic version.</param>
/// <returns>The <see cref="SemVersion"/> object.</returns>
/// <exception cref="ArgumentNullException">The <paramref name="version"/> is <see langword="null"/>.</exception>
/// <exception cref="ArgumentException">The version number has an invalid format.</exception>
/// <exception cref="OverflowException">The Major, Minor, or Patch versions are larger than <code>int.MaxValue</code>.</exception>
public static implicit operator SemVersion(string version)
#pragma warning restore CA2225 // Operator overloads have named alternates
{
return Parse(version);
}
/// <summary>
/// Compares two semantic versions for equality.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is equal to right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator ==(SemVersion left, SemVersion right)
{
return Equals(left, right);
}
/// <summary>
/// Compares two semantic versions for inequality.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is not equal to right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator !=(SemVersion left, SemVersion right)
{
return !Equals(left, right);
}
/// <summary>
/// Compares two semantic versions.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is greater than right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator >(SemVersion left, SemVersion right)
{
return Compare(left, right) > 0;
}
/// <summary>
/// Compares two semantic versions.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is greater than or equal to right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator >=(SemVersion left, SemVersion right)
{
return Equals(left, right) || Compare(left, right) > 0;
}
/// <summary>
/// Compares two semantic versions.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is less than right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator <(SemVersion left, SemVersion right)
{
return Compare(left, right) < 0;
}
/// <summary>
/// Compares two semantic versions.
/// </summary>
/// <param name="left">The left value.</param>
/// <param name="right">The right value.</param>
/// <returns>If left is less than or equal to right <see langword="true"/>, otherwise <see langword="false"/>.</returns>
public static bool operator <=(SemVersion left, SemVersion right)
{
return Equals(left, right) || Compare(left, right) < 0;
}
}
}

View file

@ -1,7 +1,7 @@
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:ModAssistant"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:sys="clr-namespace:System;assembly=mscorlib">
<sys:String x:Key="ResourceDictionaryName">i18n:en-DEBUG</sys:String>
@ -62,6 +62,7 @@
<sys:String x:Key="Mods:CheckingInstalledMods">Mods:CheckingInstalledMods</sys:String>
<sys:String x:Key="Mods:LoadingMods">Mods:LoadingMods</sys:String>
<sys:String x:Key="Mods:FinishedLoadingMods">Mods:FinishedLoadingMods</sys:String>
<sys:String x:Key="Mods:NoMods">Mods:NoMods</sys:String>
<sys:String x:Key="Mods:InstallingMod">{0} Mods:InstallingMod</sys:String>
<sys:String x:Key="Mods:InstalledMod">{0} Mods:InstalledMod</sys:String>
<sys:String x:Key="Mods:FinishedInstallingMods">Mods:FinishedInstallingMods</sys:String>
@ -69,6 +70,11 @@
<sys:String x:Key="Mods:UninstallBox:Title">{0} Mods:UninstallBox:Title</sys:String>
<sys:String x:Key="Mods:UninstallBox:Body1">{0} Mods:UninstallBox:Body1</sys:String>
<sys:String x:Key="Mods:UninstallBox:Body2">Mods:UninstallBox:Body2</sys:String>
<sys:String x:Key="Mods:FailedExtract">{0} {1} {2} {3} Mods:FailedExtract</sys:String>
<sys:String x:Key="Mods:FailedExtractMaxReached">{0} {1} Mods:FailedExtractMaxReached</sys:String>
<sys:String x:Key="Mods:SearchLabel">Mods:SearchLabel</sys:String>
<sys:String x:Key="Mods:UninstallBSIPANotFound:Title">Mods:UninstallBSIPANotFound:Title</sys:String>
<sys:String x:Key="Mods:UninstallBSIPANotFound:Body">Mods:UninstallBSIPANotFound:Body</sys:String>
<!-- About Page -->
<sys:String x:Key="About:Title">About:Title</sys:String>
@ -93,12 +99,21 @@
<sys:String x:Key="Options:SaveSelectedMods">Options:SaveSelectedMods</sys:String>
<sys:String x:Key="Options:CheckInstalledMods">Options:CheckInstalledMods</sys:String>
<sys:String x:Key="Options:SelectInstalledMods">Options:SelectInstalledMods</sys:String>
<sys:String x:Key="Options:ReinstallInstalledMods">Options:Reinstall Installed Mods</sys:String>
<sys:String x:Key="Options:EnableOneClickInstalls">Options:EnableOneClickInstalls</sys:String>
<sys:String x:Key="Options:BeatSaver">Options:BeatSaver</sys:String>
<sys:String x:Key="Options:ModelSaber">Options:ModelSaber</sys:String>
<sys:String x:Key="Options:CloseWindow">Options:CloseWindow</sys:String>
<sys:String x:Key="Options:GameType">Options:GameType</sys:String>
<sys:String x:Key="Options:GameType:Steam">Options:GameType:Steam</sys:String>
<sys:String x:Key="Options:GameType:Oculus">Options:GameType:Oculus</sys:String>
<sys:String x:Key="Options:Tools">Options:Tools</sys:String>
<sys:String x:Key="Options:InstallPlaylist">Options:InstallPlaylist</sys:String>
<sys:String x:Key="Options:InstallingPlaylist">{0} Options:InstallingPlaylist</sys:String>
<sys:String x:Key="Options:FailedPlaylistSong">{0} Options:FailedPlaylistSong</sys:String>
<sys:String x:Key="Options:FinishedPlaylist">{0} {1} Options:FinishedPlaylist</sys:String>
<sys:String x:Key="Options:ShowOCIWindow">Options:ShowOCIWindow</sys:String>
<sys:String x:Key="Options:OCIWindowYes">Options:OCIWindowYes</sys:String>
<sys:String x:Key="Options:OCIWindowClose">Options:OCIWindowClose</sys:String>
<sys:String x:Key="Options:OCIWindowNo">Options:OCIWindowNo</sys:String>
<sys:String x:Key="Options:Diagnostics">Options:Diagnostics</sys:String>
<sys:String x:Key="Options:OpenLogsButton">Options:OpenLogsButton</sys:String>
<sys:String x:Key="Options:OpenAppDataButton">Options:OpenAppDataButton</sys:String>
@ -120,6 +135,7 @@
<sys:String x:Key="Options:AllModsUninstalled">Options:AllModsUninstalled</sys:String>
<sys:String x:Key="Options:CurrentThemeRemoved">Options:CurrentThemeRemoved</sys:String>
<sys:String x:Key="Options:ThemeFolderNotFound">Options:ThemeFolderNotFound</sys:String>
<sys:String x:Key="Options:AppDataNotFound">Options:AppDataNotFound</sys:String>
<!-- Loading Page -->
<sys:String x:Key="Loading:Loading">Loading:Loading</sys:String>
@ -147,6 +163,11 @@
<sys:String x:Key="OneClick:AssetInstallFailed">OneClick:AssetInstallFailed</sys:String>
<sys:String x:Key="OneClick:ProtocolHandler:Registered">{0} OneClick:ProtocolHandler:Registered</sys:String>
<sys:String x:Key="OneClick:ProtocolHandler:Unregistered">{0} OneClick:ProtocolHandler:Unregistered</sys:String>
<sys:String x:Key="OneClick:Installing">{0} OneClick:Installing</sys:String>
<sys:String x:Key="OneClick:RatelimitSkip">{0} OneClick:RatelimitSkip</sys:String>
<sys:String x:Key="OneClick:RatelimitHit">{0} OneClick:RatelimitHit</sys:String>
<sys:String x:Key="OneClick:Failed">{0} OneClick:Failed</sys:String>
<sys:String x:Key="OneClick:Done">OneClick:Done</sys:String>
<!-- Themes Class -->
<sys:String x:Key="Themes:ThemeNotFound">Themes:ThemeNotFound</sys:String>

View file

@ -1,20 +1,20 @@
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:ModAssistant"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:sys="clr-namespace:System;assembly=mscorlib">
<sys:String x:Key="ResourceDictionaryName">i18n:en-US</sys:String>
<!-- App -->
<sys:String x:Key="App:InstallDirDialog:Title">Couldn't find your Beat Saber install folder!</sys:String>
<sys:String x:Key="App:InstallDirDialog:Title">Couldn't find your VRChat install folder!</sys:String>
<sys:String x:Key="App:InstallDirDialog:OkCancel">Press OK to try again, or Cancel to close application.</sys:String>
<sys:String x:Key="App:InvalidArgument">Invalid argument! '{0}' requires an option.</sys:String>
<sys:String x:Key="App:UnrecognizedArgument">Unrecognized argument. Closing Mod Assistant.</sys:String>
<sys:String x:Key="App:UnrecognizedArgument">Unrecognized argument. Closing VRChat Melon Assistant.</sys:String>
<sys:String x:Key="App:UnhandledException">An unhandled exception just occurred</sys:String>
<sys:String x:Key="App:Exception">Exception</sys:String>
<!-- Main Window -->
<sys:String x:Key="MainWindow:WindowTitle">ModAssistant</sys:String>
<sys:String x:Key="MainWindow:WindowTitle">VRChat Melon Assistant</sys:String>
<sys:String x:Key="MainWindow:IntroButton">Intro</sys:String>
<sys:String x:Key="MainWindow:ModsButton">Mods</sys:String>
<sys:String x:Key="MainWindow:AboutButton">About</sys:String>
@ -24,33 +24,33 @@
<sys:String x:Key="MainWindow:ModInfoButton">Mod Info</sys:String>
<sys:String x:Key="MainWindow:InstallButtonTop">Install</sys:String>
<sys:String x:Key="MainWindow:InstallButtonBottom">or Update</sys:String>
<sys:String x:Key="MainWindow:GameVersionLoadFailed">Could not load game versions, Mods tab will be unavailable.</sys:String>
<sys:String x:Key="MainWindow:GameUpdateDialog:Title">New Game Version Detected!</sys:String>
<sys:String x:Key="MainWindow:GameUpdateDialog:Line1">It looks like there's been a game update.</sys:String>
<sys:String x:Key="MainWindow:GameUpdateDialog:Line2">Please double check that the correct version is selected at the bottom left corner!</sys:String>
<sys:String x:Key="MainWindow:NoModSelected">No mod selected!</sys:String>
<sys:String x:Key="MainWindow:NoModInfoPage">{0} does not have an info page.</sys:String>
<!-- Intro Page -->
<sys:String x:Key="Intro:Title">Intro</sys:String>
<sys:String x:Key="Intro:PageTitle">Welcome to Mod Assistant</sys:String>
<sys:String x:Key="Intro:PageTitle">Welcome to VRChat Melon Assistant</sys:String>
<sys:String x:Key="Intro:Terms:Header">Please read this page entirely and carefully</sys:String>
<Span x:Key="Intro:Terms:Line1">
By using this program attest to have read and agree to the following terms:
</Span>
<Span x:Key="Intro:Terms:Line2">
Beat Saber
<Bold>does not</Bold> natively support mods. This means:
VRChat
<Bold>explicitly forbids</Bold> modding the game client in their Terms of Service. This means:
</Span>
<Span x:Key="Intro:Terms:Term0">
Modding the client
<Bold>can</Bold> lead to bans. Mods available via this installer are manually checked to minimize the chance of that happening, but <Bold>the risk is always there</Bold>.
</Span>
<Span x:Key="Intro:Terms:Term1">
Mods
<Bold>will break</Bold> every update. This is normal, and
<Bold>not</Bold> Beat Games' fault.
<Bold>not</Bold> VRChat's fault.
</Span>
<Span x:Key="Intro:Terms:Term2">
Mods
<Bold>will</Bold> cause bugs and performance issues. This is
<Bold>not</Bold> Beat Games' fault.
<Bold>not</Bold> VRChat's fault.
</Span>
<Span x:Key="Intro:Terms:Term3">
Mods are made for
@ -58,21 +58,25 @@
<Bold>free time.</Bold> Please be patient and understanding.
</Span>
<Span x:Key="Intro:ReviewsBeatGamesFault">
<Bold>DO NOT</Bold> leave negative reviews because mods broke. This is
<Bold>not</Bold> Beat Games' fault.
<LineBreak/> They are not trying to kill mods.
<Bold>DO NOT</Bold> complain to VRChat Team because mods broke. This is
<Bold>not</Bold> VRChat's fault.
<LineBreak/> They are not trying to kill mods. Most likely.
</Span>
<Span x:Key="Intro:ReportIssuesNoMods">
If you want to report an issue with the game to VRChat Team, <Bold>make sure the issue is not caused by mods.</Bold>
<LineBreak />To do so, remove all mods and/or MelonLoader and check if the issue still happens.
</Span>
<Span x:Key="Intro:ReviewsRustySpoon">
If I keep seeing people leave negative reviews
If I keep seeing people complain to the devs
<Italic>because</Italic> mods broke,
<LineBreak/>
<Bold>I will personally kill mods with a rusty spoon</Bold>
</Span>
<Span x:Key="Intro:WikiGuide">
Please read the Beginners Guide on the
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://bsmg.wiki/beginners-guide">
Wiki
</Hyperlink>.
Feel free to join the
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://discord.gg/rCqKSvR">
VRChat Modding Group discord
</Hyperlink> for more info, help, and support.
</Span>
<sys:String x:Key="Intro:AgreeButton">I Agree</sys:String>
<sys:String x:Key="Intro:DisagreeButton">Disagree</sys:String>
@ -84,6 +88,7 @@
<!-- Mods Page -->
<sys:String x:Key="Mods:Title">Mods</sys:String>
<sys:String x:Key="Mods:Header:Name">Name</sys:String>
<sys:String x:Key="Mods:Header:Author">Author</sys:String>
<sys:String x:Key="Mods:Header:Installed">Installed</sys:String>
<sys:String x:Key="Mods:Header:Latest">Latest</sys:String>
<sys:String x:Key="Mods:Header:Description">Description</sys:String>
@ -93,31 +98,34 @@
<sys:String x:Key="Mods:CheckingInstalledMods">Checking installed mods</sys:String>
<sys:String x:Key="Mods:LoadingMods">Loading Mods</sys:String>
<sys:String x:Key="Mods:FinishedLoadingMods">Finished loading mods</sys:String>
<sys:String x:Key="Mods:NoMods">No mods available</sys:String>
<sys:String x:Key="Mods:InstallingMod">Installing {0}</sys:String>
<sys:String x:Key="Mods:InstalledMod">Installed {0}</sys:String>
<sys:String x:Key="Mods:UnInstallingMelonLoader">Uninstalling MelonLoader</sys:String>
<sys:String x:Key="Mods:DownloadingMelonLoader">Downloading MelonLoader</sys:String>
<sys:String x:Key="Mods:UnpackingMelonLoader">Unpacking MelonLoader</sys:String>
<sys:String x:Key="Mods:FinishedInstallingMods">Finished installing mods</sys:String>
<sys:String x:Key="Mods:ModDownloadLinkMissing">Could not find download link for {0}</sys:String>
<sys:String x:Key="Mods:UninstallBox:Title">Uninstall {0}?</sys:String>
<sys:String x:Key="Mods:UninstallBox:Body1">Are you sure you want to remove {0}?</sys:String>
<sys:String x:Key="Mods:UninstallBox:Body2">This could break your other mods</sys:String>
<sys:String x:Key="Mods:UninstallSingleFailed">Error uninstalling mod. Make sure VRChat is not running.</sys:String>
<sys:String x:Key="Mods:UninstallMLFailed">Error uninstalling MelonLoader. Make sure VRChat is not running.</sys:String>
<sys:String x:Key="Mods:InstallMLFailed">Error installing MelonLoader. Try using the standalone installer.</sys:String>
<sys:String x:Key="Mods:FailedExtract">Failed to extract {0}, trying again in {1} seconds. ({2}/{3})</sys:String>
<sys:String x:Key="Mods:FailedExtractMaxReached">Failed to extract {0} after max attempts ({1}), skipping. This mod might not work properly so proceed at your own risk</sys:String>
<sys:String x:Key="Mods:SearchLabel">Search...</sys:String>
<!-- About Page -->
<sys:String x:Key="About:Title">About</sys:String>
<sys:String x:Key="About:PageTitle">About Mod Assistant</sys:String>
<sys:String x:Key="About:List:Header">I'm Assistant, and I made Mod Assistant for mod assistance, with a few principles in mind:</sys:String>
<sys:String x:Key="About:List:Item1">Simplicity</sys:String>
<sys:String x:Key="About:List:Item2">Portability</sys:String>
<sys:String x:Key="About:List:Item3">Single Executable</sys:String>
<sys:String x:Key="About:List:Item4">Responsible use</sys:String>
<sys:String x:Key="About:PageTitle">About VRChat Melon Assistant</sys:String>
<sys:String x:Key="About:List:Header">VRChat Melon Assistant is a fork of Assistant's Mod Assistant for Beat Saber.</sys:String>
<Span x:Key="About:SupportAssistant">
If you enjoy this program and would like to support me, please visit my
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://bs.assistant.moe/Donate/">
donation page
</Hyperlink>
or my
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://www.patreon.com/AssistantMoe">
Patreon
Visit
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://github.com/Assistant/ModAssistant">
Mod Assistant GitHub repo
</Hyperlink>
to appreciate their hard work and/or throw some money at them!
</Span>
<sys:String x:Key="About:SpecialThanks">Special Thanks ♥</sys:String>
<sys:String x:Key="About:Donate">Donate</sys:String>
@ -131,82 +139,43 @@
<sys:String x:Key="Options:SelectFolderButton">Select Folder</sys:String>
<sys:String x:Key="Options:OpenFolderButton">Open Folder</sys:String>
<sys:String x:Key="Options:SaveSelectedMods">Save Selected Mods</sys:String>
<sys:String x:Key="Options:CheckInstalledMods">Check Installed Mods</sys:String>
<sys:String x:Key="Options:CheckInstalledMods">Detect Installed Mods</sys:String>
<sys:String x:Key="Options:SelectInstalledMods">Select Installed Mods</sys:String>
<sys:String x:Key="Options:EnableOneClickInstalls">Enable OneClick™ Installs</sys:String>
<sys:String x:Key="Options:BeatSaver">BeatSaver</sys:String>
<sys:String x:Key="Options:ModelSaber">ModelSaber</sys:String>
<sys:String x:Key="Options:ReinstallInstalledMods">Reinstall Installed Mods</sys:String>
<sys:String x:Key="Options:CloseWindow">Close window when finished</sys:String>
<sys:String x:Key="Options:GameType">Game Type</sys:String>
<sys:String x:Key="Options:GameType:Steam">Steam</sys:String>
<sys:String x:Key="Options:GameType:Oculus">Oculus</sys:String>
<sys:String x:Key="Options:Tools">Tools</sys:String>
<sys:String x:Key="Options:InstallPlaylist">Install Playlist</sys:String>
<sys:String x:Key="Options:InstallingPlaylist">Installing Playlist: {0}</sys:String>
<sys:String x:Key="Options:FailedPlaylistSong">Failed song: {0}</sys:String>
<sys:String x:Key="Options:FinishedPlaylist">[{0} fails] Finished Installing Playlist: {1}</sys:String>
<sys:String x:Key="Options:Diagnostics">Diagnostics</sys:String>
<sys:String x:Key="Options:OpenLogsButton">Open Logs</sys:String>
<sys:String x:Key="Options:OpenAppDataButton">Open AppData</sys:String>
<sys:String x:Key="Options:UninstallBSIPAButton">Uninstall BSIPA</sys:String>
<sys:String x:Key="Options:RemoveAllModsButton">Remove All Mods</sys:String>
<sys:String x:Key="Options:RemoveMLButton">Remove MelonLoader</sys:String>
<sys:String x:Key="Options:InstallMLButton">Install MelonLoader</sys:String>
<sys:String x:Key="Options:ApplicationTheme">Application Theme</sys:String>
<sys:String x:Key="Options:ExportTemplateButton">Export Template</sys:String>
<sys:String x:Key="Options:UploadingLog">Uploading Log</sys:String>
<sys:String x:Key="Options:LogUrlCopied">Log URL Copied To Clipboard!</sys:String>
<sys:String x:Key="Options:LogUploadFailed">Uploading Log Failed</sys:String>
<sys:String x:Key="Options:LogUploadFailed:Title">Uploading log failed!</sys:String>
<sys:String x:Key="Options:LogUploadFailed:Body">Could not upload log file to Teknik, please try again or send the file manually.</sys:String>
<sys:String x:Key="Options:GettingModList">Getting Mod List</sys:String>
<sys:String x:Key="Options:FindingBSIPAVersion">Finding BSIPA Version</sys:String>
<sys:String x:Key="Options:BSIPAUninstalled">BSIPA Uninstalled</sys:String>
<sys:String x:Key="Options:YeetModsBox:Title">Uninstall All Mods?</sys:String>
<sys:String x:Key="Options:YeetModsBox:RemoveAllMods">Are you sure you want to remove ALL mods?</sys:String>
<sys:String x:Key="Options:YeetModsBox:CannotBeUndone">This cannot be undone.</sys:String>
<sys:String x:Key="Options:AllModsUninstalled">All Mods Uninstalled</sys:String>
<sys:String x:Key="Options:YeetMLBox:Title">Uninstall MelonLoader?</sys:String>
<sys:String x:Key="Options:YeetMLBox:RemoveAllMods">Are you sure you want to uninstall MelonLoader?</sys:String>
<sys:String x:Key="Options:YeetMLBox:CannotBeUndone">All your mods and mod data will stay intact.</sys:String>
<sys:String x:Key="Options:MLUninstalled">MelonLoader Uninstalled</sys:String>
<sys:String x:Key="Options:MLInstalled">MelonLoader Installed</sys:String>
<sys:String x:Key="Options:CurrentThemeRemoved">Current theme has been removed, reverting to default...</sys:String>
<sys:String x:Key="Options:ThemeFolderNotFound">Themes folder not found! Try exporting the template...</sys:String>
<sys:String x:Key="Options:AppDataNotFound">AppData folder not found! Try launching your game.</sys:String>
<!-- Loading Page -->
<sys:String x:Key="Loading:Loading">Loading Mods</sys:String>
<!-- Invalid Page -->
<sys:String x:Key="Invalid:Title">Invalid</sys:String>
<sys:String x:Key="Invalid:PageTitle">Invalid Installation Detected</sys:String>
<sys:String x:Key="Invalid:PageSubtitle">Your game installation is corrupted or otherwise invalid</sys:String>
<sys:String x:Key="Invalid:List:Header">This can happen if your copy of the game is pirated, or if you copied a pirated copy over your legit install</sys:String>
<Span x:Key="Invalid:List:Line1">
If your copy of the game is pirated,
<Bold>please purchase the game
<Hyperlink NavigateUri="https://beatgames.com/" local:HyperlinkExtensions.IsExternal="True">
HERE
</Hyperlink>
</Bold>.
</Span>
<Span x:Key="Invalid:List:Line2">
If your copy of the game is
<Bold>not</Bold> pirated, please
<Hyperlink NavigateUri="https://bsmg.wiki/support#clean-installation" local:HyperlinkExtensions.IsExternal="True">
do a clean install
</Hyperlink>.
</Span>
<Span x:Key="Invalid:List:Line3">
If those don't help, ask for support in the
<Span Foreground="Blue">#support</Span> channel in
<Hyperlink NavigateUri="https://discord.gg/beatsabermods" local:HyperlinkExtensions.IsExternal="True">
BSMG
</Hyperlink>.
</Span>
<sys:String x:Key="Invalid:BoughtGame1">If you used to have a pirated version but have since bought the game</sys:String>
<sys:String x:Key="Invalid:SelectFolderButton">Select Folder</sys:String>
<sys:String x:Key="Invalid:BoughtGame2">You will need to restart Mod Assistant after changing to the legit install</sys:String>
<!-- OneClick Class -->
<sys:String x:Key="OneClick:MapDownloadFailed">Could not get map details.</sys:String>
<sys:String x:Key="OneClick:SongDownloadFailed">Could not download the song.</sys:String>
<sys:String x:Key="OneClick:SongDownload:Failed">Could not download the song.</sys:String>
<sys:String x:Key="OneClick:SongDownload:NetworkIssues">There might be issues with BeatSaver or your internet connection.</sys:String>
<sys:String x:Key="OneClick:SongDownload:FailedTitle">Failed to download song ZIP</sys:String>
<sys:String x:Key="OneClick:InstallDirNotFound">Beat Saber installation path not found.</sys:String>
<sys:String x:Key="OneClick:InstalledAsset">Installed: {0}</sys:String>
<sys:String x:Key="OneClick:AssetInstallFailed">Failed to install.</sys:String>
<sys:String x:Key="OneClick:ProtocolHandler:Registered">{0} OneClick™ Install handlers registered!</sys:String>
<sys:String x:Key="OneClick:ProtocolHandler:Unregistered">{0} OneClick™ Install handlers unregistered!</sys:String>
<!-- Themes Class -->
<sys:String x:Key="Themes:ThemeNotFound">Theme not found, reverting to default theme...</sys:String>
<sys:String x:Key="Themes:ThemeSet">Theme set to {0}.</sys:String>
@ -220,9 +189,17 @@
<sys:String x:Key="Updater:DownloadFailed">Couldn't download update.</sys:String>
<!-- Utils Class -->
<sys:String x:Key="Utils:NotificationTitle">Mod Assistant</sys:String>
<sys:String x:Key="Utils:NoInstallFolder">Could not detect your Beat Saber install folder. Please select it manually.</sys:String>
<sys:String x:Key="Utils:RunAsAdmin">Mod Assistant needs to run this task as Admin. Please try again.</sys:String>
<sys:String x:Key="Utils:InstallDir:DialogTitle">Select your Beat Saber install folder</sys:String>
<sys:String x:Key="Utils:NotificationTitle">VRChat Melon Assistant</sys:String>
<sys:String x:Key="Utils:NoInstallFolder">Could not detect your VRChat install folder. Please select it manually.</sys:String>
<sys:String x:Key="Utils:RunAsAdmin">VRChat Melon Assistant needs to run this task as Admin. Please try again.</sys:String>
<sys:String x:Key="Utils:InstallDir:DialogTitle">Select your VRChat install folder</sys:String>
<sys:String x:Key="Utils:CannotOpenFolder">Can't open folder: {0}</sys:String>
<sys:String x:Key="ModInfoWindow:Title">Mod info for {0}</sys:String>
<sys:String x:Key="ModInfoWindow:Author">Author: {0}</sys:String>
<sys:String x:Key="ModInfoWindow:NoAuthor">Unknown author</sys:String>
<sys:String x:Key="ModInfoWindow:NoDescription">No description available</sys:String>
<sys:String x:Key="ModInfoWindow:DownloadLink">Download link: </sys:String>
<sys:String x:Key="ModInfoWindow:SourceCodeLink">Source code: </sys:String>
<sys:String x:Key="ModInfoWindow:InternalIds">Internal ID: {0} {1}</sys:String>
</ResourceDictionary>

View file

@ -1,9 +1,9 @@
<Window
x:Class="ModAssistant.MainWindow"
x:Class="VRCMelonAssistant.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="{DynamicResource MainWindow:WindowTitle}"
Icon="Resources/icon.ico"
@ -11,7 +11,7 @@
UIElement.PreviewMouseDown="Window_PreviewMouseDown"
mc:Ignorable="d">
<Grid>
<Rectangle Fill="{DynamicResource ModAssistantBackground}" />
<Rectangle Fill="{DynamicResource VRCMelonAssistantBackground}" />
<Rectangle>
<Rectangle.Fill>
<ImageBrush x:Name="BackgroundImage" Stretch="{DynamicResource BackgroundImageStretch}" />
@ -58,16 +58,17 @@
Margin="0,0,10,5"
Click="IntroButton_Click"
Style="{DynamicResource MainPageButton}">
<StackPanel Margin="0,8,0,0">
<StackPanel Margin="0,6,0,0">
<Image
Height="30"
VerticalAlignment="Bottom"
Source="{StaticResource info_circleDrawingImage}" />
<TextBlock
Margin="0,0,0,5"
<Viewbox Stretch="Uniform" Height="16">
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Bottom"
Padding="2,0,2,0"
Text="{DynamicResource MainWindow:IntroButton}" />
</Viewbox>
</StackPanel>
</Button>
@ -84,11 +85,12 @@
Height="30"
VerticalAlignment="Bottom"
Source="{StaticResource microchipDrawingImage}" />
<TextBlock
Margin="0,0,0,5"
<Viewbox Stretch="Uniform" Height="16">
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Bottom"
Padding="2,0,2,0"
Text="{DynamicResource MainWindow:ModsButton}" />
</Viewbox>
</StackPanel>
</Button>
@ -104,11 +106,12 @@
Height="30"
VerticalAlignment="Bottom"
Source="{StaticResource heartDrawingImage}" />
<TextBlock
Margin="0,0,0,5"
<Viewbox Stretch="Uniform" Height="16">
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Bottom"
Padding="2,0,2,0"
Text="{DynamicResource MainWindow:AboutButton}" />
</Viewbox>
</StackPanel>
</Button>
@ -124,32 +127,14 @@
Height="30"
VerticalAlignment="Bottom"
Source="{StaticResource cogDrawingImage}" />
<TextBlock
Margin="0,0,0,5"
<Viewbox Stretch="Uniform" Height="16">
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Bottom"
Padding="2,0,2,0"
Text="{DynamicResource MainWindow:OptionsButton}" />
</Viewbox>
</StackPanel>
</Button>
<StackPanel
Name="GameVersions"
Grid.Row="5"
VerticalAlignment="Bottom">
<TextBlock FontSize="10">
<TextBlock Text="{DynamicResource MainWindow:GameVersionLabel}" />
:
</TextBlock>
<ComboBox
Name="GameVersionsBox"
Margin="0,5,5,0"
SelectionChanged="GameVersionsBox_SelectionChanged" />
<ComboBox
Name="GameVersionsBoxOverride"
Margin="0,5,5,0"
Visibility="Collapsed" />
</StackPanel>
</Grid>
<StackPanel Grid.Row="1" VerticalAlignment="Center">
@ -166,8 +151,8 @@
<Grid Grid.Row="1" Grid.Column="1">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
<ColumnDefinition Width="115" />
<ColumnDefinition Width="115" />
<ColumnDefinition Width="auto"/>
<ColumnDefinition Width="auto" />
</Grid.ColumnDefinitions>
<Border
@ -184,9 +169,10 @@
<Button
Name="InfoButton"
Grid.Column="1"
Width="100"
Height="40"
Margin="0,10,0,0"
MinWidth="115"
Margin="10,10,0,0"
Padding="20,0,20,0"
HorizontalAlignment="Right"
Click="InfoButton_Click"
IsEnabled="False">
@ -197,13 +183,13 @@
Text="{DynamicResource MainWindow:ModInfoButton}" />
</StackPanel>
</Button>
<Button
Name="InstallButton"
Grid.Column="2"
Width="100"
Height="40"
Margin="0,10,0,0"
MinWidth="115"
Margin="10,10,0,0"
Padding="20,0,20,0"
HorizontalAlignment="Right"
Click="InstallButton_Click"
IsEnabled="False">

View file

@ -0,0 +1,219 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;
using VRCMelonAssistant.Pages;
using static VRCMelonAssistant.Http;
namespace VRCMelonAssistant
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public static MainWindow Instance;
public static bool ModsOpened = false;
public static bool ModsLoading = false;
public string MainText
{
get
{
return MainTextBlock.Text;
}
set
{
Dispatcher.Invoke(new Action(() => { Instance.MainTextBlock.Text = value; }));
}
}
public MainWindow()
{
InitializeComponent();
Instance = this;
const int ContentWidth = 1280;
const int ContentHeight = 720;
double ChromeWidth = SystemParameters.WindowNonClientFrameThickness.Left + SystemParameters.WindowNonClientFrameThickness.Right;
double ChromeHeight = SystemParameters.WindowNonClientFrameThickness.Top + SystemParameters.WindowNonClientFrameThickness.Bottom;
double ResizeBorder = SystemParameters.ResizeFrameVerticalBorderWidth;
Width = ChromeWidth + ContentWidth + 2 * ResizeBorder;
Height = ChromeHeight + ContentHeight + 2 * ResizeBorder;
VersionText.Text = App.Version;
Themes.LoadThemes();
Themes.FirstLoad(Properties.Settings.Default.SelectedTheme);
if (Properties.Settings.Default.Agreed)
Instance.ModsButton.IsEnabled = true;
if (!Properties.Settings.Default.Agreed || string.IsNullOrEmpty(Properties.Settings.Default.LastTab))
{
Main.Content = Intro.Instance;
}
else
{
switch (Properties.Settings.Default.LastTab)
{
case "Intro":
Main.Content = Intro.Instance;
break;
case "Mods":
ShowModsPage().NoAwait();
break;
case "About":
Main.Content = About.Instance;
break;
case "Options":
Main.Content = Options.Instance;
Themes.LoadThemes();
break;
default:
Main.Content = Intro.Instance;
break;
}
}
}
/* Force the app to shutdown when The main window is closed.
*
* Explaination:
* OneClickStatus is initialized as a static object,
* so the window will exist, even if it is unused.
* This would cause VRChat Melon Assistant to not shutdown,
* because technically a window was still open.
*/
protected override void OnClosed(EventArgs e)
{
base.OnClosed(e);
Application.Current.Shutdown();
}
internal void MarkModsPageForRefresh()
{
ModsOpened = false;
InstallButton.IsEnabled = false;
}
public async Task ShowModsPage()
{
void OpenModsPage()
{
Main.Content = Mods.Instance;
Properties.Settings.Default.LastTab = "Mods";
Properties.Settings.Default.Save();
Mods.Instance.RefreshColumns();
}
if (ModsOpened && !Mods.Instance.PendingChanges)
{
OpenModsPage();
return;
}
Main.Content = Loading.Instance;
if (ModsLoading) return;
ModsLoading = true;
await Mods.Instance.LoadMods();
ModsLoading = false;
if (ModsOpened == false) ModsOpened = true;
if (Mods.Instance.PendingChanges == true) Mods.Instance.PendingChanges = false;
if (Main.Content == Loading.Instance)
{
OpenModsPage();
}
}
private void ModsButton_Click(object sender, RoutedEventArgs e)
{
ShowModsPage().NoAwait();
}
private void IntroButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = Intro.Instance;
Properties.Settings.Default.LastTab = "Intro";
Properties.Settings.Default.Save();
}
private void AboutButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = About.Instance;
Properties.Settings.Default.LastTab = "About";
Properties.Settings.Default.Save();
}
private void OptionsButton_Click(object sender, RoutedEventArgs e)
{
Main.Content = Options.Instance;
Themes.LoadThemes();
Properties.Settings.Default.LastTab = "Options";
Properties.Settings.Default.Save();
}
private void InstallButton_Click(object sender, RoutedEventArgs e)
{
Mods.Instance.InstallMods();
}
private void InfoButton_Click(object sender, RoutedEventArgs e)
{
if ((Mods.ModListItem)Mods.Instance.ModsListView.SelectedItem == null)
{
MessageBox.Show((string)Application.Current.FindResource("MainWindow:NoModSelected"));
return;
}
Mods.ModListItem mod = ((Mods.ModListItem)Mods.Instance.ModsListView.SelectedItem);
ShowModInfoWindow(mod.ModInfo);
}
internal static void ShowModInfoWindow(Mod mod)
{
var infoWindow = new ModInfoWindow();
infoWindow.SetMod(mod);
infoWindow.Owner = Instance;
infoWindow.WindowStartupLocation = WindowStartupLocation.CenterOwner;
infoWindow.ShowDialog();
}
private void Window_PreviewMouseDown(object sender, MouseButtonEventArgs e)
{
if (About.Instance.PatUp.IsOpen)
{
About.Instance.PatUp.IsOpen = false;
About.Instance.PatButton.IsEnabled = true;
}
if (About.Instance.HugUp.IsOpen)
{
About.Instance.HugUp.IsOpen = false;
About.Instance.HugButton.IsEnabled = true;
}
}
private void Window_SizeChanged(object sender, SizeChangedEventArgs e)
{
if (Main.Content == Mods.Instance)
{
Mods.Instance.RefreshColumns();
}
}
private void BackgroundVideo_MediaEnded(object sender, RoutedEventArgs e)
{
BackgroundVideo.Position = TimeSpan.Zero;
BackgroundVideo.Play();
}
}
}

View file

@ -0,0 +1,26 @@
<Window x:Class="VRCMelonAssistant.ModInfoWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:VRCMelonAssistant"
mc:Ignorable="d"
Height="450" Width="800">
<Grid>
<Rectangle Fill="{DynamicResource VRCMelonAssistantBackground}" />
<ScrollViewer HorizontalScrollBarVisibility="Disabled" VerticalScrollBarVisibility="Auto">
<StackPanel Orientation="Vertical" Margin="10">
<TextBlock FontSize="16" Name="ModName" />
<TextBlock Name="ModVersion"/>
<TextBlock Name="ModAuthor" />
<TextBlock />
<TextBlock Name="ModDescription" TextWrapping="Wrap" />
<TextBlock />
<TextBlock Name="SourceCodeLink" />
<TextBlock Name="DownloadLink" />
<TextBlock Name="InternalIds" />
</StackPanel>
</ScrollViewer>
</Grid>
</Window>

View file

@ -0,0 +1,50 @@
using System;
using System.Windows;
using System.Windows.Documents;
namespace VRCMelonAssistant
{
public partial class ModInfoWindow : Window
{
public ModInfoWindow()
{
InitializeComponent();
}
public void SetMod(Mod mod)
{
Title = string.Format((string) FindResource("ModInfoWindow:Title"), mod.versions[0].name);
ModDescription.Text = mod.versions[0].description ?? (string) FindResource("ModInfoWindow:NoDescription");
ModName.Text = mod.versions[0].name;
ModAuthor.Text = string.Format((string) FindResource("ModInfoWindow:Author"), mod.versions[0].author ?? FindResource("ModInfoWindow:NoAuthor"));
ModVersion.Text = mod.versions[0].modVersion;
var dlLink = mod.versions[0].downloadLink;
DownloadLink.Text = (string) FindResource("ModInfoWindow:DownloadLink");
DownloadLink.Inlines.Add(new Run(" "));
if (dlLink?.StartsWith("http") == true)
DownloadLink.Inlines.Add(CreateHyperlink(dlLink));
else
DownloadLink.Inlines.Add(new Run(dlLink));
var srcLink = mod.versions[0].sourceLink;
SourceCodeLink.Text = (string) FindResource("ModInfoWindow:SourceCodeLink");
SourceCodeLink.Inlines.Add(new Run(" "));
if (srcLink?.StartsWith("http") == true)
SourceCodeLink.Inlines.Add(CreateHyperlink(srcLink));
else
SourceCodeLink.Inlines.Add(new Run(srcLink));
InternalIds.Text = string.Format((string) FindResource("ModInfoWindow:InternalIds"), mod._id, mod.versions[0]._version);
}
private static Hyperlink CreateHyperlink(string uri)
{
var link = new Hyperlink(new Run(uri)) {NavigateUri = new Uri(uri)};
link.RequestNavigate += HyperlinkExtensions.Hyperlink_RequestNavigate;
return link;
}
}
}

View file

@ -1,9 +1,9 @@
<Page
x:Class="ModAssistant.Pages.About"
x:Class="VRCMelonAssistant.Pages.About"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:wfi="clr-namespace:System.Windows.Forms.Integration;assembly=WindowsFormsIntegration"
xmlns:winForms="clr-namespace:System.Windows.Forms;assembly=System.Windows.Forms"
@ -41,34 +41,7 @@
FontSize="16"
Text="{DynamicResource About:List:Header}"
TextWrapping="Wrap" />
<TextBlock
Grid.Row="2"
Margin="15,5"
FontSize="16">
<TextBlock Text="{DynamicResource About:List:Item1}" />
</TextBlock>
<TextBlock
Grid.Row="3"
Margin="15,5"
FontSize="16">
<TextBlock Text="{DynamicResource About:List:Item2}" />
</TextBlock>
<TextBlock
Grid.Row="4"
Margin="15,5"
FontSize="16">
<TextBlock Text="{DynamicResource About:List:Item3}" />
</TextBlock>
<TextBlock
Grid.Row="5"
Margin="15,5"
FontSize="16">
<TextBlock Text="{DynamicResource About:List:Item4}" />
</TextBlock>
<TextBlock
Grid.Row="6"
@ -89,6 +62,38 @@
HorizontalAlignment="Center"
Orientation="Horizontal">
<StackPanel Margin="10" Orientation="Vertical">
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Center"
FontSize="20"
FontWeight="Bold">
<Hyperlink NavigateUri="https://github.com/Assistant/" RequestNavigate="Hyperlink_RequestNavigate">
Assistant
</Hyperlink>
</TextBlock>
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Center"
FontSize="16">
Creating the original
</TextBlock>
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Center"
FontSize="16">
Mod Assistant
</TextBlock>
<TextBlock
HorizontalAlignment="Center"
VerticalAlignment="Center"
FontSize="16">
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://ko-fi.com/N4N8JX7B">
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
<StackPanel Margin="10" Orientation="Vertical">
<TextBlock
HorizontalAlignment="Center"
@ -116,7 +121,7 @@
VerticalAlignment="Center"
FontSize="16">
<Hyperlink local:HyperlinkExtensions.IsExternal="True" NavigateUri="https://www.patreon.com/scoresaber">
<TextBlock Text="{DynamicResource About:Donate}" />
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
@ -154,7 +159,7 @@
VerticalAlignment="Center"
FontSize="16">
<Hyperlink NavigateUri="https://www.paypal.me/jackbarondev" RequestNavigate="Hyperlink_RequestNavigate">
<TextBlock Text="{DynamicResource About:Donate}" />
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
@ -180,7 +185,7 @@
VerticalAlignment="Center"
FontSize="16">
<Hyperlink NavigateUri="https://ko-fi.com/Caeden117" RequestNavigate="Hyperlink_RequestNavigate">
<TextBlock Text="{DynamicResource About:Donate}" />
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
@ -206,7 +211,7 @@
VerticalAlignment="Center"
FontSize="16">
<Hyperlink NavigateUri="https://streamlabs.com/lnterz/tip" RequestNavigate="Hyperlink_RequestNavigate">
<TextBlock Text="{DynamicResource About:Donate}" />
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
@ -238,7 +243,7 @@
VerticalAlignment="Center"
FontSize="16">
<Hyperlink NavigateUri="https://ko-fi.com/megalon" RequestNavigate="Hyperlink_RequestNavigate">
<TextBlock Text="{DynamicResource About:Donate}" />
<Run Text="{DynamicResource About:Donate}" />
</Hyperlink>
</TextBlock>
</StackPanel>
@ -251,17 +256,19 @@
Orientation="Horizontal">
<Button
x:Name="PatButton"
Width="80"
Height="30"
MinWidth="80"
Margin="0,0,5,0"
Padding="20,0,20,0"
x:FieldModifier="public"
Click="HeadpatsButton_Click"
Content="{DynamicResource About:HeadpatsButton}" />
<Button
x:Name="HugButton"
Width="80"
Height="30"
MinWidth="80"
Margin="5,0,0,0"
Padding="20,0,20,0"
x:FieldModifier="public"
Click="HugsButton_Click"
Content="{DynamicResource About:HugsButton}" />

View file

@ -4,9 +4,9 @@ using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Navigation;
using static ModAssistant.Http;
using static VRCMelonAssistant.Http;
namespace ModAssistant.Pages
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Page1.xaml
@ -51,12 +51,32 @@ namespace ModAssistant.Pages
private async Task HeadPat()
{
PatImage.Load(await WeebCDN("pats"));
try
{
PatImage.Load(await WeebCDN("pats"));
}
catch (Exception ex)
{
Application.Current.Dispatcher.Invoke(() =>
{
Utils.ShowErrorMessageBox("Oops! Can't get headpats right now!", ex);
});
}
}
private async Task Hug()
{
HugImage.Load(await WeebCDN("hugs"));
try
{
HugImage.Load(await WeebCDN("hugs"));
}
catch (Exception ex)
{
Application.Current.Dispatcher.Invoke(() =>
{
Utils.ShowErrorMessageBox("Oops! Can't get hugs right now!", ex);
});
}
}
}
}

View file

@ -1,9 +1,9 @@
<Page
x:Class="ModAssistant.Pages.Intro"
x:Class="VRCMelonAssistant.Pages.Intro"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="DynamicResource Intro:Title"
d:DesignHeight="629"
@ -23,6 +23,8 @@
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
</Grid.RowDefinitions>
@ -59,7 +61,7 @@
Margin="15,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:Terms:Term1" />
<StaticResource ResourceKey="Intro:Terms:Term0" />
</TextBlock>
<TextBlock
@ -67,7 +69,7 @@
Margin="15,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:Terms:Term2" />
<StaticResource ResourceKey="Intro:Terms:Term1" />
</TextBlock>
<TextBlock
@ -75,15 +77,15 @@
Margin="15,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:Terms:Term3" />
<StaticResource ResourceKey="Intro:Terms:Term2" />
</TextBlock>
<TextBlock
Grid.Row="7"
Margin="0,5"
Margin="15,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:ReviewsBeatGamesFault" />
<StaticResource ResourceKey="Intro:Terms:Term3" />
</TextBlock>
<TextBlock
@ -91,7 +93,7 @@
Margin="0,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:ReviewsRustySpoon" />
<StaticResource ResourceKey="Intro:ReviewsBeatGamesFault" />
</TextBlock>
<TextBlock
@ -99,20 +101,36 @@
Margin="0,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:ReportIssuesNoMods" />
</TextBlock>
<TextBlock
Grid.Row="10"
Margin="0,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:ReviewsRustySpoon" />
</TextBlock>
<TextBlock
Grid.Row="11"
Margin="0,5"
FontSize="16"
TextWrapping="Wrap">
<StaticResource ResourceKey="Intro:WikiGuide" />
</TextBlock>
<StackPanel
Grid.Row="10"
Grid.Row="12"
Margin="10"
HorizontalAlignment="Center"
VerticalAlignment="Center"
Orientation="Horizontal">
<Button
Name="Agree"
Width="100"
Height="35"
Margin="0,0,10,0"
Padding="20,0,20,0"
HorizontalAlignment="Center"
VerticalAlignment="Center"
Click="Agree_Click">
@ -124,9 +142,9 @@
</Button>
<Button
Name="Disagree"
Width="100"
Height="35"
Margin="10,0,0,0"
Padding="20,0,20,0"
HorizontalAlignment="Center"
VerticalAlignment="Center"
Click="Disagree_Click">

View file

@ -1,10 +1,9 @@
using System;
using System.Diagnostics;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Navigation;
namespace ModAssistant.Pages
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Intro.xaml
@ -30,28 +29,19 @@ namespace ModAssistant.Pages
Properties.Settings.Default.Agreed = false;
Properties.Settings.Default.Save();
MessageBox.Show((string)FindResource("Intro:ClosingApp"));
System.Windows.Application.Current.Shutdown();
Application.Current.Shutdown();
}
private void Agree_Click(object sender, RoutedEventArgs e)
{
if (string.IsNullOrEmpty(MainWindow.GameVersion))
{
string line1 = (string)FindResource("Intro:VersionDownloadFailed");
string line2 = (string)FindResource("Intro:ModsTabDisabled");
MainWindow.Instance.ModsButton.IsEnabled = true;
MessageBox.Show($"{line1}.\n{line2}");
}
else
{
MainWindow.Instance.ModsButton.IsEnabled = true;
string text = (string)FindResource("Intro:ModsTabEnabled");
Utils.SendNotify(text);
MainWindow.Instance.MainText = text;
}
string text = (string) FindResource("Intro:ModsTabEnabled");
MainWindow.Instance.MainText = text;
Properties.Settings.Default.Agreed = true;
Properties.Settings.Default.Save();
MainWindow.Instance.ShowModsPage().NoAwait();
}
}
}

View file

@ -1,9 +1,9 @@
<Page
x:Class="ModAssistant.Pages.Invalid"
x:Class="VRCMelonAssistant.Pages.Invalid"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant.Pages"
xmlns:local="clr-namespace:VRCMelonAssistant.Pages"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="{DynamicResource Invalid:Title}"
d:DesignHeight="629"

View file

@ -1,10 +1,9 @@
using System;
using System.Diagnostics;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Navigation;
namespace ModAssistant.Pages
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Invalid.xaml
@ -17,7 +16,7 @@ namespace ModAssistant.Pages
public Invalid()
{
InitializeComponent();
InstallDirectory = App.BeatSaberInstallDirectory;
InstallDirectory = App.VRChatInstallDirectory;
DirectoryTextBlock.Text = InstallDirectory;
}

View file

@ -1,9 +1,9 @@
<Page
x:Class="ModAssistant.Pages.Loading"
x:Class="VRCMelonAssistant.Pages.Loading"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant.Pages"
xmlns:local="clr-namespace:VRCMelonAssistant.Pages"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="Loading"
d:DesignHeight="450"
@ -43,18 +43,6 @@
<RowDefinition Height="auto" />
</Grid.RowDefinitions>
<Image
Grid.Row="0"
Margin="100,100,100,0"
VerticalAlignment="Center"
Source="{DynamicResource loadingInnerDrawingImage}"
Stretch="Uniform" />
<Image
Grid.Row="0"
Margin="100,100,100,0"
VerticalAlignment="Center"
Source="{DynamicResource loadingMiddleDrawingImage}"
Stretch="Uniform" />
<Image
Grid.Row="0"
Margin="100,100,100,0"

View file

@ -2,7 +2,7 @@ using System;
using System.Windows.Controls;
using System.Windows.Data;
namespace ModAssistant.Pages
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Loading.xaml

View file

@ -0,0 +1,164 @@
<Page
x:Class="VRCMelonAssistant.Pages.Mods"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:system="clr-namespace:System;assembly=mscorlib"
Title="{DynamicResource Mods:Title}"
d:DesignHeight="629"
d:DesignWidth="1182"
Loaded="Page_Loaded"
mc:Ignorable="d">
<Grid>
<Grid x:Name="NoModsGrid" Visibility="Visible">
<TextBlock HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32" FontWeight="SemiBold">
<TextBlock Text="{DynamicResource Mods:NoMods}" />.
</TextBlock>
</Grid>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="auto" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<TextBlock
Name="SearchText"
Grid.Row="0"
Height="20"
Padding="5,0,0,0"
Panel.ZIndex="1"
Background="{DynamicResource BottomStatusBarBackground}"
Foreground="{DynamicResource TextColor}"
Text="{DynamicResource Mods:SearchLabel}" />
<TextBox
Name="SearchBar"
Grid.Row="0"
Height="20"
Margin="0,-1,0,0"
Padding="3,1,0,0"
Panel.ZIndex="2"
Background="#00000000"
BorderThickness="0"
Foreground="{DynamicResource TextColor}"
TextChanged="SearchBar_TextChanged" />
<ListView
Name="ModsListView"
Grid.Row="1"
Grid.Column="0"
SelectionChanged="ModsListView_SelectionChanged"
SelectionMode="Single"
MouseDoubleClick="ModsListView_OnMouseDoubleClick">
<ListView.View>
<GridView>
<GridView.Columns>
<GridViewColumn Width="30">
<GridViewColumn.Header>
<Button
Name="SearchButton"
Margin="-5"
Padding="9,-1,9,0"
Background="#00000000"
BorderThickness="0"
Click="SearchButton_Click"
Content="🔍"
FontSize="11" />
</GridViewColumn.Header>
<GridViewColumn.CellTemplate>
<DataTemplate>
<CheckBox
Name="ModCheckBox"
Checked="ModCheckBox_Checked"
IsChecked="{Binding Path=IsSelected, Mode=TwoWay}"
IsEnabled="{Binding IsEnabled}"
Tag="{Binding ModInfo}"
Unchecked="ModCheckBox_Unchecked" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn DisplayMemberBinding="{Binding ModName}" Header="{DynamicResource Mods:Header:Name}" />
<GridViewColumn Header="{DynamicResource Mods:Header:Author}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBlock
Text="{Binding ModAuthor}"
MaxWidth="150" TextWrapping="NoWrap" TextTrimming="WordEllipsis"
/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn x:Name="InstalledColumn" Header="{DynamicResource Mods:Header:Installed}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBlock
Foreground="{Binding GetVersionColor}"
Text="{Binding InstalledVersion}"
TextDecorations="{Binding GetVersionDecoration}" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn DisplayMemberBinding="{Binding ModVersion}" Header="{DynamicResource Mods:Header:Latest}" />
<GridViewColumn x:Name="DescriptionColumn" Header="{DynamicResource Mods:Header:Description}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Margin="{Binding PromotionMargin}" Visibility="{Binding PromotionVisibility}">
<Hyperlink NavigateUri="{Binding PromotionLink, TargetNullValue=about:blank}" RequestNavigate="Hyperlink_RequestNavigate">
<Run Text="{Binding PromotionText}" />
</Hyperlink>
</TextBlock>
<TextBlock Text="{Binding ModDescription}" MouseLeftButtonDown="ModsListView_OnMouseDoubleClick" />
</StackPanel>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn
x:Name="UninstallColumn"
Width="70"
Header="{DynamicResource Mods:Header:Uninstall}">
<GridViewColumn.CellTemplate>
<DataTemplate>
<Button
Name="Uninstall"
Click="Uninstall_Click"
Content="{DynamicResource Mods:UninstallButton}"
Foreground="Red"
IsEnabled="{Binding CanDelete}"
Tag="{Binding ModInfo}"
Visibility="{Binding CanSeeDelete}" />
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
</GridView.Columns>
</GridView>
</ListView.View>
<ListView.GroupStyle>
<GroupStyle>
<GroupStyle.HeaderTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Padding="6,0,0,0">
<Run FontSize="16" FontWeight="Bold" Text="{Binding Name.Name, Mode=OneWay}" />
<Run FontSize="12" FontStyle="Italic" Text="{Binding Name.Description, Mode=OneWay}" />
</TextBlock>
</StackPanel>
</DataTemplate>
</GroupStyle.HeaderTemplate>
</GroupStyle>
</ListView.GroupStyle>
<ListView.ItemsPanel>
<ItemsPanelTemplate>
<VirtualizingStackPanel />
</ItemsPanelTemplate>
</ListView.ItemsPanel>
</ListView>
</Grid>
</Grid>
</Page>

View file

@ -0,0 +1,577 @@
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Net;
using System.Threading;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Forms;
using System.Windows.Input;
using System.Windows.Media.Animation;
using System.Windows.Navigation;
using Mono.Cecil;
using VRCMelonAssistant.Libs;
using static VRCMelonAssistant.Http;
using MessageBox = System.Windows.MessageBox;
using TextBox = System.Windows.Controls.TextBox;
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Mods.xaml
/// </summary>
public sealed partial class Mods : Page
{
public static Mods Instance = new Mods();
private static readonly ModListItem.CategoryInfo BrokenCategory = new("Broken", "These mods were broken by a game update. They will be temporarily removed and restored once they are updated for the current game version");
private static readonly ModListItem.CategoryInfo RetiredCategory = new("Retired", "These mods are either no longer needed due to VRChat updates or are no longer being maintained");
private static readonly ModListItem.CategoryInfo UncategorizedCategory = new("Uncategorized", "Mods without a category assigned");
private static readonly ModListItem.CategoryInfo UnknownCategory = new("Unknown/Unverified", "Mods not coming from VRCMG. Potentially dangerous.");
public List<string> DefaultMods = new List<string>() { "UI Expansion Kit", "Finitizer", "VRCModUpdater.Loader", "VRChatUtilityKit", "Final IK Sanity", "ActionMenuApi" };
public Mod[] AllModsList;
public List<Mod> UnknownMods = new List<Mod>();
public CollectionView view;
public bool PendingChanges;
public bool HaveInstalledMods;
private readonly SemaphoreSlim _modsLoadSem = new SemaphoreSlim(1, 1);
public List<ModListItem> ModList { get; set; }
public Mods()
{
InitializeComponent();
}
private void RefreshModsList()
{
view?.Refresh();
}
public void RefreshColumns()
{
if (MainWindow.Instance.Main.Content != Instance) return;
double viewWidth = ModsListView.ActualWidth;
double totalSize = 0;
GridViewColumn description = null;
if (ModsListView.View is GridView grid)
{
foreach (var column in grid.Columns)
{
if (column.Header?.ToString() == FindResource("Mods:Header:Description").ToString())
{
description = column;
}
else
{
totalSize += column.ActualWidth;
}
if (double.IsNaN(column.Width))
{
column.Width = column.ActualWidth;
column.Width = double.NaN;
}
}
double descriptionNewWidth = viewWidth - totalSize - 35;
description.Width = descriptionNewWidth > 200 ? descriptionNewWidth : 200;
}
}
public async Task LoadMods()
{
await _modsLoadSem.WaitAsync();
try
{
MainWindow.Instance.InstallButton.IsEnabled = false;
MainWindow.Instance.InfoButton.IsEnabled = false;
AllModsList = null;
ModList = new List<ModListItem>();
UnknownMods.Clear();
HaveInstalledMods = false;
ModsListView.Visibility = Visibility.Hidden;
MainWindow.Instance.MainText = $"{FindResource("Mods:CheckingInstalledMods")}...";
await CheckInstalledMods();
InstalledColumn.Width = double.NaN;
UninstallColumn.Width = 70;
DescriptionColumn.Width = 750;
MainWindow.Instance.MainText = $"{FindResource("Mods:LoadingMods")}...";
await PopulateModsList();
ModsListView.ItemsSource = ModList;
view = (CollectionView)CollectionViewSource.GetDefaultView(ModsListView.ItemsSource);
PropertyGroupDescription groupDescription = new PropertyGroupDescription("Category");
view.GroupDescriptions.Add(groupDescription);
this.DataContext = this;
RefreshModsList();
ModsListView.Visibility = ModList.Count == 0 ? Visibility.Hidden : Visibility.Visible;
NoModsGrid.Visibility = ModList.Count == 0 ? Visibility.Visible : Visibility.Hidden;
MainWindow.Instance.MainText = $"{FindResource("Mods:FinishedLoadingMods")}.";
MainWindow.Instance.InstallButton.IsEnabled = ModList.Count != 0;
}
finally
{
_modsLoadSem.Release();
}
}
public async Task CheckInstalledMods()
{
await GetAllMods();
await Task.Run(() =>
{
CheckInstallDir("Plugins");
CheckInstallDir("Mods");
CheckInstallDir("Plugins/Broken", isBrokenDir: true);
CheckInstallDir("Mods/Broken", isBrokenDir: true);
CheckInstallDir("Plugins/Retired", isRetiredDir: true);
CheckInstallDir("Mods/Retired", isRetiredDir: true);
});
}
public async Task GetAllMods()
{
try
{
var resp = await HttpClient.GetAsync(Utils.Constants.VRCMGModsJson);
var body = await resp.Content.ReadAsStringAsync();
AllModsList = JsonSerializer.Deserialize<Mod[]>(body);
foreach (var mod in AllModsList)
mod.category ??= HardcodedCategories.GetCategoryFor(mod) ?? "Uncategorized";
Array.Sort(AllModsList, (a, b) =>
{
var categoryCompare = String.Compare(a.category, b.category, StringComparison.Ordinal);
if (categoryCompare != 0) return categoryCompare;
return String.Compare(a.versions[0].name, b.versions[0].name, StringComparison.Ordinal);
});
}
catch (Exception e)
{
System.Windows.MessageBox.Show($"{FindResource("Mods:LoadFailed")}.\n\n" + e);
}
}
private void CheckInstallDir(string directory, bool isBrokenDir = false, bool isRetiredDir = false)
{
if (!Directory.Exists(Path.Combine(App.VRChatInstallDirectory, directory)))
{
return;
}
foreach (string file in Directory.GetFileSystemEntries(Path.Combine(App.VRChatInstallDirectory, directory), "*.dll", SearchOption.TopDirectoryOnly))
{
if (!File.Exists(file) || Path.GetExtension(file) != ".dll") continue;
var modInfo = ExtractModVersions(file);
if (modInfo.Item1 != null && modInfo.Item2 != null)
{
var haveFoundMod = false;
foreach (var mod in AllModsList)
{
if (!mod.aliases.Contains(modInfo.ModName) && mod.versions.All(it => it.name != modInfo.ModName)) continue;
HaveInstalledMods = true;
haveFoundMod = true;
mod.installedFilePath = file;
mod.installedVersion = modInfo.ModVersion;
mod.installedInBrokenDir = isBrokenDir;
mod.installedInRetiredDir = isRetiredDir;
break;
}
if (!haveFoundMod)
{
var mod = new Mod()
{
installedFilePath = file,
installedVersion = modInfo.ModVersion,
installedInBrokenDir = isBrokenDir,
installedInRetiredDir = isRetiredDir,
versions = new []
{
new Mod.ModVersion()
{
name = modInfo.ModName,
modVersion = modInfo.ModVersion,
author = modInfo.ModAuthor,
description = ""
}
}
};
UnknownMods.Add(mod);
}
}
}
}
private (string ModName, string ModVersion, string ModAuthor) ExtractModVersions(string dllPath)
{
try
{
using var asmdef = AssemblyDefinition.ReadAssembly(dllPath);
foreach (var attr in asmdef.CustomAttributes)
if (attr.AttributeType.Name == "MelonInfoAttribute" ||
attr.AttributeType.Name == "MelonModInfoAttribute")
return ((string) attr.ConstructorArguments[1].Value,
(string) attr.ConstructorArguments[2].Value, (string) attr.ConstructorArguments[3].Value);
}
catch (Exception ex)
{
var result = MessageBox.Show(
$"A mod in {Path.GetFileName(dllPath)} is invalid. Would you like to delete it to avoid this error in the future?",
"Invalid mod", MessageBoxButton.YesNo, MessageBoxImage.Warning);
if (result == MessageBoxResult.Yes)
{
try
{
File.Delete(dllPath);
}
catch (Exception ex2)
{
Utils.ShowErrorMessageBox($"Unable to delete file {dllPath}", ex2);
}
}
}
return (null, null, null);
}
public async Task PopulateModsList()
{
foreach (Mod mod in AllModsList.Where(x => !x.versions[0].IsBroken && !x.versions[0].IsRetired))
AddModToList(mod);
foreach (var mod in UnknownMods)
AddModToList(mod, UnknownCategory);
foreach (Mod mod in AllModsList.Where(x => x.versions[0].IsBroken))
AddModToList(mod);
foreach (Mod mod in AllModsList.Where(x => x.versions[0].IsRetired))
AddModToList(mod);
}
private void AddModToList(Mod mod, ModListItem.CategoryInfo categoryOverride = null)
{
bool preSelected = false;
var latestVersion = mod.versions[0];
if (DefaultMods.Contains(latestVersion.name) && !HaveInstalledMods || mod.installedFilePath != null)
{
preSelected = true;
}
ModListItem.CategoryInfo GetCategory(Mod mod)
{
if (mod.category == null) return UncategorizedCategory;
return new ModListItem.CategoryInfo(mod.category,
HardcodedCategories.GetCategoryDescription(mod.category));
}
ModListItem ListItem = new ModListItem()
{
IsSelected = preSelected,
IsEnabled = true,
ModName = latestVersion.name,
ModVersion = latestVersion.modVersion,
ModAuthor = HardcodedCategories.FixupAuthor(latestVersion.author),
ModDescription = latestVersion.description.Replace("\r\n", " ").Replace("\n", " "),
ModInfo = mod,
IsInstalled = mod.installedFilePath != null,
InstalledVersion = mod.installedVersion,
InstalledModInfo = mod,
Category = categoryOverride ?? (latestVersion.IsBroken ? BrokenCategory : (latestVersion.IsRetired ? RetiredCategory : GetCategory(mod)))
};
foreach (Promotion promo in Promotions.ActivePromotions)
{
if (latestVersion.name == promo.ModName)
{
ListItem.PromotionText = promo.Text;
ListItem.PromotionLink = promo.Link;
}
}
mod.ListItem = ListItem;
ModList.Add(ListItem);
}
public async void InstallMods()
{
MainWindow.Instance.InstallButton.IsEnabled = false;
if (!InstallHandlers.IsMelonLoaderInstalled())
await InstallHandlers.InstallMelonLoader();
foreach (Mod mod in AllModsList)
{
// Ignore mods that are newer than installed version or up-to-date
if (mod.ListItem.GetVersionComparison >= 0 && mod.installedInBrokenDir == mod.versions[0].IsBroken && mod.installedInRetiredDir == mod.versions[0].IsRetired) continue;
if (mod.ListItem.IsSelected)
{
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstallingMod"), mod.versions[0].name)}...";
await InstallHandlers.InstallMod(mod);
MainWindow.Instance.MainText = $"{string.Format((string)FindResource("Mods:InstalledMod"), mod.versions[0].name)}.";
}
}
MainWindow.Instance.MainText = $"{FindResource("Mods:FinishedInstallingMods")}.";
MainWindow.Instance.InstallButton.IsEnabled = true;
RefreshModsList();
}
private void ModCheckBox_Checked(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.CheckBox).Tag as Mod);
mod.ListItem.IsSelected = true;
}
private void ModCheckBox_Unchecked(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.CheckBox).Tag as Mod);
mod.ListItem.IsSelected = false;
}
public class Category
{
public string CategoryName { get; set; }
public List<ModListItem> Mods = new List<ModListItem>();
}
public class ModListItem
{
public string ModName { get; set; }
public string ModVersion { get; set; }
public string ModAuthor { get; set; }
public string ModDescription { get; set; }
public bool PreviousState { get; set; }
public bool IsEnabled { get; set; }
public bool IsSelected { get; set; }
public Mod ModInfo { get; set; }
public CategoryInfo Category { get; set; }
public Mod InstalledModInfo { get; set; }
public bool IsInstalled { get; set; }
private SemVersion _installedVersion { get; set; }
public string InstalledVersion
{
get
{
if (!IsInstalled || _installedVersion == null) return "-";
return _installedVersion.ToString();
}
set
{
if (SemVersion.TryParse(value, out SemVersion tempInstalledVersion))
{
_installedVersion = tempInstalledVersion;
}
else
{
_installedVersion = null;
}
}
}
public string GetVersionColor
{
get
{
if (!IsInstalled || _installedVersion == null) return "Black";
return _installedVersion >= ModVersion ? "Green" : "Red";
}
}
public string GetVersionDecoration
{
get
{
if (!IsInstalled || _installedVersion == null) return "None";
return _installedVersion >= ModVersion ? "None" : "Strikethrough";
}
}
public int GetVersionComparison
{
get
{
if (!IsInstalled || _installedVersion == null || _installedVersion < ModVersion) return -1;
if (_installedVersion > ModVersion) return 1;
return 0;
}
}
public bool CanDelete => IsInstalled;
public string CanSeeDelete => IsInstalled ? "Visible" : "Hidden";
public string PromotionText { get; set; }
public string PromotionLink { get; set; }
public string PromotionMargin
{
get
{
if (string.IsNullOrEmpty(PromotionText)) return "0";
return "0,0,5,0";
}
}
public Visibility PromotionVisibility => string.IsNullOrEmpty(PromotionText) ? Visibility.Collapsed : Visibility.Visible;
public record CategoryInfo(string Name, string Description)
{
public string Name { get; } = Name;
public string Description { get; } = Description;
}
}
private void ModsListView_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
if ((Mods.ModListItem)Instance.ModsListView.SelectedItem == null)
{
MainWindow.Instance.InfoButton.IsEnabled = false;
}
else
{
MainWindow.Instance.InfoButton.IsEnabled = true;
}
}
private void Uninstall_Click(object sender, RoutedEventArgs e)
{
Mod mod = ((sender as System.Windows.Controls.Button).Tag as Mod);
string title = string.Format((string)FindResource("Mods:UninstallBox:Title"), mod.versions[0].name);
string body1 = string.Format((string)FindResource("Mods:UninstallBox:Body1"), mod.versions[0].name);
string body2 = string.Format((string)FindResource("Mods:UninstallBox:Body2"), mod.versions[0].name);
var result = System.Windows.Forms.MessageBox.Show($"{body1}\n{body2}", title, MessageBoxButtons.YesNo);
if (result == DialogResult.Yes)
{
UninstallModFromList(mod);
}
}
private void UninstallModFromList(Mod mod)
{
UninstallMod(mod.ListItem.InstalledModInfo);
mod.ListItem.IsInstalled = false;
mod.ListItem.InstalledVersion = null;
mod.ListItem.IsSelected = false;
RefreshModsList();
view.Refresh();
}
public void UninstallMod(Mod mod)
{
try
{
File.Delete(mod.installedFilePath);
}
catch (Exception ex)
{
System.Windows.MessageBox.Show($"{FindResource("Mods:UninstallSingleFailed")}.\n\n" + ex);
}
}
private void Hyperlink_RequestNavigate(object sender, RequestNavigateEventArgs e)
{
Process.Start(new ProcessStartInfo(e.Uri.AbsoluteUri));
e.Handled = true;
}
private void Page_Loaded(object sender, RoutedEventArgs e)
{
RefreshColumns();
}
private void SearchButton_Click(object sender, RoutedEventArgs e)
{
if (SearchBar.Height == 0)
{
SearchBar.Focus();
Animate(SearchBar, 0, 20, new TimeSpan(0, 0, 0, 0, 300));
Animate(SearchText, 0, 20, new TimeSpan(0, 0, 0, 0, 300));
ModsListView.Items.Filter = new Predicate<object>(SearchFilter);
}
else
{
Animate(SearchBar, 20, 0, new TimeSpan(0, 0, 0, 0, 300));
Animate(SearchText, 20, 0, new TimeSpan(0, 0, 0, 0, 300));
ModsListView.Items.Filter = null;
}
}
private void SearchBar_TextChanged(object sender, TextChangedEventArgs e)
{
ModsListView.Items.Filter = new Predicate<object>(SearchFilter);
if (SearchBar.Text.Length > 0)
{
SearchText.Text = null;
}
else
{
SearchText.Text = (string)FindResource("Mods:SearchLabel");
}
}
private bool SearchFilter(object mod)
{
ModListItem item = mod as ModListItem;
if (item.ModName.ToLower().Contains(SearchBar.Text.ToLower())) return true;
if (item.ModDescription.ToLower().Contains(SearchBar.Text.ToLower())) return true;
if (item.ModName.ToLower().Replace(" ", string.Empty).Contains(SearchBar.Text.ToLower().Replace(" ", string.Empty))) return true;
if (item.ModDescription.ToLower().Replace(" ", string.Empty).Contains(SearchBar.Text.ToLower().Replace(" ", string.Empty))) return true;
return false;
}
private void Animate(TextBlock target, double oldHeight, double newHeight, TimeSpan duration)
{
target.Height = oldHeight;
DoubleAnimation animation = new DoubleAnimation(newHeight, duration);
target.BeginAnimation(HeightProperty, animation);
}
private void Animate(TextBox target, double oldHeight, double newHeight, TimeSpan duration)
{
target.Height = oldHeight;
DoubleAnimation animation = new DoubleAnimation(newHeight, duration);
target.BeginAnimation(HeightProperty, animation);
}
private void ModsListView_OnMouseDoubleClick(object sender, MouseButtonEventArgs e)
{
if (e.ClickCount != 2) return;
var selectedMod = ModsListView.SelectedItem as ModListItem;
if (selectedMod == null) return;
MainWindow.ShowModInfoWindow(selectedMod.ModInfo);
}
}
}

View file

@ -1,9 +1,9 @@
<Page
x:Class="ModAssistant.Pages.Options"
x:Class="VRCMelonAssistant.Pages.Options"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:ModAssistant.Pages"
xmlns:local="clr-namespace:VRCMelonAssistant.Pages"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="{DynamicResource Options:Title}"
d:DesignHeight="629"
@ -28,6 +28,10 @@
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
<RowDefinition Height="auto" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="auto" />
@ -44,6 +48,23 @@
FontWeight="Bold"
Text="{DynamicResource Options:PageTitle}" />
<TextBlock
Grid.Row="0"
Grid.Column="2"
Margin="5"
HorizontalAlignment="Right"
FontSize="22"
FontWeight="Bold"
Text="A 文" />
<ComboBox
Name="LanguageSelectComboBox"
Grid.Row="0"
Grid.Column="3"
Height="30"
HorizontalAlignment="Stretch"
VerticalContentAlignment="Center"
SelectionChanged="LanguageSelectComboBox_SelectionChanged" />
<TextBlock
Grid.Row="1"
Margin="5"
@ -71,115 +92,20 @@
<Button
Grid.Row="2"
Grid.Column="2"
Width="93"
Height="30"
Margin="3"
Margin="5"
Padding="5"
Click="SelectDirButton_Click"
Content="{DynamicResource Options:SelectFolderButton}" />
<Button
Grid.Row="2"
Grid.Column="3"
Width="93"
Height="30"
Margin="3"
Margin="5"
Padding="5"
Click="OpenDirButton_Click"
Content="{DynamicResource Options:OpenFolderButton}" />
<TextBlock
Grid.Row="3"
Margin="5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold"
Text="{DynamicResource Options:SaveSelectedMods}" />
<CheckBox
Name="SaveSelected"
Grid.Row="3"
Grid.Column="1"
HorizontalAlignment="Left"
VerticalAlignment="Center"
Checked="SaveSelected_Checked"
IsChecked="{Binding SaveSelection, Mode=TwoWay}"
Unchecked="SaveSelected_Unchecked" />
<TextBlock
Grid.Row="4"
Margin="5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold"
Text="{DynamicResource Options:CheckInstalledMods}" />
<CheckBox
Name="CheckInstalled"
Grid.Row="4"
Grid.Column="1"
HorizontalAlignment="Left"
VerticalAlignment="Center"
Checked="CheckInstalled_Checked"
IsChecked="{Binding CheckInstalledMods, Mode=TwoWay}"
Unchecked="CheckInstalled_Unchecked" />
<TextBlock
Grid.Row="5"
Margin="50,5,5,5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold"
Text="{DynamicResource Options:SelectInstalledMods}" />
<CheckBox
Name="SelectInstalled"
Grid.Row="5"
Grid.Column="1"
HorizontalAlignment="Left"
VerticalAlignment="Center"
Checked="SelectInstalled_Checked"
IsChecked="{Binding SelectInstalledMods, Mode=TwoWay}"
Unchecked="SelectInstalled_Unchecked" />
<TextBlock
Grid.Row="6"
Margin="5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold">
<TextBlock Text="{DynamicResource Options:EnableOneClickInstalls}" />
: ↳
</TextBlock>
<TextBlock
Grid.Row="7"
Margin="50,5,5,5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold"
Text="{DynamicResource Options:BeatSaver}" />
<CheckBox
Name="BeatSaverProtocolHandler"
Grid.Row="7"
Grid.Column="1"
HorizontalAlignment="Left"
VerticalAlignment="Center"
Checked="BeatSaverProtocolHandler_Checked"
IsChecked="{Binding BeatSaverProtocolHandlerEnabled}"
Unchecked="BeatSaverProtocolHandler_Unchecked" />
<TextBlock
Grid.Row="8"
Margin="50,5,5,5"
HorizontalAlignment="Left"
FontSize="16"
FontWeight="Bold"
Text="{DynamicResource Options:ModelSaber}" />
<CheckBox
Name="ModelSaberProtocolHandler"
Grid.Row="8"
Grid.Column="1"
HorizontalAlignment="Left"
VerticalAlignment="Center"
Checked="ModelSaberProtocolHandler_Checked"
IsChecked="{Binding ModelSaberProtocolHandlerEnabled}"
Unchecked="ModelSaberProtocolHandler_Unchecked" />
<StackPanel
Grid.Row="12"
Margin="5"
@ -226,61 +152,64 @@
Name="ApplicationThemeExportTemplate"
Grid.Row="13"
Grid.Column="2"
Width="93"
Height="30"
Margin="3"
Margin="5"
Padding="5"
Click="ApplicationThemeExportTemplate_Click"
Content="{DynamicResource Options:ExportTemplateButton}" />
<Button
Name="ApplicationThemeOpenThemesFolder"
Grid.Row="13"
Grid.Column="3"
Width="93"
Height="30"
Margin="3"
Margin="5"
Padding="5"
Click="ApplicationThemeOpenThemesFolder_Click"
Content="{DynamicResource Options:OpenFolderButton}" />
<TextBlock
Grid.Row="14"
Grid.Row="16"
Margin="15,5,5,5"
HorizontalAlignment="Left"
FontSize="24"
FontWeight="Bold"
Text="{DynamicResource Options:Diagnostics}" />
<StackPanel
Grid.Row="15"
Grid.Row="17"
Grid.ColumnSpan="4"
Margin="0"
HorizontalAlignment="Left"
Orientation="Horizontal">
<Button
Width="80"
Height="30"
Margin="5"
Click="OpenLogsDirButton_Click"
Content="{DynamicResource Options:OpenLogsButton}" />
<Button
x:Name="OpenAppData"
Width="100"
Height="30"
Margin="5"
Padding="5"
Click="OpenAppDataButton_Click"
Content="{DynamicResource Options:OpenAppDataButton}" />
<Button
x:Name="YeetBSIPA"
Width="100"
Height="30"
Margin="5"
Click="YeetBSIPAButton_Click"
Content="{DynamicResource Options:UninstallBSIPAButton}" />
<Button
Width="110"
Height="30"
Margin="5"
Padding="5"
Background="{DynamicResource ButtonDangerBackground}"
Click="YeetModsButton_Click">
<TextBlock Foreground="White" Text="{DynamicResource Options:RemoveAllModsButton}" />
</Button>
<Button
Height="30"
Margin="5"
Padding="5"
Background="{DynamicResource ButtonDangerBackground}"
Click="YeetMelonLoaderButton_Click">
<TextBlock Foreground="White" Text="{DynamicResource Options:RemoveMLButton}" />
</Button>
<Button
Height="30"
Margin="5"
Padding="5"
Click="InstallMelonLoaderButton_Click">
<TextBlock Foreground="White" Text="{DynamicResource Options:InstallMLButton}" />
</Button>
</StackPanel>
</Grid>

View file

@ -0,0 +1,187 @@
using System;
using System.Collections.Generic;
using System.Configuration;
using System.Diagnostics;
using System.IO;
using System.Net;
using System.Net.Http;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using Path = System.IO.Path;
namespace VRCMelonAssistant.Pages
{
/// <summary>
/// Interaction logic for Options.xaml
/// </summary>
public partial class Options : Page
{
public static Options Instance = new Options();
public string InstallDirectory { get; set; }
public string InstallType { get; set; }
public bool CloseWindowOnFinish { get; set; }
public string LogURL { get; private set; }
public Options()
{
InitializeComponent();
this.DataContext = this;
}
private void SelectDirButton_Click(object sender, RoutedEventArgs e)
{
Utils.GetManualDir();
DirectoryTextBlock.Text = InstallDirectory;
GameTypeTextBlock.Text = InstallType;
}
private void OpenDirButton_Click(object sender, RoutedEventArgs e)
{
Utils.OpenFolder(InstallDirectory);
}
private void Test_Click(object sender, RoutedEventArgs e)
{
MessageBox.Show(Utils.GetSteamDir());
}
private void CloseWindowOnFinish_Checked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.CloseWindowOnFinish = true;
App.CloseWindowOnFinish = true;
CloseWindowOnFinish = true;
Properties.Settings.Default.Save();
}
private void CloseWindowOnFinish_Unchecked(object sender, RoutedEventArgs e)
{
Properties.Settings.Default.CloseWindowOnFinish = false;
App.CloseWindowOnFinish = false;
CloseWindowOnFinish = false;
Properties.Settings.Default.Save();
}
private void OpenAppDataButton_Click(object sender, RoutedEventArgs e)
{
string location = Path.Combine(
Environment.GetFolderPath(Environment.SpecialFolder.UserProfile),
"AppData", "LocalLow", "VRChat");
if (Directory.Exists(location))
{
Utils.OpenFolder(location);
}
else
{
MessageBox.Show((string)Application.Current.FindResource("Options:AppDataNotFound"));
}
}
private async void YeetModsButton_Click(object sender, RoutedEventArgs e)
{
string title = (string)Application.Current.FindResource("Options:YeetModsBox:Title");
string line1 = (string)Application.Current.FindResource("Options:YeetModsBox:RemoveAllMods");
string line2 = (string)Application.Current.FindResource("Options:YeetModsBox:CannotBeUndone");
var resp = System.Windows.Forms.MessageBox.Show($"{line1}\n{line2}", title, System.Windows.Forms.MessageBoxButtons.YesNo);
if (resp == System.Windows.Forms.DialogResult.Yes)
{
var modsDir = Path.Combine(App.VRChatInstallDirectory, "Mods");
if (Directory.Exists(modsDir))
Directory.Delete(modsDir, true);
var pluginsDir = Path.Combine(App.VRChatInstallDirectory, "Plugins");
if (Directory.Exists(pluginsDir))
Directory.Delete(pluginsDir, true);
Directory.CreateDirectory(modsDir);
Directory.CreateDirectory(pluginsDir);
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:AllModsUninstalled")}...";
}
}
private async void YeetMelonLoaderButton_Click(object sender, RoutedEventArgs e)
{
string title = (string)Application.Current.FindResource("Options:YeetMLBox:Title");
string line1 = (string)Application.Current.FindResource("Options:YeetMLBox:RemoveAllMods");
string line2 = (string)Application.Current.FindResource("Options:YeetMLBox:CannotBeUndone");
var resp = System.Windows.Forms.MessageBox.Show($"{line1}\n{line2}", title, System.Windows.Forms.MessageBoxButtons.YesNo);
if (resp == System.Windows.Forms.DialogResult.Yes)
{
InstallHandlers.RemoveMelonLoader();
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:MLUninstalled")}...";
}
}
private async void InstallMelonLoaderButton_Click(object sender, RoutedEventArgs e)
{
await InstallHandlers.InstallMelonLoader();
MainWindow.Instance.MainText = $"{Application.Current.FindResource("Options:MLInstalled")}...";
}
private void ApplicationThemeComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
if ((sender as ComboBox).SelectedItem == null)
{
Themes.ApplyWindowsTheme();
MainWindow.Instance.MainText = (string)Application.Current.FindResource("Options:CurrentThemeRemoved");
}
else
{
Themes.ApplyTheme((sender as ComboBox).SelectedItem.ToString());
}
}
public void LanguageSelectComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
if ((sender as ComboBox).SelectedItem == null)
{
// Apply default language
Console.WriteLine("Applying default language");
Languages.LoadLanguage("en");
}
else
{
// Get the matching language from the LoadedLanguages array, then try and use it
var languageName = (sender as ComboBox).SelectedItem.ToString();
var selectedLanguage = Languages.LoadedLanguages.Find(language => language.NativeName.CompareTo(languageName) == 0);
if (Languages.LoadLanguage(selectedLanguage.Name))
{
Properties.Settings.Default.LanguageCode = selectedLanguage.Name;
Properties.Settings.Default.Save();
if (Languages.FirstRun)
{
Languages.FirstRun = false;
}
else
{
Process.Start(Utils.ExePath, App.Arguments);
Application.Current.Dispatcher.Invoke(() => { Application.Current.Shutdown(); });
}
}
}
}
private void ApplicationThemeExportTemplate_Click(object sender, RoutedEventArgs e)
{
Themes.WriteThemeToDisk("Ugly Kulu-Ya-Ku");
Themes.LoadThemes();
}
private void ApplicationThemeOpenThemesFolder_Click(object sender, RoutedEventArgs e)
{
if (Directory.Exists(Themes.ThemeDirectory))
{
Utils.OpenFolder(Themes.ThemeDirectory);
}
else
{
MessageBox.Show((string)Application.Current.FindResource("Options:ThemeFolderNotFound"));
}
}
}
}

View file

@ -7,12 +7,12 @@ using System.Windows;
// General Information about an assembly is controlled through the following
// set of attributes. Change these attribute values to modify the information
// associated with an assembly.
[assembly: AssemblyTitle("ModAssistant")]
[assembly: AssemblyTitle("VRChat Melon Assistant")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("")]
[assembly: AssemblyProduct("ModAssistant")]
[assembly: AssemblyCopyright("Copyright © Assistant 2019")]
[assembly: AssemblyProduct("VRChat Melon Assistant")]
[assembly: AssemblyCopyright("Copyright © knah 2021; based on ModAssistant © Assistant 2019-2021")]
[assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")]
@ -51,5 +51,5 @@ using System.Windows;
// You can specify all the values or you can default the Build and Revision Numbers
// by using the '*' as shown below:
// [assembly: AssemblyVersion("1.0.*")]
[assembly: AssemblyVersion("1.1.0.0")]
[assembly: AssemblyFileVersion("1.1.0.0")]
[assembly: AssemblyVersion("1.1.26.1029")]
[assembly: AssemblyFileVersion("1.1.26.1029+vrc")]

View file

@ -8,7 +8,7 @@
// </auto-generated>
//------------------------------------------------------------------------------
namespace ModAssistant.Properties
namespace VRCMelonAssistant.Properties
{
@ -44,7 +44,7 @@ namespace ModAssistant.Properties
{
if ((resourceMan == null))
{
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("ModAssistant.Properties.Resources", typeof(Resources).Assembly);
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("VRCMelonAssistant.Properties.Resources", typeof(Resources).Assembly);
resourceMan = temp;
}
return resourceMan;

View file

@ -1,4 +1,4 @@
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
// <auto-generated>
// This code was generated by a tool.
// Runtime Version:4.0.30319.42000
@ -8,21 +8,21 @@
// </auto-generated>
//------------------------------------------------------------------------------
namespace ModAssistant.Properties {
namespace VRCMelonAssistant.Properties {
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "16.4.0.0")]
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "15.9.0.0")]
public sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase {
private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings())));
public static Settings Default {
get {
return defaultInstance;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
@ -34,7 +34,7 @@ namespace ModAssistant.Properties {
this["InstallFolder"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
@ -46,43 +46,7 @@ namespace ModAssistant.Properties {
this["StoreType"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("False")]
public bool SaveSelected {
get {
return ((bool)(this["SaveSelected"]));
}
set {
this["SaveSelected"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("True")]
public bool CheckInstalled {
get {
return ((bool)(this["CheckInstalled"]));
}
set {
this["CheckInstalled"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("False")]
public string SavedMods {
get {
return ((string)(this["SavedMods"]));
}
set {
this["SavedMods"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("False")]
@ -94,43 +58,7 @@ namespace ModAssistant.Properties {
this["Agreed"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("False")]
public bool SelectInstalled {
get {
return ((bool)(this["SelectInstalled"]));
}
set {
this["SelectInstalled"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
public string GameVersion {
get {
return ((string)(this["GameVersion"]));
}
set {
this["GameVersion"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
public string AllGameVersions {
get {
return ((string)(this["AllGameVersions"]));
}
set {
this["AllGameVersions"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("True")]
@ -142,7 +70,7 @@ namespace ModAssistant.Properties {
this["UpgradeRequired"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
@ -154,7 +82,7 @@ namespace ModAssistant.Properties {
this["LastTab"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
@ -166,5 +94,29 @@ namespace ModAssistant.Properties {
this["SelectedTheme"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("False")]
public bool CloseWindowOnFinish {
get {
return ((bool)(this["CloseWindowOnFinish"]));
}
set {
this["CloseWindowOnFinish"] = value;
}
}
[global::System.Configuration.UserScopedSettingAttribute()]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Configuration.DefaultSettingValueAttribute("")]
public string LanguageCode {
get {
return ((string)(this["LanguageCode"]));
}
set {
this["LanguageCode"] = value;
}
}
}
}

View file

@ -1,5 +1,5 @@
<?xml version='1.0' encoding='utf-8'?>
<SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)" GeneratedClassNamespace="ModAssistant.Properties" GeneratedClassName="Settings">
<?xml version="1.0" encoding="UTF-8"?>
<SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)" GeneratedClassNamespace="VRCMelonAssistant.Properties" GeneratedClassName="Settings">
<Profiles />
<Settings>
<Setting Name="InstallFolder" Type="System.String" Scope="User">
@ -8,27 +8,9 @@
<Setting Name="StoreType" Type="System.String" Scope="User">
<Value Profile="(Default)" />
</Setting>
<Setting Name="SaveSelected" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">False</Value>
</Setting>
<Setting Name="CheckInstalled" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">True</Value>
</Setting>
<Setting Name="SavedMods" Type="System.String" Scope="User">
<Value Profile="(Default)">False</Value>
</Setting>
<Setting Name="Agreed" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">False</Value>
</Setting>
<Setting Name="SelectInstalled" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">False</Value>
</Setting>
<Setting Name="GameVersion" Type="System.String" Scope="User">
<Value Profile="(Default)" />
</Setting>
<Setting Name="AllGameVersions" Type="System.String" Scope="User">
<Value Profile="(Default)" />
</Setting>
<Setting Name="UpgradeRequired" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">True</Value>
</Setting>
@ -38,5 +20,12 @@
<Setting Name="SelectedTheme" Type="System.String" Scope="User">
<Value Profile="(Default)" />
</Setting>
<Setting Name="CloseWindowOnFinish" Type="System.Boolean" Scope="User">
<Value Profile="(Default)">False</Value>
</Setting>
<Setting Name="LanguageCode" Type="System.String" Scope="User">
<Value Profile="(Default)" />
</Setting>
</Settings>
</SettingsFile>

View file

@ -0,0 +1,8 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:VRCMelonAssistant"
xmlns:system="clr-namespace:System;assembly=mscorlib"
xmlns:pages="clr-namespace:VRCMelonAssistant.Pages">
</ResourceDictionary>

View file

@ -26,19 +26,23 @@
</DrawingGroup>
<DrawingImage x:Key="microchipDrawingImage" Drawing="{StaticResource microchipDrawingGroup}" />
<Geometry x:Key="loadingInnerGeometry">F1 M4000,4000z M0,0z M1833.4,1475.6C1827.5,1490 1818.3,1503.3 1805.8,1514.3 1762.7,1551.9 1697.3,1547.4 1659.7,1504.3 1628.8,1468.8 1626.4,1418.3 1650.5,1380.6L1170.6,1131.3 1336.5,1597.2C1394,1576.7 1460.7,1591.9 1503.2,1640.7 1558.7,1704.4 1552.1,1801.1 1488.4,1856.6 1473.5,1869.6 1456.7,1879.1 1439.1,1885.3L1814.9,2940.7 2747.8,2941 2875.1,2016.8 1833.4,1475.6z</Geometry>
<DrawingGroup x:Key="loadingInnerDrawingGroup" ClipGeometry="M0,0 V4000 H4000 V0 H0 Z">
<GeometryDrawing Brush="{DynamicResource LoadingIconColor}" Geometry="{StaticResource loadingInnerGeometry}" />
</DrawingGroup>
<DrawingImage x:Key="loadingInnerDrawingImage" Drawing="{DynamicResource loadingInnerDrawingGroup}" />
<Geometry x:Key="loadingMiddleGeometry">F1 M4000,4000z M0,0z M3384.4,1915.4C3385.5,2231.2 3278.1,2552.4 3057.5,2818.2 2570.1,3405.4 1714.7,3499.3 1146.9,3028 579.1,2556.7 514.1,1698.6 1001.5,1111.5 1222.2,845.7 1518.2,681 1828.8,624 1491,666 1167.9,831.7 933.3,1114.2 444.1,1703.5 525.3,2577.8 1114.5,3067 1703.7,3556.2 2578.1,3475.1 3067.3,2885.8 3301.9,2603.2 3405.2,2255.1 3384.4,1915.4z</Geometry>
<DrawingGroup x:Key="loadingMiddleDrawingGroup" ClipGeometry="M0,0 V4000 H4000 V0 H0 Z">
<GeometryDrawing Brush="{DynamicResource LoadingIconColor}" Geometry="{StaticResource loadingMiddleGeometry}" />
</DrawingGroup>
<DrawingImage x:Key="loadingMiddleDrawingImage" Drawing="{DynamicResource loadingMiddleDrawingGroup}" />
<Geometry x:Key="loadingOuterGeometry">F1 M4000,4000z M0,0z M3710.9,2000L3870,1840.9 3667.5,1638.4C3501.6,870.1 2818.3,294.3 2000.3,294.3 1182.4,294.3 499.1,870 333.2,1638.2L333,1637.9 130,1840.9 289.1,2000 130,2159.1 333,2362.1 333.2,2361.8C499.2,3130 1182.5,3705.6 2000.3,3705.6 2818.3,3705.6 3501.6,3129.8 3667.5,2361.5L3870,2159 3710.9,2000z M2000.3,3489.8C1177.5,3489.8 510.5,2822.8 510.5,2000 510.5,1177.2 1177.5,510.2 2000.3,510.2 2823.1,510.2 3490.1,1177.2 3490.1,2000 3490.1,2822.8 2823.1,3489.8 2000.3,3489.8z</Geometry>
<Geometry x:Key="loadingOuterGeometry">M53.0751762,0.2930124c-0.3906021-0.3907-1.0234032-0.3907-1.4141006,0l-4.7492027,4.7500997
c-0.0007973,0.0008001-0.0019989,0.0009999-0.002697,0.0016999c0,0-0.0009995,0.0019002-0.0017014,0.0027003L0.2929758,51.6696129
c-0.3906,0.3907013-0.3906,1.0235977,0,1.414299c7.2772999,7.2787018,16.8319988,10.9170036,26.3915997,10.9161034
c9.5555973-0.0010033,19.1151981-3.6394043,26.3906002-10.9161034c7.044899-7.0452003,10.9247971-16.4199982,10.9247971-26.3955002
C63.9999733,16.7130127,60.1200752,7.3392124,53.0751762,0.2930124z M25.684576,55.2598114
c-6.6767998-0.2311974-13.2832003-2.7827988-18.5009995-7.6531982L25.684576,29.1025124V55.2598114z M27.684576,29.1020126
l18.3379002,18.3422012c0.0578003,0.0578995,0.129097,0.0895004,0.1959991,0.1301994
c-5.2234001,4.8905983-11.8432007,7.4529991-18.5338993,7.6852989V29.1020126z M47.5670738,46.2267113
c-0.040699-0.0671997-0.0724983-0.1386986-0.1306-0.1968002L29.0985756,27.6877117h26.1591988
C55.0209732,34.6395111,52.3183746,41.1540108,47.5670738,46.2267113z M29.0990753,25.6873131L47.599575,7.1839123
c4.730999,5.0665998,7.4213982,11.5671005,7.6581993,18.5034008H29.0990753z M51.6610756,51.6696129
c-6.6669998,6.6681976-15.5371017,10.339798-24.9764996,10.339798c-9.1055012,0-17.6741009-3.4241982-24.2534008-9.6498985
l3.3382998-3.3386002c5.8694005,5.5032005,13.3885002,8.2630005,20.9151001,8.262001
c7.834898-0.0009995,15.6737995-2.9849014,21.6385975-8.9508018c5.777401-5.7775002,8.9589996-13.4643974,8.9589996-21.6436996
c0-7.8449001-2.9330978-15.230299-8.2684822-20.9188004l3.3373833-3.3378999
c6.2243004,6.5804005,9.6488991,15.1505995,9.6488991,24.2566986C61.9999733,36.129612,58.3280754,45.0014114,51.6610756,51.6696129
z</Geometry>
<DrawingGroup x:Key="loadingOuterDrawingGroup" ClipGeometry="M0,0 V4000 H4000 V0 H0 Z">
<GeometryDrawing Brush="{DynamicResource LoadingIconColor}" Geometry="{StaticResource loadingOuterGeometry}" />
</DrawingGroup>

Binary file not shown.

After

Width:  |  Height:  |  Size: 361 KiB

View file

@ -0,0 +1,8 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<Style TargetType="Menu">
<Setter Property="Background" Value="{DynamicResource ButtonBackground}" />
<Setter Property="BorderBrush" Value="{DynamicResource ButtonBackground}" />
<Setter Property="BorderThickness" Value="1" />
<Setter Property="Foreground" Value="{DynamicResource TextColor}" />
</Style>
</ResourceDictionary>

View file

@ -0,0 +1,66 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<Style TargetType="MenuItem">
<Setter Property="Background" Value="{DynamicResource ButtonBackground}" />
<Setter Property="BorderBrush" Value="{DynamicResource ButtonBackground}" />
<Setter Property="BorderThickness" Value="1" />
<Setter Property="Foreground" Value="{DynamicResource TextColor}" />
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type MenuItem}">
<Border
x:Name="templateRoot"
BorderBrush="{DynamicResource ButtonOutline}"
BorderThickness="{TemplateBinding BorderThickness}"
Padding="4"
Background="{TemplateBinding Background}"
SnapsToDevicePixels="True">
<Grid VerticalAlignment="Center">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto"/>
<ColumnDefinition Width="Auto"/>
</Grid.ColumnDefinitions>
<ContentPresenter x:Name="Icon" Content="{TemplateBinding Icon}" ContentSource="Icon" HorizontalAlignment="Center" Height="16" Margin="3" SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" VerticalAlignment="Center" Width="16"/>
<Path x:Name="GlyphPanel" Data="F1M10,1.2L4.7,9.1 4.5,9.1 0,5.2 1.3,3.5 4.3,6.1 8.3,0 10,1.2z" Fill="{TemplateBinding Foreground}" FlowDirection="LeftToRight" Margin="3" Visibility="Collapsed" VerticalAlignment="Center"/>
<ContentPresenter ContentTemplate="{TemplateBinding HeaderTemplate}" Content="{TemplateBinding Header}" Grid.Column="1" ContentStringFormat="{TemplateBinding HeaderStringFormat}" ContentSource="Header" Margin="{TemplateBinding Padding}" RecognizesAccessKey="True" SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}"/>
<Popup x:Name="PART_Popup" AllowsTransparency="True" Focusable="False" IsOpen="{Binding IsSubmenuOpen, RelativeSource={RelativeSource TemplatedParent}}" PopupAnimation="{DynamicResource {x:Static SystemParameters.MenuPopupAnimationKey}}" Placement="Bottom">
<ScrollViewer x:Name="SubMenuScrollViewer" Style="{DynamicResource {ComponentResourceKey ResourceId=MenuScrollViewer, TypeInTargetAssembly={x:Type FrameworkElement}}}">
<Grid RenderOptions.ClearTypeHint="Enabled">
<Canvas HorizontalAlignment="Left" Height="0" VerticalAlignment="Top" Width="1">
<Rectangle x:Name="OpaqueRect" Fill="Black" Height="{Binding ActualHeight, ElementName=SubMenuBorder}" Width="{Binding ActualWidth, ElementName=SubMenuBorder}"/>
</Canvas>
<Rectangle Fill="#FFD7D7D7" HorizontalAlignment="Left" Margin="29,2,0,2" Width="1" />
<ItemsPresenter x:Name="ItemsPresenter" Margin="0,0,0,0" KeyboardNavigation.DirectionalNavigation="Cycle" Grid.IsSharedSizeScope="True" SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" KeyboardNavigation.TabNavigation="Cycle"/>
</Grid>
</ScrollViewer>
</Popup>
</Grid>
</Border>
<ControlTemplate.Triggers>
<Trigger Property="IsSuspendingPopupAnimation" Value="True">
<Setter Property="PopupAnimation" TargetName="PART_Popup" Value="None"/>
</Trigger>
<Trigger Property="Icon" Value="{x:Null}">
<Setter Property="Visibility" TargetName="Icon" Value="Collapsed"/>
</Trigger>
<Trigger Property="IsChecked" Value="True">
<Setter Property="Visibility" TargetName="GlyphPanel" Value="Visible"/>
<Setter Property="Visibility" TargetName="Icon" Value="Collapsed"/>
</Trigger>
<Trigger Property="IsHighlighted" Value="True">
<Setter Property="Background" Value="{DynamicResource ButtonHighlightedBackground}" />
<Setter Property="Foreground" Value="{DynamicResource TextHighlighted}" />
</Trigger>
<Trigger Property="IsEnabled" Value="False">
<Setter Property="TextElement.Foreground" TargetName="templateRoot" Value="#FF707070"/>
<Setter Property="Fill" TargetName="GlyphPanel" Value="#FF707070"/>
</Trigger>
<Trigger Property="CanContentScroll" SourceName="SubMenuScrollViewer" Value="False">
<Setter Property="Canvas.Top" TargetName="OpaqueRect" Value="{Binding VerticalOffset, ElementName=SubMenuScrollViewer}"/>
<Setter Property="Canvas.Left" TargetName="OpaqueRect" Value="{Binding HorizontalOffset, ElementName=SubMenuScrollViewer}"/>
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ResourceDictionary>

View file

@ -1,5 +1,5 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<Style x:Key="ModAssistantRepeatButton" TargetType="{x:Type RepeatButton}">
<Style x:Key="VRCMelonAssistantRepeatButton" TargetType="{x:Type RepeatButton}">
<Setter Property="BorderThickness" Value="1" />
<Setter Property="HorizontalContentAlignment" Value="Center" />
<Setter Property="VerticalContentAlignment" Value="Center" />
@ -39,7 +39,7 @@
</Setter.Value>
</Setter>
</Style>
<Style x:Key="ModAssistantSmallRepeatButton" TargetType="{x:Type RepeatButton}">
<Style x:Key="VRCMelonAssistantSmallRepeatButton" TargetType="{x:Type RepeatButton}">
<Setter Property="FrameworkElement.OverridesDefaultStyle" Value="True" />
<Setter Property="Background" Value="{DynamicResource ScrollBarBackground}" />
<Setter Property="Focusable" Value="False" />

View file

@ -25,7 +25,7 @@
Name="PART_LineUpButton"
Command="{x:Static ScrollBar.LineUpCommand}"
IsEnabled="{TemplateBinding IsMouseOver}"
Style="{DynamicResource ResourceKey=ModAssistantRepeatButton}">
Style="{DynamicResource ResourceKey=VRCMelonAssistantRepeatButton}">
<Path
Name="ArrowTop"
Margin="3,4,3,3"
@ -42,16 +42,16 @@
<RepeatButton
Background="{DynamicResource ScrollBarBackground}"
Command="{x:Static ScrollBar.PageUpCommand}"
Style="{DynamicResource ResourceKey=ModAssistantSmallRepeatButton}" />
Style="{DynamicResource ResourceKey=VRCMelonAssistantSmallRepeatButton}" />
</Track.DecreaseRepeatButton>
<Track.IncreaseRepeatButton>
<RepeatButton
Background="{DynamicResource ScrollBarBackground}"
Command="{x:Static ScrollBar.PageDownCommand}"
Style="{DynamicResource ResourceKey=ModAssistantSmallRepeatButton}" />
Style="{DynamicResource ResourceKey=VRCMelonAssistantSmallRepeatButton}" />
</Track.IncreaseRepeatButton>
<Track.Thumb>
<Thumb Background="{DynamicResource ScrollBarHandle}" Style="{DynamicResource ResourceKey=ModAssistantThumb}" />
<Thumb Background="{DynamicResource ScrollBarHandle}" Style="{DynamicResource ResourceKey=VRCMelonAssistantThumb}" />
</Track.Thumb>
</Track>
<RepeatButton
@ -59,7 +59,7 @@
Grid.Row="2"
Command="{x:Static ScrollBar.LineDownCommand}"
IsEnabled="{TemplateBinding IsMouseOver}"
Style="{DynamicResource ResourceKey=ModAssistantRepeatButton}">
Style="{DynamicResource ResourceKey=VRCMelonAssistantRepeatButton}">
<Path
Name="ArrowBottom"
Margin="3,4,3,3"
@ -130,7 +130,7 @@
Name="PART_LineLeftButton"
Command="{x:Static ScrollBar.LineLeftCommand}"
IsEnabled="{TemplateBinding IsMouseOver}"
Style="{DynamicResource ResourceKey=ModAssistantRepeatButton}">
Style="{DynamicResource ResourceKey=VRCMelonAssistantRepeatButton}">
<Path
Name="ArrowLeft"
Margin="3"
@ -146,16 +146,16 @@
<RepeatButton
Background="{DynamicResource ScrollBarBackground}"
Command="{x:Static ScrollBar.PageLeftCommand}"
Style="{DynamicResource ResourceKey=ModAssistantSmallRepeatButton}" />
Style="{DynamicResource ResourceKey=VRCMelonAssistantSmallRepeatButton}" />
</Track.DecreaseRepeatButton>
<Track.IncreaseRepeatButton>
<RepeatButton
Background="{DynamicResource ScrollBarBackground}"
Command="{x:Static ScrollBar.PageRightCommand}"
Style="{DynamicResource ResourceKey=ModAssistantSmallRepeatButton}" />
Style="{DynamicResource ResourceKey=VRCMelonAssistantSmallRepeatButton}" />
</Track.IncreaseRepeatButton>
<Track.Thumb>
<Thumb Background="{DynamicResource ScrollBarHandle}" Style="{DynamicResource ResourceKey=ModAssistantThumb}" />
<Thumb Background="{DynamicResource ScrollBarHandle}" Style="{DynamicResource ResourceKey=VRCMelonAssistantThumb}" />
</Track.Thumb>
</Track>
<RepeatButton
@ -163,7 +163,7 @@
Grid.Column="2"
Command="{x:Static ScrollBar.LineRightCommand}"
IsEnabled="{TemplateBinding IsMouseOver}"
Style="{DynamicResource ResourceKey=ModAssistantRepeatButton}">
Style="{DynamicResource ResourceKey=VRCMelonAssistantRepeatButton}">
<Path
Name="ArrowRight"
Margin="3"

View file

@ -1,5 +1,5 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<Style x:Key="ModAssistantThumb" TargetType="{x:Type Thumb}">
<Style x:Key="VRCMelonAssistantThumb" TargetType="{x:Type Thumb}">
<Setter Property="FrameworkElement.OverridesDefaultStyle" Value="True" />
<Setter Property="IsTabStop" Value="False" />
<Setter Property="Template">

View file

@ -1,6 +1,6 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<!-- ModAssistant Dark Theme by Caeden117 and lolPants -->
<!-- VRChat Melon Assistant Dark Theme by Caeden117 and lolPants -->
<!-- Standard Styles -->
<Color x:Key="StandardContent">#E0E0E0</Color>
@ -74,7 +74,7 @@
<GridLength x:Key="ScrollBarButtonHeight">0</GridLength>
<!-- Various important elements that need to be controlled independently -->
<SolidColorBrush x:Key="ModAssistantBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="VRCMelonAssistantBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="FrameBackgroundColor" Color="#CC0F0F0F" />
<SolidColorBrush x:Key="BottomStatusBarBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="BottomStatusBarOutline" Color="{DynamicResource ResourceKey=StandardBorder}" />

View file

@ -1,6 +1,6 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<!-- ModAssistant Light Pink Theme by Caeden117 -->
<!-- VRChat Melon Assistant Light Pink Theme by Caeden117 -->
<!-- Default text -->
<SolidColorBrush x:Key="TextColor" Color="#101010" />
@ -64,7 +64,7 @@
<GridLength x:Key="ScrollBarButtonHeight">0</GridLength>
<!-- Various important elements that need to be controlled independently -->
<SolidColorBrush x:Key="ModAssistantBackground" Color="#FFFAE5FF" />
<SolidColorBrush x:Key="VRCMelonAssistantBackground" Color="#FFFAE5FF" />
<SolidColorBrush x:Key="FrameBackgroundColor" Color="#CCFAE5FF" />
<SolidColorBrush x:Key="BottomStatusBarBackground" Color="#FFE288D6" />
<SolidColorBrush x:Key="DirectoryBackground" Color="#FFE288D6" />

View file

@ -1,6 +1,6 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<!-- ModAssistant Light Theme by Caeden117 and lolPants -->
<!-- VRChat Melon Assistant Light Theme by Caeden117 and lolPants -->
<!-- Standard Styles -->
<Color x:Key="StandardContent">#101010</Color>
@ -73,7 +73,7 @@
<GridLength x:Key="ScrollBarButtonHeight">0</GridLength>
<!-- Various important elements that need to be controlled independently -->
<SolidColorBrush x:Key="ModAssistantBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="VRCMelonAssistantBackground" Color="{DynamicResource ResourceKey=StandardSecondary}" />
<SolidColorBrush x:Key="FrameBackgroundColor" Color="#CCF3F3F3" />
<SolidColorBrush x:Key="BottomStatusBarBackground" Color="{DynamicResource ResourceKey=StandardPrimary}" />
<SolidColorBrush x:Key="BottomStatusBarOutline" Color="{DynamicResource ResourceKey=StandardBorder}" />

View file

@ -1,16 +1,16 @@
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<!--
ModAssistant Ugly Kulu-Ya-Ku Theme by Assistant
Feel free to use this as a template for designing your own themes for Mod Assistant.
VRChat Melon Assistant Ugly Kulu-Ya-Ku Theme by Assistant
Feel free to use this as a template for designing your own themes for VRChat Melon Assistant.
The Color fields take in Hexadecimal RGB (#RRGGBB) or Hexadecimal ARGB (#AARRGGBB).
They can also take in common names like "Gray", "Maroon", "Magenta", "LimeGreen", etc.
For more information, see: https://docs.microsoft.com/en-us/dotnet/api/system.windows.media.colors
You may set color variables like below in the Standard Styles section, or simply set the colors directly.
To reload your theme, exit and re-enter the Options menu.
Inspired by https://www.nexusmods.com/monsterhunterworld/mods/2080
william requested I make this theme. One time william requested a hug from pook. pook denied said request.
I then witnessed william hounding pook, until he siezed them by breaking their foot.
@ -89,7 +89,7 @@
<GridLength x:Key="ScrollBarButtonHeight">0</GridLength>
<!-- Various important elements that need to be controlled independently -->
<SolidColorBrush x:Key="ModAssistantBackground" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="VRCMelonAssistantBackground" Color="{DynamicResource ResourceKey=StandardIcon}" />
<SolidColorBrush x:Key="FrameBackgroundColor" Color="#CCFF51B6" />
<SolidColorBrush x:Key="BottomStatusBarBackground" Color="#BF489A" />
<SolidColorBrush x:Key="BottomStatusBarOutline" Color="{DynamicResource ResourceKey=StandardBorder}" />

View file

@ -1,18 +1,20 @@
<?xml version="1.0" encoding="utf-8"?>
<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<Import Project="..\packages\ILRepack.MSBuild.Task.2.0.13\build\ILRepack.MSBuild.Task.props" Condition="Exists('..\packages\ILRepack.MSBuild.Task.2.0.13\build\ILRepack.MSBuild.Task.props')" />
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
<PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
<ProjectGuid>{6A224B82-40DA-40B3-94DC-EFBEC2BDDA39}</ProjectGuid>
<OutputType>WinExe</OutputType>
<RootNamespace>ModAssistant</RootNamespace>
<AssemblyName>ModAssistant</AssemblyName>
<TargetFrameworkVersion>v4.6.1</TargetFrameworkVersion>
<RootNamespace>VRCMelonAssistant</RootNamespace>
<AssemblyName>VRCMelonAssistant</AssemblyName>
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
<FileAlignment>512</FileAlignment>
<ProjectTypeGuids>{60dc8134-eba5-43b8-bcc9-bb4bc16c2548};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
<WarningLevel>4</WarningLevel>
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
<LangVersion>latest</LangVersion>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<PlatformTarget>AnyCPU</PlatformTarget>
@ -37,12 +39,50 @@
<ApplicationIcon>Resources\icon.ico</ApplicationIcon>
</PropertyGroup>
<ItemGroup>
<Reference Include="Microsoft.Build.Framework, Version=15.1.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
<HintPath>..\packages\Microsoft.Build.Framework.15.9.20\lib\net46\Microsoft.Build.Framework.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Microsoft.Build.Utilities.Core, Version=15.1.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
<HintPath>..\packages\Microsoft.Build.Utilities.Core.15.9.20\lib\net46\Microsoft.Build.Utilities.Core.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Microsoft.VisualStudio.Setup.Configuration.Interop, Version=1.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
<HintPath>..\packages\Microsoft.VisualStudio.Setup.Configuration.Interop.1.16.30\lib\net35\Microsoft.VisualStudio.Setup.Configuration.Interop.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Mono.Cecil, Version=0.11.3.0, Culture=neutral, PublicKeyToken=50cebf1cceb9d05e">
<HintPath>..\packages\Mono.Cecil.0.11.3\lib\net40\Mono.Cecil.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Mono.Cecil.Mdb, Version=0.11.3.0, Culture=neutral, PublicKeyToken=50cebf1cceb9d05e">
<HintPath>..\packages\Mono.Cecil.0.11.3\lib\net40\Mono.Cecil.Mdb.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Mono.Cecil.Pdb, Version=0.11.3.0, Culture=neutral, PublicKeyToken=50cebf1cceb9d05e">
<HintPath>..\packages\Mono.Cecil.0.11.3\lib\net40\Mono.Cecil.Pdb.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="Mono.Cecil.Rocks, Version=0.11.3.0, Culture=neutral, PublicKeyToken=50cebf1cceb9d05e">
<HintPath>..\packages\Mono.Cecil.0.11.3\lib\net40\Mono.Cecil.Rocks.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="mscorlib" />
<Reference Include="System" />
<Reference Include="System.Collections.Immutable, Version=1.2.3.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
<HintPath>..\packages\System.Collections.Immutable.1.5.0\lib\netstandard2.0\System.Collections.Immutable.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="System.Configuration" />
<Reference Include="System.Data" />
<Reference Include="System.Drawing" />
<Reference Include="System.IO.Compression" />
<Reference Include="System.IO.Compression.FileSystem" />
<Reference Include="System.Management" />
<Reference Include="System.Runtime.InteropServices.RuntimeInformation, Version=4.0.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
<HintPath>..\packages\System.Runtime.InteropServices.RuntimeInformation.4.3.0\lib\net45\System.Runtime.InteropServices.RuntimeInformation.dll</HintPath>
<Private>True</Private>
</Reference>
<Reference Include="System.Web" />
<Reference Include="System.Web.Extensions" />
<Reference Include="System.Windows.Forms" />
@ -65,20 +105,24 @@
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</ApplicationDefinition>
<Compile Include="Classes\HardcodedCategories.cs" />
<Compile Include="Classes\Http.cs" />
<Compile Include="Classes\HyperlinkExtensions.cs" />
<Compile Include="Classes\InstallHandlers.cs" />
<Compile Include="Classes\Languages.cs" />
<Compile Include="Classes\Promotions.cs" />
<Compile Include="Classes\Diagnostics.cs" />
<Compile Include="Classes\Themes.cs" />
<Compile Include="Classes\Updater.cs" />
<Compile Include="Libs\semver\SemVersion.cs" />
<Compile Include="Libs\semver\IntExtensions.cs" />
<Compile Include="ModInfoWindow.xaml.cs">
<DependentUpon>ModInfoWindow.xaml</DependentUpon>
</Compile>
<Compile Include="Pages\Intro.xaml.cs">
<DependentUpon>Intro.xaml</DependentUpon>
</Compile>
<Compile Include="Classes\Mod.cs" />
<Page Include="Localisation\en-US.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
</Page>
<Page Include="Localisation\en-DEBUG.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
@ -96,6 +140,11 @@
<DependentUpon>About.xaml</DependentUpon>
</Compile>
<Compile Include="Classes\Utils.cs" />
<Page Include="Localisation\en.xaml">
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</Page>
<Page Include="ModInfoWindow.xaml" />
<Page Include="Pages\Intro.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
@ -108,7 +157,6 @@
<DependentUpon>App.xaml</DependentUpon>
<SubType>Code</SubType>
</Compile>
<Compile Include="Classes\OneClickInstaller.cs" />
<Compile Include="MainWindow.xaml.cs">
<DependentUpon>MainWindow.xaml</DependentUpon>
<SubType>Code</SubType>
@ -133,6 +181,7 @@
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
</Page>
<Page Include="Resources\AppResources.xaml" />
<Page Include="Resources\Icons.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
@ -169,6 +218,14 @@
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
</Page>
<Page Include="Styles\MenuItem.xaml">
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</Page>
<Page Include="Styles\Menu.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
</Page>
<Page Include="Styles\RepeatButton.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
@ -189,10 +246,6 @@
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
</Page>
<Page Include="Themes\BSMG.xaml">
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</Page>
<Page Include="Themes\Dark.xaml">
<SubType>Designer</SubType>
<Generator>MSBuild:Compile</Generator>
@ -235,6 +288,7 @@
<Generator>ResXFileCodeGenerator</Generator>
<LastGenOutput>Resources.Designer.cs</LastGenOutput>
</EmbeddedResource>
<None Include="packages.config" />
<None Include="Properties\Settings.settings">
<Generator>PublicSettingsSingleFileGenerator</Generator>
<LastGenOutput>Settings.Designer.cs</LastGenOutput>
@ -246,9 +300,20 @@
<ItemGroup>
<Resource Include="Resources\icon.ico" />
</ItemGroup>
<ItemGroup />
<ItemGroup>
<EmbeddedResource Include="Themes\BSMG\Sidebar.png" />
</ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
<PropertyGroup>
<ErrorText>This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}.</ErrorText>
</PropertyGroup>
<Error Condition="!Exists('..\packages\ILRepack.MSBuild.Task.2.0.13\build\ILRepack.MSBuild.Task.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\ILRepack.MSBuild.Task.2.0.13\build\ILRepack.MSBuild.Task.props'))" />
</Target>
<Target Name="ILRepack" AfterTargets="Build" Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<PropertyGroup>
<WorkingDirectory>$(MSBuildThisFileDirectory)bin\$(Configuration)\$(TargetFramework)</WorkingDirectory>
</PropertyGroup>
<ILRepack OutputType="$(OutputType)" MainAssembly="$(AssemblyName).exe" OutputAssembly="$(AssemblyName).exe" InputAssemblies="$(WorkingDirectory)\Mono.Cecil.dll" WilcardInputAssemblies="true" WorkingDirectory="$(WorkingDirectory)" />
</Target>
</Project>

View file

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="utf-8"?>
<packages>
<package id="ILRepack.MSBuild.Task" version="2.0.13" targetFramework="net48" />
<package id="Microsoft.Build.Framework" version="15.9.20" targetFramework="net48" />
<package id="Microsoft.Build.Utilities.Core" version="15.9.20" targetFramework="net48" />
<package id="Microsoft.VisualStudio.Setup.Configuration.Interop" version="1.16.30" targetFramework="net48" developmentDependency="true" />
<package id="Mono.Cecil" version="0.11.3" targetFramework="net461" />
<package id="System.Collections.Immutable" version="1.5.0" targetFramework="net48" />
<package id="System.Runtime.InteropServices.RuntimeInformation" version="4.3.0" targetFramework="net48" />
</packages>

7
tools/README.md Normal file
View file

@ -0,0 +1,7 @@
# ModAssistant Tools
> These are tools used to build / improve ModAssistant. **They are not to be distributed with the compiled binary.**
## Translation Stubs
Use `generate_translation_stubs.py` to read the English locale file and generate missing strings for all other locales. Requires Python 3.6 or above.
Simply run `python ./tools/generate_translation_stubs.py` and commit the stubs.

View file

@ -0,0 +1,144 @@
# -*- coding: utf-8 -*-
import os
import re
from shutil import copyfile
# --------------- Configuration ---------------- #
master_file = "./ModAssistant/Localisation/en.xaml"
source_dir = "./ModAssistant/Localisation"
target_dir = source_dir
comment = " <!-- NEEDS TRANSLATING -->"
# --------------- Configuration ---------------- #
def read_file(filepath):
"""
Read file content
:param filepath:
:return:
"""
with open(filepath, "r", encoding="utf-8") as file:
data = file.read()
return data
def find_files(root_directory):
"""
Find all files to be processed
:param root_directory:
:return:
"""
for root, ds, fs in os.walk(root_directory):
for fn in fs:
fullname = os.path.join(root, fn)
master_name = os.path.basename(master_file)
if master_name in fullname:
continue
yield fullname
def search(regex, m_string, group=1):
"""
Search for the first item that matches by regular expression
:param regex:
:param m_string:
:return:
"""
found = re.compile(regex, re.M).search(m_string)
result = ""
if found:
result = found.group(group)
return result
def write_file(filepath, content):
"""
Write text content to file
:param filepath:
:param content:
:return:
"""
if not os.path.exists(target_dir):
os.makedirs(target_dir, exist_ok=True)
with open(filepath, "w", encoding="utf-8") as f:
f.write(content)
def find_items(data):
keys_regex = r"<\s*([^\s>]+)\s.*Key[^>]+>"
keys_matches = re.finditer(keys_regex, data, re.M)
items = []
for n, m in enumerate(keys_matches, start=1):
if len(items):
last_dict = items[-1]
if len(last_dict):
last_dict['end'] = m.start()
items.append({"match": m[0], "name": m[1], "start": m.end()})
items[-1]['end'] = len(data)
for item in items:
item_text = data[item['start']: item['end']]
end_tag = "</%s>" % item['name']
full_tag = item['match'] + item_text[:item_text.rfind(end_tag) + len(end_tag)]
item['end_tag'] = end_tag
item['full_tag'] = full_tag
return items
def wrapper_key(match, content):
"""
Try to find the comment behind the key
:param match:
:param content:
:return:
"""
regex = "(\s*" + re.escape(match) + "(\ *<\!\-\-[^\-]+\-\->)?\s*)"
with_comment = search(r"" + str(regex), content, 1)
# print(regex)
# print(match)
# print(with_comment)
# print("-----")
if with_comment:
match = with_comment
return match
def main():
"""
Cycle processing of xaml files except master_file
:return:
"""
master_data = read_file(master_file)
print("The master file is", master_file)
items = find_items(master_data)
for f in find_files(source_dir):
content = master_data + ""
xml_data = read_file(f)
xml_dict = {}
for xml_item in find_items(xml_data):
xml_dict[xml_item['match']] = xml_item['full_tag']
for item in items:
# print(f)
# if "OneClick:Done" in item['match'] and "fr.xaml" in f:
# print("----")
# pass
if item['match'] in xml_dict.keys():
match = wrapper_key(xml_dict[item['match']], xml_data)
master_match = wrapper_key(item['full_tag'], content)
# print(master_match)
content = content.replace(master_match, match)
else:
pre_blanks = search(r"(\s*)" + re.escape(item['full_tag']), master_data)
content = re.sub(r"\s*" + re.escape(item['full_tag']), pre_blanks + item['full_tag'] + comment, content)
# Put the processed files in the "dist" directory
filepath = f.replace(source_dir, target_dir)
write_file(filepath, content)
print("Processing", f)
# Copy "master_file" to the target directory
if source_dir != target_dir:
copyfile(master_file, master_file.replace(source_dir, target_dir))
if __name__ == '__main__':
main()
print("done'd!")