MainWindowViewModel.cs 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539
  1. using ARMeilleure.Translation.PTC;
  2. using Avalonia;
  3. using Avalonia.Controls;
  4. using Avalonia.Input;
  5. using Avalonia.Media;
  6. using Avalonia.Threading;
  7. using DynamicData;
  8. using DynamicData.Binding;
  9. using LibHac.Fs;
  10. using LibHac.FsSystem;
  11. using LibHac.Ncm;
  12. using Ryujinx.Ava.Common;
  13. using Ryujinx.Ava.Common.Locale;
  14. using Ryujinx.Ava.Input;
  15. using Ryujinx.Ava.Ui.Controls;
  16. using Ryujinx.Ava.Ui.Windows;
  17. using Ryujinx.Common;
  18. using Ryujinx.Common.Configuration;
  19. using Ryujinx.Common.Logging;
  20. using Ryujinx.HLE;
  21. using Ryujinx.HLE.FileSystem;
  22. using Ryujinx.HLE.HOS;
  23. using Ryujinx.Modules;
  24. using Ryujinx.Ui.App.Common;
  25. using Ryujinx.Ui.Common;
  26. using Ryujinx.Ui.Common.Configuration;
  27. using Ryujinx.Ui.Common.Helper;
  28. using System;
  29. using System.Collections.Generic;
  30. using System.Collections.ObjectModel;
  31. using System.Globalization;
  32. using System.IO;
  33. using System.Threading;
  34. using System.Threading.Tasks;
  35. using Path = System.IO.Path;
  36. using ShaderCacheLoadingState = Ryujinx.Graphics.Gpu.Shader.ShaderCacheState;
  37. namespace Ryujinx.Ava.Ui.ViewModels
  38. {
  39. internal class MainWindowViewModel : BaseModel
  40. {
  41. private const int HotKeyPressDelayMs = 500;
  42. private readonly MainWindow _owner;
  43. private ObservableCollection<ApplicationData> _applications;
  44. private string _aspectStatusText;
  45. private string _loadHeading;
  46. private string _cacheLoadStatus;
  47. private string _searchText;
  48. private Timer _searchTimer;
  49. private string _dockedStatusText;
  50. private string _fifoStatusText;
  51. private string _gameStatusText;
  52. private string _volumeStatusText;
  53. private string _gpuStatusText;
  54. private bool _isAmiiboRequested;
  55. private bool _isGameRunning;
  56. private bool _isLoading;
  57. private int _progressMaximum;
  58. private int _progressValue;
  59. private long _lastFullscreenToggle = Environment.TickCount64;
  60. private bool _showLoadProgress;
  61. private bool _showMenuAndStatusBar = true;
  62. private bool _showStatusSeparator;
  63. private Brush _progressBarForegroundColor;
  64. private Brush _progressBarBackgroundColor;
  65. private Brush _vsyncColor;
  66. private byte[] _selectedIcon;
  67. private bool _isAppletMenuActive;
  68. private int _statusBarProgressMaximum;
  69. private int _statusBarProgressValue;
  70. private bool _isPaused;
  71. private bool _showContent = true;
  72. private bool _isLoadingIndeterminate = true;
  73. private bool _showAll;
  74. private string _lastScannedAmiiboId;
  75. private ReadOnlyObservableCollection<ApplicationData> _appsObservableList;
  76. public ApplicationLibrary ApplicationLibrary => _owner.ApplicationLibrary;
  77. public string TitleName { get; internal set; }
  78. public MainWindowViewModel(MainWindow owner) : this()
  79. {
  80. _owner = owner;
  81. }
  82. public MainWindowViewModel()
  83. {
  84. Applications = new ObservableCollection<ApplicationData>();
  85. Applications.ToObservableChangeSet()
  86. .Filter(Filter)
  87. .Sort(GetComparer())
  88. .Bind(out _appsObservableList).AsObservableList();
  89. if (Program.PreviewerDetached)
  90. {
  91. LoadConfigurableHotKeys();
  92. Volume = ConfigurationState.Instance.System.AudioVolume;
  93. }
  94. }
  95. public void Initialize()
  96. {
  97. ApplicationLibrary.ApplicationCountUpdated += ApplicationLibrary_ApplicationCountUpdated;
  98. ApplicationLibrary.ApplicationAdded += ApplicationLibrary_ApplicationAdded;
  99. Ptc.PtcStateChanged -= ProgressHandler;
  100. Ptc.PtcStateChanged += ProgressHandler;
  101. }
  102. public string SearchText
  103. {
  104. get => _searchText;
  105. set
  106. {
  107. _searchText = value;
  108. _searchTimer?.Dispose();
  109. _searchTimer = new Timer(TimerCallback, null, 1000, 0);
  110. }
  111. }
  112. private void TimerCallback(object obj)
  113. {
  114. RefreshView();
  115. _searchTimer.Dispose();
  116. _searchTimer = null;
  117. }
  118. public ReadOnlyObservableCollection<ApplicationData> AppsObservableList
  119. {
  120. get => _appsObservableList;
  121. set
  122. {
  123. _appsObservableList = value;
  124. OnPropertyChanged();
  125. }
  126. }
  127. public bool IsPaused
  128. {
  129. get => _isPaused;
  130. set
  131. {
  132. _isPaused = value;
  133. OnPropertyChanged();
  134. }
  135. }
  136. public bool EnableNonGameRunningControls => !IsGameRunning;
  137. public bool ShowFirmwareStatus => !ShowLoadProgress;
  138. public bool IsGameRunning
  139. {
  140. get => _isGameRunning;
  141. set
  142. {
  143. _isGameRunning = value;
  144. if (!value)
  145. {
  146. ShowMenuAndStatusBar = false;
  147. }
  148. OnPropertyChanged();
  149. OnPropertyChanged(nameof(EnableNonGameRunningControls));
  150. OnPropertyChanged(nameof(ShowFirmwareStatus));
  151. }
  152. }
  153. public bool IsAmiiboRequested
  154. {
  155. get => _isAmiiboRequested && _isGameRunning;
  156. set
  157. {
  158. _isAmiiboRequested = value;
  159. OnPropertyChanged();
  160. }
  161. }
  162. public bool ShowLoadProgress
  163. {
  164. get => _showLoadProgress;
  165. set
  166. {
  167. _showLoadProgress = value;
  168. OnPropertyChanged();
  169. OnPropertyChanged(nameof(ShowFirmwareStatus));
  170. }
  171. }
  172. public string GameStatusText
  173. {
  174. get => _gameStatusText;
  175. set
  176. {
  177. _gameStatusText = value;
  178. OnPropertyChanged();
  179. }
  180. }
  181. private string _showUikey = "F4";
  182. private string _pauseKey = "F5";
  183. private string _screenshotkey = "F8";
  184. private float _volume;
  185. private string _backendText;
  186. public ApplicationData SelectedApplication
  187. {
  188. get
  189. {
  190. return Glyph switch
  191. {
  192. Glyph.List => _owner.GameList.SelectedApplication,
  193. Glyph.Grid => _owner.GameGrid.SelectedApplication,
  194. _ => null,
  195. };
  196. }
  197. }
  198. public string LoadHeading
  199. {
  200. get => _loadHeading;
  201. set
  202. {
  203. _loadHeading = value;
  204. OnPropertyChanged();
  205. }
  206. }
  207. public string CacheLoadStatus
  208. {
  209. get => _cacheLoadStatus;
  210. set
  211. {
  212. _cacheLoadStatus = value;
  213. OnPropertyChanged();
  214. }
  215. }
  216. public Brush ProgressBarBackgroundColor
  217. {
  218. get => _progressBarBackgroundColor;
  219. set
  220. {
  221. _progressBarBackgroundColor = value;
  222. OnPropertyChanged();
  223. }
  224. }
  225. public Brush ProgressBarForegroundColor
  226. {
  227. get => _progressBarForegroundColor;
  228. set
  229. {
  230. _progressBarForegroundColor = value;
  231. OnPropertyChanged();
  232. }
  233. }
  234. public Brush VsyncColor
  235. {
  236. get => _vsyncColor;
  237. set
  238. {
  239. _vsyncColor = value;
  240. OnPropertyChanged();
  241. }
  242. }
  243. public byte[] SelectedIcon
  244. {
  245. get => _selectedIcon;
  246. set
  247. {
  248. _selectedIcon = value;
  249. OnPropertyChanged();
  250. }
  251. }
  252. public int ProgressMaximum
  253. {
  254. get => _progressMaximum;
  255. set
  256. {
  257. _progressMaximum = value;
  258. OnPropertyChanged();
  259. }
  260. }
  261. public int ProgressValue
  262. {
  263. get => _progressValue;
  264. set
  265. {
  266. _progressValue = value;
  267. OnPropertyChanged();
  268. }
  269. }
  270. public int StatusBarProgressMaximum
  271. {
  272. get => _statusBarProgressMaximum;
  273. set
  274. {
  275. _statusBarProgressMaximum = value;
  276. OnPropertyChanged();
  277. }
  278. }
  279. public int StatusBarProgressValue
  280. {
  281. get => _statusBarProgressValue;
  282. set
  283. {
  284. _statusBarProgressValue = value;
  285. OnPropertyChanged();
  286. }
  287. }
  288. public string FifoStatusText
  289. {
  290. get => _fifoStatusText;
  291. set
  292. {
  293. _fifoStatusText = value;
  294. OnPropertyChanged();
  295. }
  296. }
  297. public string GpuNameText
  298. {
  299. get => _gpuStatusText;
  300. set
  301. {
  302. _gpuStatusText = value;
  303. OnPropertyChanged();
  304. }
  305. }
  306. public string BackendText
  307. {
  308. get => _backendText;
  309. set
  310. {
  311. _backendText = value;
  312. OnPropertyChanged();
  313. }
  314. }
  315. public string DockedStatusText
  316. {
  317. get => _dockedStatusText;
  318. set
  319. {
  320. _dockedStatusText = value;
  321. OnPropertyChanged();
  322. }
  323. }
  324. public string AspectRatioStatusText
  325. {
  326. get => _aspectStatusText;
  327. set
  328. {
  329. _aspectStatusText = value;
  330. OnPropertyChanged();
  331. }
  332. }
  333. public string VolumeStatusText
  334. {
  335. get => _volumeStatusText;
  336. set
  337. {
  338. _volumeStatusText = value;
  339. OnPropertyChanged();
  340. }
  341. }
  342. public bool VolumeMuted => _volume == 0;
  343. public float Volume
  344. {
  345. get => _volume;
  346. set
  347. {
  348. _volume = value;
  349. if (_isGameRunning)
  350. {
  351. _owner.AppHost.Device.SetVolume(_volume);
  352. }
  353. OnPropertyChanged(nameof(VolumeStatusText));
  354. OnPropertyChanged(nameof(VolumeMuted));
  355. OnPropertyChanged();
  356. }
  357. }
  358. public bool ShowStatusSeparator
  359. {
  360. get => _showStatusSeparator;
  361. set
  362. {
  363. _showStatusSeparator = value;
  364. OnPropertyChanged();
  365. }
  366. }
  367. public bool ShowMenuAndStatusBar
  368. {
  369. get => _showMenuAndStatusBar;
  370. set
  371. {
  372. _showMenuAndStatusBar = value;
  373. OnPropertyChanged();
  374. }
  375. }
  376. public bool IsLoadingIndeterminate
  377. {
  378. get => _isLoadingIndeterminate;
  379. set
  380. {
  381. _isLoadingIndeterminate = value;
  382. OnPropertyChanged();
  383. }
  384. }
  385. public bool ShowContent
  386. {
  387. get => _showContent;
  388. set
  389. {
  390. _showContent = value;
  391. OnPropertyChanged();
  392. }
  393. }
  394. public bool IsAppletMenuActive
  395. {
  396. get => _isAppletMenuActive && EnableNonGameRunningControls;
  397. set
  398. {
  399. _isAppletMenuActive = value;
  400. OnPropertyChanged();
  401. }
  402. }
  403. public bool IsGrid => Glyph == Glyph.Grid;
  404. public bool IsList => Glyph == Glyph.List;
  405. internal void Sort(bool isAscending)
  406. {
  407. IsAscending = isAscending;
  408. RefreshView();
  409. }
  410. internal void Sort(ApplicationSort sort)
  411. {
  412. SortMode = sort;
  413. RefreshView();
  414. }
  415. private IComparer<ApplicationData> GetComparer()
  416. {
  417. return SortMode switch
  418. {
  419. ApplicationSort.LastPlayed => new Models.Generic.LastPlayedSortComparer(IsAscending),
  420. ApplicationSort.FileSize => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.FileSizeBytes)
  421. : SortExpressionComparer<ApplicationData>.Descending(app => app.FileSizeBytes),
  422. ApplicationSort.TotalTimePlayed => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.TimePlayedNum)
  423. : SortExpressionComparer<ApplicationData>.Descending(app => app.TimePlayedNum),
  424. ApplicationSort.Title => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.TitleName)
  425. : SortExpressionComparer<ApplicationData>.Descending(app => app.TitleName),
  426. ApplicationSort.Favorite => !IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.Favorite)
  427. : SortExpressionComparer<ApplicationData>.Descending(app => app.Favorite),
  428. ApplicationSort.Developer => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.Developer)
  429. : SortExpressionComparer<ApplicationData>.Descending(app => app.Developer),
  430. ApplicationSort.FileType => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.FileExtension)
  431. : SortExpressionComparer<ApplicationData>.Descending(app => app.FileExtension),
  432. ApplicationSort.Path => IsAscending ? SortExpressionComparer<ApplicationData>.Ascending(app => app.Path)
  433. : SortExpressionComparer<ApplicationData>.Descending(app => app.Path),
  434. _ => null,
  435. };
  436. }
  437. private void RefreshView()
  438. {
  439. RefreshGrid();
  440. }
  441. private void RefreshGrid()
  442. {
  443. Applications.ToObservableChangeSet()
  444. .Filter(Filter)
  445. .Sort(GetComparer())
  446. .Bind(out _appsObservableList).AsObservableList();
  447. OnPropertyChanged(nameof(AppsObservableList));
  448. }
  449. public bool StartGamesInFullscreen
  450. {
  451. get => ConfigurationState.Instance.Ui.StartFullscreen;
  452. set
  453. {
  454. ConfigurationState.Instance.Ui.StartFullscreen.Value = value;
  455. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  456. OnPropertyChanged();
  457. }
  458. }
  459. public bool ShowConsole
  460. {
  461. get => ConfigurationState.Instance.Ui.ShowConsole;
  462. set
  463. {
  464. ConfigurationState.Instance.Ui.ShowConsole.Value = value;
  465. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  466. OnPropertyChanged();
  467. }
  468. }
  469. public bool ShowConsoleVisible
  470. {
  471. get => ConsoleHelper.SetConsoleWindowStateSupported;
  472. }
  473. public ObservableCollection<ApplicationData> Applications
  474. {
  475. get => _applications;
  476. set
  477. {
  478. _applications = value;
  479. OnPropertyChanged();
  480. }
  481. }
  482. public Glyph Glyph
  483. {
  484. get => (Glyph)ConfigurationState.Instance.Ui.GameListViewMode.Value;
  485. set
  486. {
  487. ConfigurationState.Instance.Ui.GameListViewMode.Value = (int)value;
  488. OnPropertyChanged();
  489. OnPropertyChanged(nameof(IsGrid));
  490. OnPropertyChanged(nameof(IsList));
  491. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  492. }
  493. }
  494. public bool ShowNames
  495. {
  496. get => ConfigurationState.Instance.Ui.ShowNames && ConfigurationState.Instance.Ui.GridSize > 1; set
  497. {
  498. ConfigurationState.Instance.Ui.ShowNames.Value = value;
  499. OnPropertyChanged();
  500. OnPropertyChanged(nameof(GridSizeScale));
  501. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  502. }
  503. }
  504. internal ApplicationSort SortMode
  505. {
  506. get => (ApplicationSort)ConfigurationState.Instance.Ui.ApplicationSort.Value;
  507. private set
  508. {
  509. ConfigurationState.Instance.Ui.ApplicationSort.Value = (int)value;
  510. OnPropertyChanged();
  511. OnPropertyChanged(nameof(SortName));
  512. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  513. }
  514. }
  515. public bool IsSortedByFavorite => SortMode == ApplicationSort.Favorite;
  516. public bool IsSortedByTitle => SortMode == ApplicationSort.Title;
  517. public bool IsSortedByDeveloper => SortMode == ApplicationSort.Developer;
  518. public bool IsSortedByLastPlayed => SortMode == ApplicationSort.LastPlayed;
  519. public bool IsSortedByTimePlayed => SortMode == ApplicationSort.TotalTimePlayed;
  520. public bool IsSortedByType => SortMode == ApplicationSort.FileType;
  521. public bool IsSortedBySize => SortMode == ApplicationSort.FileSize;
  522. public bool IsSortedByPath => SortMode == ApplicationSort.Path;
  523. public string SortName
  524. {
  525. get
  526. {
  527. return SortMode switch
  528. {
  529. ApplicationSort.Title => LocaleManager.Instance["GameListHeaderApplication"],
  530. ApplicationSort.Developer => LocaleManager.Instance["GameListHeaderDeveloper"],
  531. ApplicationSort.LastPlayed => LocaleManager.Instance["GameListHeaderLastPlayed"],
  532. ApplicationSort.TotalTimePlayed => LocaleManager.Instance["GameListHeaderTimePlayed"],
  533. ApplicationSort.FileType => LocaleManager.Instance["GameListHeaderFileExtension"],
  534. ApplicationSort.FileSize => LocaleManager.Instance["GameListHeaderFileSize"],
  535. ApplicationSort.Path => LocaleManager.Instance["GameListHeaderPath"],
  536. ApplicationSort.Favorite => LocaleManager.Instance["CommonFavorite"],
  537. _ => string.Empty,
  538. };
  539. }
  540. }
  541. public bool IsAscending
  542. {
  543. get => ConfigurationState.Instance.Ui.IsAscendingOrder;
  544. private set
  545. {
  546. ConfigurationState.Instance.Ui.IsAscendingOrder.Value = value;
  547. OnPropertyChanged();
  548. OnPropertyChanged(nameof(SortMode));
  549. OnPropertyChanged(nameof(SortName));
  550. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  551. }
  552. }
  553. public KeyGesture ShowUiKey
  554. {
  555. get => KeyGesture.Parse(_showUikey); set
  556. {
  557. _showUikey = value.ToString();
  558. OnPropertyChanged();
  559. }
  560. }
  561. public KeyGesture ScreenshotKey
  562. {
  563. get => KeyGesture.Parse(_screenshotkey); set
  564. {
  565. _screenshotkey = value.ToString();
  566. OnPropertyChanged();
  567. }
  568. }
  569. public KeyGesture PauseKey
  570. {
  571. get => KeyGesture.Parse(_pauseKey); set
  572. {
  573. _pauseKey = value.ToString();
  574. OnPropertyChanged();
  575. }
  576. }
  577. public bool IsGridSmall => ConfigurationState.Instance.Ui.GridSize == 1;
  578. public bool IsGridMedium => ConfigurationState.Instance.Ui.GridSize == 2;
  579. public bool IsGridLarge => ConfigurationState.Instance.Ui.GridSize == 3;
  580. public bool IsGridHuge => ConfigurationState.Instance.Ui.GridSize == 4;
  581. public int GridSizeScale
  582. {
  583. get => ConfigurationState.Instance.Ui.GridSize;
  584. set
  585. {
  586. ConfigurationState.Instance.Ui.GridSize.Value = value;
  587. if (value < 2)
  588. {
  589. ShowNames = false;
  590. }
  591. OnPropertyChanged();
  592. OnPropertyChanged(nameof(IsGridSmall));
  593. OnPropertyChanged(nameof(IsGridMedium));
  594. OnPropertyChanged(nameof(IsGridLarge));
  595. OnPropertyChanged(nameof(IsGridHuge));
  596. OnPropertyChanged(nameof(ShowNames));
  597. ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
  598. }
  599. }
  600. public async void OpenAmiiboWindow()
  601. {
  602. if (!_isAmiiboRequested)
  603. {
  604. return;
  605. }
  606. if (_owner.AppHost.Device.System.SearchingForAmiibo(out int deviceId))
  607. {
  608. string titleId = _owner.AppHost.Device.Application.TitleIdText.ToUpper();
  609. AmiiboWindow window = new(_showAll, _lastScannedAmiiboId, titleId);
  610. await window.ShowDialog(_owner);
  611. if (window.IsScanned)
  612. {
  613. _showAll = window.ViewModel.ShowAllAmiibo;
  614. _lastScannedAmiiboId = window.ScannedAmiibo.GetId();
  615. _owner.AppHost.Device.System.ScanAmiibo(deviceId, _lastScannedAmiiboId, window.ViewModel.UseRandomUuid);
  616. }
  617. }
  618. }
  619. public void HandleShaderProgress(Switch emulationContext)
  620. {
  621. emulationContext.Gpu.ShaderCacheStateChanged -= ProgressHandler;
  622. emulationContext.Gpu.ShaderCacheStateChanged += ProgressHandler;
  623. }
  624. private bool Filter(object arg)
  625. {
  626. if (arg is ApplicationData app)
  627. {
  628. return string.IsNullOrWhiteSpace(_searchText) || app.TitleName.ToLower().Contains(_searchText.ToLower());
  629. }
  630. return false;
  631. }
  632. private void ApplicationLibrary_ApplicationAdded(object sender, ApplicationAddedEventArgs e)
  633. {
  634. AddApplication(e.AppData);
  635. }
  636. private void ApplicationLibrary_ApplicationCountUpdated(object sender, ApplicationCountUpdatedEventArgs e)
  637. {
  638. StatusBarProgressValue = e.NumAppsLoaded;
  639. StatusBarProgressMaximum = e.NumAppsFound;
  640. LocaleManager.Instance.UpdateDynamicValue("StatusBarGamesLoaded", StatusBarProgressValue, StatusBarProgressMaximum);
  641. Dispatcher.UIThread.Post(() =>
  642. {
  643. if (e.NumAppsFound == 0)
  644. {
  645. _owner.LoadProgressBar.IsVisible = false;
  646. }
  647. if (e.NumAppsLoaded == e.NumAppsFound)
  648. {
  649. _owner.LoadProgressBar.IsVisible = false;
  650. }
  651. });
  652. }
  653. public void AddApplication(ApplicationData applicationData)
  654. {
  655. Dispatcher.UIThread.InvokeAsync(() =>
  656. {
  657. Applications.Add(applicationData);
  658. });
  659. }
  660. public async void LoadApplications()
  661. {
  662. await Dispatcher.UIThread.InvokeAsync(() =>
  663. {
  664. Applications.Clear();
  665. _owner.LoadProgressBar.IsVisible = true;
  666. StatusBarProgressMaximum = 0;
  667. StatusBarProgressValue = 0;
  668. LocaleManager.Instance.UpdateDynamicValue("StatusBarGamesLoaded", 0, 0);
  669. });
  670. ReloadGameList();
  671. }
  672. private void ReloadGameList()
  673. {
  674. if (_isLoading)
  675. {
  676. return;
  677. }
  678. _isLoading = true;
  679. Thread thread = new(() =>
  680. {
  681. ApplicationLibrary.LoadApplications(ConfigurationState.Instance.Ui.GameDirs.Value, ConfigurationState.Instance.System.Language);
  682. _isLoading = false;
  683. })
  684. { Name = "GUI.AppListLoadThread", Priority = ThreadPriority.AboveNormal };
  685. thread.Start();
  686. }
  687. public async void OpenFile()
  688. {
  689. OpenFileDialog dialog = new()
  690. {
  691. Title = LocaleManager.Instance["OpenFileDialogTitle"]
  692. };
  693. dialog.Filters.Add(new FileDialogFilter
  694. {
  695. Name = LocaleManager.Instance["AllSupportedFormats"],
  696. Extensions =
  697. {
  698. "nsp",
  699. "pfs0",
  700. "xci",
  701. "nca",
  702. "nro",
  703. "nso"
  704. }
  705. });
  706. dialog.Filters.Add(new FileDialogFilter { Name = "NSP", Extensions = { "nsp" } });
  707. dialog.Filters.Add(new FileDialogFilter { Name = "PFS0", Extensions = { "pfs0" } });
  708. dialog.Filters.Add(new FileDialogFilter { Name = "XCI", Extensions = { "xci" } });
  709. dialog.Filters.Add(new FileDialogFilter { Name = "NCA", Extensions = { "nca" } });
  710. dialog.Filters.Add(new FileDialogFilter { Name = "NRO", Extensions = { "nro" } });
  711. dialog.Filters.Add(new FileDialogFilter { Name = "NSO", Extensions = { "nso" } });
  712. string[] files = await dialog.ShowAsync(_owner);
  713. if (files != null && files.Length > 0)
  714. {
  715. _owner.LoadApplication(files[0]);
  716. }
  717. }
  718. public async void OpenFolder()
  719. {
  720. OpenFolderDialog dialog = new()
  721. {
  722. Title = LocaleManager.Instance["OpenFolderDialogTitle"]
  723. };
  724. string folder = await dialog.ShowAsync(_owner);
  725. if (!string.IsNullOrWhiteSpace(folder) && Directory.Exists(folder))
  726. {
  727. _owner.LoadApplication(folder);
  728. }
  729. }
  730. public void LoadConfigurableHotKeys()
  731. {
  732. if (AvaloniaKeyboardMappingHelper.TryGetAvaKey((Ryujinx.Input.Key)ConfigurationState.Instance.Hid.Hotkeys.Value.ShowUi, out var showUiKey))
  733. {
  734. ShowUiKey = new KeyGesture(showUiKey, KeyModifiers.None);
  735. }
  736. if (AvaloniaKeyboardMappingHelper.TryGetAvaKey((Ryujinx.Input.Key)ConfigurationState.Instance.Hid.Hotkeys.Value.Screenshot, out var screenshotKey))
  737. {
  738. ScreenshotKey = new KeyGesture(screenshotKey, KeyModifiers.None);
  739. }
  740. if (AvaloniaKeyboardMappingHelper.TryGetAvaKey((Ryujinx.Input.Key)ConfigurationState.Instance.Hid.Hotkeys.Value.Pause, out var pauseKey))
  741. {
  742. PauseKey = new KeyGesture(pauseKey, KeyModifiers.None);
  743. }
  744. }
  745. public void TakeScreenshot()
  746. {
  747. _owner.AppHost.ScreenshotRequested = true;
  748. }
  749. public void HideUi()
  750. {
  751. ShowMenuAndStatusBar = false;
  752. }
  753. public void SetListMode()
  754. {
  755. Glyph = Glyph.List;
  756. }
  757. public void SetGridMode()
  758. {
  759. Glyph = Glyph.Grid;
  760. }
  761. public void OpenMiiApplet()
  762. {
  763. string contentPath = _owner.ContentManager.GetInstalledContentPath(0x0100000000001009, StorageId.BuiltInSystem, NcaContentType.Program);
  764. if (!string.IsNullOrWhiteSpace(contentPath))
  765. {
  766. _owner.LoadApplication(contentPath, false, "Mii Applet");
  767. }
  768. }
  769. public static void OpenRyujinxFolder()
  770. {
  771. OpenHelper.OpenFolder(AppDataManager.BaseDirPath);
  772. }
  773. public static void OpenLogsFolder()
  774. {
  775. string logPath = Path.Combine(ReleaseInformations.GetBaseApplicationDirectory(), "Logs");
  776. new DirectoryInfo(logPath).Create();
  777. OpenHelper.OpenFolder(logPath);
  778. }
  779. public void ToggleFullscreen()
  780. {
  781. if (Environment.TickCount64 - _lastFullscreenToggle < HotKeyPressDelayMs)
  782. {
  783. return;
  784. }
  785. _lastFullscreenToggle = Environment.TickCount64;
  786. if (_owner.WindowState == WindowState.FullScreen)
  787. {
  788. _owner.WindowState = WindowState.Normal;
  789. if (IsGameRunning)
  790. {
  791. ShowMenuAndStatusBar = true;
  792. }
  793. }
  794. else
  795. {
  796. _owner.WindowState = WindowState.FullScreen;
  797. if (IsGameRunning)
  798. {
  799. ShowMenuAndStatusBar = false;
  800. }
  801. }
  802. OnPropertyChanged(nameof(IsFullScreen));
  803. }
  804. public bool IsFullScreen => _owner.WindowState == WindowState.FullScreen;
  805. public void ToggleDockMode()
  806. {
  807. if (IsGameRunning)
  808. {
  809. ConfigurationState.Instance.System.EnableDockedMode.Value = !ConfigurationState.Instance.System.EnableDockedMode.Value;
  810. }
  811. }
  812. public async void ExitCurrentState()
  813. {
  814. if (_owner.WindowState == WindowState.FullScreen)
  815. {
  816. ToggleFullscreen();
  817. }
  818. else if (IsGameRunning)
  819. {
  820. await Task.Delay(100);
  821. _owner.AppHost?.ShowExitPrompt();
  822. }
  823. }
  824. public async void OpenSettings()
  825. {
  826. _owner.SettingsWindow = new(_owner.VirtualFileSystem, _owner.ContentManager);
  827. await _owner.SettingsWindow.ShowDialog(_owner);
  828. LoadConfigurableHotKeys();
  829. }
  830. public async void ManageProfiles()
  831. {
  832. await NavigationDialogHost.Show(_owner.AccountManager, _owner.ContentManager, _owner.VirtualFileSystem, _owner.LibHacHorizonManager.RyujinxClient);
  833. }
  834. public async void OpenAboutWindow()
  835. {
  836. await new AboutWindow().ShowDialog(_owner);
  837. }
  838. public void ChangeLanguage(object obj)
  839. {
  840. LocaleManager.Instance.LoadDefaultLanguage();
  841. LocaleManager.Instance.LoadLanguage((string)obj);
  842. }
  843. private void ProgressHandler<T>(T state, int current, int total) where T : Enum
  844. {
  845. try
  846. {
  847. ProgressMaximum = total;
  848. ProgressValue = current;
  849. switch (state)
  850. {
  851. case PtcLoadingState ptcState:
  852. CacheLoadStatus = $"{current} / {total}";
  853. switch (ptcState)
  854. {
  855. case PtcLoadingState.Start:
  856. case PtcLoadingState.Loading:
  857. LoadHeading = LocaleManager.Instance["CompilingPPTC"];
  858. IsLoadingIndeterminate = false;
  859. break;
  860. case PtcLoadingState.Loaded:
  861. LoadHeading = string.Format(LocaleManager.Instance["LoadingHeading"], TitleName);
  862. IsLoadingIndeterminate = true;
  863. CacheLoadStatus = "";
  864. break;
  865. }
  866. break;
  867. case ShaderCacheLoadingState shaderCacheState:
  868. CacheLoadStatus = $"{current} / {total}";
  869. switch (shaderCacheState)
  870. {
  871. case ShaderCacheLoadingState.Start:
  872. case ShaderCacheLoadingState.Loading:
  873. LoadHeading = LocaleManager.Instance["CompilingShaders"];
  874. IsLoadingIndeterminate = false;
  875. break;
  876. case ShaderCacheLoadingState.Loaded:
  877. LoadHeading = string.Format(LocaleManager.Instance["LoadingHeading"], TitleName);
  878. IsLoadingIndeterminate = true;
  879. CacheLoadStatus = "";
  880. break;
  881. }
  882. break;
  883. default:
  884. throw new ArgumentException($"Unknown Progress Handler type {typeof(T)}");
  885. }
  886. }
  887. catch (Exception) { }
  888. }
  889. public void OpenUserSaveDirectory()
  890. {
  891. ApplicationData selection = SelectedApplication;
  892. if (selection != null)
  893. {
  894. Task.Run(() =>
  895. {
  896. if (!ulong.TryParse(selection.TitleId, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out ulong titleIdNumber))
  897. {
  898. Dispatcher.UIThread.Post(async () =>
  899. {
  900. await ContentDialogHelper.CreateErrorDialog(LocaleManager.Instance["DialogRyujinxErrorMessage"], LocaleManager.Instance["DialogInvalidTitleIdErrorMessage"]);
  901. });
  902. return;
  903. }
  904. UserId userId = new((ulong)_owner.AccountManager.LastOpenedUser.UserId.High, (ulong)_owner.AccountManager.LastOpenedUser.UserId.Low);
  905. SaveDataFilter saveDataFilter = SaveDataFilter.Make(titleIdNumber, saveType: default, userId, saveDataId: default, index: default);
  906. OpenSaveDirectory(in saveDataFilter, selection, titleIdNumber);
  907. });
  908. }
  909. }
  910. public void ToggleFavorite()
  911. {
  912. ApplicationData selection = SelectedApplication;
  913. if (selection != null)
  914. {
  915. selection.Favorite = !selection.Favorite;
  916. ApplicationLibrary.LoadAndSaveMetaData(selection.TitleId, appMetadata =>
  917. {
  918. appMetadata.Favorite = selection.Favorite;
  919. });
  920. RefreshView();
  921. }
  922. }
  923. public void OpenModsDirectory()
  924. {
  925. ApplicationData selection = SelectedApplication;
  926. if (selection != null)
  927. {
  928. string modsBasePath = _owner.VirtualFileSystem.ModLoader.GetModsBasePath();
  929. string titleModsPath = _owner.VirtualFileSystem.ModLoader.GetTitleDir(modsBasePath, selection.TitleId);
  930. OpenHelper.OpenFolder(titleModsPath);
  931. }
  932. }
  933. public void OpenSdModsDirectory()
  934. {
  935. ApplicationData selection = SelectedApplication;
  936. if (selection != null)
  937. {
  938. string sdModsBasePath = _owner.VirtualFileSystem.ModLoader.GetSdModsBasePath();
  939. string titleModsPath = _owner.VirtualFileSystem.ModLoader.GetTitleDir(sdModsBasePath, selection.TitleId);
  940. OpenHelper.OpenFolder(titleModsPath);
  941. }
  942. }
  943. public void OpenPtcDirectory()
  944. {
  945. ApplicationData selection = SelectedApplication;
  946. if (selection != null)
  947. {
  948. string ptcDir = Path.Combine(AppDataManager.GamesDirPath, selection.TitleId, "cache", "cpu");
  949. string mainPath = Path.Combine(ptcDir, "0");
  950. string backupPath = Path.Combine(ptcDir, "1");
  951. if (!Directory.Exists(ptcDir))
  952. {
  953. Directory.CreateDirectory(ptcDir);
  954. Directory.CreateDirectory(mainPath);
  955. Directory.CreateDirectory(backupPath);
  956. }
  957. OpenHelper.OpenFolder(ptcDir);
  958. }
  959. }
  960. public async void PurgePtcCache()
  961. {
  962. ApplicationData selection = SelectedApplication;
  963. if (selection != null)
  964. {
  965. DirectoryInfo mainDir = new(Path.Combine(AppDataManager.GamesDirPath, selection.TitleId, "cache", "cpu", "0"));
  966. DirectoryInfo backupDir = new(Path.Combine(AppDataManager.GamesDirPath, selection.TitleId, "cache", "cpu", "1"));
  967. // FIXME: Found a way to reproduce the bold effect on the title name (fork?).
  968. UserResult result = await ContentDialogHelper.CreateConfirmationDialog(LocaleManager.Instance["DialogWarning"],
  969. string.Format(LocaleManager.Instance["DialogPPTCDeletionMessage"], selection.TitleName),
  970. LocaleManager.Instance["InputDialogYes"],
  971. LocaleManager.Instance["InputDialogNo"],
  972. LocaleManager.Instance["RyujinxConfirm"]);
  973. List<FileInfo> cacheFiles = new();
  974. if (mainDir.Exists)
  975. {
  976. cacheFiles.AddRange(mainDir.EnumerateFiles("*.cache"));
  977. }
  978. if (backupDir.Exists)
  979. {
  980. cacheFiles.AddRange(backupDir.EnumerateFiles("*.cache"));
  981. }
  982. if (cacheFiles.Count > 0 && result == UserResult.Yes)
  983. {
  984. foreach (FileInfo file in cacheFiles)
  985. {
  986. try
  987. {
  988. file.Delete();
  989. }
  990. catch (Exception e)
  991. {
  992. await ContentDialogHelper.CreateErrorDialog(string.Format(LocaleManager.Instance["DialogPPTCDeletionErrorMessage"], file.Name, e));
  993. }
  994. }
  995. }
  996. }
  997. }
  998. public void OpenShaderCacheDirectory()
  999. {
  1000. ApplicationData selection = SelectedApplication;
  1001. if (selection != null)
  1002. {
  1003. string shaderCacheDir = Path.Combine(AppDataManager.GamesDirPath, selection.TitleId, "cache", "shader");
  1004. if (!Directory.Exists(shaderCacheDir))
  1005. {
  1006. Directory.CreateDirectory(shaderCacheDir);
  1007. }
  1008. OpenHelper.OpenFolder(shaderCacheDir);
  1009. }
  1010. }
  1011. public void SimulateWakeUpMessage()
  1012. {
  1013. _owner.AppHost.Device.System.SimulateWakeUpMessage();
  1014. }
  1015. public async void PurgeShaderCache()
  1016. {
  1017. ApplicationData selection = SelectedApplication;
  1018. if (selection != null)
  1019. {
  1020. DirectoryInfo shaderCacheDir = new(Path.Combine(AppDataManager.GamesDirPath, selection.TitleId, "cache", "shader"));
  1021. // FIXME: Found a way to reproduce the bold effect on the title name (fork?).
  1022. UserResult result = await ContentDialogHelper.CreateConfirmationDialog(LocaleManager.Instance["DialogWarning"],
  1023. string.Format(LocaleManager.Instance["DialogShaderDeletionMessage"], selection.TitleName),
  1024. LocaleManager.Instance["InputDialogYes"],
  1025. LocaleManager.Instance["InputDialogNo"],
  1026. LocaleManager.Instance["RyujinxConfirm"]);
  1027. List<DirectoryInfo> oldCacheDirectories = new();
  1028. List<FileInfo> newCacheFiles = new();
  1029. if (shaderCacheDir.Exists)
  1030. {
  1031. oldCacheDirectories.AddRange(shaderCacheDir.EnumerateDirectories("*"));
  1032. newCacheFiles.AddRange(shaderCacheDir.GetFiles("*.toc"));
  1033. newCacheFiles.AddRange(shaderCacheDir.GetFiles("*.data"));
  1034. }
  1035. if ((oldCacheDirectories.Count > 0 || newCacheFiles.Count > 0) && result == UserResult.Yes)
  1036. {
  1037. foreach (DirectoryInfo directory in oldCacheDirectories)
  1038. {
  1039. try
  1040. {
  1041. directory.Delete(true);
  1042. }
  1043. catch (Exception e)
  1044. {
  1045. await ContentDialogHelper.CreateErrorDialog(string.Format(LocaleManager.Instance["DialogPPTCDeletionErrorMessage"], directory.Name, e));
  1046. }
  1047. }
  1048. }
  1049. foreach (FileInfo file in newCacheFiles)
  1050. {
  1051. try
  1052. {
  1053. file.Delete();
  1054. }
  1055. catch (Exception e)
  1056. {
  1057. await ContentDialogHelper.CreateErrorDialog(string.Format(LocaleManager.Instance["ShaderCachePurgeError"], file.Name, e));
  1058. }
  1059. }
  1060. }
  1061. }
  1062. public async void CheckForUpdates()
  1063. {
  1064. if (Updater.CanUpdate(true, _owner))
  1065. {
  1066. await Updater.BeginParse(_owner, true);
  1067. }
  1068. }
  1069. public async void OpenTitleUpdateManager()
  1070. {
  1071. ApplicationData selection = SelectedApplication;
  1072. if (selection != null)
  1073. {
  1074. await new TitleUpdateWindow(_owner.VirtualFileSystem, ulong.Parse(selection.TitleId, NumberStyles.HexNumber), selection.TitleName).ShowDialog(_owner);
  1075. }
  1076. }
  1077. public async void OpenDownloadableContentManager()
  1078. {
  1079. ApplicationData selection = SelectedApplication;
  1080. if (selection != null)
  1081. {
  1082. await new DownloadableContentManagerWindow(_owner.VirtualFileSystem, ulong.Parse(selection.TitleId, NumberStyles.HexNumber), selection.TitleName).ShowDialog(_owner);
  1083. }
  1084. }
  1085. public async void OpenCheatManager()
  1086. {
  1087. ApplicationData selection = SelectedApplication;
  1088. if (selection != null)
  1089. {
  1090. await new CheatWindow(_owner.VirtualFileSystem, selection.TitleId, selection.TitleName).ShowDialog(_owner);
  1091. }
  1092. }
  1093. public async void OpenCheatManagerForCurrentApp()
  1094. {
  1095. if (!IsGameRunning)
  1096. {
  1097. return;
  1098. }
  1099. ApplicationLoader application = _owner.AppHost.Device.Application;
  1100. if (application != null)
  1101. {
  1102. await new CheatWindow(_owner.VirtualFileSystem, application.TitleIdText, application.TitleName).ShowDialog(_owner);
  1103. _owner.AppHost.Device.EnableCheats();
  1104. }
  1105. }
  1106. public void OpenDeviceSaveDirectory()
  1107. {
  1108. ApplicationData selection = SelectedApplication;
  1109. if (selection != null)
  1110. {
  1111. Task.Run(() =>
  1112. {
  1113. if (!ulong.TryParse(selection.TitleId, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out ulong titleIdNumber))
  1114. {
  1115. Dispatcher.UIThread.Post(async () =>
  1116. {
  1117. await ContentDialogHelper.CreateErrorDialog(LocaleManager.Instance["DialogRyujinxErrorMessage"], LocaleManager.Instance["DialogInvalidTitleIdErrorMessage"]);
  1118. });
  1119. return;
  1120. }
  1121. var saveDataFilter = SaveDataFilter.Make(titleIdNumber, SaveDataType.Device, userId: default, saveDataId: default, index: default);
  1122. OpenSaveDirectory(in saveDataFilter, selection, titleIdNumber);
  1123. });
  1124. }
  1125. }
  1126. public void OpenBcatSaveDirectory()
  1127. {
  1128. ApplicationData selection = SelectedApplication;
  1129. if (selection != null)
  1130. {
  1131. Task.Run(() =>
  1132. {
  1133. if (!ulong.TryParse(selection.TitleId, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out ulong titleIdNumber))
  1134. {
  1135. Dispatcher.UIThread.Post(async () =>
  1136. {
  1137. await ContentDialogHelper.CreateErrorDialog(LocaleManager.Instance["DialogRyujinxErrorMessage"], LocaleManager.Instance["DialogInvalidTitleIdErrorMessage"]);
  1138. });
  1139. return;
  1140. }
  1141. var saveDataFilter = SaveDataFilter.Make(titleIdNumber, SaveDataType.Bcat, userId: default, saveDataId: default, index: default);
  1142. OpenSaveDirectory(in saveDataFilter, selection, titleIdNumber);
  1143. });
  1144. }
  1145. }
  1146. private void OpenSaveDirectory(in SaveDataFilter filter, ApplicationData data, ulong titleId)
  1147. {
  1148. ApplicationHelper.OpenSaveDir(in filter, titleId, data.ControlHolder, data.TitleName);
  1149. }
  1150. private async void ExtractLogo()
  1151. {
  1152. var selection = SelectedApplication;
  1153. if (selection != null)
  1154. {
  1155. await ApplicationHelper.ExtractSection(NcaSectionType.Logo, selection.Path);
  1156. }
  1157. }
  1158. private async void ExtractRomFs()
  1159. {
  1160. var selection = SelectedApplication;
  1161. if (selection != null)
  1162. {
  1163. await ApplicationHelper.ExtractSection(NcaSectionType.Data, selection.Path);
  1164. }
  1165. }
  1166. private async void ExtractExeFs()
  1167. {
  1168. var selection = SelectedApplication;
  1169. if (selection != null)
  1170. {
  1171. await ApplicationHelper.ExtractSection(NcaSectionType.Code, selection.Path);
  1172. }
  1173. }
  1174. public void CloseWindow()
  1175. {
  1176. _owner.Close();
  1177. }
  1178. private async Task HandleFirmwareInstallation(string filename)
  1179. {
  1180. try
  1181. {
  1182. SystemVersion firmwareVersion = _owner.ContentManager.VerifyFirmwarePackage(filename);
  1183. if (firmwareVersion == null)
  1184. {
  1185. await ContentDialogHelper.CreateErrorDialog(string.Format(LocaleManager.Instance["DialogFirmwareInstallerFirmwareNotFoundErrorMessage"], filename));
  1186. return;
  1187. }
  1188. string dialogTitle = string.Format(LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallTitle"], firmwareVersion.VersionString);
  1189. SystemVersion currentVersion = _owner.ContentManager.GetCurrentFirmwareVersion();
  1190. string dialogMessage = string.Format(LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallMessage"], firmwareVersion.VersionString);
  1191. if (currentVersion != null)
  1192. {
  1193. dialogMessage += string.Format(LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallSubMessage"], currentVersion.VersionString);
  1194. }
  1195. dialogMessage += LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallConfirmMessage"];
  1196. UserResult result = await ContentDialogHelper.CreateConfirmationDialog(
  1197. dialogTitle,
  1198. dialogMessage,
  1199. LocaleManager.Instance["InputDialogYes"],
  1200. LocaleManager.Instance["InputDialogNo"],
  1201. LocaleManager.Instance["RyujinxConfirm"]);
  1202. UpdateWaitWindow waitingDialog = ContentDialogHelper.CreateWaitingDialog(dialogTitle, LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallWaitMessage"]);
  1203. if (result == UserResult.Yes)
  1204. {
  1205. Logger.Info?.Print(LogClass.Application, $"Installing firmware {firmwareVersion.VersionString}");
  1206. Thread thread = new(() =>
  1207. {
  1208. Dispatcher.UIThread.InvokeAsync(delegate
  1209. {
  1210. waitingDialog.Show();
  1211. });
  1212. try
  1213. {
  1214. _owner.ContentManager.InstallFirmware(filename);
  1215. Dispatcher.UIThread.InvokeAsync(async delegate
  1216. {
  1217. waitingDialog.Close();
  1218. string message = string.Format(LocaleManager.Instance["DialogFirmwareInstallerFirmwareInstallSuccessMessage"], firmwareVersion.VersionString);
  1219. await ContentDialogHelper.CreateInfoDialog(dialogTitle, message, LocaleManager.Instance["InputDialogOk"], "", LocaleManager.Instance["RyujinxInfo"]);
  1220. Logger.Info?.Print(LogClass.Application, message);
  1221. // Purge Applet Cache.
  1222. DirectoryInfo miiEditorCacheFolder = new DirectoryInfo(Path.Combine(AppDataManager.GamesDirPath, "0100000000001009", "cache"));
  1223. if (miiEditorCacheFolder.Exists)
  1224. {
  1225. miiEditorCacheFolder.Delete(true);
  1226. }
  1227. });
  1228. }
  1229. catch (Exception ex)
  1230. {
  1231. Dispatcher.UIThread.InvokeAsync(async () =>
  1232. {
  1233. waitingDialog.Close();
  1234. await ContentDialogHelper.CreateErrorDialog(ex.Message);
  1235. });
  1236. }
  1237. finally
  1238. {
  1239. _owner.RefreshFirmwareStatus();
  1240. }
  1241. });
  1242. thread.Name = "GUI.FirmwareInstallerThread";
  1243. thread.Start();
  1244. }
  1245. }
  1246. catch (LibHac.Common.Keys.MissingKeyException ex)
  1247. {
  1248. Logger.Error?.Print(LogClass.Application, ex.ToString());
  1249. Dispatcher.UIThread.Post(async () => await UserErrorDialog.ShowUserErrorDialog(UserError.NoKeys, _owner));
  1250. }
  1251. catch (Exception ex)
  1252. {
  1253. await ContentDialogHelper.CreateErrorDialog(ex.Message);
  1254. }
  1255. }
  1256. public async void InstallFirmwareFromFile()
  1257. {
  1258. OpenFileDialog dialog = new() { AllowMultiple = false };
  1259. dialog.Filters.Add(new FileDialogFilter { Name = LocaleManager.Instance["FileDialogAllTypes"], Extensions = { "xci", "zip" } });
  1260. dialog.Filters.Add(new FileDialogFilter { Name = "XCI", Extensions = { "xci" } });
  1261. dialog.Filters.Add(new FileDialogFilter { Name = "ZIP", Extensions = { "zip" } });
  1262. string[] file = await dialog.ShowAsync(_owner);
  1263. if (file != null && file.Length > 0)
  1264. {
  1265. await HandleFirmwareInstallation(file[0]);
  1266. }
  1267. }
  1268. public async void InstallFirmwareFromFolder()
  1269. {
  1270. OpenFolderDialog dialog = new();
  1271. string folder = await dialog.ShowAsync(_owner);
  1272. if (!string.IsNullOrWhiteSpace(folder))
  1273. {
  1274. await HandleFirmwareInstallation(folder);
  1275. }
  1276. }
  1277. }
  1278. }