2015-08-31 21:35:33 -07:00
// Copyright 2015 Citra Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
# pragma once
# include <atomic>
2018-04-15 15:42:58 -07:00
# include <map>
# include <unordered_map>
2018-04-18 13:01:45 -07:00
# include <QCoreApplication>
2016-04-13 14:04:05 -07:00
# include <QImage>
2018-04-15 15:42:58 -07:00
# include <QObject>
# include <QPainter>
2015-08-31 21:35:33 -07:00
# include <QRunnable>
# include <QStandardItem>
# include <QString>
# include "citra_qt/util/util.h"
2018-05-19 02:21:26 -07:00
# include "common/file_util.h"
2018-04-15 15:42:58 -07:00
# include "common/logging/log.h"
2016-09-17 17:38:01 -07:00
# include "common/string_util.h"
2016-05-17 16:42:45 -07:00
# include "core/loader/smdh.h"
2016-04-13 14:04:05 -07:00
/**
2017-02-26 17:58:51 -08:00
* Gets the game icon from SMDH data .
* @ param smdh SMDH data
2016-04-13 14:04:05 -07:00
* @ param large If true , returns large icon ( 48 x48 ) , otherwise returns small icon ( 24 x24 )
* @ return QPixmap game icon
*/
2016-05-17 16:42:45 -07:00
static QPixmap GetQPixmapFromSMDH ( const Loader : : SMDH & smdh , bool large ) {
std : : vector < u16 > icon_data = smdh . GetIcon ( large ) ;
const uchar * data = reinterpret_cast < const uchar * > ( icon_data . data ( ) ) ;
int size = large ? 48 : 24 ;
QImage icon ( data , size , size , QImage : : Format : : Format_RGB16 ) ;
2016-04-13 14:04:05 -07:00
return QPixmap : : fromImage ( icon ) ;
}
/**
* Gets the default icon ( for games without valid SMDH )
* @ param large If true , returns large icon ( 48 x48 ) , otherwise returns small icon ( 24 x24 )
* @ return QPixmap default icon
*/
static QPixmap GetDefaultIcon ( bool large ) {
int size = large ? 48 : 24 ;
QPixmap icon ( size , size ) ;
icon . fill ( Qt : : transparent ) ;
return icon ;
}
2018-04-15 15:42:58 -07:00
/**
* Creates a circle pixmap from a specified color
* @ param color The color the pixmap shall have
* @ return QPixmap circle pixmap
*/
static QPixmap CreateCirclePixmapFromColor ( const QColor & color ) {
QPixmap circle_pixmap ( 16 , 16 ) ;
circle_pixmap . fill ( Qt : : transparent ) ;
QPainter painter ( & circle_pixmap ) ;
painter . setPen ( color ) ;
painter . setBrush ( color ) ;
painter . drawEllipse ( 0 , 0 , 15 , 15 ) ;
return circle_pixmap ;
}
2016-04-13 14:04:05 -07:00
/**
2017-02-26 17:58:51 -08:00
* Gets the short game title from SMDH data .
* @ param smdh SMDH data
2016-04-13 14:04:05 -07:00
* @ param language title language
* @ return QString short title
*/
2016-09-17 17:38:01 -07:00
static QString GetQStringShortTitleFromSMDH ( const Loader : : SMDH & smdh ,
Loader : : SMDH : : TitleLanguage language ) {
2016-05-17 16:42:45 -07:00
return QString : : fromUtf16 ( smdh . GetShortTitle ( language ) . data ( ) ) ;
2016-04-13 14:04:05 -07:00
}
2015-08-31 21:35:33 -07:00
2018-05-01 10:57:01 -07:00
/**
* Gets the game region from SMDH data .
* @ param smdh SMDH data
* @ return QString region
*/
static QString GetRegionFromSMDH ( const Loader : : SMDH & smdh ) {
const Loader : : SMDH : : GameRegion region = smdh . GetRegion ( ) ;
switch ( region ) {
case Loader : : SMDH : : GameRegion : : Invalid :
return QObject : : tr ( " Invalid region " ) ;
case Loader : : SMDH : : GameRegion : : Japan :
return QObject : : tr ( " Japan " ) ;
case Loader : : SMDH : : GameRegion : : NorthAmerica :
return QObject : : tr ( " North America " ) ;
case Loader : : SMDH : : GameRegion : : Europe :
return QObject : : tr ( " Europe " ) ;
case Loader : : SMDH : : GameRegion : : Australia :
return QObject : : tr ( " Australia " ) ;
case Loader : : SMDH : : GameRegion : : China :
return QObject : : tr ( " China " ) ;
case Loader : : SMDH : : GameRegion : : Korea :
return QObject : : tr ( " Korea " ) ;
case Loader : : SMDH : : GameRegion : : Taiwan :
return QObject : : tr ( " Taiwan " ) ;
case Loader : : SMDH : : GameRegion : : RegionFree :
return QObject : : tr ( " Region free " ) ;
default :
return QObject : : tr ( " Invalid Region " ) ;
}
}
2018-04-15 15:42:58 -07:00
struct CompatStatus {
QString color ;
2018-04-17 09:14:39 -07:00
const char * text ;
const char * tooltip ;
2018-04-15 15:42:58 -07:00
} ;
// When this is put in a class, MSVS builds crash when closing Citra
// clang-format off
const static inline std : : map < QString , CompatStatus > status_data = {
2018-04-17 09:14:39 -07:00
{ " 0 " , { " #5c93ed " , QT_TRANSLATE_NOOP ( " GameList " , " Perfect " ) , QT_TRANSLATE_NOOP ( " GameList " , " Game functions flawless with no audio or graphical glitches, all tested functionality works as intended without \n any workarounds needed. " ) } } ,
{ " 1 " , { " #47d35c " , QT_TRANSLATE_NOOP ( " GameList " , " Great " ) , QT_TRANSLATE_NOOP ( " GameList " , " Game functions with minor graphical or audio glitches and is playable from start to finish. May require some \n workarounds. " ) } } ,
{ " 2 " , { " #94b242 " , QT_TRANSLATE_NOOP ( " GameList " , " Okay " ) , QT_TRANSLATE_NOOP ( " GameList " , " Game functions with major graphical or audio glitches, but game is playable from start to finish with \n workarounds. " ) } } ,
{ " 3 " , { " #f2d624 " , QT_TRANSLATE_NOOP ( " GameList " , " Bad " ) , QT_TRANSLATE_NOOP ( " GameList " , " Game functions, but with major graphical or audio glitches. Unable to progress in specific areas due to glitches \n even with workarounds. " ) } } ,
{ " 4 " , { " #FF0000 " , QT_TRANSLATE_NOOP ( " GameList " , " Intro/Menu " ) , QT_TRANSLATE_NOOP ( " GameList " , " Game is completely unplayable due to major graphical or audio glitches. Unable to progress past the Start \n Screen. " ) } } ,
{ " 5 " , { " #828282 " , QT_TRANSLATE_NOOP ( " GameList " , " Won't Boot " ) , QT_TRANSLATE_NOOP ( " GameList " , " The game crashes when attempting to startup. " ) } } ,
{ " 99 " , { " #000000 " , QT_TRANSLATE_NOOP ( " GameList " , " Not Tested " ) , QT_TRANSLATE_NOOP ( " GameList " , " The game has not yet been tested. " ) } } , } ;
2018-04-15 15:42:58 -07:00
// clang-format on
2015-08-31 21:35:33 -07:00
2018-04-15 15:42:58 -07:00
class GameListItem : public QStandardItem {
2015-08-31 21:35:33 -07:00
public :
2016-09-18 18:01:46 -07:00
GameListItem ( ) : QStandardItem ( ) { }
GameListItem ( const QString & string ) : QStandardItem ( string ) { }
virtual ~ GameListItem ( ) override { }
2015-08-31 21:35:33 -07:00
} ;
/**
* A specialization of GameListItem for path values .
* This class ensures that for every full path value it holds , a correct string representation
* of just the filename ( with no extension ) will be displayed to the user .
2016-10-20 07:26:59 -07:00
* If this class receives valid SMDH data , it will also display game icons and titles .
2015-08-31 21:35:33 -07:00
*/
class GameListItemPath : public GameListItem {
public :
static const int FullPathRole = Qt : : UserRole + 1 ;
2016-04-13 14:04:05 -07:00
static const int TitleRole = Qt : : UserRole + 2 ;
2016-12-15 01:55:03 -08:00
static const int ProgramIdRole = Qt : : UserRole + 3 ;
2015-08-31 21:35:33 -07:00
2016-09-18 18:01:46 -07:00
GameListItemPath ( ) : GameListItem ( ) { }
2018-01-23 19:32:27 -08:00
GameListItemPath ( const QString & game_path , const std : : vector < u8 > & smdh_data , u64 program_id )
2016-12-15 01:55:03 -08:00
: GameListItem ( ) {
2015-08-31 21:35:33 -07:00
setData ( game_path , FullPathRole ) ;
2016-12-15 01:55:03 -08:00
setData ( qulonglong ( program_id ) , ProgramIdRole ) ;
2016-04-13 14:04:05 -07:00
2018-01-23 19:32:27 -08:00
if ( ! Loader : : IsValidSMDH ( smdh_data ) ) {
2016-04-13 14:04:05 -07:00
// SMDH is not valid, set a default icon
setData ( GetDefaultIcon ( true ) , Qt : : DecorationRole ) ;
return ;
}
2018-01-24 08:17:04 -08:00
2018-01-24 08:16:40 -08:00
Loader : : SMDH smdh ;
2018-01-23 19:32:27 -08:00
memcpy ( & smdh , smdh_data . data ( ) , sizeof ( Loader : : SMDH ) ) ;
2016-04-13 14:04:05 -07:00
// Get icon from SMDH
2016-05-17 16:42:45 -07:00
setData ( GetQPixmapFromSMDH ( smdh , true ) , Qt : : DecorationRole ) ;
2016-04-13 14:04:05 -07:00
2018-01-20 09:33:14 -08:00
// Get title from SMDH
2016-09-17 17:38:01 -07:00
setData ( GetQStringShortTitleFromSMDH ( smdh , Loader : : SMDH : : TitleLanguage : : English ) ,
TitleRole ) ;
2015-08-31 21:35:33 -07:00
}
2016-04-13 14:04:05 -07:00
QVariant data ( int role ) const override {
if ( role = = Qt : : DisplayRole ) {
2018-05-19 02:21:26 -07:00
std : : string path , filename , extension ;
Common : : SplitPath ( data ( FullPathRole ) . toString ( ) . toStdString ( ) , & path , & filename ,
& extension ) ;
2016-04-13 14:04:05 -07:00
QString title = data ( TitleRole ) . toString ( ) ;
2018-05-19 02:21:26 -07:00
QString second_name = QString : : fromStdString ( filename + extension ) ;
QRegExp installed_system_pattern (
QString : : fromStdString (
FileUtil : : GetUserPath ( D_SDMC_IDX ) +
" Nintendo "
" 3DS/00000000000000000000000000000000/00000000000000000000000000000000/ "
" title/000400(0|1)0/[0-9a-f]{8}/content/ " )
. replace ( " \\ " , " \\ \\ " ) ) ;
if ( installed_system_pattern . exactMatch ( QString : : fromStdString ( path ) ) ) {
// Use a different mechanism for system / installed titles showing program ID
second_name =
" 000 " + QString : : number ( data ( ProgramIdRole ) . toULongLong ( ) , 16 ) . toUpper ( ) ;
}
return title + ( title . isEmpty ( ) ? " " : " \n " ) + second_name ;
2015-08-31 21:35:33 -07:00
} else {
2016-04-13 14:04:05 -07:00
return GameListItem : : data ( role ) ;
2015-08-31 21:35:33 -07:00
}
}
} ;
2018-04-15 15:42:58 -07:00
class GameListItemCompat : public GameListItem {
public :
static const int CompatNumberRole = Qt : : UserRole + 1 ;
GameListItemCompat ( ) = default ;
explicit GameListItemCompat ( const QString compatiblity ) {
auto iterator = status_data . find ( compatiblity ) ;
if ( iterator = = status_data . end ( ) ) {
NGLOG_WARNING ( Frontend , " Invalid compatibility number {} " , compatiblity . toStdString ( ) ) ;
return ;
}
CompatStatus status = iterator - > second ;
setData ( compatiblity , CompatNumberRole ) ;
2018-04-17 09:14:39 -07:00
setText ( QCoreApplication : : translate ( " GameList " , status . text ) ) ;
setToolTip ( QCoreApplication : : translate ( " GameList " , status . tooltip ) ) ;
2018-04-15 15:42:58 -07:00
setData ( CreateCirclePixmapFromColor ( status . color ) , Qt : : DecorationRole ) ;
}
bool operator < ( const QStandardItem & other ) const override {
return data ( CompatNumberRole ) < other . data ( CompatNumberRole ) ;
}
} ;
2018-05-01 10:57:01 -07:00
class GameListItemRegion : public GameListItem {
public :
GameListItemRegion ( ) = default ;
explicit GameListItemRegion ( const std : : vector < u8 > & smdh_data ) {
if ( ! Loader : : IsValidSMDH ( smdh_data ) ) {
setText ( QObject : : tr ( " Invalid region " ) ) ;
return ;
}
Loader : : SMDH smdh ;
memcpy ( & smdh , smdh_data . data ( ) , sizeof ( Loader : : SMDH ) ) ;
setText ( GetRegionFromSMDH ( smdh ) ) ;
}
} ;
2015-08-31 21:35:33 -07:00
/**
* A specialization of GameListItem for size values .
* This class ensures that for every numerical size value it holds ( in bytes ) , a correct
* human - readable string representation will be displayed to the user .
*/
class GameListItemSize : public GameListItem {
public :
static const int SizeRole = Qt : : UserRole + 1 ;
2016-09-18 18:01:46 -07:00
GameListItemSize ( ) : GameListItem ( ) { }
2016-09-17 17:38:01 -07:00
GameListItemSize ( const qulonglong size_bytes ) : GameListItem ( ) {
2015-08-31 21:35:33 -07:00
setData ( size_bytes , SizeRole ) ;
}
2016-09-17 17:38:01 -07:00
void setData ( const QVariant & value , int role ) override {
2015-08-31 21:35:33 -07:00
// By specializing setData for SizeRole, we can ensure that the numerical and string
// representations of the data are always accurate and in the correct format.
if ( role = = SizeRole ) {
qulonglong size_bytes = value . toULongLong ( ) ;
GameListItem : : setData ( ReadableByteSize ( size_bytes ) , Qt : : DisplayRole ) ;
GameListItem : : setData ( value , SizeRole ) ;
} else {
GameListItem : : setData ( value , role ) ;
}
}
/**
* This operator is , in practice , only used by the TreeView sorting systems .
2016-09-17 17:38:01 -07:00
* Override it so that it will correctly sort by numerical value instead of by string
* representation .
2015-08-31 21:35:33 -07:00
*/
2016-09-17 17:38:01 -07:00
bool operator < ( const QStandardItem & other ) const override {
2015-08-31 21:35:33 -07:00
return data ( SizeRole ) . toULongLong ( ) < other . data ( SizeRole ) . toULongLong ( ) ;
}
} ;
/**
* Asynchronous worker object for populating the game list .
* Communicates with other threads through Qt ' s signal / slot system .
*/
class GameListWorker : public QObject , public QRunnable {
Q_OBJECT
public :
2018-05-09 18:57:57 -07:00
GameListWorker (
QString dir_path , bool deep_scan ,
const std : : unordered_map < std : : string , std : : pair < QString , QString > > & compatibility_list )
2018-04-15 15:42:58 -07:00
: QObject ( ) , QRunnable ( ) , dir_path ( dir_path ) , deep_scan ( deep_scan ) ,
compatibility_list ( compatibility_list ) { }
2015-08-31 21:35:33 -07:00
public slots :
/// Starts the processing of directory tree information.
void run ( ) override ;
/// Tells the worker that it should no longer continue processing. Thread-safe.
void Cancel ( ) ;
signals :
/**
* The ` EntryReady ` signal is emitted once an entry has been prepared and is ready
* to be added to the game list .
* @ param entry_items a list with ` QStandardItem ` s that make up the columns of the new entry .
*/
void EntryReady ( QList < QStandardItem * > entry_items ) ;
2017-04-17 19:53:40 -07:00
/**
* After the worker has traversed the game directory looking for entries , this signal is emmited
* with a list of folders that should be watched for changes as well .
*/
void Finished ( QStringList watch_list ) ;
2015-08-31 21:35:33 -07:00
private :
2017-04-17 19:53:40 -07:00
QStringList watch_list ;
2015-08-31 21:35:33 -07:00
QString dir_path ;
bool deep_scan ;
2018-05-09 18:57:57 -07:00
const std : : unordered_map < std : : string , std : : pair < QString , QString > > & compatibility_list ;
2015-08-31 21:35:33 -07:00
std : : atomic_bool stop_processing ;
2015-09-05 23:59:04 -07:00
void AddFstEntriesToGameList ( const std : : string & dir_path , unsigned int recursion = 0 ) ;
2015-08-31 21:35:33 -07:00
} ;