2017-07-13 10:26:03 +02:00
/*
* cmodlistview_moc . cpp , part of VCMI engine
*
* Authors : listed in file AUTHORS in main folder
*
* License : GNU General Public License v2 .0 or later
* Full text of license available in license . txt file , in main folder
*
*/
2013-08-22 17:22:49 +03:00
# include "StdInc.h"
2013-08-24 23:11:51 +03:00
# include "cmodlistview_moc.h"
# include "ui_cmodlistview_moc.h"
2014-03-29 15:46:41 +03:00
# include "imageviewer_moc.h"
2019-07-30 09:28:42 +02:00
# include "../mainwindow_moc.h"
2013-08-22 17:22:49 +03:00
# include <QJsonArray>
# include <QCryptographicHash>
2022-05-28 15:32:20 +02:00
# include <QRegularExpression>
2013-08-22 17:22:49 +03:00
2024-11-12 22:00:21 +02:00
# include "modstatemodel.h"
# include "modstateitemmodel_moc.h"
# include "modstatecontroller.h"
2013-08-24 23:11:51 +03:00
# include "cdownloadmanager_moc.h"
2024-08-30 21:17:18 +02:00
# include "chroniclesextractor.h"
2024-04-21 01:32:31 +02:00
# include "../settingsView/csettingsview_moc.h"
2024-10-07 07:47:00 +02:00
# include "../vcmiqt/launcherdirs.h"
# include "../vcmiqt/jsonutils.h"
2024-04-21 16:56:39 +02:00
# include "../helper.h"
2013-08-22 17:22:49 +03:00
2024-04-21 01:32:31 +02:00
# include "../../lib/VCMIDirs.h"
2013-09-02 20:26:52 +03:00
# include "../../lib/CConfigHandler.h"
2024-07-20 14:55:17 +02:00
# include "../../lib/texts/Languages.h"
2023-09-01 02:12:41 +02:00
# include "../../lib/modding/CModVersion.h"
2024-08-31 00:44:20 +02:00
# include "../../lib/filesystem/Filesystem.h"
2013-08-22 17:22:49 +03:00
2024-08-31 23:44:36 +02:00
# include <future>
2013-08-22 17:22:49 +03:00
void CModListView : : setupModModel ( )
{
2024-11-13 19:25:59 +02:00
static const QString repositoryCachePath = CLauncherDirs : : downloadsPath ( ) + " /repositoryCache.json " ;
const auto & cachedRepositoryData = JsonUtils : : jsonFromFile ( repositoryCachePath ) ;
2024-11-12 22:00:21 +02:00
modStateModel = std : : make_shared < ModStateModel > ( ) ;
2024-11-13 19:25:59 +02:00
if ( ! cachedRepositoryData . isNull ( ) )
modStateModel - > appendRepositories ( cachedRepositoryData ) ;
modModel = new ModStateItemModel ( modStateModel , this ) ;
2024-11-12 22:00:21 +02:00
manager = std : : make_unique < ModStateController > ( modStateModel ) ;
2013-08-22 17:22:49 +03:00
}
2022-12-25 13:19:16 +02:00
void CModListView : : changeEvent ( QEvent * event )
{
2022-12-29 16:37:38 +02:00
if ( event - > type ( ) = = QEvent : : LanguageChange )
2022-12-25 13:19:16 +02:00
{
ui - > retranslateUi ( this ) ;
modModel - > reloadRepositories ( ) ;
}
2022-12-28 17:58:39 +02:00
QWidget : : changeEvent ( event ) ;
2022-12-25 13:19:16 +02:00
}
2023-12-25 19:53:02 +02:00
void CModListView : : dragEnterEvent ( QDragEnterEvent * event )
{
2023-12-25 23:41:15 +02:00
if ( event - > mimeData ( ) - > hasUrls ( ) )
for ( const auto & url : event - > mimeData ( ) - > urls ( ) )
2024-08-30 21:17:18 +02:00
for ( const auto & ending : QStringList ( { " .zip " , " .h3m " , " .h3c " , " .vmap " , " .vcmp " , " .json " , " .exe " } ) )
2023-12-25 23:41:15 +02:00
if ( url . fileName ( ) . endsWith ( ending , Qt : : CaseInsensitive ) )
{
event - > acceptProposedAction ( ) ;
return ;
}
2023-12-25 19:53:02 +02:00
}
void CModListView : : dropEvent ( QDropEvent * event )
{
const QMimeData * mimeData = event - > mimeData ( ) ;
if ( mimeData - > hasUrls ( ) )
{
2023-12-25 20:30:40 +02:00
const QList < QUrl > urlList = mimeData - > urls ( ) ;
2023-12-25 23:41:15 +02:00
for ( const auto & url : urlList )
2024-07-08 15:10:44 +02:00
manualInstallFile ( url . toLocalFile ( ) ) ;
2023-12-25 19:53:02 +02:00
}
}
2013-08-22 17:22:49 +03:00
void CModListView : : setupFilterModel ( )
{
2018-04-28 10:56:01 +02:00
filterModel = new CModFilterModel ( modModel , this ) ;
2013-08-22 17:22:49 +03:00
filterModel - > setFilterKeyColumn ( - 1 ) ; // filter across all columns
filterModel - > setSortCaseSensitivity ( Qt : : CaseInsensitive ) ; // to make it more user-friendly
2014-03-20 20:06:25 +03:00
filterModel - > setDynamicSortFilter ( true ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : setupModsView ( )
{
ui - > allModsView - > setModel ( filterModel ) ;
// input data is not sorted - sort it before display
ui - > allModsView - > sortByColumn ( ModFields : : TYPE , Qt : : AscendingOrder ) ;
2014-03-20 20:06:25 +03:00
ui - > allModsView - > header ( ) - > setSectionResizeMode ( ModFields : : STATUS_ENABLED , QHeaderView : : Fixed ) ;
2018-04-13 07:34:58 +02:00
ui - > allModsView - > header ( ) - > setSectionResizeMode ( ModFields : : STATUS_UPDATE , QHeaderView : : Fixed ) ;
2014-03-20 20:06:25 +03:00
2019-07-30 09:28:42 +02:00
QSettings s ( Ui : : teamName , Ui : : appName ) ;
auto state = s . value ( " AllModsView/State " ) . toByteArray ( ) ;
2019-07-30 11:00:44 +02:00
if ( ! state . isNull ( ) ) //read last saved settings
2019-07-30 09:28:42 +02:00
{
ui - > allModsView - > header ( ) - > restoreState ( state ) ;
}
else //default //TODO: default high-DPI scaling
{
2024-06-08 18:00:45 +02:00
ui - > allModsView - > setColumnWidth ( ModFields : : NAME , 220 ) ;
2019-07-30 09:28:42 +02:00
ui - > allModsView - > setColumnWidth ( ModFields : : TYPE , 75 ) ;
}
2023-03-27 19:11:41 +02:00
ui - > allModsView - > resizeColumnToContents ( ModFields : : STATUS_ENABLED ) ;
ui - > allModsView - > resizeColumnToContents ( ModFields : : STATUS_UPDATE ) ;
2019-07-30 09:28:42 +02:00
2014-03-20 20:06:25 +03:00
ui - > allModsView - > setUniformRowHeights ( true ) ;
2018-04-13 07:34:58 +02:00
connect ( ui - > allModsView - > selectionModel ( ) , SIGNAL ( currentRowChanged ( const QModelIndex & , const QModelIndex & ) ) ,
this , SLOT ( modSelected ( const QModelIndex & , const QModelIndex & ) ) ) ;
2013-08-22 17:22:49 +03:00
2018-04-13 07:34:58 +02:00
connect ( filterModel , SIGNAL ( modelReset ( ) ) ,
this , SLOT ( modelReset ( ) ) ) ;
2013-08-24 23:11:51 +03:00
2018-04-13 07:34:58 +02:00
connect ( modModel , SIGNAL ( dataChanged ( QModelIndex , QModelIndex ) ) ,
this , SLOT ( dataChanged ( QModelIndex , QModelIndex ) ) ) ;
2013-08-22 17:22:49 +03:00
}
2018-04-13 07:34:58 +02:00
CModListView : : CModListView ( QWidget * parent )
2023-03-12 18:33:29 +02:00
: QWidget ( parent )
, ui ( new Ui : : CModListView )
2013-08-22 17:22:49 +03:00
{
ui - > setupUi ( this ) ;
2023-12-25 19:53:02 +02:00
setAcceptDrops ( true ) ;
2024-03-02 23:24:00 +02:00
ui - > uninstallButton - > setIcon ( QIcon { " :/icons/mod-delete.png " } ) ;
ui - > enableButton - > setIcon ( QIcon { " :/icons/mod-enabled.png " } ) ;
ui - > disableButton - > setIcon ( QIcon { " :/icons/mod-disabled.png " } ) ;
ui - > updateButton - > setIcon ( QIcon { " :/icons/mod-update.png " } ) ;
ui - > installButton - > setIcon ( QIcon { " :/icons/mod-download.png " } ) ;
2024-07-16 16:13:58 +02:00
ui - > splitter - > setStyleSheet ( " QSplitter::handle {background: palette('window');} " ) ;
2013-08-22 17:22:49 +03:00
setupModModel ( ) ;
setupFilterModel ( ) ;
setupModsView ( ) ;
ui - > progressWidget - > setVisible ( false ) ;
dlManager = nullptr ;
2023-03-11 00:57:55 +02:00
2018-04-13 07:34:58 +02:00
if ( settings [ " launcher " ] [ " autoCheckRepositories " ] . Bool ( ) )
2014-11-03 17:47:37 +02:00
loadRepositories ( ) ;
2024-05-01 20:05:01 +02:00
# ifdef VCMI_MOBILE
2022-11-22 01:36:27 +02:00
for ( auto * scrollWidget : {
( QAbstractItemView * ) ui - > allModsView ,
( QAbstractItemView * ) ui - > screenshotsList } )
{
2024-05-01 20:05:01 +02:00
Helper : : enableScrollBySwiping ( scrollWidget ) ;
2022-11-22 01:36:27 +02:00
scrollWidget - > setVerticalScrollMode ( QAbstractItemView : : ScrollPerPixel ) ;
scrollWidget - > setVerticalScrollBarPolicy ( Qt : : ScrollBarAlwaysOff ) ;
}
# endif
2013-08-24 23:11:51 +03:00
}
2013-08-22 17:22:49 +03:00
2013-08-24 23:11:51 +03:00
void CModListView : : loadRepositories ( )
{
2023-06-29 14:41:12 +02:00
QStringList repositories ;
if ( settings [ " launcher " ] [ " defaultRepositoryEnabled " ] . Bool ( ) )
repositories . push_back ( QString : : fromStdString ( settings [ " launcher " ] [ " defaultRepositoryURL " ] . String ( ) ) ) ;
if ( settings [ " launcher " ] [ " extraRepositoryEnabled " ] . Bool ( ) )
repositories . push_back ( QString : : fromStdString ( settings [ " launcher " ] [ " extraRepositoryURL " ] . String ( ) ) ) ;
2024-11-13 22:27:51 +02:00
for ( const auto & entry : repositories )
2013-08-22 17:22:49 +03:00
{
2023-06-29 14:41:12 +02:00
if ( entry . isEmpty ( ) )
continue ;
2013-08-22 17:22:49 +03:00
// URL must be encoded to something else to get rid of symbols illegal in file names
2023-06-29 14:41:12 +02:00
auto hashed = QCryptographicHash : : hash ( entry . toUtf8 ( ) , QCryptographicHash : : Md5 ) ;
2013-08-22 17:22:49 +03:00
auto hashedStr = QString : : fromUtf8 ( hashed . toHex ( ) ) ;
2024-05-01 12:46:39 +02:00
downloadFile ( hashedStr + " .json " , entry , tr ( " mods repository index " ) ) ;
2013-08-22 17:22:49 +03:00
}
}
CModListView : : ~ CModListView ( )
{
2019-07-30 09:28:42 +02:00
QSettings s ( Ui : : teamName , Ui : : appName ) ;
s . setValue ( " AllModsView/State " , ui - > allModsView - > header ( ) - > saveState ( ) ) ;
2013-08-22 17:22:49 +03:00
delete ui ;
}
static QString replaceIfNotEmpty ( QVariant value , QString pattern )
{
2018-04-13 07:34:58 +02:00
if ( value . canConvert < QString > ( ) )
2024-11-14 18:55:15 +02:00
{
if ( value . toString ( ) . isEmpty ( ) )
return " " ;
else
return pattern . arg ( value . toString ( ) ) ;
}
if ( value . canConvert < QStringList > ( ) )
{
if ( value . toStringList ( ) . isEmpty ( ) )
return " " ;
else
return pattern . arg ( value . toStringList ( ) . join ( " , " ) ) ;
}
2013-08-22 17:22:49 +03:00
// all valid types of data should have been filtered by code above
assert ( ! value . isValid ( ) ) ;
return " " ;
}
static QString replaceIfNotEmpty ( QStringList value , QString pattern )
{
2018-04-13 07:34:58 +02:00
if ( ! value . empty ( ) )
2013-08-22 17:22:49 +03:00
return pattern . arg ( value . join ( " , " ) ) ;
return " " ;
}
2024-11-13 22:27:51 +02:00
QString CModListView : : genChangelogText ( const ModState & mod )
2014-03-23 15:08:01 +03:00
{
QString headerTemplate = " <p><span style= \" font-weight:600; \" >%1: </span></p> " ;
QString entryBegin = " <p align= \" justify \" ><ul> " ;
QString entryEnd = " </ul></p> " ;
QString entryLine = " <li>%1</li> " ;
//QString versionSeparator = "<hr/>";
QString result ;
2024-11-12 22:00:21 +02:00
QMap < QString , QStringList > changelog = mod . getChangelog ( ) ;
2014-03-23 15:08:01 +03:00
QList < QString > versions = changelog . keys ( ) ;
std : : sort ( versions . begin ( ) , versions . end ( ) , [ ] ( QString lesser , QString greater )
{
2023-09-01 02:12:41 +02:00
return CModVersion : : fromString ( lesser . toStdString ( ) ) < CModVersion : : fromString ( greater . toStdString ( ) ) ;
2014-03-23 15:08:01 +03:00
} ) ;
2023-05-27 22:12:03 +02:00
std : : reverse ( versions . begin ( ) , versions . end ( ) ) ;
2014-03-23 15:08:01 +03:00
2024-11-13 22:27:51 +02:00
for ( const auto & version : versions )
2014-03-23 15:08:01 +03:00
{
result + = headerTemplate . arg ( version ) ;
result + = entryBegin ;
2024-11-13 22:27:51 +02:00
for ( const auto & line : changelog . value ( version ) )
2014-03-23 15:08:01 +03:00
result + = entryLine . arg ( line ) ;
result + = entryEnd ;
}
return result ;
}
2024-11-14 20:43:41 +02:00
QStringList CModListView : : getModNames ( QString queryingModID , QStringList input )
2023-03-26 23:32:24 +02:00
{
QStringList result ;
2024-11-14 20:43:41 +02:00
auto queryingMod = modStateModel - > getMod ( queryingModID ) ;
2023-03-27 00:10:44 +02:00
for ( const auto & modID : input )
2023-03-26 23:32:24 +02:00
{
2024-11-13 19:25:59 +02:00
// Missing mod - use mod ID
// TODO: for submods show parent ID instead
if ( ! modStateModel - > isModExists ( modID ) )
{
result + = modID ;
continue ;
}
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( modID ) ;
2023-03-26 23:32:24 +02:00
2024-11-14 20:43:41 +02:00
if ( queryingMod . getParentID ( ) = = modID )
continue ;
2024-11-14 18:55:15 +02:00
if ( mod . isHidden ( ) )
continue ;
2024-11-12 22:00:21 +02:00
QString displayName = mod . getName ( ) ;
2024-10-17 16:19:59 +02:00
if ( displayName . isEmpty ( ) )
displayName = modID . toLower ( ) ;
2024-11-14 20:43:41 +02:00
if ( mod . isSubmod ( ) & & queryingMod . getParentID ( ) ! = mod . getParentID ( ) )
2024-10-17 16:19:59 +02:00
{
2024-11-12 22:00:21 +02:00
auto parentModID = mod . getTopParentID ( ) ;
auto parentMod = modStateModel - > getMod ( parentModID ) ;
QString parentDisplayName = parentMod . getName ( ) ;
2024-10-17 16:19:59 +02:00
if ( parentDisplayName . isEmpty ( ) )
parentDisplayName = parentModID . toLower ( ) ;
2024-11-12 22:00:21 +02:00
if ( modStateModel - > isModInstalled ( modID ) )
2024-10-17 16:19:59 +02:00
displayName = QString ( " %1 (%2) " ) . arg ( displayName , parentDisplayName ) ;
else
displayName = parentDisplayName ;
}
result + = displayName ;
2023-03-26 23:32:24 +02:00
}
return result ;
}
2024-11-13 22:27:51 +02:00
QString CModListView : : genModInfoText ( const ModState & mod )
2013-08-22 17:22:49 +03:00
{
QString prefix = " <p><span style= \" font-weight:600; \" >%1: </span> " ; // shared prefix
2022-09-29 16:21:29 +02:00
QString redPrefix = " <p><span style= \" font-weight:600; color:red \" >%1: </span> " ; // shared prefix
2013-08-22 17:22:49 +03:00
QString lineTemplate = prefix + " %2</p> " ;
2018-04-13 07:34:58 +02:00
QString urlTemplate = prefix + " <a href= \" %2 \" >%3</a></p> " ;
2013-08-22 17:22:49 +03:00
QString textTemplate = prefix + " </p><p align= \" justify \" >%2</p> " ;
2014-03-20 20:06:25 +03:00
QString listTemplate = " <p align= \" justify \" >%1: %2</p> " ;
QString noteTemplate = " <p align= \" justify \" >%1</p> " ;
2022-12-25 13:19:16 +02:00
QString incompatibleString = redPrefix + tr ( " Mod is incompatible " ) + " </p> " ;
2022-09-29 16:21:29 +02:00
QString supportedVersions = redPrefix + " %2 %3 %4</p> " ;
2013-08-22 17:22:49 +03:00
QString result ;
2024-11-12 22:00:21 +02:00
result + = replaceIfNotEmpty ( mod . getName ( ) , lineTemplate . arg ( tr ( " Mod name " ) ) ) ;
2024-11-14 20:01:49 +02:00
if ( mod . isUpdateAvailable ( ) )
{
result + = replaceIfNotEmpty ( mod . getInstalledVersion ( ) , lineTemplate . arg ( tr ( " Installed version " ) ) ) ;
result + = replaceIfNotEmpty ( mod . getRepositoryVersion ( ) , lineTemplate . arg ( tr ( " Latest version " ) ) ) ;
}
else
2024-11-14 20:22:02 +02:00
result + = replaceIfNotEmpty ( mod . getVersion ( ) , lineTemplate . arg ( tr ( " Installed version " ) ) ) ;
2024-11-12 22:00:21 +02:00
2024-11-14 20:22:02 +02:00
if ( mod . isInstalled ( ) )
result + = replaceIfNotEmpty ( modStateModel - > getInstalledModSizeFormatted ( mod . getID ( ) ) , lineTemplate . arg ( tr ( " Size " ) ) ) ;
2014-03-23 15:08:01 +03:00
2024-11-12 22:00:21 +02:00
if ( ( ! mod . isInstalled ( ) | | mod . isUpdateAvailable ( ) ) & & ! mod . getDownloadSizeFormatted ( ) . isEmpty ( ) )
result + = replaceIfNotEmpty ( mod . getDownloadSizeFormatted ( ) , lineTemplate . arg ( tr ( " Download size " ) ) ) ;
2023-09-01 16:43:26 +02:00
2024-11-12 22:00:21 +02:00
result + = replaceIfNotEmpty ( mod . getAuthors ( ) , lineTemplate . arg ( tr ( " Authors " ) ) ) ;
2014-03-23 15:08:01 +03:00
2024-11-13 19:25:59 +02:00
if ( ! mod . getLicenseName ( ) . isEmpty ( ) )
2024-11-12 22:00:21 +02:00
result + = urlTemplate . arg ( tr ( " License " ) ) . arg ( mod . getLicenseUrl ( ) ) . arg ( mod . getLicenseName ( ) ) ;
2014-03-23 15:08:01 +03:00
2024-11-13 19:25:59 +02:00
if ( ! mod . getContact ( ) . isEmpty ( ) )
2024-11-12 22:00:21 +02:00
result + = urlTemplate . arg ( tr ( " Contact " ) ) . arg ( mod . getContact ( ) ) . arg ( mod . getContact ( ) ) ;
2014-03-23 15:08:01 +03:00
2022-09-29 16:21:29 +02:00
//compatibility info
2023-03-26 23:26:58 +02:00
if ( ! mod . isCompatible ( ) )
2022-09-29 16:21:29 +02:00
{
2024-11-12 22:00:21 +02:00
auto compatibilityInfo = mod . getCompatibleVersionRange ( ) ;
auto minStr = compatibilityInfo . first ;
auto maxStr = compatibilityInfo . second ;
2022-09-29 16:21:29 +02:00
result + = incompatibleString . arg ( tr ( " Compatibility " ) ) ;
if ( minStr = = maxStr )
result + = supportedVersions . arg ( tr ( " Required VCMI version " ) , minStr , " " , " " ) ;
else
{
if ( minStr . isEmpty ( ) | | maxStr . isEmpty ( ) )
{
if ( minStr . isEmpty ( ) )
2024-05-01 11:43:20 +02:00
result + = supportedVersions . arg ( tr ( " Supported VCMI version " ) , maxStr , " , " , tr ( " please upgrade mod " ) ) ;
2022-09-29 16:21:29 +02:00
else
2024-05-01 12:46:39 +02:00
result + = supportedVersions . arg ( tr ( " Required VCMI version " ) , minStr , " " , tr ( " or newer " ) ) ;
2022-09-29 16:21:29 +02:00
}
else
result + = supportedVersions . arg ( tr ( " Supported VCMI versions " ) , minStr , " - " , maxStr ) ;
}
}
2024-11-12 22:00:21 +02:00
QVariant baseLanguageVariant = mod . getBaseLanguage ( ) ;
2023-03-26 23:26:58 +02:00
QString baseLanguageID = baseLanguageVariant . isValid ( ) ? baseLanguageVariant . toString ( ) : " english " ;
2024-11-12 22:00:21 +02:00
QStringList supportedLanguages = mod . getSupportedLanguages ( ) ;
2023-03-26 23:26:58 +02:00
2024-11-12 22:00:21 +02:00
if ( supportedLanguages . size ( ) > 1 )
2023-03-26 23:26:58 +02:00
{
2024-11-12 22:00:21 +02:00
QStringList supportedLanguagesTranslated ;
2023-03-26 23:26:58 +02:00
2024-11-12 22:00:21 +02:00
for ( const auto & languageID : supportedLanguages )
supportedLanguagesTranslated + = QApplication : : translate ( " Language " , Languages : : getLanguageOptions ( languageID . toStdString ( ) ) . nameEnglish . c_str ( ) ) ;
2023-03-26 23:26:58 +02:00
2024-11-12 22:00:21 +02:00
result + = replaceIfNotEmpty ( supportedLanguagesTranslated , lineTemplate . arg ( tr ( " Languages " ) ) ) ;
2023-03-26 23:26:58 +02:00
}
2024-11-14 20:43:41 +02:00
result + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , mod . getDependencies ( ) ) , lineTemplate . arg ( tr ( " Required mods " ) ) ) ;
result + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , mod . getConflicts ( ) ) , lineTemplate . arg ( tr ( " Conflicting mods " ) ) ) ;
2024-11-12 22:00:21 +02:00
result + = replaceIfNotEmpty ( mod . getDescription ( ) , textTemplate . arg ( tr ( " Description " ) ) ) ;
2013-08-22 17:22:49 +03:00
result + = " <p></p> " ; // to get some empty space
2023-03-30 22:23:14 +02:00
QString unknownDeps = tr ( " This mod can not be installed or enabled because the following dependencies are not present " ) ;
QString blockingMods = tr ( " This mod can not be enabled because the following mods are incompatible with it " ) ;
QString hasActiveDependentMods = tr ( " This mod cannot be disabled because it is required by the following mods " ) ;
QString hasDependentMods = tr ( " This mod cannot be uninstalled or updated because it is required by the following mods " ) ;
QString thisIsSubmod = tr ( " This is a submod and it cannot be installed or uninstalled separately from its parent mod " ) ;
2013-08-22 17:22:49 +03:00
QString notes ;
2024-11-14 20:43:41 +02:00
notes + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , findInvalidDependencies ( mod . getID ( ) ) ) , listTemplate . arg ( unknownDeps ) ) ;
notes + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , findBlockingMods ( mod . getID ( ) ) ) , listTemplate . arg ( blockingMods ) ) ;
if ( modStateModel - > isModEnabled ( mod . getID ( ) ) )
notes + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , findDependentMods ( mod . getID ( ) , true ) ) , listTemplate . arg ( hasActiveDependentMods ) ) ;
2018-04-13 07:34:58 +02:00
if ( mod . isInstalled ( ) )
2024-11-14 20:43:41 +02:00
notes + = replaceIfNotEmpty ( getModNames ( mod . getID ( ) , findDependentMods ( mod . getID ( ) , false ) ) , listTemplate . arg ( hasDependentMods ) ) ;
2014-03-20 20:06:25 +03:00
2023-10-21 22:55:20 +02:00
if ( mod . isSubmod ( ) )
2014-03-20 20:06:25 +03:00
notes + = noteTemplate . arg ( thisIsSubmod ) ;
2013-08-22 17:22:49 +03:00
2018-04-13 07:34:58 +02:00
if ( notes . size ( ) )
2014-03-20 20:06:25 +03:00
result + = textTemplate . arg ( tr ( " Notes " ) ) . arg ( notes ) ;
2013-08-22 17:22:49 +03:00
return result ;
}
void CModListView : : disableModInfo ( )
{
2014-08-04 14:03:57 +03:00
ui - > disableButton - > setVisible ( false ) ;
ui - > enableButton - > setVisible ( false ) ;
ui - > installButton - > setVisible ( false ) ;
ui - > uninstallButton - > setVisible ( false ) ;
ui - > updateButton - > setVisible ( false ) ;
2013-08-22 17:22:49 +03:00
}
2014-03-20 20:06:25 +03:00
void CModListView : : dataChanged ( const QModelIndex & topleft , const QModelIndex & bottomRight )
2013-08-22 17:22:49 +03:00
{
2014-03-20 20:06:25 +03:00
selectMod ( ui - > allModsView - > currentIndex ( ) ) ;
}
void CModListView : : selectMod ( const QModelIndex & index )
{
2024-11-14 20:01:49 +02:00
ui - > tabWidget - > setCurrentIndex ( 0 ) ;
2018-04-13 07:34:58 +02:00
if ( ! index . isValid ( ) )
2013-08-22 17:22:49 +03:00
{
disableModInfo ( ) ;
}
else
{
2024-03-02 23:24:00 +02:00
const auto modName = index . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( modName ) ;
2013-08-22 17:22:49 +03:00
2024-11-14 20:01:49 +02:00
ui - > tabWidget - > setTabEnabled ( 1 , ! mod . getChangelog ( ) . isEmpty ( ) ) ;
ui - > tabWidget - > setTabEnabled ( 2 , ! mod . getScreenshots ( ) . isEmpty ( ) ) ;
2014-03-23 15:08:01 +03:00
ui - > modInfoBrowser - > setHtml ( genModInfoText ( mod ) ) ;
ui - > changelogBrowser - > setHtml ( genChangelogText ( mod ) ) ;
2013-08-22 17:22:49 +03:00
2024-05-01 20:05:01 +02:00
Helper : : enableScrollBySwiping ( ui - > modInfoBrowser ) ;
Helper : : enableScrollBySwiping ( ui - > changelogBrowser ) ;
2024-11-14 22:56:19 +02:00
//FIXME: this function should be recursive
//FIXME: ensure that this is also reflected correctly in "Notes" section of mod description
2024-03-02 23:24:00 +02:00
bool hasInvalidDeps = ! findInvalidDependencies ( modName ) . empty ( ) ;
2024-11-14 22:56:19 +02:00
//bool hasBlockingMods = !findBlockingMods(modName).empty();
2024-03-02 23:24:00 +02:00
bool hasDependentMods = ! findDependentMods ( modName , true ) . empty ( ) ;
2013-08-22 17:22:49 +03:00
2024-11-14 20:43:41 +02:00
ui - > disableButton - > setVisible ( modStateModel - > isModEnabled ( mod . getID ( ) ) ) ;
ui - > enableButton - > setVisible ( ! modStateModel - > isModEnabled ( mod . getID ( ) ) ) ;
2023-10-21 22:55:20 +02:00
ui - > installButton - > setVisible ( mod . isAvailable ( ) & & ! mod . isSubmod ( ) ) ;
ui - > uninstallButton - > setVisible ( mod . isInstalled ( ) & & ! mod . isSubmod ( ) ) ;
2024-11-12 22:00:21 +02:00
ui - > updateButton - > setVisible ( mod . isUpdateAvailable ( ) ) ;
2013-08-22 17:22:49 +03:00
// Block buttons if action is not allowed at this time
// TODO: automate handling of some of these cases instead of forcing player
// to resolve all conflicts manually.
2024-11-14 22:56:19 +02:00
ui - > disableButton - > setEnabled ( true ) ;
ui - > enableButton - > setEnabled ( ! hasInvalidDeps ) ;
2013-08-22 17:22:49 +03:00
ui - > installButton - > setEnabled ( ! hasInvalidDeps ) ;
2024-11-12 22:00:21 +02:00
ui - > uninstallButton - > setEnabled ( ! hasDependentMods & & ! mod . isHidden ( ) ) ;
2013-08-22 17:22:49 +03:00
ui - > updateButton - > setEnabled ( ! hasInvalidDeps & & ! hasDependentMods ) ;
2014-03-23 15:08:01 +03:00
loadScreenshots ( ) ;
2013-08-22 17:22:49 +03:00
}
}
2018-04-13 07:34:58 +02:00
void CModListView : : modSelected ( const QModelIndex & current , const QModelIndex & )
2013-08-22 17:22:49 +03:00
{
2014-03-20 20:06:25 +03:00
selectMod ( current ) ;
2013-08-22 17:22:49 +03:00
}
2018-04-13 07:34:58 +02:00
void CModListView : : on_allModsView_activated ( const QModelIndex & index )
2013-08-22 17:22:49 +03:00
{
2014-03-20 20:06:25 +03:00
selectMod ( index ) ;
2023-01-26 01:01:41 +02:00
loadScreenshots ( ) ;
2013-08-22 17:22:49 +03:00
}
2018-04-13 07:34:58 +02:00
void CModListView : : on_lineEdit_textChanged ( const QString & arg1 )
2013-08-22 17:22:49 +03:00
{
2022-05-28 15:32:20 +02:00
# if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
auto baseStr = QRegularExpression : : wildcardToRegularExpression ( arg1 , QRegularExpression : : UnanchoredWildcardConversion ) ;
# else
auto baseStr = QRegularExpression : : wildcardToRegularExpression ( arg1 ) ;
//Hack due to lack QRegularExpression::UnanchoredWildcardConversion in Qt5
baseStr . chop ( 3 ) ;
baseStr . remove ( 0 , 5 ) ;
# endif
QRegularExpression regExp { baseStr , QRegularExpression : : CaseInsensitiveOption } ;
filterModel - > setFilterRegularExpression ( regExp ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : on_comboBox_currentIndexChanged ( int index )
{
2024-11-12 22:00:21 +02:00
auto enumIndex = static_cast < ModFilterMask > ( index ) ;
filterModel - > setTypeFilter ( enumIndex ) ;
2013-08-22 17:22:49 +03:00
}
QStringList CModListView : : findInvalidDependencies ( QString mod )
{
QStringList ret ;
2024-11-12 22:00:21 +02:00
for ( QString requirement : modStateModel - > getMod ( mod ) . getDependencies ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-13 19:25:59 +02:00
if ( ! modStateModel - > isModExists ( requirement ) )
2024-04-20 22:48:54 +02:00
ret + = requirement ;
2013-08-22 17:22:49 +03:00
}
return ret ;
}
2018-10-29 17:56:14 +02:00
QStringList CModListView : : findBlockingMods ( QString modUnderTest )
2013-08-22 17:22:49 +03:00
{
QStringList ret ;
2024-11-12 22:00:21 +02:00
auto required = modStateModel - > getMod ( modUnderTest ) . getDependencies ( ) ;
2013-08-22 17:22:49 +03:00
2024-11-12 22:00:21 +02:00
for ( QString name : modStateModel - > getAllMods ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( name ) ;
2013-08-22 17:22:49 +03:00
2024-11-14 20:43:41 +02:00
if ( modStateModel - > isModEnabled ( mod . getID ( ) ) )
2013-08-22 17:22:49 +03:00
{
// one of enabled mods have requirement (or this mod) marked as conflict
2024-11-13 22:27:51 +02:00
for ( const auto & conflict : mod . getConflicts ( ) )
2018-04-13 07:34:58 +02:00
{
if ( required . contains ( conflict ) )
2013-08-22 17:22:49 +03:00
ret . push_back ( name ) ;
2018-04-13 07:34:58 +02:00
}
2013-08-22 17:22:49 +03:00
}
}
return ret ;
}
QStringList CModListView : : findDependentMods ( QString mod , bool excludeDisabled )
{
QStringList ret ;
2024-11-12 22:00:21 +02:00
for ( QString modName : modStateModel - > getAllMods ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-12 22:00:21 +02:00
auto current = modStateModel - > getMod ( modName ) ;
2013-08-22 17:22:49 +03:00
2024-01-23 12:57:38 +02:00
if ( ! current . isInstalled ( ) | | ! current . isVisible ( ) )
2013-09-08 16:02:34 +03:00
continue ;
2023-11-19 20:44:28 +02:00
if ( current . getDependencies ( ) . contains ( mod , Qt : : CaseInsensitive ) )
2018-04-13 07:34:58 +02:00
{
2024-11-14 20:43:41 +02:00
if ( ! ( modStateModel - > isModEnabled ( modName ) & & excludeDisabled ) )
2018-04-13 07:34:58 +02:00
ret + = modName ;
}
2013-08-22 17:22:49 +03:00
}
return ret ;
}
void CModListView : : on_enableButton_clicked ( )
{
2014-03-20 20:06:25 +03:00
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2022-12-26 02:34:10 +02:00
enableModByName ( modName ) ;
checkManagerErrors ( ) ;
}
2013-08-22 17:22:49 +03:00
2022-12-26 02:34:10 +02:00
void CModListView : : enableModByName ( QString modName )
{
2024-11-14 22:56:19 +02:00
manager - > enableMod ( modName ) ;
modModel - > reloadRepositories ( ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : on_disableButton_clicked ( )
{
2014-03-20 20:06:25 +03:00
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2013-08-22 17:22:49 +03:00
2022-12-26 02:34:10 +02:00
disableModByName ( modName ) ;
checkManagerErrors ( ) ;
}
void CModListView : : disableModByName ( QString modName )
{
2024-11-14 22:56:19 +02:00
manager - > disableMod ( modName ) ;
modModel - > reloadRepositories ( ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : on_updateButton_clicked ( )
{
2014-03-20 20:06:25 +03:00
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2013-08-22 17:22:49 +03:00
assert ( findInvalidDependencies ( modName ) . empty ( ) ) ;
2024-11-13 22:27:51 +02:00
for ( const auto & name : modStateModel - > getMod ( modName ) . getDependencies ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( name ) ;
2013-08-22 17:22:49 +03:00
// update required mod, install missing (can be new dependency)
2024-11-12 22:00:21 +02:00
if ( mod . isUpdateAvailable ( ) | | ! mod . isInstalled ( ) )
downloadFile ( name + " .zip " , mod . getDownloadUrl ( ) , name , mod . getDownloadSizeMegabytes ( ) ) ;
2013-08-22 17:22:49 +03:00
}
}
void CModListView : : on_uninstallButton_clicked ( )
{
2014-03-20 20:06:25 +03:00
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2013-08-22 17:22:49 +03:00
// NOTE: perhaps add "manually installed" flag and uninstall those dependencies that don't have it?
2024-11-12 22:00:21 +02:00
if ( modStateModel - > isModExists ( modName ) & & modStateModel - > getMod ( modName ) . isInstalled ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-14 20:43:41 +02:00
if ( modStateModel - > isModEnabled ( modName ) )
2013-11-14 16:21:09 +03:00
manager - > disableMod ( modName ) ;
2013-08-22 17:22:49 +03:00
manager - > uninstallMod ( modName ) ;
}
2022-12-26 02:34:10 +02:00
2013-09-06 00:25:03 +03:00
checkManagerErrors ( ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : on_installButton_clicked ( )
{
2014-03-20 20:06:25 +03:00
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2013-08-22 17:22:49 +03:00
assert ( findInvalidDependencies ( modName ) . empty ( ) ) ;
2024-11-12 22:00:21 +02:00
2024-11-13 22:27:51 +02:00
for ( const auto & name : modStateModel - > getMod ( modName ) . getDependencies ( ) )
2013-08-22 17:22:49 +03:00
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( name ) ;
2024-06-07 17:03:45 +02:00
if ( mod . isAvailable ( ) )
2024-11-12 22:00:21 +02:00
downloadFile ( name + " .zip " , mod . getDownloadUrl ( ) , name , mod . getDownloadSizeMegabytes ( ) ) ;
2024-11-14 20:43:41 +02:00
else if ( modStateModel - > isModEnabled ( name ) )
2024-04-20 22:48:54 +02:00
enableModByName ( name ) ;
}
2024-11-13 22:27:51 +02:00
for ( const auto & name : modStateModel - > getMod ( modName ) . getConflicts ( ) )
2024-04-20 22:48:54 +02:00
{
2024-11-14 20:43:41 +02:00
if ( modStateModel - > isModEnabled ( name ) )
2024-04-20 22:48:54 +02:00
{
//TODO: consider reverse dependencies disabling
//TODO: consider if it may be possible for subdependencies to block disabling conflicting mod?
//TODO: consider if it may be possible to get subconflicts that will block disabling conflicting mod?
disableModByName ( name ) ;
}
2013-08-22 17:22:49 +03:00
}
}
2024-04-20 00:26:58 +02:00
void CModListView : : on_installFromFileButton_clicked ( )
{
2024-07-10 20:41:26 +02:00
// iOS can't display modal dialogs when called directly on button press
// https://bugreports.qt.io/browse/QTBUG-98651
QTimer : : singleShot ( 0 , this , [ this ]
2024-04-20 00:26:58 +02:00
{
2024-09-02 22:51:30 +02:00
QString filter = tr ( " All supported files " ) + " (*.h3m *.vmap *.h3c *.vcmp *.zip *.json *.exe);; " +
tr ( " Maps " ) + " (*.h3m *.vmap);; " +
tr ( " Campaigns " ) + " (*.h3c *.vcmp);; " +
tr ( " Configs " ) + " (*.json);; " +
tr ( " Mods " ) + " (*.zip);; " +
tr ( " Gog files " ) + " (*.exe) " ;
2024-09-01 13:39:40 +02:00
# if defined(VCMI_MOBILE)
filter = tr ( " All files (*.*) " ) ; //Workaround for sometimes incorrect mime for some extensions (e.g. for exe)
# endif
2024-08-30 21:17:18 +02:00
QStringList files = QFileDialog : : getOpenFileNames ( this , tr ( " Select files (configs, mods, maps, campaigns, gog files) to install... " ) , QDir : : homePath ( ) , filter ) ;
2024-07-10 20:41:26 +02:00
for ( const auto & file : files )
{
manualInstallFile ( file ) ;
}
} ) ;
2024-04-20 00:26:58 +02:00
}
2024-07-08 15:10:44 +02:00
void CModListView : : manualInstallFile ( QString filePath )
2024-04-21 01:32:31 +02:00
{
2024-07-08 15:10:44 +02:00
QString fileName = QFileInfo { filePath } . fileName ( ) ;
if ( filePath . endsWith ( " .zip " , Qt : : CaseInsensitive ) )
2024-04-21 01:32:31 +02:00
downloadFile ( fileName . toLower ( )
// mod name currently comes from zip file -> remove suffixes from github zip download
. replace ( QRegularExpression ( " -[0-9a-f]{40} " ) , " " )
. replace ( QRegularExpression ( " -vcmi-.+ \\ .zip " ) , " .zip " )
. replace ( " -main.zip " , " .zip " )
2024-07-08 15:10:44 +02:00
, QUrl : : fromLocalFile ( filePath ) , " mods " ) ;
else if ( filePath . endsWith ( " .json " , Qt : : CaseInsensitive ) )
2024-04-21 01:32:31 +02:00
{
QDir configDir ( QString : : fromStdString ( VCMIDirs : : get ( ) . userConfigPath ( ) . string ( ) ) ) ;
2024-04-21 12:13:50 +02:00
QStringList configFile = configDir . entryList ( { fileName } , QDir : : Filter : : Files ) ; // case insensitive check
2024-04-21 01:32:31 +02:00
if ( ! configFile . empty ( ) )
{
2024-04-21 16:56:39 +02:00
auto dialogResult = QMessageBox : : warning ( this , tr ( " Replace config file? " ) , tr ( " Do you want to replace %1? " ) . arg ( configFile [ 0 ] ) , QMessageBox : : Yes | QMessageBox : : No , QMessageBox : : No ) ;
if ( dialogResult = = QMessageBox : : Yes )
2024-04-21 01:32:31 +02:00
{
2024-04-21 16:56:39 +02:00
const auto configFilePath = configDir . filePath ( configFile [ 0 ] ) ;
QFile : : remove ( configFilePath ) ;
2024-07-08 15:10:44 +02:00
QFile : : copy ( filePath , configFilePath ) ;
2024-04-21 01:32:31 +02:00
// reload settings
2024-04-21 16:56:39 +02:00
Helper : : loadSettings ( ) ;
2024-11-13 22:27:51 +02:00
for ( const auto widget : qApp - > allWidgets ( ) )
2024-04-21 12:13:50 +02:00
if ( auto settingsView = qobject_cast < CSettingsView * > ( widget ) )
2024-04-21 01:32:31 +02:00
settingsView - > loadSettings ( ) ;
2024-11-13 22:27:51 +02:00
// TODO: rescan local mods
2024-04-21 01:32:31 +02:00
}
}
}
else
2024-07-08 15:10:44 +02:00
downloadFile ( fileName , QUrl : : fromLocalFile ( filePath ) , fileName ) ;
2024-04-21 01:32:31 +02:00
}
2023-09-01 16:43:26 +02:00
void CModListView : : downloadFile ( QString file , QString url , QString description , qint64 size )
2024-07-08 15:10:44 +02:00
{
downloadFile ( file , QUrl { url } , description , size ) ;
}
void CModListView : : downloadFile ( QString file , QUrl url , QString description , qint64 size )
2013-08-22 17:22:49 +03:00
{
2018-04-13 07:34:58 +02:00
if ( ! dlManager )
2013-08-22 17:22:49 +03:00
{
dlManager = new CDownloadManager ( ) ;
ui - > progressWidget - > setVisible ( true ) ;
connect ( dlManager , SIGNAL ( downloadProgress ( qint64 , qint64 ) ) ,
2018-04-13 07:34:58 +02:00
this , SLOT ( downloadProgress ( qint64 , qint64 ) ) ) ;
2013-08-22 17:22:49 +03:00
connect ( dlManager , SIGNAL ( finished ( QStringList , QStringList , QStringList ) ) ,
2018-04-13 07:34:58 +02:00
this , SLOT ( downloadFinished ( QStringList , QStringList , QStringList ) ) ) ;
2024-07-08 15:10:44 +02:00
2023-09-19 23:38:00 +02:00
connect ( manager . get ( ) , SIGNAL ( extractionProgress ( qint64 , qint64 ) ) ,
2023-12-18 00:34:00 +02:00
this , SLOT ( extractionProgress ( qint64 , qint64 ) ) ) ;
2013-08-22 17:22:49 +03:00
2024-11-12 22:00:21 +02:00
connect ( modModel , & ModStateItemModel : : dataChanged , filterModel , & QAbstractItemModel : : dataChanged ) ;
2013-08-22 17:22:49 +03:00
2024-07-08 15:10:44 +02:00
const auto progressBarFormat = tr ( " Downloading %1. %p% (%v MB out of %m MB) finished " ) . arg ( description ) ;
2013-08-22 17:22:49 +03:00
ui - > progressBar - > setFormat ( progressBarFormat ) ;
}
2024-07-08 15:10:44 +02:00
dlManager - > downloadFile ( url , file , size ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : downloadProgress ( qint64 current , qint64 max )
{
2023-09-01 22:18:23 +02:00
// display progress, in megabytes
2023-09-19 23:38:00 +02:00
ui - > progressBar - > setVisible ( true ) ;
2023-09-01 22:18:23 +02:00
ui - > progressBar - > setMaximum ( max / ( 1024 * 1024 ) ) ;
ui - > progressBar - > setValue ( current / ( 1024 * 1024 ) ) ;
2013-08-22 17:22:49 +03:00
}
2023-12-18 00:34:00 +02:00
void CModListView : : extractionProgress ( qint64 current , qint64 max )
{
// display progress, in extracted files
ui - > progressBar - > setVisible ( true ) ;
ui - > progressBar - > setMaximum ( max ) ;
ui - > progressBar - > setValue ( current ) ;
}
2013-08-22 17:22:49 +03:00
void CModListView : : downloadFinished ( QStringList savedFiles , QStringList failedFiles , QStringList errors )
{
2023-09-01 22:18:23 +02:00
QString title = tr ( " Download failed " ) ;
QString firstLine = tr ( " Unable to download all files. \n \n Encountered errors: \n \n " ) ;
QString lastLine = tr ( " \n \n Install successfully downloaded? " ) ;
2022-09-29 15:38:03 +02:00
bool doInstallFiles = false ;
2013-08-22 17:22:49 +03:00
// if all files were d/loaded there should be no errors. And on failure there must be an error
assert ( failedFiles . empty ( ) = = errors . empty ( ) ) ;
2018-04-13 07:34:58 +02:00
if ( savedFiles . empty ( ) )
2013-08-22 17:22:49 +03:00
{
// no successfully downloaded mods
2018-04-13 07:34:58 +02:00
QMessageBox : : warning ( this , title , firstLine + errors . join ( " \n " ) , QMessageBox : : Ok , QMessageBox : : Ok ) ;
2013-08-22 17:22:49 +03:00
}
2018-04-13 07:34:58 +02:00
else if ( ! failedFiles . empty ( ) )
2013-08-22 17:22:49 +03:00
{
// some mods were not downloaded
int result = QMessageBox : : warning ( this , title , firstLine + errors . join ( " \n " ) + lastLine ,
QMessageBox : : Yes | QMessageBox : : No , QMessageBox : : No ) ;
2018-04-13 07:34:58 +02:00
if ( result = = QMessageBox : : Yes )
2022-09-29 15:38:03 +02:00
doInstallFiles = true ;
2013-08-22 17:22:49 +03:00
}
else
{
// everything OK
2022-09-29 15:38:03 +02:00
doInstallFiles = true ;
2013-08-22 17:22:49 +03:00
}
dlManager - > deleteLater ( ) ;
dlManager = nullptr ;
2023-09-19 23:38:00 +02:00
ui - > progressBar - > setMaximum ( 0 ) ;
ui - > progressBar - > setValue ( 0 ) ;
2022-09-29 15:38:03 +02:00
if ( doInstallFiles )
installFiles ( savedFiles ) ;
2022-12-26 02:34:10 +02:00
2023-09-19 23:38:00 +02:00
hideProgressBar ( ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : hideProgressBar ( )
{
2018-04-13 07:34:58 +02:00
if ( dlManager = = nullptr ) // it was not recreated meanwhile
2013-08-22 17:22:49 +03:00
{
ui - > progressWidget - > setVisible ( false ) ;
ui - > progressBar - > setMaximum ( 0 ) ;
ui - > progressBar - > setValue ( 0 ) ;
}
}
void CModListView : : installFiles ( QStringList files )
{
QStringList mods ;
2023-12-25 19:53:02 +02:00
QStringList maps ;
2014-03-23 15:08:01 +03:00
QStringList images ;
2024-08-30 21:17:18 +02:00
QStringList exe ;
2024-11-13 19:25:59 +02:00
JsonNode repository ;
2013-08-22 17:22:49 +03:00
// TODO: some better way to separate zip's with mods and downloaded repository files
2018-04-13 07:34:58 +02:00
for ( QString filename : files )
2013-08-22 17:22:49 +03:00
{
2023-12-25 23:41:15 +02:00
if ( filename . endsWith ( " .zip " , Qt : : CaseInsensitive ) )
2013-08-22 17:22:49 +03:00
mods . push_back ( filename ) ;
2023-12-25 23:41:15 +02:00
else if ( filename . endsWith ( " .h3m " , Qt : : CaseInsensitive ) | | filename . endsWith ( " .h3c " , Qt : : CaseInsensitive ) | | filename . endsWith ( " .vmap " , Qt : : CaseInsensitive ) | | filename . endsWith ( " .vcmp " , Qt : : CaseInsensitive ) )
2023-12-25 19:53:02 +02:00
maps . push_back ( filename ) ;
2024-08-30 21:17:18 +02:00
if ( filename . endsWith ( " .exe " , Qt : : CaseInsensitive ) )
exe . push_back ( filename ) ;
2023-12-25 23:41:15 +02:00
else if ( filename . endsWith ( " .json " , Qt : : CaseInsensitive ) )
2022-09-29 15:38:03 +02:00
{
//download and merge additional files
2024-11-13 19:25:59 +02:00
const auto & repoData = JsonUtils : : jsonFromFile ( filename ) ;
if ( repoData [ " name " ] . isNull ( ) )
2022-09-29 15:38:03 +02:00
{
2024-11-13 19:25:59 +02:00
// This is main repository index. Download all referenced mods
for ( const auto & [ modName , modJson ] : repoData . Struct ( ) )
2022-09-29 15:38:03 +02:00
{
2024-11-13 19:25:59 +02:00
auto modNameLower = boost : : algorithm : : to_lower_copy ( modName ) ;
auto modJsonUrl = modJson [ " mod " ] ;
if ( ! modJsonUrl . isNull ( ) )
downloadFile ( QString : : fromStdString ( modName + " .json " ) , QString : : fromStdString ( modJsonUrl . String ( ) ) , tr ( " mods repository index " ) ) ;
2024-11-14 17:41:22 +02:00
repository [ modNameLower ] = modJson ;
2022-09-29 15:38:03 +02:00
}
}
else
{
2024-11-13 19:25:59 +02:00
// This is json of a single mod. Extract name of mod and add it to repo
auto modName = QFileInfo ( filename ) . baseName ( ) . toStdString ( ) ;
auto modNameLower = boost : : algorithm : : to_lower_copy ( modName ) ;
repository [ modNameLower ] = repoData ;
2022-09-29 15:38:03 +02:00
}
}
2023-12-25 23:41:15 +02:00
else if ( filename . endsWith ( " .png " , Qt : : CaseInsensitive ) )
2014-03-23 15:08:01 +03:00
images . push_back ( filename ) ;
2013-08-22 17:22:49 +03:00
}
2023-09-05 12:33:26 +02:00
2024-11-13 19:25:59 +02:00
if ( ! repository . isNull ( ) )
{
manager - > appendRepositories ( repository ) ;
modModel - > reloadRepositories ( ) ;
static const QString repositoryCachePath = CLauncherDirs : : downloadsPath ( ) + " /repositoryCache.json " ;
JsonUtils : : jsonToFile ( repositoryCachePath , modStateModel - > getRepositoryData ( ) ) ;
}
2023-09-05 12:33:26 +02:00
2018-04-13 07:34:58 +02:00
if ( ! mods . empty ( ) )
2013-08-22 17:22:49 +03:00
installMods ( mods ) ;
2014-03-23 15:08:01 +03:00
2023-12-25 19:53:02 +02:00
if ( ! maps . empty ( ) )
installMaps ( maps ) ;
2024-08-30 21:17:18 +02:00
if ( ! exe . empty ( ) )
2024-08-30 22:20:33 +02:00
{
2024-08-31 18:46:45 +02:00
ui - > progressBar - > setFormat ( tr ( " Installing chronicles " ) ) ;
2024-08-31 00:44:20 +02:00
2024-08-31 18:46:45 +02:00
float prog = 0.0 ;
auto futureExtract = std : : async ( std : : launch : : async , [ this , exe , & prog ] ( )
{
2024-08-31 22:05:36 +02:00
ChroniclesExtractor ce ( this , [ & prog ] ( float progress ) { prog = progress ; } ) ;
2024-08-31 18:46:45 +02:00
ce . installChronicles ( exe ) ;
return true ;
} ) ;
while ( futureExtract . wait_for ( std : : chrono : : milliseconds ( 10 ) ) ! = std : : future_status : : ready )
{
2024-09-01 02:16:03 +02:00
emit extractionProgress ( static_cast < int > ( prog * 1000.f ) , 1000 ) ;
2024-08-31 18:46:45 +02:00
qApp - > processEvents ( ) ;
}
if ( futureExtract . get ( ) )
{
//update
CResourceHandler : : get ( " initial " ) - > updateFilteredFiles ( [ ] ( const std : : string & ) { return true ; } ) ;
2024-11-13 22:27:51 +02:00
// TODO: rescan local mods
2024-08-31 18:46:45 +02:00
}
2024-08-30 22:20:33 +02:00
}
2024-08-30 21:17:18 +02:00
2018-04-13 07:34:58 +02:00
if ( ! images . empty ( ) )
2014-03-23 15:08:01 +03:00
loadScreenshots ( ) ;
2013-08-22 17:22:49 +03:00
}
void CModListView : : installMods ( QStringList archives )
{
QStringList modNames ;
2018-04-13 07:34:58 +02:00
for ( QString archive : archives )
2013-08-22 17:22:49 +03:00
{
// get basename out of full file name
// remove path remove extension
QString modName = archive . section ( ' / ' , - 1 , - 1 ) . section ( ' . ' , 0 , 0 ) ;
modNames . push_back ( modName ) ;
}
2013-12-08 13:07:06 +03:00
QStringList modsToEnable ;
2013-08-22 17:22:49 +03:00
// disable mod(s), to properly recalculate dependencies, if changed
2018-04-13 07:34:58 +02:00
for ( QString mod : boost : : adaptors : : reverse ( modNames ) )
2013-09-06 00:25:03 +03:00
{
2024-11-12 22:00:21 +02:00
ModState entry = modStateModel - > getMod ( mod ) ;
2018-04-13 07:34:58 +02:00
if ( entry . isInstalled ( ) )
2013-12-08 13:07:06 +03:00
{
// enable mod if installed and enabled
2024-11-14 20:43:41 +02:00
if ( modStateModel - > isModEnabled ( mod ) )
2013-12-08 13:07:06 +03:00
modsToEnable . push_back ( mod ) ;
}
else
{
// enable mod if m
2018-04-13 07:34:58 +02:00
if ( settings [ " launcher " ] [ " enableInstalledMods " ] . Bool ( ) )
2013-12-08 13:07:06 +03:00
modsToEnable . push_back ( mod ) ;
}
2013-09-06 00:25:03 +03:00
}
2013-08-22 17:22:49 +03:00
// uninstall old version of mod, if installed
2018-04-13 07:34:58 +02:00
for ( QString mod : boost : : adaptors : : reverse ( modNames ) )
2013-09-06 00:25:03 +03:00
{
2024-11-12 22:00:21 +02:00
if ( modStateModel - > getMod ( mod ) . isInstalled ( ) )
2013-09-06 00:25:03 +03:00
manager - > uninstallMod ( mod ) ;
}
2013-08-22 17:22:49 +03:00
2018-04-13 07:34:58 +02:00
for ( int i = 0 ; i < modNames . size ( ) ; i + + )
2023-09-19 23:38:00 +02:00
{
ui - > progressBar - > setFormat ( tr ( " Installing mod %1 " ) . arg ( modNames [ i ] ) ) ;
2013-08-22 17:22:49 +03:00
manager - > installMod ( modNames [ i ] , archives [ i ] ) ;
2023-09-19 23:38:00 +02:00
}
2013-08-22 17:22:49 +03:00
2018-04-13 07:34:58 +02:00
for ( QString mod : modsToEnable )
2014-03-23 15:08:01 +03:00
{
2024-11-14 22:56:19 +02:00
manager - > enableMod ( mod ) ; // TODO: make it as a single action, so if mod 1 depends on mod 2 it would still activate
2014-03-23 15:08:01 +03:00
}
2013-08-22 17:22:49 +03:00
2023-09-11 17:10:07 +02:00
checkManagerErrors ( ) ;
2018-04-13 07:34:58 +02:00
for ( QString archive : archives )
2013-08-22 17:22:49 +03:00
QFile : : remove ( archive ) ;
}
2023-12-25 23:41:15 +02:00
void CModListView : : installMaps ( QStringList maps )
2023-12-25 19:53:02 +02:00
{
2024-02-06 00:33:47 +02:00
const auto destDir = CLauncherDirs : : mapsPath ( ) + QChar { ' / ' } ;
2023-12-25 19:53:02 +02:00
2023-12-25 23:41:15 +02:00
for ( QString map : maps )
2023-12-25 19:53:02 +02:00
{
2023-12-25 23:41:15 +02:00
QFile ( map ) . rename ( destDir + map . section ( ' / ' , - 1 , - 1 ) ) ;
2023-12-25 19:53:02 +02:00
}
}
2014-11-03 17:47:37 +02:00
void CModListView : : on_refreshButton_clicked ( )
{
loadRepositories ( ) ;
}
2013-08-22 17:22:49 +03:00
void CModListView : : on_pushButton_clicked ( )
{
delete dlManager ;
dlManager = nullptr ;
hideProgressBar ( ) ;
}
void CModListView : : modelReset ( )
{
2023-01-26 01:01:41 +02:00
selectMod ( filterModel - > rowCount ( ) > 0 ? filterModel - > index ( 0 , 0 ) : QModelIndex ( ) ) ;
2013-08-24 23:11:51 +03:00
}
2013-09-06 00:25:03 +03:00
void CModListView : : checkManagerErrors ( )
{
QString errors = manager - > getErrors ( ) . join ( ' \n ' ) ;
2018-04-13 07:34:58 +02:00
if ( errors . size ( ) ! = 0 )
2013-09-06 00:25:03 +03:00
{
2023-09-01 22:18:23 +02:00
QString title = tr ( " Operation failed " ) ;
QString description = tr ( " Encountered errors: \n " ) + errors ;
2018-04-13 07:34:58 +02:00
QMessageBox : : warning ( this , title , description , QMessageBox : : Ok , QMessageBox : : Ok ) ;
2013-09-06 00:25:03 +03:00
}
}
2014-03-23 15:08:01 +03:00
void CModListView : : on_tabWidget_currentChanged ( int index )
{
loadScreenshots ( ) ;
}
void CModListView : : loadScreenshots ( )
{
2023-01-26 01:01:41 +02:00
if ( ui - > tabWidget - > currentIndex ( ) = = 2 )
2014-03-23 15:08:01 +03:00
{
2024-07-19 12:38:42 +02:00
if ( ! ui - > allModsView - > currentIndex ( ) . isValid ( ) )
2024-07-18 22:28:47 +02:00
{
// select the first mod, so we can access its data
ui - > allModsView - > setCurrentIndex ( filterModel - > index ( 0 , 0 ) ) ;
}
2014-03-23 15:08:01 +03:00
ui - > screenshotsList - > clear ( ) ;
QString modName = ui - > allModsView - > currentIndex ( ) . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2024-11-12 22:00:21 +02:00
assert ( modStateModel - > isModExists ( modName ) ) ; //should be filtered out by check above
2014-03-23 15:08:01 +03:00
2024-11-12 22:00:21 +02:00
for ( QString url : modStateModel - > getMod ( modName ) . getScreenshots ( ) )
2014-03-23 15:08:01 +03:00
{
// URL must be encoded to something else to get rid of symbols illegal in file names
2024-02-06 00:33:47 +02:00
const auto hashed = QCryptographicHash : : hash ( url . toUtf8 ( ) , QCryptographicHash : : Md5 ) ;
const auto fileName = QString { QLatin1String { " %1.png " } } . arg ( QLatin1String { hashed . toHex ( ) } ) ;
2014-03-23 15:08:01 +03:00
2024-02-06 00:33:47 +02:00
const auto fullPath = QString { QLatin1String { " %1/%2 " } } . arg ( CLauncherDirs : : downloadsPath ( ) , fileName ) ;
2014-03-23 15:08:01 +03:00
QPixmap pixmap ( fullPath ) ;
2018-04-13 07:34:58 +02:00
if ( pixmap . isNull ( ) )
2014-03-23 15:08:01 +03:00
{
// image file not exists or corrupted - try to redownload
2024-05-01 11:43:20 +02:00
downloadFile ( fileName , url , tr ( " screenshots " ) ) ;
2014-03-23 15:08:01 +03:00
}
else
{
// managed to load cached image
QIcon icon ( pixmap ) ;
2024-01-16 23:40:48 +02:00
auto * item = new QListWidgetItem ( icon , QString ( tr ( " Screenshot %1 " ) ) . arg ( ui - > screenshotsList - > count ( ) + 1 ) ) ;
2014-03-23 15:08:01 +03:00
ui - > screenshotsList - > addItem ( item ) ;
}
}
}
}
2018-04-13 07:34:58 +02:00
void CModListView : : on_screenshotsList_clicked ( const QModelIndex & index )
2014-03-23 15:08:01 +03:00
{
2018-04-13 07:34:58 +02:00
if ( index . isValid ( ) )
2014-03-23 15:08:01 +03:00
{
QIcon icon = ui - > screenshotsList - > item ( index . row ( ) ) - > icon ( ) ;
auto pixmap = icon . pixmap ( icon . availableSizes ( ) [ 0 ] ) ;
ImageViewer : : showPixmap ( pixmap , this ) ;
}
}
2014-08-04 14:03:57 +03:00
2023-03-11 00:57:55 +02:00
void CModListView : : doInstallMod ( const QString & modName )
{
assert ( findInvalidDependencies ( modName ) . empty ( ) ) ;
2024-11-13 22:27:51 +02:00
for ( const auto & name : modStateModel - > getMod ( modName ) . getDependencies ( ) )
2023-03-11 00:57:55 +02:00
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( name ) ;
2023-03-11 00:57:55 +02:00
if ( ! mod . isInstalled ( ) )
2024-11-12 22:00:21 +02:00
downloadFile ( name + " .zip " , mod . getDownloadUrl ( ) , name , mod . getDownloadSizeMegabytes ( ) ) ;
2023-03-11 00:57:55 +02:00
}
}
2023-03-26 19:11:59 +02:00
bool CModListView : : isModAvailable ( const QString & modName )
2023-03-11 00:57:55 +02:00
{
2024-11-12 22:00:21 +02:00
return ! modStateModel - > isModInstalled ( modName ) ;
2023-03-11 00:57:55 +02:00
}
2023-03-14 13:37:22 +02:00
bool CModListView : : isModEnabled ( const QString & modName )
{
2024-11-14 20:43:41 +02:00
return modStateModel - > isModEnabled ( modName ) ;
2023-03-14 13:37:22 +02:00
}
2024-09-14 21:34:39 +02:00
bool CModListView : : isModInstalled ( const QString & modName )
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( modName ) ;
2024-09-14 21:34:39 +02:00
return mod . isInstalled ( ) ;
}
2023-03-11 00:57:55 +02:00
QString CModListView : : getTranslationModName ( const QString & language )
{
2024-11-12 22:00:21 +02:00
for ( const auto & modName : modStateModel - > getAllMods ( ) )
2023-03-11 00:57:55 +02:00
{
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( modName ) ;
2023-03-11 00:57:55 +02:00
2023-03-14 15:59:33 +02:00
if ( ! mod . isTranslation ( ) )
2023-03-11 00:57:55 +02:00
continue ;
2024-11-12 22:00:21 +02:00
if ( mod . getBaseLanguage ( ) ! = language )
2023-03-11 00:57:55 +02:00
continue ;
return modName ;
}
return QString ( ) ;
}
2023-09-01 01:29:50 +02:00
void CModListView : : on_allModsView_doubleClicked ( const QModelIndex & index )
{
if ( ! index . isValid ( ) )
return ;
2023-09-01 15:40:52 +02:00
auto modName = index . data ( ModRoles : : ModNameRole ) . toString ( ) ;
2024-11-12 22:00:21 +02:00
auto mod = modStateModel - > getMod ( modName ) ;
2023-09-01 15:40:52 +02:00
bool hasInvalidDeps = ! findInvalidDependencies ( modName ) . empty ( ) ;
bool hasBlockingMods = ! findBlockingMods ( modName ) . empty ( ) ;
bool hasDependentMods = ! findDependentMods ( modName , true ) . empty ( ) ;
2023-10-21 22:55:20 +02:00
if ( ! hasInvalidDeps & & mod . isAvailable ( ) & & ! mod . isSubmod ( ) )
2023-09-01 01:29:50 +02:00
{
on_installButton_clicked ( ) ;
return ;
}
2024-11-12 22:00:21 +02:00
if ( ! hasInvalidDeps & & ! hasDependentMods & & mod . isUpdateAvailable ( ) & & index . column ( ) = = ModFields : : STATUS_UPDATE )
2023-09-01 01:29:50 +02:00
{
on_updateButton_clicked ( ) ;
2023-09-01 15:40:52 +02:00
return ;
}
if ( index . column ( ) = = ModFields : : NAME )
{
if ( ui - > allModsView - > isExpanded ( index ) )
ui - > allModsView - > collapse ( index ) ;
else
ui - > allModsView - > expand ( index ) ;
return ;
2023-09-01 01:29:50 +02:00
}
2024-11-14 20:43:41 +02:00
if ( ! hasBlockingMods & & ! hasInvalidDeps & & ! modStateModel - > isModEnabled ( modName ) )
2023-09-01 01:29:50 +02:00
{
on_enableButton_clicked ( ) ;
return ;
}
2024-11-14 20:43:41 +02:00
if ( ! hasDependentMods & & ! mod . isVisible ( ) & & modStateModel - > isModEnabled ( modName ) )
2023-09-01 01:29:50 +02:00
{
on_disableButton_clicked ( ) ;
2023-09-01 15:40:52 +02:00
return ;
2023-09-01 01:29:50 +02:00
}
}