MainWindowViewModel.cs 50 KB

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