2019-09-02 12:03:57 -04:00
using Gtk ;
2020-01-05 08:49:44 -03:00
using JsonPrettyPrinterPlus ;
2019-09-02 12:03:57 -04:00
using Ryujinx.Audio ;
using Ryujinx.Common.Logging ;
2020-01-05 08:49:44 -03:00
using Ryujinx.Configuration ;
2019-11-29 01:32:51 -03:00
using Ryujinx.Graphics.Gal ;
2019-12-21 16:52:31 -03:00
using Ryujinx.Graphics.Gal.OpenGL ;
2020-01-05 08:49:44 -03:00
using Ryujinx.HLE.FileSystem ;
2019-09-02 12:03:57 -04:00
using Ryujinx.Profiler ;
using System ;
2020-01-05 08:49:44 -03:00
using System.Diagnostics ;
2019-09-02 12:03:57 -04:00
using System.IO ;
using System.Reflection ;
using System.Text ;
using System.Threading ;
2019-12-21 16:52:31 -03:00
using System.Threading.Tasks ;
2019-11-29 01:32:51 -03:00
using Utf8Json ;
using Utf8Json.Resolvers ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
using GUI = Gtk . Builder . ObjectAttribute ;
namespace Ryujinx.Ui
2019-09-02 12:03:57 -04:00
{
public class MainWindow : Window
{
2019-11-29 01:32:51 -03:00
private static HLE . Switch _device ;
2019-09-02 12:03:57 -04:00
private static IGalRenderer _renderer ;
private static IAalOutput _audioOut ;
2019-11-29 01:32:51 -03:00
private static GlScreen _screen ;
2019-09-02 12:03:57 -04:00
private static ListStore _tableStore ;
2019-11-29 01:32:51 -03:00
private static bool _updatingGameTable ;
private static bool _gameLoaded ;
private static bool _ending ;
private static TreeView _treeView ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
#pragma warning disable CS0649
#pragma warning disable IDE0044
2019-09-02 12:03:57 -04:00
[GUI] Window _mainWin ;
[GUI] CheckMenuItem _fullScreen ;
[GUI] MenuItem _stopEmulation ;
2019-11-29 01:32:51 -03:00
[GUI] CheckMenuItem _favToggle ;
2019-09-02 12:03:57 -04:00
[GUI] CheckMenuItem _iconToggle ;
2019-11-29 01:32:51 -03:00
[GUI] CheckMenuItem _appToggle ;
2019-09-02 12:03:57 -04: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 23:49:51 -03:00
[GUI] TreeSelection _gameTableSelection ;
2019-11-29 01:32:51 -03:00
[GUI] Label _progressLabel ;
[GUI] LevelBar _progressBar ;
#pragma warning restore CS0649
#pragma warning restore IDE0044
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
public MainWindow ( ) : this ( new Builder ( "Ryujinx.Ui.MainWindow.glade" ) ) { }
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
private MainWindow ( Builder builder ) : base ( builder . GetObject ( "_mainWin" ) . Handle )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
builder . Autoconnect ( this ) ;
DeleteEvent + = Window_Close ;
ApplicationLibrary . ApplicationAdded + = Application_Added ;
2019-12-21 23:49:51 -03:00
_gameTable . ButtonReleaseEvent + = Row_Clicked ;
2020-01-05 08:49:44 -03:00
bool continueWithStartup = Migration . PromptIfMigrationNeededForStartup ( this , out bool migrationNeeded ) ;
if ( ! continueWithStartup )
{
End ( ) ;
}
2019-09-02 12:03:57 -04:00
_renderer = new OglRenderer ( ) ;
_audioOut = InitializeAudioEngine ( ) ;
2019-12-21 16:52:31 -03:00
// TODO: Initialization and dispose of HLE.Switch when starting/stoping emulation.
_device = InitializeSwitchInstance ( ) ;
2019-09-02 12:03:57 -04:00
2020-01-05 08:49:44 -03:00
if ( migrationNeeded )
{
bool migrationSuccessful = Migration . DoMigrationForStartup ( this , _device ) ;
if ( ! migrationSuccessful )
{
End ( ) ;
}
}
2019-11-29 01:32:51 -03:00
_treeView = _gameTable ;
2019-09-02 12:03:57 -04:00
ApplyTheme ( ) ;
2019-11-29 01:32:51 -03:00
_mainWin . Icon = new Gdk . Pixbuf ( Assembly . GetExecutingAssembly ( ) , "Ryujinx.Ui.assets.Icon.png" ) ;
2019-09-02 12:03:57 -04:00
_stopEmulation . Sensitive = false ;
2019-12-21 16:52:31 -03: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-29 01:32:51 -03:00
_gameTable . Model = _tableStore = new ListStore (
2019-12-21 16:52:31 -03:00
typeof ( bool ) ,
typeof ( Gdk . Pixbuf ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
typeof ( string ) ,
2019-11-29 01:32:51 -03:00
typeof ( string ) ) ;
2019-12-21 16:52:31 -03:00
2019-11-29 01:32:51 -03: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
}
internal static void ApplyTheme ( )
{
2019-12-21 16:52:31 -03:00
if ( ! ConfigurationState . Instance . Ui . EnableCustomTheme )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
return ;
}
2019-09-02 12:03:57 -04:00
2019-12-21 16:52:31 -03:00
if ( File . Exists ( ConfigurationState . Instance . Ui . CustomThemePath ) & & ( System . IO . Path . GetExtension ( ConfigurationState . Instance . Ui . CustomThemePath ) = = ".css" ) )
2019-11-29 01:32:51 -03:00
{
CssProvider cssProvider = new CssProvider ( ) ;
2019-09-02 12:03:57 -04:00
2019-12-21 16:52:31 -03:00
cssProvider . LoadFromPath ( ConfigurationState . Instance . Ui . CustomThemePath ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
StyleContext . AddProviderForScreen ( Gdk . Screen . Default , cssProvider , 800 ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
else
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
Logger . PrintWarning ( LogClass . Application , $"The \" custom_theme_path \ " section in \"Config.json\" contains an invalid path: \"{ConfigurationState.Instance.Ui.CustomThemePath}\"." ) ;
2019-11-29 01:32:51 -03:00
}
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void UpdateColumns ( )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
foreach ( TreeViewColumn column in _gameTable . Columns )
{
_gameTable . RemoveColumn ( column ) ;
}
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
CellRendererToggle favToggle = new CellRendererToggle ( ) ;
favToggle . Toggled + = FavToggle_Toggled ;
2019-12-21 16:52:31 -03: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-29 01:32:51 -03:00
foreach ( TreeViewColumn column in _gameTable . Columns )
2019-09-02 12:03:57 -04:00
{
2019-12-21 23:49:51 -03: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 12:03:57 -04:00
}
2019-12-21 16:52:31 -03:00
}
private HLE . Switch InitializeSwitchInstance ( )
{
HLE . Switch instance = new HLE . Switch ( _renderer , _audioOut ) ;
instance . Initialize ( ) ;
return instance ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
internal static async Task UpdateGameTable ( )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
if ( _updatingGameTable )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
return ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
_updatingGameTable = true ;
_tableStore . Clear ( ) ;
2020-01-05 08:49:44 -03:00
await Task . Run ( ( ) = > ApplicationLibrary . LoadApplications ( ConfigurationState . Instance . Ui . GameDirs ,
_device . System . KeySet , _device . System . State . DesiredTitleLanguage , _device . System . FsClient ,
_device . FileSystem ) ) ;
2019-11-29 01:32:51 -03:00
_updatingGameTable = false ;
2019-09-02 12:03:57 -04:00
}
2019-09-07 21:59:41 -04:00
internal void LoadApplication ( string path )
2019-09-02 12:03:57 -04:00
{
if ( _gameLoaded )
{
2019-11-29 01:32:51 -03:00
GtkDialog . CreateErrorDialog ( "A game has already been loaded. Please close the emulator and try again" ) ;
2019-09-02 12:03:57 -04:00
}
else
{
2019-09-19 20:59:48 -03:00
Logger . RestartTime ( ) ;
2019-12-21 16:52:31 -03:00
// TODO: Move this somewhere else + reloadable?
GraphicsConfig . ShadersDumpPath = ConfigurationState . Instance . Graphics . ShadersDumpPath ;
2019-09-02 12:03:57 -04: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." ) ;
_device . LoadCart ( path , romFsFiles [ 0 ] ) ;
}
else
{
Logger . PrintInfo ( LogClass . Application , "Loading as cart WITHOUT RomFS." ) ;
_device . LoadCart ( path ) ;
}
}
else if ( File . Exists ( path ) )
{
switch ( System . IO . Path . GetExtension ( path ) . ToLowerInvariant ( ) )
{
case ".xci" :
Logger . PrintInfo ( LogClass . Application , "Loading as XCI." ) ;
_device . LoadXci ( path ) ;
break ;
case ".nca" :
Logger . PrintInfo ( LogClass . Application , "Loading as NCA." ) ;
_device . LoadNca ( path ) ;
break ;
case ".nsp" :
case ".pfs0" :
Logger . PrintInfo ( LogClass . Application , "Loading as NSP." ) ;
_device . LoadNsp ( path ) ;
break ;
default :
Logger . PrintInfo ( LogClass . Application , "Loading as homebrew." ) ;
try
{
_device . LoadProgram ( path ) ;
}
catch ( ArgumentOutOfRangeException )
{
2019-09-19 20:59:48 -03:00
Logger . PrintError ( LogClass . Application , "The file which you have specified is unsupported by Ryujinx." ) ;
2019-09-02 12:03:57 -04:00
}
break ;
}
}
else
{
2019-09-19 20:59:48 -03:00
Logger . PrintWarning ( LogClass . Application , "Please specify a valid XCI/NCA/NSP/PFS0/NRO file." ) ;
2019-09-02 12:03:57 -04:00
End ( ) ;
}
2019-11-29 01:32:51 -03:00
#if MACOS_BUILD
CreateGameWindow ( ) ;
#else
new Thread ( CreateGameWindow ) . Start ( ) ;
#endif
2019-09-02 12:03:57 -04:00
_gameLoaded = true ;
_stopEmulation . Sensitive = true ;
2019-12-21 16:52:31 -03:00
DiscordIntegrationModule . SwitchToPlayingState ( _device . System . TitleId , _device . System . TitleName ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
string metadataFolder = System . IO . Path . Combine ( new VirtualFileSystem ( ) . GetBasePath ( ) , "games" , _device . System . TitleId , "gui" ) ;
string metadataFile = System . IO . Path . Combine ( metadataFolder , "metadata.json" ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
IJsonFormatterResolver resolver = CompositeResolver . Create ( new [ ] { StandardResolver . AllowPrivateSnakeCase } ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
ApplicationMetadata appMetadata ;
if ( ! File . Exists ( metadataFile ) )
{
Directory . CreateDirectory ( metadataFolder ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
appMetadata = new ApplicationMetadata
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
Favorite = false ,
TimePlayed = 0 ,
LastPlayed = "Never"
} ;
byte [ ] data = JsonSerializer . Serialize ( appMetadata , resolver ) ;
File . WriteAllText ( metadataFile , Encoding . UTF8 . GetString ( data , 0 , data . Length ) . PrettyPrintJson ( ) ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
using ( Stream stream = File . OpenRead ( metadataFile ) )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
appMetadata = JsonSerializer . Deserialize < ApplicationMetadata > ( stream , resolver ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
appMetadata . LastPlayed = DateTime . UtcNow . ToString ( ) ;
byte [ ] saveData = JsonSerializer . Serialize ( appMetadata , resolver ) ;
File . WriteAllText ( metadataFile , Encoding . UTF8 . GetString ( saveData , 0 , saveData . Length ) . PrettyPrintJson ( ) ) ;
2019-09-02 12:03:57 -04:00
}
}
private static void CreateGameWindow ( )
{
2019-12-21 16:52:31 -03:00
_device . Hid . InitializePrimaryController ( ConfigurationState . Instance . Hid . ControllerType ) ;
2019-11-29 01:32:51 -03:00
using ( _screen = new GlScreen ( _device , _renderer ) )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
_screen . MainLoop ( ) ;
2019-09-02 12:03:57 -04:00
End ( ) ;
}
}
private static void End ( )
{
2019-11-29 01:32:51 -03:00
if ( _ending )
{
return ;
}
_ending = true ;
2019-09-02 12:03:57 -04:00
if ( _gameLoaded )
{
2019-11-29 01:32:51 -03:00
string metadataFolder = System . IO . Path . Combine ( new VirtualFileSystem ( ) . GetBasePath ( ) , "games" , _device . System . TitleId , "gui" ) ;
string metadataFile = System . IO . Path . Combine ( metadataFolder , "metadata.json" ) ;
IJsonFormatterResolver resolver = CompositeResolver . Create ( new [ ] { StandardResolver . AllowPrivateSnakeCase } ) ;
ApplicationMetadata appMetadata ;
if ( ! File . Exists ( metadataFile ) )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
Directory . CreateDirectory ( metadataFolder ) ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
appMetadata = new ApplicationMetadata
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
Favorite = false ,
TimePlayed = 0 ,
LastPlayed = "Never"
} ;
2019-09-02 12:03:57 -04:00
2019-11-29 01:32:51 -03:00
byte [ ] data = JsonSerializer . Serialize ( appMetadata , resolver ) ;
File . WriteAllText ( metadataFile , Encoding . UTF8 . GetString ( data , 0 , data . Length ) . PrettyPrintJson ( ) ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
using ( Stream stream = File . OpenRead ( metadataFile ) )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
appMetadata = JsonSerializer . Deserialize < ApplicationMetadata > ( stream , resolver ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
DateTime lastPlayedDateTime = DateTime . Parse ( appMetadata . LastPlayed ) ;
double sessionTimePlayed = DateTime . UtcNow . Subtract ( lastPlayedDateTime ) . TotalSeconds ;
appMetadata . TimePlayed + = Math . Round ( sessionTimePlayed , MidpointRounding . AwayFromZero ) ;
byte [ ] saveData = JsonSerializer . Serialize ( appMetadata , resolver ) ;
File . WriteAllText ( metadataFile , Encoding . UTF8 . GetString ( saveData , 0 , saveData . Length ) . PrettyPrintJson ( ) ) ;
2019-09-02 12:03:57 -04:00
}
Profile . FinishProfiling ( ) ;
2020-01-05 08:49:44 -03:00
_device ? . Dispose ( ) ;
_audioOut ? . Dispose ( ) ;
2019-09-02 12:03:57 -04:00
Logger . Shutdown ( ) ;
Environment . Exit ( 0 ) ;
}
/// <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 ( )
{
if ( SoundIoAudioOut . IsSupported )
{
return new SoundIoAudioOut ( ) ;
}
else if ( OpenALAudioOut . IsSupported )
{
return new OpenALAudioOut ( ) ;
}
else
{
return new DummyAudioOut ( ) ;
}
}
//Events
2019-12-21 23:49:51 -03:00
private void Application_Added ( object sender , ApplicationAddedEventArgs args )
2019-11-29 01:32:51 -03:00
{
Application . Invoke ( delegate
{
_tableStore . AppendValues (
2019-12-21 23:49:51 -03: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-29 01:32:51 -03:00
} ) ;
}
private void FavToggle_Toggled ( object sender , ToggledArgs args )
{
_tableStore . GetIter ( out TreeIter treeIter , new TreePath ( args . Path ) ) ;
string titleId = _tableStore . GetValue ( treeIter , 2 ) . ToString ( ) . Split ( "\n" ) [ 1 ] . ToLower ( ) ;
string metadataPath = System . IO . Path . Combine ( new VirtualFileSystem ( ) . GetBasePath ( ) , "games" , titleId , "gui" , "metadata.json" ) ;
IJsonFormatterResolver resolver = CompositeResolver . Create ( new [ ] { StandardResolver . AllowPrivateSnakeCase } ) ;
ApplicationMetadata appMetadata ;
using ( Stream stream = File . OpenRead ( metadataPath ) )
{
appMetadata = JsonSerializer . Deserialize < ApplicationMetadata > ( stream , resolver ) ;
}
if ( ( bool ) _tableStore . GetValue ( treeIter , 0 ) )
{
_tableStore . SetValue ( treeIter , 0 , false ) ;
appMetadata . Favorite = false ;
}
else
{
_tableStore . SetValue ( treeIter , 0 , true ) ;
appMetadata . Favorite = true ;
}
byte [ ] saveData = JsonSerializer . Serialize ( appMetadata , resolver ) ;
File . WriteAllText ( metadataPath , Encoding . UTF8 . GetString ( saveData , 0 , saveData . Length ) . PrettyPrintJson ( ) ) ;
}
private void Row_Activated ( object sender , RowActivatedArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 23:49:51 -03:00
_gameTableSelection . GetSelected ( out TreeIter treeIter ) ;
2019-11-29 01:32:51 -03:00
string path = ( string ) _tableStore . GetValue ( treeIter , 9 ) ;
2019-09-02 12:03:57 -04:00
LoadApplication ( path ) ;
}
2019-12-21 23:49:51 -03: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-05 08:49:44 -03:00
GameTableContextMenu contextMenu = new GameTableContextMenu ( _tableStore , treeIter , _device . System . FsClient ) ;
2019-12-21 23:49:51 -03:00
contextMenu . ShowAll ( ) ;
contextMenu . PopupAtPointer ( null ) ;
}
2019-11-29 01:32:51 -03:00
private void Load_Application_File ( object sender , EventArgs args )
2019-09-02 12:03:57 -04: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-29 01:32:51 -03:00
fileChooser . Dispose ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Load_Application_Folder ( object sender , EventArgs args )
2019-09-02 12:03:57 -04: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-29 01:32:51 -03:00
fileChooser . Dispose ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Open_Ryu_Folder ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
Process . Start ( new ProcessStartInfo ( )
{
2019-11-29 01:32:51 -03:00
FileName = new VirtualFileSystem ( ) . GetBasePath ( ) ,
2019-09-02 12:03:57 -04:00
UseShellExecute = true ,
Verb = "open"
} ) ;
}
2019-11-29 01:32:51 -03:00
private void Exit_Pressed ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
_screen ? . Exit ( ) ;
2019-09-02 12:03:57 -04:00
End ( ) ;
}
2019-11-29 01:32:51 -03:00
private void Window_Close ( object sender , DeleteEventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
_screen ? . Exit ( ) ;
2019-09-02 12:03:57 -04:00
End ( ) ;
}
2019-11-29 01:32:51 -03:00
private void StopEmulation_Pressed ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
// TODO: Write logic to kill running game
2019-11-29 01:32:51 -03:00
_gameLoaded = false ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void FullScreen_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
if ( _fullScreen . Active )
{
Fullscreen ( ) ;
}
else
{
Unfullscreen ( ) ;
}
}
2019-11-29 01:32:51 -03:00
private void Settings_Pressed ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
SwitchSettings settingsWin = new SwitchSettings ( ) ;
2019-11-29 01:32:51 -03:00
settingsWin . Show ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Update_Pressed ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
string ryuUpdater = System . IO . Path . Combine ( new VirtualFileSystem ( ) . GetBasePath ( ) , "RyuUpdater.exe" ) ;
2019-09-02 12:03:57 -04:00
try
{
Process . Start ( new ProcessStartInfo ( ryuUpdater , "/U" ) { UseShellExecute = true } ) ;
}
catch ( System . ComponentModel . Win32Exception )
{
2019-11-29 01:32:51 -03:00
GtkDialog . CreateErrorDialog ( "Update canceled by user or updater was not found" ) ;
2019-09-02 12:03:57 -04:00
}
}
2019-11-29 01:32:51 -03:00
private void About_Pressed ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-11-29 01:32:51 -03:00
AboutWindow aboutWin = new AboutWindow ( ) ;
aboutWin . Show ( ) ;
}
private void Fav_Toggled ( object sender , EventArgs args )
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . FavColumn . Value = _favToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Icon_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . IconColumn . Value = _iconToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Title_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . AppColumn . Value = _appToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Developer_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . DevColumn . Value = _developerToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Version_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . VersionColumn . Value = _versionToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void TimePlayed_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . TimePlayedColumn . Value = _timePlayedToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void LastPlayed_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . LastPlayedColumn . Value = _lastPlayedToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void FileExt_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . FileExtColumn . Value = _fileExtToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void FileSize_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . FileSizeColumn . Value = _fileSizeToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03:00
UpdateColumns ( ) ;
2019-09-02 12:03:57 -04:00
}
2019-11-29 01:32:51 -03:00
private void Path_Toggled ( object sender , EventArgs args )
2019-09-02 12:03:57 -04:00
{
2019-12-21 16:52:31 -03:00
ConfigurationState . Instance . Ui . GuiColumns . PathColumn . Value = _pathToggle . Active ;
2019-11-29 01:32:51 -03:00
2019-12-21 16:52:31 -03:00
SaveConfig ( ) ;
2019-11-29 01:32:51 -03: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 12:03:57 -04:00
}
2019-12-21 16:52:31 -03:00
public static void SaveConfig ( )
{
ConfigurationState . Instance . ToFileFormat ( ) . SaveConfig ( System . IO . Path . Combine ( AppDomain . CurrentDomain . BaseDirectory , "Config.json" ) ) ;
}
2019-09-02 12:03:57 -04:00
}
}