2022-01-18 16:14:17 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
//
|
|
|
|
// EmulationStation Desktop Edition
|
|
|
|
// GamelistBase.cpp
|
|
|
|
//
|
|
|
|
// Gamelist base class with utility functions and other low-level logic.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "views/GamelistBase.h"
|
|
|
|
|
|
|
|
#include "CollectionSystemsManager.h"
|
|
|
|
#include "FileFilterIndex.h"
|
|
|
|
#include "UIModeController.h"
|
|
|
|
#include "guis/GuiGamelistOptions.h"
|
|
|
|
#include "views/ViewController.h"
|
|
|
|
|
2022-01-19 17:01:54 +00:00
|
|
|
GamelistBase::GamelistBase(FileData* root)
|
|
|
|
: mRoot {root}
|
2022-01-18 16:14:17 +00:00
|
|
|
, mRandomGame {nullptr}
|
2022-01-18 21:04:05 +00:00
|
|
|
, mLastUpdated {nullptr}
|
|
|
|
, mGameCount {0}
|
|
|
|
, mFavoritesGameCount {0}
|
|
|
|
, mFilteredGameCount {0}
|
|
|
|
, mFilteredGameCountAll {0}
|
|
|
|
, mIsFiltered {false}
|
|
|
|
, mIsFolder {false}
|
|
|
|
, mVideoPlaying {false}
|
2022-01-18 16:14:17 +00:00
|
|
|
{
|
2022-02-11 22:38:23 +00:00
|
|
|
setSize(Renderer::getScreenWidth(), Renderer::getScreenHeight());
|
2022-01-18 19:42:50 +00:00
|
|
|
|
|
|
|
mList.setSize(mSize.x, mSize.y * 0.8f);
|
|
|
|
mList.setPosition(0.0f, mSize.y * 0.2f);
|
|
|
|
mList.setDefaultZIndex(20.0f);
|
|
|
|
addChild(&mList);
|
|
|
|
|
|
|
|
populateList(root->getChildrenListToDisplay(), root);
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GamelistBase::~GamelistBase()
|
|
|
|
{
|
|
|
|
//
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::setCursor(FileData* cursor)
|
|
|
|
{
|
|
|
|
if (!mList.setCursor(cursor) && (!cursor->isPlaceHolder())) {
|
|
|
|
populateList(cursor->getParent()->getChildrenListToDisplay(), cursor->getParent());
|
|
|
|
mList.setCursor(cursor);
|
|
|
|
|
|
|
|
// Update our cursor stack in case our cursor just got set to some folder
|
|
|
|
// we weren't in before.
|
|
|
|
if (mCursorStack.empty() || mCursorStack.top() != cursor->getParent()) {
|
|
|
|
std::stack<FileData*> tmp;
|
|
|
|
FileData* ptr = cursor->getParent();
|
|
|
|
|
|
|
|
while (ptr && ptr != mRoot) {
|
|
|
|
tmp.push(ptr);
|
|
|
|
ptr = ptr->getParent();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flip the stack and put it in mCursorStack.
|
|
|
|
mCursorStack = std::stack<FileData*>();
|
|
|
|
while (!tmp.empty()) {
|
|
|
|
mCursorStack.push(tmp.top());
|
|
|
|
tmp.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GamelistBase::input(InputConfig* config, Input input)
|
|
|
|
{
|
|
|
|
if (input.value != 0) {
|
|
|
|
if (config->isMappedTo("a", input)) {
|
|
|
|
FileData* cursor = getCursor();
|
|
|
|
if (cursor->getType() == GAME) {
|
|
|
|
onPauseVideo();
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
stopListScrolling();
|
|
|
|
launch(cursor);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// It's a folder.
|
|
|
|
if (cursor->getChildren().size() > 0) {
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SELECTSOUND);
|
|
|
|
mCursorStack.push(cursor);
|
|
|
|
populateList(cursor->getChildrenListToDisplay(), cursor);
|
|
|
|
|
|
|
|
FileData* newCursor = nullptr;
|
|
|
|
std::vector<FileData*> listEntries = cursor->getChildrenListToDisplay();
|
|
|
|
// Check if there is an entry in the cursor stack history matching any entry
|
|
|
|
// in the currect folder. If so, select that entry.
|
|
|
|
for (auto it = mCursorStackHistory.begin(); // Line break.
|
|
|
|
it != mCursorStackHistory.end(); ++it) {
|
|
|
|
if (std::find(listEntries.begin(), listEntries.end(), *it) !=
|
|
|
|
listEntries.end()) {
|
|
|
|
newCursor = *it;
|
|
|
|
mCursorStackHistory.erase(it);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was no match in the cursor history, simply select the first entry.
|
|
|
|
if (!newCursor)
|
|
|
|
newCursor = getCursor();
|
|
|
|
setCursor(newCursor);
|
|
|
|
if (mRoot->getSystem()->getThemeFolder() == "custom-collections")
|
|
|
|
updateHelpPrompts();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("b", input)) {
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
if (mCursorStack.size()) {
|
|
|
|
// Save the position to the cursor stack history.
|
|
|
|
mCursorStackHistory.push_back(getCursor());
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(BACKSOUND);
|
|
|
|
populateList(mCursorStack.top()->getParent()->getChildrenListToDisplay(),
|
|
|
|
mCursorStack.top()->getParent());
|
|
|
|
setCursor(mCursorStack.top());
|
|
|
|
if (mCursorStack.size() > 0)
|
|
|
|
mCursorStack.pop();
|
|
|
|
if (mRoot->getSystem()->getThemeFolder() == "custom-collections")
|
|
|
|
updateHelpPrompts();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(BACKSOUND);
|
|
|
|
onPauseVideo();
|
|
|
|
onFocusLost();
|
|
|
|
stopListScrolling();
|
|
|
|
SystemData* systemToView = getCursor()->getSystem();
|
|
|
|
if (systemToView->isCustomCollection() &&
|
|
|
|
systemToView->getRootFolder()->getParent())
|
|
|
|
ViewController::getInstance()->goToSystemView(
|
|
|
|
systemToView->getRootFolder()->getParent()->getSystem(), true);
|
|
|
|
else
|
|
|
|
ViewController::getInstance()->goToSystemView(systemToView, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("x", input)) {
|
|
|
|
if (getCursor()->getType() == PLACEHOLDER) {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("x", input) &&
|
|
|
|
mRoot->getSystem()->getThemeFolder() == "custom-collections" &&
|
|
|
|
mCursorStack.empty() &&
|
|
|
|
ViewController::getInstance()->getState().viewing ==
|
|
|
|
ViewController::GAMELIST) {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
// Jump to the randomly selected game.
|
|
|
|
if (mRandomGame) {
|
|
|
|
stopListScrolling();
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
mWindow->startMediaViewer(mRandomGame);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mRoot->getSystem()->isGameSystem()) {
|
|
|
|
stopListScrolling();
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
mWindow->startMediaViewer(getCursor());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (config->isMappedLike(getQuickSystemSelectRightButton(), input)) {
|
|
|
|
if (Settings::getInstance()->getBool("QuickSystemSelect") &&
|
|
|
|
SystemData::sSystemVector.size() > 1) {
|
|
|
|
onPauseVideo();
|
|
|
|
onFocusLost();
|
|
|
|
stopListScrolling();
|
|
|
|
ViewController::getInstance()->goToNextGamelist();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (config->isMappedLike(getQuickSystemSelectLeftButton(), input)) {
|
|
|
|
if (Settings::getInstance()->getBool("QuickSystemSelect") &&
|
|
|
|
SystemData::sSystemVector.size() > 1) {
|
|
|
|
onPauseVideo();
|
|
|
|
onFocusLost();
|
|
|
|
stopListScrolling();
|
|
|
|
ViewController::getInstance()->goToPrevGamelist();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (Settings::getInstance()->getBool("RandomAddButton") &&
|
|
|
|
(config->isMappedTo("leftthumbstickclick", input) ||
|
|
|
|
config->isMappedTo("rightthumbstickclick", input))) {
|
|
|
|
if (mRoot->getSystem()->isGameSystem() && getCursor()->getType() != PLACEHOLDER) {
|
|
|
|
stopListScrolling();
|
|
|
|
// Jump to a random game.
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
FileData* randomGame = getCursor()->getSystem()->getRandomGame(getCursor());
|
|
|
|
if (randomGame)
|
|
|
|
setCursor(randomGame);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("y", input) &&
|
|
|
|
mRoot->getSystem()->getThemeFolder() == "custom-collections" &&
|
|
|
|
!CollectionSystemsManager::getInstance()->isEditing() && mCursorStack.empty() &&
|
|
|
|
ViewController::getInstance()->getState().viewing == ViewController::GAMELIST) {
|
|
|
|
// Jump to the randomly selected game.
|
|
|
|
if (mRandomGame) {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SELECTSOUND);
|
|
|
|
// If there is already an mCursorStackHistory entry for the collection, then
|
|
|
|
// remove it so we don't get multiple entries.
|
|
|
|
std::vector<FileData*> listEntries =
|
|
|
|
mRandomGame->getSystem()->getRootFolder()->getChildrenListToDisplay();
|
|
|
|
for (auto it = mCursorStackHistory.begin(); it != mCursorStackHistory.end(); ++it) {
|
|
|
|
if (std::find(listEntries.begin(), listEntries.end(), *it) !=
|
|
|
|
listEntries.end()) {
|
|
|
|
mCursorStackHistory.erase(it);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setCursor(mRandomGame);
|
|
|
|
updateHelpPrompts();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("y", input) &&
|
|
|
|
!Settings::getInstance()->getBool("FavoritesAddButton") &&
|
|
|
|
!CollectionSystemsManager::getInstance()->isEditing()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("y", input) &&
|
|
|
|
!UIModeController::getInstance()->isUIModeKid() &&
|
|
|
|
!UIModeController::getInstance()->isUIModeKiosk()) {
|
|
|
|
// Notify the user if attempting to add a custom collection to a custom collection.
|
|
|
|
if (CollectionSystemsManager::getInstance()->isEditing() &&
|
|
|
|
mRoot->getSystem()->isGameSystem() && getCursor()->getType() != PLACEHOLDER &&
|
|
|
|
getCursor()->getParent()->getPath() == "collections") {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(FAVORITESOUND);
|
|
|
|
mWindow->queueInfoPopup("CAN'T ADD CUSTOM COLLECTIONS TO CUSTOM COLLECTIONS", 4000);
|
|
|
|
}
|
|
|
|
// Notify the user if attempting to add a placeholder to a custom collection.
|
|
|
|
if (CollectionSystemsManager::getInstance()->isEditing() &&
|
|
|
|
mRoot->getSystem()->isGameSystem() && getCursor()->getType() == PLACEHOLDER) {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(FAVORITESOUND);
|
|
|
|
mWindow->queueInfoPopup("CAN'T ADD PLACEHOLDERS TO CUSTOM COLLECTIONS", 4000);
|
|
|
|
}
|
|
|
|
else if (mRoot->getSystem()->isGameSystem() && getCursor()->getType() != PLACEHOLDER &&
|
|
|
|
getCursor()->getParent()->getPath() != "collections") {
|
|
|
|
if (getCursor()->getType() == GAME || getCursor()->getType() == FOLDER)
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(FAVORITESOUND);
|
|
|
|
// When marking or unmarking a game as favorite, don't jump to the new position
|
|
|
|
// it gets after the gamelist sorting. Instead retain the cursor position in the
|
|
|
|
// list using the logic below.
|
|
|
|
FileData* entryToUpdate = getCursor();
|
|
|
|
SystemData* system = getCursor()->getSystem();
|
|
|
|
bool favoritesSorting;
|
|
|
|
bool removedLastFavorite = false;
|
|
|
|
bool selectLastEntry = false;
|
|
|
|
bool isEditing = CollectionSystemsManager::getInstance()->isEditing();
|
|
|
|
bool foldersOnTop = Settings::getInstance()->getBool("FoldersOnTop");
|
|
|
|
// If the current list only contains folders, then treat it as if the folders
|
|
|
|
// are not sorted on top, this way the logic should work exactly as for mixed
|
|
|
|
// lists or files-only lists.
|
|
|
|
if (getCursor()->getType() == FOLDER && foldersOnTop == true)
|
|
|
|
foldersOnTop = !getCursor()->getParent()->getOnlyFoldersFlag();
|
|
|
|
|
|
|
|
if (mRoot->getSystem()->isCustomCollection() ||
|
|
|
|
mRoot->getSystem()->getThemeFolder() == "custom-collections")
|
|
|
|
favoritesSorting = Settings::getInstance()->getBool("FavFirstCustom");
|
|
|
|
else
|
|
|
|
favoritesSorting = Settings::getInstance()->getBool("FavoritesFirst");
|
|
|
|
|
|
|
|
if (favoritesSorting && mRoot->getSystem()->getName() != "recent" && !isEditing) {
|
|
|
|
FileData* entryToSelect;
|
|
|
|
// Add favorite flag.
|
|
|
|
if (!getCursor()->getFavorite()) {
|
|
|
|
// If it's a folder and folders are sorted on top, select the current entry.
|
|
|
|
if (foldersOnTop && getCursor()->getType() == FOLDER) {
|
|
|
|
entryToSelect = getCursor();
|
|
|
|
}
|
|
|
|
// If it's the first entry to be marked as favorite, select the next entry.
|
|
|
|
else if (getCursor() == getFirstEntry()) {
|
|
|
|
entryToSelect = getNextEntry();
|
|
|
|
}
|
|
|
|
else if (getCursor() == getLastEntry() &&
|
|
|
|
getPreviousEntry()->getFavorite()) {
|
|
|
|
entryToSelect = getLastEntry();
|
|
|
|
selectLastEntry = true;
|
|
|
|
}
|
|
|
|
// If we are on the favorite marking boundary, select the next entry.
|
|
|
|
else if (getCursor()->getFavorite() != getPreviousEntry()->getFavorite()) {
|
|
|
|
entryToSelect = getNextEntry();
|
|
|
|
}
|
|
|
|
// If we mark the second entry as favorite and the first entry is not a
|
|
|
|
// favorite, then select this entry if they are of the same type.
|
|
|
|
else if (getPreviousEntry() == getFirstEntry() &&
|
|
|
|
getCursor()->getType() == getPreviousEntry()->getType()) {
|
|
|
|
entryToSelect = getPreviousEntry();
|
|
|
|
}
|
|
|
|
// For all other scenarios try to select the next entry, and if it doesn't
|
|
|
|
// exist, select the previous entry.
|
|
|
|
else {
|
|
|
|
entryToSelect =
|
|
|
|
getCursor() != getNextEntry() ? getNextEntry() : getPreviousEntry();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Remove favorite flag.
|
|
|
|
else {
|
|
|
|
// If it's a folder and folders are sorted on top, select the current entry.
|
|
|
|
if (foldersOnTop && getCursor()->getType() == FOLDER) {
|
|
|
|
entryToSelect = getCursor();
|
|
|
|
}
|
|
|
|
// If it's the last entry, select the previous entry.
|
|
|
|
else if (getCursor() == getLastEntry()) {
|
|
|
|
entryToSelect = getPreviousEntry();
|
|
|
|
}
|
|
|
|
// If we are on the favorite marking boundary, select the previous entry,
|
|
|
|
// unless folders are sorted on top and the previous entry is a folder.
|
|
|
|
else if (foldersOnTop &&
|
|
|
|
getCursor()->getFavorite() != getNextEntry()->getFavorite()) {
|
|
|
|
entryToSelect = getPreviousEntry()->getType() == FOLDER ?
|
|
|
|
getCursor() :
|
|
|
|
getPreviousEntry();
|
|
|
|
}
|
|
|
|
// If we are on the favorite marking boundary, select the previous entry.
|
|
|
|
else if (getCursor()->getFavorite() != getNextEntry()->getFavorite()) {
|
|
|
|
entryToSelect = getPreviousEntry();
|
|
|
|
}
|
|
|
|
// For all other scenarios try to select the next entry, and if it doesn't
|
|
|
|
// exist, select the previous entry.
|
|
|
|
else {
|
|
|
|
entryToSelect =
|
|
|
|
getCursor() != getNextEntry() ? getNextEntry() : getPreviousEntry();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we removed the last favorite marking, set the flag to jump to the
|
|
|
|
// first list entry after the sorting has been performed.
|
|
|
|
if (foldersOnTop && getCursor() == getFirstGameEntry() &&
|
|
|
|
!getNextEntry()->getFavorite())
|
|
|
|
removedLastFavorite = true;
|
|
|
|
else if (getCursor() == getFirstEntry() && !getNextEntry()->getFavorite())
|
|
|
|
removedLastFavorite = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
setCursor(entryToSelect);
|
|
|
|
system = entryToUpdate->getSystem();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Marking folders as favorites don't make them part of any collections,
|
|
|
|
// so it makes more sense to handle it here than to add the function to
|
|
|
|
// CollectionSystemsManager.
|
|
|
|
if (entryToUpdate->getType() == FOLDER) {
|
|
|
|
if (isEditing) {
|
|
|
|
mWindow->queueInfoPopup("CAN'T ADD FOLDERS TO CUSTOM COLLECTIONS", 4000);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
MetaDataList* md = &entryToUpdate->getSourceFileData()->metadata;
|
|
|
|
if (md->get("favorite") == "false") {
|
|
|
|
md->set("favorite", "true");
|
|
|
|
mWindow->queueInfoPopup(
|
|
|
|
"MARKED FOLDER '" +
|
|
|
|
Utils::String::toUpper(Utils::String::removeParenthesis(
|
|
|
|
entryToUpdate->getName())) +
|
|
|
|
"' AS FAVORITE",
|
|
|
|
4000);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
md->set("favorite", "false");
|
|
|
|
mWindow->queueInfoPopup(
|
|
|
|
"REMOVED FAVORITE MARKING FOR FOLDER '" +
|
|
|
|
Utils::String::toUpper(Utils::String::removeParenthesis(
|
|
|
|
entryToUpdate->getName())) +
|
|
|
|
"'",
|
|
|
|
4000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
entryToUpdate->getSourceFileData()->getSystem()->onMetaDataSavePoint();
|
|
|
|
|
|
|
|
getCursor()->getParent()->sort(
|
|
|
|
mRoot->getSortTypeFromString(mRoot->getSortTypeString()),
|
|
|
|
Settings::getInstance()->getBool("FavoritesFirst"));
|
|
|
|
|
|
|
|
ViewController::getInstance()->onFileChanged(getCursor(), false);
|
|
|
|
|
|
|
|
// Always jump to the first entry in the gamelist if the last favorite
|
|
|
|
// was unmarked. We couldn't do this earlier as we didn't have the list
|
|
|
|
// sorted yet.
|
|
|
|
if (removedLastFavorite) {
|
2022-01-18 19:42:50 +00:00
|
|
|
ViewController::getInstance()
|
|
|
|
->getGamelistView(entryToUpdate->getSystem())
|
|
|
|
->setCursor(ViewController::getInstance()
|
|
|
|
->getGamelistView(entryToUpdate->getSystem())
|
|
|
|
->getFirstEntry());
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (isEditing && entryToUpdate->metadata.get("nogamecount") == "true") {
|
|
|
|
mWindow->queueInfoPopup("CAN'T ADD ENTRIES THAT ARE NOT COUNTED "
|
|
|
|
"AS GAMES TO CUSTOM COLLECTIONS",
|
|
|
|
4000);
|
|
|
|
}
|
|
|
|
else if (CollectionSystemsManager::getInstance()->toggleGameInCollection(
|
|
|
|
entryToUpdate)) {
|
|
|
|
// As the toggling of the game destroyed this object, we need to get the view
|
|
|
|
// from ViewController instead of using the reference that existed before the
|
|
|
|
// destruction. Otherwise we get random crashes.
|
2022-01-18 19:42:50 +00:00
|
|
|
GamelistView* view =
|
|
|
|
ViewController::getInstance()->getGamelistView(system).get();
|
2022-01-18 16:14:17 +00:00
|
|
|
// Jump to the first entry in the gamelist if the last favorite was unmarked.
|
|
|
|
if (foldersOnTop && removedLastFavorite &&
|
|
|
|
!entryToUpdate->getSystem()->isCustomCollection()) {
|
2022-01-18 19:42:50 +00:00
|
|
|
ViewController::getInstance()
|
|
|
|
->getGamelistView(entryToUpdate->getSystem())
|
|
|
|
->setCursor(ViewController::getInstance()
|
|
|
|
->getGamelistView(entryToUpdate->getSystem())
|
|
|
|
->getFirstGameEntry());
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
else if (removedLastFavorite &&
|
|
|
|
!entryToUpdate->getSystem()->isCustomCollection()) {
|
|
|
|
setCursor(getFirstEntry());
|
2022-01-18 19:42:50 +00:00
|
|
|
view->setCursor(view->getFirstEntry());
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
else if (selectLastEntry) {
|
|
|
|
setCursor(getLastEntry());
|
2022-01-18 19:42:50 +00:00
|
|
|
view->setCursor(view->getLastEntry());
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
// Display the indication icons which show what games are part of the
|
|
|
|
// custom collection currently being edited. This is done cheaply using
|
|
|
|
// onFileChanged() which will trigger populateList().
|
|
|
|
if (isEditing) {
|
|
|
|
for (auto it = SystemData::sSystemVector.begin();
|
|
|
|
it != SystemData::sSystemVector.end(); ++it) {
|
2022-01-18 19:42:50 +00:00
|
|
|
ViewController::getInstance()->getGamelistView((*it))->onFileChanged(
|
|
|
|
ViewController::getInstance()->getGamelistView((*it))->getCursor(),
|
|
|
|
false);
|
2022-01-18 16:14:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (config->isMappedTo("y", input) && getCursor()->isPlaceHolder()) {
|
|
|
|
NavigationSounds::getInstance().playThemeNavigationSound(SCROLLSOUND);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Select button opens GuiGamelistOptions.
|
|
|
|
if (!UIModeController::getInstance()->isUIModeKid() && // Line break.
|
|
|
|
config->isMappedTo("back", input) && input.value) {
|
|
|
|
ViewController::getInstance()->cancelViewTransitions();
|
|
|
|
stopListScrolling();
|
2022-01-19 17:01:54 +00:00
|
|
|
mWindow->pushGui(new GuiGamelistOptions(this->mRoot->getSystem()));
|
2022-01-18 16:14:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ctrl-R reloads the view when debugging.
|
|
|
|
else if (Settings::getInstance()->getBool("Debug") &&
|
|
|
|
config->getDeviceId() == DEVICE_KEYBOARD &&
|
|
|
|
(SDL_GetModState() & (KMOD_LCTRL | KMOD_RCTRL)) && input.id == SDLK_r &&
|
|
|
|
input.value != 0) {
|
2022-01-18 19:42:50 +00:00
|
|
|
LOG(LogDebug) << "GamelistView::input(): Reloading view";
|
|
|
|
ViewController::getInstance()->reloadGamelistView(this->mRoot->getSystem(), true);
|
2022-01-18 16:14:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return GuiComponent::input(config, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::populateList(const std::vector<FileData*>& files, FileData* firstEntry)
|
|
|
|
{
|
|
|
|
mFirstGameEntry = nullptr;
|
|
|
|
bool favoriteStar {true};
|
|
|
|
bool isEditing {false};
|
|
|
|
std::string editingCollection;
|
|
|
|
std::string inCollectionPrefix;
|
|
|
|
|
|
|
|
if (CollectionSystemsManager::getInstance()->isEditing()) {
|
|
|
|
editingCollection = CollectionSystemsManager::getInstance()->getEditingCollection();
|
|
|
|
isEditing = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the settings that control whether a unicode star character should be added
|
|
|
|
// as a prefix to the game name.
|
|
|
|
if (files.size() > 0) {
|
|
|
|
if (files.front()->getSystem()->isCustomCollection())
|
|
|
|
favoriteStar = Settings::getInstance()->getBool("FavStarCustom");
|
|
|
|
else
|
|
|
|
favoriteStar = Settings::getInstance()->getBool("FavoritesStar");
|
|
|
|
}
|
|
|
|
|
|
|
|
mList.clear();
|
|
|
|
|
|
|
|
if (files.size() > 0) {
|
|
|
|
for (auto it = files.cbegin(); it != files.cend(); ++it) {
|
|
|
|
if (!mFirstGameEntry && (*it)->getType() == GAME)
|
|
|
|
mFirstGameEntry = (*it);
|
|
|
|
// Add a leading tick mark icon to the game name if it's part of the custom collection
|
|
|
|
// currently being edited.
|
|
|
|
if (isEditing && (*it)->getType() == GAME) {
|
|
|
|
if (CollectionSystemsManager::getInstance()->inCustomCollection(editingCollection,
|
|
|
|
(*it))) {
|
|
|
|
if (Settings::getInstance()->getBool("SpecialCharsASCII"))
|
|
|
|
inCollectionPrefix = "! ";
|
|
|
|
else
|
|
|
|
inCollectionPrefix = ViewController::TICKMARK_CHAR + " ";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
inCollectionPrefix = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((*it)->getFavorite() && favoriteStar &&
|
|
|
|
mRoot->getSystem()->getName() != "favorites") {
|
|
|
|
if (Settings::getInstance()->getBool("SpecialCharsASCII"))
|
|
|
|
mList.add(inCollectionPrefix + "* " + (*it)->getName(), *it,
|
|
|
|
((*it)->getType() == FOLDER));
|
|
|
|
else
|
|
|
|
mList.add(inCollectionPrefix + ViewController::FAVORITE_CHAR + " " +
|
|
|
|
(*it)->getName(),
|
|
|
|
*it, ((*it)->getType() == FOLDER));
|
|
|
|
}
|
|
|
|
else if ((*it)->getType() == FOLDER && mRoot->getSystem()->getName() != "collections") {
|
|
|
|
if (Settings::getInstance()->getBool("SpecialCharsASCII"))
|
|
|
|
mList.add("# " + (*it)->getName(), *it, true);
|
|
|
|
else
|
|
|
|
mList.add(ViewController::FOLDER_CHAR + " " + (*it)->getName(), *it, true);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mList.add(inCollectionPrefix + (*it)->getName(), *it, ((*it)->getType() == FOLDER));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
addPlaceholder(firstEntry);
|
|
|
|
}
|
|
|
|
|
|
|
|
generateGamelistInfo(getCursor(), firstEntry);
|
|
|
|
generateFirstLetterIndex(files);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::addPlaceholder(FileData* firstEntry)
|
|
|
|
{
|
|
|
|
// Empty list, add a placeholder.
|
|
|
|
FileData* placeholder;
|
|
|
|
|
|
|
|
if (firstEntry && firstEntry->getSystem()->isGroupedCustomCollection())
|
|
|
|
placeholder = firstEntry->getSystem()->getPlaceholder();
|
|
|
|
else
|
|
|
|
placeholder = this->mRoot->getSystem()->getPlaceholder();
|
|
|
|
|
|
|
|
mList.add(placeholder->getName(), placeholder, (placeholder->getType() == PLACEHOLDER));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::generateFirstLetterIndex(const std::vector<FileData*>& files)
|
|
|
|
{
|
|
|
|
std::string firstChar;
|
|
|
|
|
|
|
|
bool onlyFavorites {true};
|
|
|
|
bool onlyFolders {true};
|
|
|
|
bool hasFavorites {false};
|
|
|
|
bool hasFolders {false};
|
|
|
|
bool favoritesSorting {false};
|
|
|
|
|
|
|
|
mFirstLetterIndex.clear();
|
|
|
|
|
|
|
|
if (files.size() > 0 && files.front()->getSystem()->isCustomCollection())
|
|
|
|
favoritesSorting = Settings::getInstance()->getBool("FavFirstCustom");
|
|
|
|
else
|
|
|
|
favoritesSorting = Settings::getInstance()->getBool("FavoritesFirst");
|
|
|
|
|
|
|
|
bool foldersOnTop {Settings::getInstance()->getBool("FoldersOnTop")};
|
|
|
|
|
|
|
|
// Find out if there are only favorites and/or only folders in the list.
|
|
|
|
for (auto it = files.begin(); it != files.end(); ++it) {
|
|
|
|
if (!((*it)->getFavorite()))
|
|
|
|
onlyFavorites = false;
|
|
|
|
if (!((*it)->getType() == FOLDER))
|
|
|
|
onlyFolders = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the index.
|
|
|
|
for (auto it = files.begin(); it != files.end(); ++it) {
|
|
|
|
if ((*it)->getType() == FOLDER && (*it)->getFavorite() && favoritesSorting &&
|
|
|
|
!onlyFavorites) {
|
|
|
|
hasFavorites = true;
|
|
|
|
}
|
|
|
|
else if ((*it)->getType() == FOLDER && foldersOnTop && !onlyFolders) {
|
|
|
|
hasFolders = true;
|
|
|
|
}
|
|
|
|
else if ((*it)->getType() == GAME && (*it)->getFavorite() && favoritesSorting &&
|
|
|
|
!onlyFavorites) {
|
|
|
|
hasFavorites = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mFirstLetterIndex.push_back(Utils::String::getFirstCharacter((*it)->getSortName()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort and make each entry unique.
|
|
|
|
std::sort(mFirstLetterIndex.begin(), mFirstLetterIndex.end());
|
|
|
|
auto last = std::unique(mFirstLetterIndex.begin(), mFirstLetterIndex.end());
|
|
|
|
mFirstLetterIndex.erase(last, mFirstLetterIndex.end());
|
|
|
|
|
|
|
|
// If there are any favorites and/or folders in the list, insert their respective
|
|
|
|
// Unicode characters at the beginning of the vector.
|
|
|
|
if (hasFavorites)
|
|
|
|
mFirstLetterIndex.insert(mFirstLetterIndex.begin(), ViewController::FAVORITE_CHAR);
|
|
|
|
|
|
|
|
if (hasFolders)
|
|
|
|
mFirstLetterIndex.insert(mFirstLetterIndex.begin(), ViewController::FOLDER_CHAR);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::generateGamelistInfo(FileData* cursor, FileData* firstEntry)
|
|
|
|
{
|
|
|
|
// Generate data needed for the gamelistInfo field, which is displayed from the
|
|
|
|
// gamelist interfaces (Detailed/Video/Grid).
|
|
|
|
mIsFiltered = false;
|
|
|
|
mIsFolder = false;
|
|
|
|
FileData* rootFolder {firstEntry->getSystem()->getRootFolder()};
|
|
|
|
|
|
|
|
std::pair<unsigned int, unsigned int> gameCount;
|
|
|
|
FileFilterIndex* idx {rootFolder->getSystem()->getIndex()};
|
|
|
|
|
|
|
|
// For the 'recent' collection we need to recount the games as the collection was
|
|
|
|
// trimmed down to 50 items. If we don't do this, the game count will not be correct
|
|
|
|
// as it would include all the games prior to trimming.
|
|
|
|
if (mRoot->getPath() == "recent")
|
|
|
|
mRoot->countGames(gameCount);
|
|
|
|
|
|
|
|
gameCount = rootFolder->getGameCount();
|
|
|
|
|
|
|
|
mGameCount = gameCount.first;
|
|
|
|
mFavoritesGameCount = gameCount.second;
|
|
|
|
mFilteredGameCount = 0;
|
|
|
|
mFilteredGameCountAll = 0;
|
|
|
|
|
|
|
|
if (idx->isFiltered()) {
|
|
|
|
mIsFiltered = true;
|
|
|
|
mFilteredGameCount =
|
|
|
|
static_cast<unsigned int>(rootFolder->getFilesRecursive(GAME, true, false).size());
|
|
|
|
// Also count the games that are set to not be counted as games, as the filter may
|
|
|
|
// apply to such entries as well and this will be indicated with a separate '+ XX'
|
|
|
|
// in the GamelistInfo field.
|
|
|
|
mFilteredGameCountAll =
|
|
|
|
static_cast<unsigned int>(rootFolder->getFilesRecursive(GAME, true, true).size());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (firstEntry->getParent() && firstEntry->getParent()->getType() == FOLDER)
|
|
|
|
mIsFolder = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::remove(FileData* game, bool deleteFile)
|
|
|
|
{
|
2022-01-22 20:42:43 +00:00
|
|
|
// Optionally delete the game file on the filesystem.
|
2022-01-18 16:14:17 +00:00
|
|
|
if (deleteFile)
|
|
|
|
Utils::FileSystem::removeFile(game->getPath());
|
|
|
|
|
|
|
|
FileData* parent {game->getParent()};
|
|
|
|
// Select next element in list, or previous if none.
|
|
|
|
if (getCursor() == game) {
|
|
|
|
std::vector<FileData*> siblings {parent->getChildrenListToDisplay()};
|
|
|
|
auto gameIter = std::find(siblings.cbegin(), siblings.cend(), game);
|
|
|
|
unsigned int gamePos {
|
|
|
|
static_cast<unsigned int>(std::distance(siblings.cbegin(), gameIter))};
|
|
|
|
if (gameIter != siblings.cend()) {
|
|
|
|
if ((gamePos + 1) < siblings.size())
|
|
|
|
setCursor(siblings.at(gamePos + 1));
|
|
|
|
else if (gamePos > 1)
|
|
|
|
setCursor(siblings.at(gamePos - 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mList.remove(game);
|
|
|
|
|
|
|
|
if (mList.size() == 0)
|
|
|
|
addPlaceholder(nullptr);
|
|
|
|
|
|
|
|
// If a game has been deleted, immediately remove the entry from gamelist.xml
|
|
|
|
// regardless of the value of the setting SaveGamelistsMode.
|
|
|
|
game->setDeletionFlag(true);
|
|
|
|
parent->getSystem()->writeMetaData();
|
|
|
|
|
|
|
|
// Remove before repopulating (removes from parent), then update the view.
|
|
|
|
delete game;
|
|
|
|
|
|
|
|
if (deleteFile) {
|
|
|
|
parent->sort(parent->getSortTypeFromString(parent->getSortTypeString()),
|
|
|
|
Settings::getInstance()->getBool("FavoritesFirst"));
|
|
|
|
onFileChanged(parent, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GamelistBase::removeMedia(FileData* game)
|
|
|
|
{
|
|
|
|
std::string systemMediaDir {FileData::getMediaDirectory() + game->getSystem()->getName()};
|
|
|
|
std::string mediaType;
|
|
|
|
std::string path;
|
|
|
|
|
|
|
|
// Stop the video player, especially important on Windows as the file would otherwise be locked.
|
|
|
|
onStopVideo();
|
|
|
|
|
|
|
|
// If there are no media files left in the directory after the deletion, then remove
|
|
|
|
// the directory too. Remove any empty parent directories as well.
|
|
|
|
auto removeEmptyDirFunc = [](std::string systemMediaDir, std::string mediaType,
|
|
|
|
std::string path) {
|
|
|
|
std::string parentPath {Utils::FileSystem::getParent(path)};
|
|
|
|
while (parentPath != systemMediaDir + "/" + mediaType) {
|
|
|
|
if (Utils::FileSystem::getDirContent(parentPath).size() == 0) {
|
|
|
|
Utils::FileSystem::removeDirectory(parentPath);
|
|
|
|
parentPath = Utils::FileSystem::getParent(parentPath);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Remove all game media files on the filesystem.
|
|
|
|
while (Utils::FileSystem::exists(game->getVideoPath())) {
|
|
|
|
mediaType = "videos";
|
|
|
|
path = game->getVideoPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getMiximagePath())) {
|
|
|
|
mediaType = "miximages";
|
|
|
|
path = game->getMiximagePath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getScreenshotPath())) {
|
|
|
|
mediaType = "screenshots";
|
|
|
|
path = game->getScreenshotPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getTitleScreenPath())) {
|
|
|
|
mediaType = "titlescreens";
|
|
|
|
path = game->getTitleScreenPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getCoverPath())) {
|
|
|
|
mediaType = "covers";
|
|
|
|
path = game->getCoverPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getBackCoverPath())) {
|
|
|
|
mediaType = "backcovers";
|
|
|
|
path = game->getBackCoverPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getFanArtPath())) {
|
|
|
|
mediaType = "fanart";
|
|
|
|
path = game->getFanArtPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getMarqueePath())) {
|
|
|
|
mediaType = "marquees";
|
|
|
|
path = game->getMarqueePath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->get3DBoxPath())) {
|
|
|
|
mediaType = "3dboxes";
|
|
|
|
path = game->get3DBoxPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getPhysicalMediaPath())) {
|
|
|
|
mediaType = "physicalmedia";
|
|
|
|
path = game->getPhysicalMediaPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Utils::FileSystem::exists(game->getThumbnailPath())) {
|
|
|
|
mediaType = "thumbnails";
|
|
|
|
path = game->getThumbnailPath();
|
|
|
|
Utils::FileSystem::removeFile(path);
|
|
|
|
removeEmptyDirFunc(systemMediaDir, mediaType, path);
|
|
|
|
}
|
|
|
|
}
|