2019-09-02 11:03:57 -05:00
using Gtk ;
2020-01-05 05:49:44 -06:00
using JsonPrettyPrinterPlus ;
2019-09-02 11:03:57 -05:00
using Ryujinx.Audio ;
using Ryujinx.Common.Logging ;
2020-01-05 05:49:44 -06:00
using Ryujinx.Configuration ;
2020-01-21 16:23:11 -06:00
using Ryujinx.Graphics.GAL ;
2019-10-13 01:02:07 -05:00
using Ryujinx.Graphics.OpenGL ;
2020-01-05 05:49:44 -06:00
using Ryujinx.HLE.FileSystem ;
2020-01-21 16:23:11 -06:00
using Ryujinx.HLE.FileSystem.Content ;
2019-09-02 11:03:57 -05:00
using Ryujinx.Profiler ;
using System ;
2020-01-05 05:49:44 -06:00
using System.Diagnostics ;
2019-09-02 11:03:57 -05:00
using System.IO ;
using System.Reflection ;
using System.Text ;
using System.Threading ;
2019-12-21 13:52:31 -06:00
using System.Threading.Tasks ;
2019-11-28 22:32:51 -06:00
using Utf8Json ;
using Utf8Json.Resolvers ;
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
using GUI = Gtk . Builder . ObjectAttribute ;
namespace Ryujinx.Ui
2019-09-02 11:03:57 -05:00
{
public class MainWindow : Window
{
2020-01-21 16:23:11 -06:00
private static VirtualFileSystem _virtualFileSystem ;
private static ContentManager _contentManager ;
2019-09-02 11:03:57 -05:00
2020-01-21 16:23:11 -06:00
private static HLE . Switch _emulationContext ;
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
private static GlScreen _screen ;
2019-09-02 11:03:57 -05:00
private static ListStore _tableStore ;
2019-11-28 22:32:51 -06:00
private static bool _updatingGameTable ;
private static bool _gameLoaded ;
private static bool _ending ;
private static TreeView _treeView ;
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
#pragma warning disable CS0649
#pragma warning disable IDE0044
2019-09-02 11:03:57 -05:00
[GUI] Window _mainWin ;
[GUI] CheckMenuItem _fullScreen ;
[GUI] MenuItem _stopEmulation ;
2019-11-28 22:32:51 -06:00
[GUI] CheckMenuItem _favToggle ;
2020-01-11 20:10:55 -06:00
[GUI] MenuItem _firmwareInstallFile ;
[GUI] MenuItem _firmwareInstallDirectory ;
2019-09-02 11:03:57 -05:00
[GUI] CheckMenuItem _iconToggle ;
2019-11-28 22:32:51 -06:00
[GUI] CheckMenuItem _appToggle ;
2019-09-02 11:03:57 -05:00
[GUI] CheckMenuItem _developerToggle ;
[GUI] CheckMenuItem _versionToggle ;
[GUI] CheckMenuItem _timePlayedToggle ;
[GUI] CheckMenuItem _lastPlayedToggle ;
[GUI] CheckMenuItem _fileExtToggle ;
[GUI] CheckMenuItem _fileSizeToggle ;
[GUI] CheckMenuItem _pathToggle ;
[GUI] TreeView _gameTable ;
2019-12-21 20:49:51 -06:00
[GUI] TreeSelection _gameTableSelection ;
2019-11-28 22:32:51 -06:00
[GUI] Label _progressLabel ;
2020-01-11 20:10:55 -06:00
[GUI] Label _firmwareVersionLabel ;
2019-11-28 22:32:51 -06:00
[GUI] LevelBar _progressBar ;
#pragma warning restore CS0649
#pragma warning restore IDE0044
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
public MainWindow ( ) : this ( new Builder ( "Ryujinx.Ui.MainWindow.glade" ) ) { }
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
private MainWindow ( Builder builder ) : base ( builder . GetObject ( "_mainWin" ) . Handle )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
builder . Autoconnect ( this ) ;
DeleteEvent + = Window_Close ;
ApplicationLibrary . ApplicationAdded + = Application_Added ;
2019-12-21 20:49:51 -06:00
_gameTable . ButtonReleaseEvent + = Row_Clicked ;
2020-01-21 16:23:11 -06:00
// First we check that a migration isn't needed. (because VirtualFileSystem will create the new directory otherwise)
2020-01-05 05:49:44 -06:00
bool continueWithStartup = Migration . PromptIfMigrationNeededForStartup ( this , out bool migrationNeeded ) ;
2019-10-13 01:02:07 -05:00
if ( ! continueWithStartup )
2020-01-05 05:49:44 -06:00
{
2020-01-21 16:23:11 -06:00
End ( null ) ;
2020-01-05 05:49:44 -06:00
}
2020-01-21 16:23:11 -06:00
_virtualFileSystem = new VirtualFileSystem ( ) ;
_contentManager = new ContentManager ( _virtualFileSystem ) ;
2019-09-02 11:03:57 -05:00
2020-01-05 05:49:44 -06:00
if ( migrationNeeded )
{
2020-01-21 16:23:11 -06:00
bool migrationSuccessful = Migration . DoMigrationForStartup ( this , _virtualFileSystem ) ;
2020-01-05 05:49:44 -06:00
if ( ! migrationSuccessful )
{
2020-01-21 16:23:11 -06:00
End ( null ) ;
2020-01-05 05:49:44 -06:00
}
}
2020-01-21 16:23:11 -06:00
// Make sure that everything is loaded.
_virtualFileSystem . Reload ( ) ;
2019-11-28 22:32:51 -06:00
_treeView = _gameTable ;
2019-09-02 11:03:57 -05:00
ApplyTheme ( ) ;
2019-11-28 22:32:51 -06:00
_mainWin . Icon = new Gdk . Pixbuf ( Assembly . GetExecutingAssembly ( ) , "Ryujinx.Ui.assets.Icon.png" ) ;
2019-09-02 11:03:57 -05:00
_stopEmulation . Sensitive = false ;
2019-12-21 13:52:31 -06:00
if ( ConfigurationState . Instance . Ui . GuiColumns . FavColumn ) _favToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . IconColumn ) _iconToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . AppColumn ) _appToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . DevColumn ) _developerToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . VersionColumn ) _versionToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . TimePlayedColumn ) _timePlayedToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . LastPlayedColumn ) _lastPlayedToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . FileExtColumn ) _fileExtToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . FileSizeColumn ) _fileSizeToggle . Active = true ;
if ( ConfigurationState . Instance . Ui . GuiColumns . PathColumn ) _pathToggle . Active = true ;
2019-11-28 22:32:51 -06:00
_gameTable . Model = _tableStore = new ListStore (
2019-12-21 13:52:31 -06:00
typeof ( bool ) ,
typeof ( Gdk . Pixbuf ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
2019-11-28 22:32:51 -06:00
typeof ( string ) ) ;
2019-12-21 13:52:31 -06:00
2019-11-28 22:32:51 -06:00
_tableStore . SetSortFunc ( 5 , TimePlayedSort ) ;
_tableStore . SetSortFunc ( 6 , LastPlayedSort ) ;
_tableStore . SetSortFunc ( 8 , FileSizeSort ) ;
_tableStore . SetSortColumnId ( 0 , SortType . Descending ) ;
UpdateColumns ( ) ;
#pragma warning disable CS4014
UpdateGameTable ( ) ;
#pragma warning restore CS4014
2020-01-11 20:10:55 -06:00
Task . Run ( RefreshFirmwareLabel ) ;
2019-11-28 22:32:51 -06:00
}
internal static void ApplyTheme ( )
{
2019-12-21 13:52:31 -06:00
if ( ! ConfigurationState . Instance . Ui . EnableCustomTheme )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
return ;
}
2019-09-02 11:03:57 -05:00
2019-12-21 13:52:31 -06:00
if ( File . Exists ( ConfigurationState . Instance . Ui . CustomThemePath ) & & ( System . IO . Path . GetExtension ( ConfigurationState . Instance . Ui . CustomThemePath ) = = ".css" ) )
2019-11-28 22:32:51 -06:00
{
CssProvider cssProvider = new CssProvider ( ) ;
2019-09-02 11:03:57 -05:00
2019-12-21 13:52:31 -06:00
cssProvider . LoadFromPath ( ConfigurationState . Instance . Ui . CustomThemePath ) ;
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
StyleContext . AddProviderForScreen ( Gdk . Screen . Default , cssProvider , 800 ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
else
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
Logger . PrintWarning ( LogClass . Application , $"The \" custom_theme_path \ " section in \"Config.json\" contains an invalid path: \"{ConfigurationState.Instance.Ui.CustomThemePath}\"." ) ;
2019-11-28 22:32:51 -06:00
}
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void UpdateColumns ( )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
foreach ( TreeViewColumn column in _gameTable . Columns )
{
_gameTable . RemoveColumn ( column ) ;
}
2019-09-02 11:03:57 -05:00
2019-11-28 22:32:51 -06:00
CellRendererToggle favToggle = new CellRendererToggle ( ) ;
favToggle . Toggled + = FavToggle_Toggled ;
2019-12-21 13:52:31 -06:00
if ( ConfigurationState . Instance . Ui . GuiColumns . FavColumn ) _gameTable . AppendColumn ( "Fav" , favToggle , "active" , 0 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . IconColumn ) _gameTable . AppendColumn ( "Icon" , new CellRendererPixbuf ( ) , "pixbuf" , 1 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . AppColumn ) _gameTable . AppendColumn ( "Application" , new CellRendererText ( ) , "text" , 2 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . DevColumn ) _gameTable . AppendColumn ( "Developer" , new CellRendererText ( ) , "text" , 3 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . VersionColumn ) _gameTable . AppendColumn ( "Version" , new CellRendererText ( ) , "text" , 4 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . TimePlayedColumn ) _gameTable . AppendColumn ( "Time Played" , new CellRendererText ( ) , "text" , 5 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . LastPlayedColumn ) _gameTable . AppendColumn ( "Last Played" , new CellRendererText ( ) , "text" , 6 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . FileExtColumn ) _gameTable . AppendColumn ( "File Ext" , new CellRendererText ( ) , "text" , 7 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . FileSizeColumn ) _gameTable . AppendColumn ( "File Size" , new CellRendererText ( ) , "text" , 8 ) ;
if ( ConfigurationState . Instance . Ui . GuiColumns . PathColumn ) _gameTable . AppendColumn ( "Path" , new CellRendererText ( ) , "text" , 9 ) ;
2019-11-28 22:32:51 -06:00
foreach ( TreeViewColumn column in _gameTable . Columns )
2019-09-02 11:03:57 -05:00
{
2019-12-21 20:49:51 -06:00
if ( column . Title = = "Fav" & & ConfigurationState . Instance . Ui . GuiColumns . FavColumn ) column . SortColumnId = 0 ;
else if ( column . Title = = "Application" & & ConfigurationState . Instance . Ui . GuiColumns . AppColumn ) column . SortColumnId = 2 ;
else if ( column . Title = = "Developer" & & ConfigurationState . Instance . Ui . GuiColumns . DevColumn ) column . SortColumnId = 3 ;
else if ( column . Title = = "Version" & & ConfigurationState . Instance . Ui . GuiColumns . VersionColumn ) column . SortColumnId = 4 ;
else if ( column . Title = = "Time Played" & & ConfigurationState . Instance . Ui . GuiColumns . TimePlayedColumn ) column . SortColumnId = 5 ;
else if ( column . Title = = "Last Played" & & ConfigurationState . Instance . Ui . GuiColumns . LastPlayedColumn ) column . SortColumnId = 6 ;
else if ( column . Title = = "File Ext" & & ConfigurationState . Instance . Ui . GuiColumns . FileExtColumn ) column . SortColumnId = 7 ;
else if ( column . Title = = "File Size" & & ConfigurationState . Instance . Ui . GuiColumns . FileSizeColumn ) column . SortColumnId = 8 ;
else if ( column . Title = = "Path" & & ConfigurationState . Instance . Ui . GuiColumns . PathColumn ) column . SortColumnId = 9 ;
2019-09-02 11:03:57 -05:00
}
2019-12-21 13:52:31 -06:00
}
private HLE . Switch InitializeSwitchInstance ( )
{
2020-01-21 16:23:11 -06:00
_virtualFileSystem . Reload ( ) ;
HLE . Switch instance = new HLE . Switch ( _virtualFileSystem , _contentManager , InitializeRenderer ( ) , InitializeAudioEngine ( ) ) ;
2019-12-21 13:52:31 -06:00
instance . Initialize ( ) ;
return instance ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
internal static async Task UpdateGameTable ( )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
if ( _updatingGameTable )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
return ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
_updatingGameTable = true ;
_tableStore . Clear ( ) ;
2020-01-05 05:49:44 -06:00
await Task . Run ( ( ) = > ApplicationLibrary . LoadApplications ( ConfigurationState . Instance . Ui . GameDirs ,
2020-01-21 16:23:11 -06:00
_virtualFileSystem , ConfigurationState . Instance . System . Language ) ) ;
2019-11-28 22:32:51 -06:00
_updatingGameTable = false ;
2019-09-02 11:03:57 -05:00
}
2019-09-07 20:59:41 -05:00
internal void LoadApplication ( string path )
2019-09-02 11:03:57 -05:00
{
if ( _gameLoaded )
{
2019-11-28 22:32:51 -06:00
GtkDialog . CreateErrorDialog ( "A game has already been loaded. Please close the emulator and try again" ) ;
2019-09-02 11:03:57 -05:00
}
else
{
2019-09-19 18:59:48 -05:00
Logger . RestartTime ( ) ;
2020-01-21 16:23:11 -06:00
HLE . Switch device = InitializeSwitchInstance ( ) ;
2019-12-21 13:52:31 -06:00
// TODO: Move this somewhere else + reloadable?
2020-01-21 16:23:11 -06:00
Graphics . Gpu . GraphicsConfig . ShadersDumpPath = ConfigurationState . Instance . Graphics . ShadersDumpPath ;
2019-12-21 13:52:31 -06:00
2019-09-02 11:03:57 -05:00
if ( Directory . Exists ( path ) )
{
string [ ] romFsFiles = Directory . GetFiles ( path , "*.istorage" ) ;
if ( romFsFiles . Length = = 0 )
{
romFsFiles = Directory . GetFiles ( path , "*.romfs" ) ;
}
if ( romFsFiles . Length > 0 )
{
Logger . PrintInfo ( LogClass . Application , "Loading as cart with RomFS." ) ;
2020-01-21 16:23:11 -06:00
device . LoadCart ( path , romFsFiles [ 0 ] ) ;
2019-09-02 11:03:57 -05:00
}
else
{
Logger . PrintInfo ( LogClass . Application , "Loading as cart WITHOUT RomFS." ) ;
2020-01-21 16:23:11 -06:00
device . LoadCart ( path ) ;
2019-09-02 11:03:57 -05:00
}
}
else if ( File . Exists ( path ) )
{
switch ( System . IO . Path . GetExtension ( path ) . ToLowerInvariant ( ) )
{
case ".xci" :
Logger . PrintInfo ( LogClass . Application , "Loading as XCI." ) ;
2020-01-21 16:23:11 -06:00
device . LoadXci ( path ) ;
2019-09-02 11:03:57 -05:00
break ;
case ".nca" :
Logger . PrintInfo ( LogClass . Application , "Loading as NCA." ) ;
2020-01-21 16:23:11 -06:00
device . LoadNca ( path ) ;
2019-09-02 11:03:57 -05:00
break ;
case ".nsp" :
case ".pfs0" :
Logger . PrintInfo ( LogClass . Application , "Loading as NSP." ) ;
2020-01-21 16:23:11 -06:00
device . LoadNsp ( path ) ;
2019-09-02 11:03:57 -05:00
break ;
default :
Logger . PrintInfo ( LogClass . Application , "Loading as homebrew." ) ;
try
{
2020-01-21 16:23:11 -06:00
device . LoadProgram ( path ) ;
2019-09-02 11:03:57 -05:00
}
catch ( ArgumentOutOfRangeException )
{
2019-09-19 18:59:48 -05:00
Logger . PrintError ( LogClass . Application , "The file which you have specified is unsupported by Ryujinx." ) ;
2019-09-02 11:03:57 -05:00
}
break ;
}
}
else
{
2019-09-19 18:59:48 -05:00
Logger . PrintWarning ( LogClass . Application , "Please specify a valid XCI/NCA/NSP/PFS0/NRO file." ) ;
2020-01-21 16:23:11 -06:00
End ( device ) ;
2019-09-02 11:03:57 -05:00
}
2020-01-21 16:23:11 -06:00
_emulationContext = device ;
2019-11-28 22:32:51 -06:00
#if MACOS_BUILD
2020-01-21 16:23:11 -06:00
CreateGameWindow ( device ) ;
2019-11-28 22:32:51 -06:00
#else
2020-01-21 16:23:11 -06:00
new Thread ( ( ) = > CreateGameWindow ( device ) ) . Start ( ) ;
2019-11-28 22:32:51 -06:00
#endif
2019-09-02 11:03:57 -05:00
_gameLoaded = true ;
_stopEmulation . Sensitive = true ;
2020-01-11 20:10:55 -06:00
_firmwareInstallFile . Sensitive = false ;
_firmwareInstallDirectory . Sensitive = false ;
2020-01-21 16:23:11 -06:00
DiscordIntegrationModule . SwitchToPlayingState ( device . System . TitleIdText , device . System . TitleName ) ;
2019-09-02 11:03:57 -05:00
2020-01-21 16:23:11 -06:00
ApplicationLibrary . LoadAndSaveMetaData ( device . System . TitleIdText , appMetadata = >
2019-09-02 11:03:57 -05:00
{
2020-01-11 21:01:04 -06:00
appMetadata . LastPlayed = DateTime . UtcNow . ToString ( ) ;
} ) ;
2019-09-02 11:03:57 -05:00
}
}
2020-01-21 16:23:11 -06:00
private void CreateGameWindow ( HLE . Switch device )
2019-09-02 11:03:57 -05:00
{
2020-01-21 16:23:11 -06:00
device . Hid . InitializePrimaryController ( ConfigurationState . Instance . Hid . ControllerType ) ;
2019-12-21 13:52:31 -06:00
2020-01-21 16:23:11 -06:00
using ( _screen = new GlScreen ( device ) )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
_screen . MainLoop ( ) ;
2020-01-21 16:23:11 -06:00
}
device . Dispose ( ) ;
_emulationContext = null ;
_screen = null ;
_gameLoaded = false ;
DiscordIntegrationModule . SwitchToMainMenu ( ) ;
Application . Invoke ( delegate
{
_stopEmulation . Sensitive = false ;
_firmwareInstallFile . Sensitive = true ;
_firmwareInstallDirectory . Sensitive = true ;
} ) ;
}
private static void UpdateGameMetadata ( string titleId )
{
if ( _gameLoaded )
{
ApplicationLibrary . LoadAndSaveMetaData ( titleId , appMetadata = >
{
DateTime lastPlayedDateTime = DateTime . Parse ( appMetadata . LastPlayed ) ;
double sessionTimePlayed = DateTime . UtcNow . Subtract ( lastPlayedDateTime ) . TotalSeconds ;
2019-09-02 11:03:57 -05:00
2020-01-21 16:23:11 -06:00
appMetadata . TimePlayed + = Math . Round ( sessionTimePlayed , MidpointRounding . AwayFromZero ) ;
} ) ;
2019-09-02 11:03:57 -05:00
}
}
2020-01-21 16:23:11 -06:00
private void End ( HLE . Switch device )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
if ( _ending )
{
return ;
}
_ending = true ;
2020-01-21 16:23:11 -06:00
if ( device ! = null )
2019-09-02 11:03:57 -05:00
{
2020-01-21 16:23:11 -06:00
UpdateGameMetadata ( device . System . TitleIdText ) ;
2019-09-02 11:03:57 -05:00
}
2020-01-21 16:23:11 -06:00
Dispose ( ) ;
2019-09-02 11:03:57 -05:00
Profile . FinishProfiling ( ) ;
2020-01-21 16:23:11 -06:00
device ? . Dispose ( ) ;
DiscordIntegrationModule . Exit ( ) ;
2019-09-02 11:03:57 -05:00
Logger . Shutdown ( ) ;
2020-01-21 16:23:11 -06:00
Application . Quit ( ) ;
}
private static IRenderer InitializeRenderer ( )
{
return new Renderer ( ) ;
2019-09-02 11:03:57 -05:00
}
/// <summary>
/// Picks an <see cref="IAalOutput"/> audio output renderer supported on this machine
/// </summary>
/// <returns>An <see cref="IAalOutput"/> supported by this machine</returns>
private static IAalOutput InitializeAudioEngine ( )
{
2020-01-01 09:39:09 -06:00
if ( OpenALAudioOut . IsSupported )
2019-09-02 11:03:57 -05:00
{
2020-01-01 09:39:09 -06:00
return new OpenALAudioOut ( ) ;
2019-09-02 11:03:57 -05:00
}
2020-01-01 09:39:09 -06:00
else if ( SoundIoAudioOut . IsSupported )
2019-09-02 11:03:57 -05:00
{
2020-01-01 09:39:09 -06:00
return new SoundIoAudioOut ( ) ;
2019-09-02 11:03:57 -05:00
}
else
{
return new DummyAudioOut ( ) ;
}
}
//Events
2019-12-21 20:49:51 -06:00
private void Application_Added ( object sender , ApplicationAddedEventArgs args )
2019-11-28 22:32:51 -06:00
{
Application . Invoke ( delegate
{
_tableStore . AppendValues (
2019-12-21 20:49:51 -06:00
args . AppData . Favorite ,
new Gdk . Pixbuf ( args . AppData . Icon , 75 , 75 ) ,
$"{args.AppData.TitleName}\n{args.AppData.TitleId.ToUpper()}" ,
args . AppData . Developer ,
args . AppData . Version ,
args . AppData . TimePlayed ,
args . AppData . LastPlayed ,
args . AppData . FileExtension ,
args . AppData . FileSize ,
args . AppData . Path ) ;
_progressLabel . Text = $"{args.NumAppsLoaded}/{args.NumAppsFound} Games Loaded" ;
_progressBar . Value = ( float ) args . NumAppsLoaded / args . NumAppsFound ;
2019-11-28 22:32:51 -06:00
} ) ;
}
private void FavToggle_Toggled ( object sender , ToggledArgs args )
{
_tableStore . GetIter ( out TreeIter treeIter , new TreePath ( args . Path ) ) ;
2020-01-11 21:01:04 -06:00
string titleId = _tableStore . GetValue ( treeIter , 2 ) . ToString ( ) . Split ( "\n" ) [ 1 ] . ToLower ( ) ;
2019-11-28 22:32:51 -06:00
2020-01-11 21:01:04 -06:00
bool newToggleValue = ! ( bool ) _tableStore . GetValue ( treeIter , 0 ) ;
2019-10-13 01:02:07 -05:00
2020-01-11 21:01:04 -06:00
_tableStore . SetValue ( treeIter , 0 , newToggleValue ) ;
2019-11-28 22:32:51 -06:00
2020-01-11 21:01:04 -06:00
ApplicationLibrary . LoadAndSaveMetaData ( titleId , appMetadata = >
2019-11-28 22:32:51 -06:00
{
2020-01-11 21:01:04 -06:00
appMetadata . Favorite = newToggleValue ;
} ) ;
2019-11-28 22:32:51 -06:00
}
private void Row_Activated ( object sender , RowActivatedArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 20:49:51 -06:00
_gameTableSelection . GetSelected ( out TreeIter treeIter ) ;
2019-11-28 22:32:51 -06:00
string path = ( string ) _tableStore . GetValue ( treeIter , 9 ) ;
2019-09-02 11:03:57 -05:00
LoadApplication ( path ) ;
}
2019-12-21 20:49:51 -06:00
private void Row_Clicked ( object sender , ButtonReleaseEventArgs args )
{
if ( args . Event . Button ! = 3 ) return ;
_gameTableSelection . GetSelected ( out TreeIter treeIter ) ;
if ( treeIter . UserData = = IntPtr . Zero ) return ;
2020-01-21 16:23:11 -06:00
GameTableContextMenu contextMenu = new GameTableContextMenu ( _tableStore , treeIter , _virtualFileSystem . FsClient ) ;
2019-12-21 20:49:51 -06:00
contextMenu . ShowAll ( ) ;
contextMenu . PopupAtPointer ( null ) ;
}
2019-11-28 22:32:51 -06:00
private void Load_Application_File ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
FileChooserDialog fileChooser = new FileChooserDialog ( "Choose the file to open" , this , FileChooserAction . Open , "Cancel" , ResponseType . Cancel , "Open" , ResponseType . Accept ) ;
fileChooser . Filter = new FileFilter ( ) ;
fileChooser . Filter . AddPattern ( "*.nsp" ) ;
fileChooser . Filter . AddPattern ( "*.pfs0" ) ;
fileChooser . Filter . AddPattern ( "*.xci" ) ;
fileChooser . Filter . AddPattern ( "*.nca" ) ;
fileChooser . Filter . AddPattern ( "*.nro" ) ;
fileChooser . Filter . AddPattern ( "*.nso" ) ;
if ( fileChooser . Run ( ) = = ( int ) ResponseType . Accept )
{
LoadApplication ( fileChooser . Filename ) ;
}
2019-11-28 22:32:51 -06:00
fileChooser . Dispose ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Load_Application_Folder ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
FileChooserDialog fileChooser = new FileChooserDialog ( "Choose the folder to open" , this , FileChooserAction . SelectFolder , "Cancel" , ResponseType . Cancel , "Open" , ResponseType . Accept ) ;
if ( fileChooser . Run ( ) = = ( int ) ResponseType . Accept )
{
LoadApplication ( fileChooser . Filename ) ;
}
2019-11-28 22:32:51 -06:00
fileChooser . Dispose ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Open_Ryu_Folder ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
Process . Start ( new ProcessStartInfo ( )
{
2019-11-28 22:32:51 -06:00
FileName = new VirtualFileSystem ( ) . GetBasePath ( ) ,
2019-09-02 11:03:57 -05:00
UseShellExecute = true ,
Verb = "open"
} ) ;
}
2019-11-28 22:32:51 -06:00
private void Exit_Pressed ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
_screen ? . Exit ( ) ;
2020-01-21 16:23:11 -06:00
End ( _emulationContext ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Window_Close ( object sender , DeleteEventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
_screen ? . Exit ( ) ;
2020-01-21 16:23:11 -06:00
End ( _emulationContext ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void StopEmulation_Pressed ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2020-01-21 16:23:11 -06:00
_screen ? . Exit ( ) ;
2019-09-02 11:03:57 -05:00
}
2020-01-11 20:10:55 -06:00
private void Installer_File_Pressed ( object o , EventArgs args )
{
FileChooserDialog fileChooser = new FileChooserDialog ( "Choose the firmware file to open" ,
this ,
FileChooserAction . Open ,
"Cancel" ,
ResponseType . Cancel ,
"Open" ,
ResponseType . Accept ) ;
fileChooser . Filter = new FileFilter ( ) ;
fileChooser . Filter . AddPattern ( "*.zip" ) ;
fileChooser . Filter . AddPattern ( "*.xci" ) ;
HandleInstallerDialog ( fileChooser ) ;
}
private void Installer_Directory_Pressed ( object o , EventArgs args )
{
FileChooserDialog directoryChooser = new FileChooserDialog ( "Choose the firmware directory to open" ,
this ,
FileChooserAction . SelectFolder ,
"Cancel" ,
ResponseType . Cancel ,
"Open" ,
ResponseType . Accept ) ;
HandleInstallerDialog ( directoryChooser ) ;
}
private void RefreshFirmwareLabel ( )
{
2020-01-21 16:23:11 -06:00
var currentFirmware = _contentManager . GetCurrentFirmwareVersion ( ) ;
2020-01-11 20:10:55 -06:00
GLib . Idle . Add ( new GLib . IdleHandler ( ( ) = >
{
_firmwareVersionLabel . Text = currentFirmware ! = null ? currentFirmware . VersionString : "0.0.0" ;
return false ;
} ) ) ;
}
private void HandleInstallerDialog ( FileChooserDialog fileChooser )
{
if ( fileChooser . Run ( ) = = ( int ) ResponseType . Accept )
{
MessageDialog dialog = null ;
try
{
string filename = fileChooser . Filename ;
fileChooser . Dispose ( ) ;
2020-01-21 16:23:11 -06:00
var firmwareVersion = _contentManager . VerifyFirmwarePackage ( filename ) ;
2020-01-11 20:10:55 -06:00
if ( firmwareVersion = = null )
{
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Info , ButtonsType . Ok , false , "" ) ;
dialog . Text = "Firmware not found." ;
dialog . SecondaryText = $"A valid system firmware was not found in {filename}." ;
Logger . PrintError ( LogClass . Application , $"A valid system firmware was not found in {filename}." ) ;
dialog . Run ( ) ;
dialog . Hide ( ) ;
dialog . Dispose ( ) ;
return ;
}
2020-01-21 16:23:11 -06:00
var currentVersion = _contentManager . GetCurrentFirmwareVersion ( ) ;
2020-01-11 20:10:55 -06:00
string dialogMessage = $"System version {firmwareVersion.VersionString} will be installed." ;
if ( currentVersion ! = null )
{
dialogMessage + = $"This will replace the current system version {currentVersion.VersionString}. " ;
}
dialogMessage + = "Do you want to continue?" ;
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Question , ButtonsType . YesNo , false , "" ) ;
dialog . Text = $"Install Firmware {firmwareVersion.VersionString}" ;
dialog . SecondaryText = dialogMessage ;
int response = dialog . Run ( ) ;
dialog . Dispose ( ) ;
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Info , ButtonsType . None , false , "" ) ;
dialog . Text = $"Install Firmware {firmwareVersion.VersionString}" ;
dialog . SecondaryText = "Installing firmware..." ;
if ( response = = ( int ) ResponseType . Yes )
{
Logger . PrintInfo ( LogClass . Application , $"Installing firmware {firmwareVersion.VersionString}" ) ;
Thread thread = new Thread ( ( ) = >
{
GLib . Idle . Add ( new GLib . IdleHandler ( ( ) = >
{
dialog . Run ( ) ;
return false ;
} ) ) ;
try
{
2020-01-21 16:23:11 -06:00
_contentManager . InstallFirmware ( filename ) ;
2020-01-11 20:10:55 -06:00
GLib . Idle . Add ( new GLib . IdleHandler ( ( ) = >
{
dialog . Dispose ( ) ;
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Info , ButtonsType . Ok , false , "" ) ;
dialog . Text = $"Install Firmware {firmwareVersion.VersionString}" ;
dialog . SecondaryText = $"System version {firmwareVersion.VersionString} successfully installed." ;
Logger . PrintInfo ( LogClass . Application , $"System version {firmwareVersion.VersionString} successfully installed." ) ;
dialog . Run ( ) ;
dialog . Dispose ( ) ;
return false ;
} ) ) ;
}
catch ( Exception ex )
{
GLib . Idle . Add ( new GLib . IdleHandler ( ( ) = >
{
dialog . Dispose ( ) ;
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Info , ButtonsType . Ok , false , "" ) ;
dialog . Text = $"Install Firmware {firmwareVersion.VersionString} Failed." ;
dialog . SecondaryText = $"An error occured while installing system version {firmwareVersion.VersionString}." +
" Please check logs for more info." ;
Logger . PrintError ( LogClass . Application , ex . Message ) ;
dialog . Run ( ) ;
dialog . Dispose ( ) ;
return false ;
} ) ) ;
}
finally
{
RefreshFirmwareLabel ( ) ;
}
} ) ;
2020-01-12 18:21:54 -06:00
thread . Name = "GUI.FirmwareInstallerThread" ;
2020-01-11 20:10:55 -06:00
thread . Start ( ) ;
}
else
{
dialog . Dispose ( ) ;
}
}
catch ( Exception ex )
{
if ( dialog ! = null )
{
dialog . Dispose ( ) ;
}
dialog = new MessageDialog ( this , DialogFlags . Modal , MessageType . Info , ButtonsType . Ok , false , "" ) ;
dialog . Text = "Parsing Firmware Failed." ;
dialog . SecondaryText = "An error occured while parsing firmware. Please check the logs for more info." ;
Logger . PrintError ( LogClass . Application , ex . Message ) ;
dialog . Run ( ) ;
dialog . Dispose ( ) ;
}
}
else
{
fileChooser . Dispose ( ) ;
}
}
private void FullScreen_Toggled ( object o , EventArgs args )
2019-09-02 11:03:57 -05:00
{
if ( _fullScreen . Active )
{
Fullscreen ( ) ;
}
else
{
Unfullscreen ( ) ;
}
}
2019-11-28 22:32:51 -06:00
private void Settings_Pressed ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
SwitchSettings settingsWin = new SwitchSettings ( ) ;
2019-11-28 22:32:51 -06:00
settingsWin . Show ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Update_Pressed ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
string ryuUpdater = System . IO . Path . Combine ( new VirtualFileSystem ( ) . GetBasePath ( ) , "RyuUpdater.exe" ) ;
2019-09-02 11:03:57 -05:00
try
{
Process . Start ( new ProcessStartInfo ( ryuUpdater , "/U" ) { UseShellExecute = true } ) ;
}
catch ( System . ComponentModel . Win32Exception )
{
2019-11-28 22:32:51 -06:00
GtkDialog . CreateErrorDialog ( "Update canceled by user or updater was not found" ) ;
2019-09-02 11:03:57 -05:00
}
}
2019-11-28 22:32:51 -06:00
private void About_Pressed ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-11-28 22:32:51 -06:00
AboutWindow aboutWin = new AboutWindow ( ) ;
aboutWin . Show ( ) ;
}
private void Fav_Toggled ( object sender , EventArgs args )
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . FavColumn . Value = _favToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Icon_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . IconColumn . Value = _iconToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Title_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . AppColumn . Value = _appToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Developer_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . DevColumn . Value = _developerToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Version_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . VersionColumn . Value = _versionToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void TimePlayed_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . TimePlayedColumn . Value = _timePlayedToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void LastPlayed_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . LastPlayedColumn . Value = _lastPlayedToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void FileExt_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . FileExtColumn . Value = _fileExtToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void FileSize_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . FileSizeColumn . Value = _fileSizeToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
2019-09-02 11:03:57 -05:00
}
2019-11-28 22:32:51 -06:00
private void Path_Toggled ( object sender , EventArgs args )
2019-09-02 11:03:57 -05:00
{
2019-12-21 13:52:31 -06:00
ConfigurationState . Instance . Ui . GuiColumns . PathColumn . Value = _pathToggle . Active ;
2019-11-28 22:32:51 -06:00
2019-12-21 13:52:31 -06:00
SaveConfig ( ) ;
2019-11-28 22:32:51 -06:00
UpdateColumns ( ) ;
}
private void RefreshList_Pressed ( object sender , ButtonReleaseEventArgs args )
{
#pragma warning disable CS4014
UpdateGameTable ( ) ;
#pragma warning restore CS4014
}
private static int TimePlayedSort ( ITreeModel model , TreeIter a , TreeIter b )
{
string aValue = model . GetValue ( a , 5 ) . ToString ( ) ;
string bValue = model . GetValue ( b , 5 ) . ToString ( ) ;
if ( aValue . Length > 4 & & aValue . Substring ( aValue . Length - 4 ) = = "mins" )
{
aValue = ( float . Parse ( aValue . Substring ( 0 , aValue . Length - 5 ) ) * 60 ) . ToString ( ) ;
}
else if ( aValue . Length > 3 & & aValue . Substring ( aValue . Length - 3 ) = = "hrs" )
{
aValue = ( float . Parse ( aValue . Substring ( 0 , aValue . Length - 4 ) ) * 3600 ) . ToString ( ) ;
}
else if ( aValue . Length > 4 & & aValue . Substring ( aValue . Length - 4 ) = = "days" )
{
aValue = ( float . Parse ( aValue . Substring ( 0 , aValue . Length - 5 ) ) * 86400 ) . ToString ( ) ;
}
else
{
aValue = aValue . Substring ( 0 , aValue . Length - 1 ) ;
}
if ( bValue . Length > 4 & & bValue . Substring ( bValue . Length - 4 ) = = "mins" )
{
bValue = ( float . Parse ( bValue . Substring ( 0 , bValue . Length - 5 ) ) * 60 ) . ToString ( ) ;
}
else if ( bValue . Length > 3 & & bValue . Substring ( bValue . Length - 3 ) = = "hrs" )
{
bValue = ( float . Parse ( bValue . Substring ( 0 , bValue . Length - 4 ) ) * 3600 ) . ToString ( ) ;
}
else if ( bValue . Length > 4 & & bValue . Substring ( bValue . Length - 4 ) = = "days" )
{
bValue = ( float . Parse ( bValue . Substring ( 0 , bValue . Length - 5 ) ) * 86400 ) . ToString ( ) ;
}
else
{
bValue = bValue . Substring ( 0 , bValue . Length - 1 ) ;
}
if ( float . Parse ( aValue ) > float . Parse ( bValue ) )
{
return - 1 ;
}
else if ( float . Parse ( bValue ) > float . Parse ( aValue ) )
{
return 1 ;
}
else
{
return 0 ;
}
}
private static int LastPlayedSort ( ITreeModel model , TreeIter a , TreeIter b )
{
string aValue = model . GetValue ( a , 6 ) . ToString ( ) ;
string bValue = model . GetValue ( b , 6 ) . ToString ( ) ;
if ( aValue = = "Never" )
{
aValue = DateTime . UnixEpoch . ToString ( ) ;
}
if ( bValue = = "Never" )
{
bValue = DateTime . UnixEpoch . ToString ( ) ;
}
return DateTime . Compare ( DateTime . Parse ( bValue ) , DateTime . Parse ( aValue ) ) ;
}
private static int FileSizeSort ( ITreeModel model , TreeIter a , TreeIter b )
{
string aValue = model . GetValue ( a , 8 ) . ToString ( ) ;
string bValue = model . GetValue ( b , 8 ) . ToString ( ) ;
if ( aValue . Substring ( aValue . Length - 2 ) = = "GB" )
{
aValue = ( float . Parse ( aValue [ 0. . ^ 2 ] ) * 1024 ) . ToString ( ) ;
}
else
{
aValue = aValue [ 0. . ^ 2 ] ;
}
if ( bValue . Substring ( bValue . Length - 2 ) = = "GB" )
{
bValue = ( float . Parse ( bValue [ 0. . ^ 2 ] ) * 1024 ) . ToString ( ) ;
}
else
{
bValue = bValue [ 0. . ^ 2 ] ;
}
if ( float . Parse ( aValue ) > float . Parse ( bValue ) )
{
return - 1 ;
}
else if ( float . Parse ( bValue ) > float . Parse ( aValue ) )
{
return 1 ;
}
else
{
return 0 ;
}
2019-09-02 11:03:57 -05:00
}
2019-12-21 13:52:31 -06:00
public static void SaveConfig ( )
{
ConfigurationState . Instance . ToFileFormat ( ) . SaveConfig ( System . IO . Path . Combine ( AppDomain . CurrentDomain . BaseDirectory , "Config.json" ) ) ;
}
2019-09-02 11:03:57 -05:00
}
}