MainWindowViewModel.cs 47 KB

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