// SPDX-License-Identifier: MIT // // EmulationStation Desktop Edition // ViewController.cpp // // Handles overall system navigation including animations and transitions. // Creates the gamelist views and handles refresh and reloads of these when needed // (for example when metadata has been changed or when a list sorting has taken place). // Initiates the launching of games, calling FileData to do the actual launch. // Displays a dialog when there are no games found on startup. // #include "views/ViewController.h" #include "FileFilterIndex.h" #include "InputManager.h" #include "Log.h" #include "Settings.h" #include "Sound.h" #include "SystemData.h" #include "SystemView.h" #include "UIModeController.h" #include "Window.h" #include "animations/Animation.h" #include "animations/LambdaAnimation.h" #include "animations/MoveCameraAnimation.h" #include "guis/GuiMenu.h" #include "guis/GuiTextEditKeyboardPopup.h" #include "guis/GuiTextEditPopup.h" #include "views/GamelistView.h" #include "views/SystemView.h" ViewController::ViewController() noexcept : mRenderer {Renderer::getInstance()} , mNoGamesMessageBox {nullptr} , mCurrentView {nullptr} , mPreviousView {nullptr} , mSkipView {nullptr} , mGameToLaunch {nullptr} , mCamera {Renderer::getIdentity()} , mSystemViewTransition {false} , mWrappedViews {false} , mFadeOpacity {0} , mCancelledTransition {false} , mLockInput {false} , mNextSystem {false} { mState.viewing = NOTHING; mState.viewstyle = AUTOMATIC; } ViewController* ViewController::getInstance() { static ViewController instance; return &instance; } void ViewController::invalidSystemsFileDialog() { std::string errorMessage = "COULDN'T PARSE THE SYSTEMS CONFIGURATION FILE.\n" "IF YOU HAVE A CUSTOMIZED es_systems.xml FILE, THEN\n" "SOMETHING IS LIKELY WRONG WITH YOUR XML SYNTAX.\n" "IF YOU DON'T HAVE A CUSTOM SYSTEMS FILE, THEN THE\n" "EMULATIONSTATION INSTALLATION IS BROKEN. SEE THE\n" "APPLICATION LOG FILE es_log.txt FOR ADDITIONAL INFO."; mWindow->pushGui(new GuiMsgBox( HelpStyle(), errorMessage.c_str(), "QUIT", [] { SDL_Event quit; quit.type = SDL_QUIT; SDL_PushEvent(&quit); }, "", nullptr, "", nullptr, true)); } void ViewController::noGamesDialog() { mNoGamesErrorMessage = "NO GAME FILES WERE FOUND. EITHER PLACE YOUR GAMES IN\n" "THE CURRENTLY CONFIGURED ROM DIRECTORY OR CHANGE\n" "ITS PATH USING THE BUTTON BELOW. OPTIONALLY THE ROM\n" "DIRECTORY STRUCTURE CAN BE GENERATED WHICH WILL\n" "CREATE A TEXT FILE FOR EACH SYSTEM PROVIDING SOME\n" "INFORMATION SUCH AS THE SUPPORTED FILE EXTENSIONS.\n" "THIS IS THE CURRENTLY CONFIGURED ROM DIRECTORY:\n"; #if defined(_WIN64) mRomDirectory = Utils::String::replace(FileData::getROMDirectory(), "/", "\\"); #else mRomDirectory = FileData::getROMDirectory(); #endif mNoGamesMessageBox = new GuiMsgBox( HelpStyle(), mNoGamesErrorMessage + mRomDirectory, "CHANGE ROM DIRECTORY", [this] { std::string currentROMDirectory; #if defined(_WIN64) currentROMDirectory = Utils::String::replace(FileData::getROMDirectory(), "/", "\\"); #else currentROMDirectory = FileData::getROMDirectory(); #endif if (Settings::getInstance()->getBool("VirtualKeyboard")) { mWindow->pushGui(new GuiTextEditKeyboardPopup( HelpStyle(), "ENTER ROM DIRECTORY PATH", currentROMDirectory, [this](const std::string& newROMDirectory) { Settings::getInstance()->setString("ROMDirectory", Utils::String::trim(newROMDirectory)); Settings::getInstance()->saveFile(); #if defined(_WIN64) mRomDirectory = Utils::String::replace(FileData::getROMDirectory(), "/", "\\"); #else mRomDirectory = FileData::getROMDirectory(); #endif mNoGamesMessageBox->changeText(mNoGamesErrorMessage + mRomDirectory); mWindow->pushGui(new GuiMsgBox(HelpStyle(), "ROM DIRECTORY SETTING SAVED, RESTART\n" "THE APPLICATION TO RESCAN THE SYSTEMS", "OK", nullptr, "", nullptr, "", nullptr, true)); }, false, "SAVE", "SAVE CHANGES?", "Currently configured path:", currentROMDirectory, "LOAD CURRENTLY CONFIGURED PATH", "CLEAR (LEAVE BLANK TO RESET TO DEFAULT PATH)")); } else { mWindow->pushGui(new GuiTextEditPopup( HelpStyle(), "ENTER ROM DIRECTORY PATH", currentROMDirectory, [this](const std::string& newROMDirectory) { Settings::getInstance()->setString("ROMDirectory", Utils::String::trim(newROMDirectory)); Settings::getInstance()->saveFile(); #if defined(_WIN64) mRomDirectory = Utils::String::replace(FileData::getROMDirectory(), "/", "\\"); #else mRomDirectory = FileData::getROMDirectory(); #endif mNoGamesMessageBox->changeText(mNoGamesErrorMessage + mRomDirectory); mWindow->pushGui(new GuiMsgBox(HelpStyle(), "ROM DIRECTORY SETTING SAVED, RESTART\n" "THE APPLICATION TO RESCAN THE SYSTEMS", "OK", nullptr, "", nullptr, "", nullptr, true)); }, false, "SAVE", "SAVE CHANGES?", "Currently configured path:", currentROMDirectory, "LOAD CURRENTLY CONFIGURED PATH", "CLEAR (LEAVE BLANK TO RESET TO DEFAULT PATH)")); } }, "CREATE DIRECTORIES", [this] { mWindow->pushGui(new GuiMsgBox( HelpStyle(), "THIS WILL CREATE DIRECTORIES FOR ALL THE\n" "GAME SYSTEMS DEFINED IN es_systems.xml\n\n" "THIS MAY CREATE A LOT OF FOLDERS SO IT'S\n" "ADVICED TO REMOVE THE ONES YOU DON'T NEED\n\n" "PROCEED?", "YES", [this] { if (!SystemData::createSystemDirectories()) { mWindow->pushGui(new GuiMsgBox(HelpStyle(), "THE SYSTEM DIRECTORIES WERE SUCCESSFULLY\n" "GENERATED, EXIT THE APPLICATION AND PLACE\n" "YOUR GAMES IN THE NEWLY CREATED FOLDERS", "OK", nullptr, "", nullptr, "", nullptr, true)); } else { mWindow->pushGui(new GuiMsgBox(HelpStyle(), "ERROR CREATING THE SYSTEM DIRECTORIES,\n" "PERMISSION PROBLEMS OR DISK FULL?\n\n" "SEE THE LOG FILE FOR MORE DETAILS", "OK", nullptr, "", nullptr, "", nullptr, true)); } }, "NO", nullptr, "", nullptr, true)); }, "QUIT", [] { SDL_Event quit; quit.type = SDL_QUIT; SDL_PushEvent(&quit); }, true, false); mWindow->pushGui(mNoGamesMessageBox); } void ViewController::invalidAlternativeEmulatorDialog() { mWindow->pushGui(new GuiMsgBox(getHelpStyle(), "AT LEAST ONE OF YOUR SYSTEMS HAS AN\n" "INVALID ALTERNATIVE EMULATOR CONFIGURED\n" "WITH NO MATCHING ENTRY IN THE SYSTEMS\n" "CONFIGURATION FILE, PLEASE REVIEW YOUR\n" "SETUP USING THE 'ALTERNATIVE EMULATORS'\n" "INTERFACE IN THE 'OTHER SETTINGS' MENU")); } void ViewController::goToStart(bool playTransition) { // Needed to avoid segfaults during emergency shutdown. if (mRenderer->getSDLWindow() == nullptr) return; #if defined(__APPLE__) // The startup animations are very choppy on macOS as of moving to SDL 2.0.18 so the // best user experience is to simply disable them on this OS. if (mState.viewing == NOTHING) playTransition = false; #endif // If the system view does not exist, then create it. We do this here as it would // otherwise not be done if jumping directly into a specific game system on startup. if (!mSystemListView) getSystemListView(); // If a specific system is requested, go directly to its game list. auto requestedSystem = Settings::getInstance()->getString("StartupSystem"); if ("" != requestedSystem && "retropie" != requestedSystem) { for (auto it = SystemData::sSystemVector.cbegin(); // Line break. it != SystemData::sSystemVector.cend(); ++it) { if ((*it)->getName() == requestedSystem) { goToGamelist(*it); if (!playTransition) cancelViewTransitions(); return; } } // Requested system doesn't exist. Settings::getInstance()->setString("StartupSystem", ""); } // Get the first system entry. goToSystemView(getSystemListView()->getFirstSystem(), false); } void ViewController::ReloadAndGoToStart() { mWindow->renderLoadingScreen("Loading..."); reloadAll(); if (mState.viewing == GAMELIST) { goToSystemView(SystemData::sSystemVector.front(), false); goToSystem(SystemData::sSystemVector.front(), false); } else { goToSystem(SystemData::sSystemVector.front(), false); } } bool ViewController::isCameraMoving() { if (mCurrentView) { if (mCamera[3].x - -mCurrentView->getPosition().x != 0.0f || mCamera[3].y - -mCurrentView->getPosition().y != 0.0f) return true; } return false; } void ViewController::cancelViewTransitions() { if (Settings::getInstance()->getString("TransitionStyle") == "slide") { if (isCameraMoving()) { mCamera[3].x = -mCurrentView->getPosition().x; mCamera[3].y = -mCurrentView->getPosition().y; stopAllAnimations(); } // mSkipView is used when skipping through the gamelists in quick succession. // Without this, the game video (or static image) would not get rendered during // the slide transition animation. else if (mSkipView) { mSkipView.reset(); mSkipView = nullptr; } } else if (Settings::getInstance()->getString("TransitionStyle") == "fade") { if (isAnimationPlaying(0)) { finishAnimation(0); mCancelledTransition = true; mFadeOpacity = 0; mWindow->invalidateCachedBackground(); } } } void ViewController::stopScrolling() { if (mRenderer->getSDLWindow() == nullptr) return; mSystemListView->stopScrolling(); mCurrentView->stopListScrolling(); if (mSystemListView->isSystemAnimationPlaying(0)) mSystemListView->finishSystemAnimation(0); } int ViewController::getSystemId(SystemData* system) { std::vector& sysVec = SystemData::sSystemVector; return static_cast(std::find(sysVec.cbegin(), sysVec.cend(), system) - sysVec.cbegin()); } void ViewController::restoreViewPosition() { if (mPreviousView) { glm::vec3 restorePosition {mPreviousView->getPosition()}; restorePosition.x = mWrapPreviousPositionX; mPreviousView->setPosition(restorePosition); mWrapPreviousPositionX = 0; mWrappedViews = false; } } void ViewController::goToSystemView(SystemData* system, bool playTransition) { bool applicationStartup = false; if (mState.viewing == NOTHING) applicationStartup = true; // Restore the X position for the view, if it was previously moved. if (mWrappedViews) restoreViewPosition(); if (mPreviousView) { mPreviousView.reset(); mPreviousView = nullptr; } if (mCurrentView != nullptr) mCurrentView->onTransition(); mPreviousView = mCurrentView; if (system->isGroupedCustomCollection()) system = system->getRootFolder()->getParent()->getSystem(); mState.viewing = SYSTEM_SELECT; mState.system = system; mSystemViewTransition = true; auto systemList = getSystemListView(); systemList->setPosition(getSystemId(system) * Renderer::getScreenWidth(), systemList->getPosition().y); systemList->goToSystem(system, false); mCurrentView = systemList; mCurrentView->onShow(); // Application startup animation. if (applicationStartup) { mCamera = glm::translate(mCamera, -mCurrentView->getPosition()); if (Settings::getInstance()->getString("TransitionStyle") == "slide") { if (getSystemListView()->getPrimaryType() == SystemView::PrimaryType::CAROUSEL) { if (getSystemListView()->getCarouselType() == CarouselComponent::CarouselType::HORIZONTAL || getSystemListView()->getCarouselType() == CarouselComponent::CarouselType::HORIZONTAL_WHEEL) mCamera[3].y += Renderer::getScreenHeight(); else mCamera[3].x -= Renderer::getScreenWidth(); } else if (getSystemListView()->getPrimaryType() == SystemView::PrimaryType::TEXTLIST) { mCamera[3].y += Renderer::getScreenHeight(); } updateHelpPrompts(); } else if (Settings::getInstance()->getString("TransitionStyle") == "fade") { if (getSystemListView()->getPrimaryType() == SystemView::PrimaryType::CAROUSEL) { if (getSystemListView()->getCarouselType() == CarouselComponent::CarouselType::HORIZONTAL || getSystemListView()->getCarouselType() == CarouselComponent::CarouselType::HORIZONTAL_WHEEL) mCamera[3].y += Renderer::getScreenHeight(); else mCamera[3].x += Renderer::getScreenWidth(); } else if (getSystemListView()->getPrimaryType() == SystemView::PrimaryType::TEXTLIST) { mCamera[3].y += Renderer::getScreenHeight(); } } else { updateHelpPrompts(); } } #if defined(__APPLE__) // The startup animations are very choppy on macOS as of moving to SDL 2.0.18 so the // best user experience is to simply disable them on this OS. if (applicationStartup) playViewTransition(true); else if (playTransition) #else if (playTransition || applicationStartup) #endif playViewTransition(); else playViewTransition(true); } void ViewController::goToSystem(SystemData* system, bool animate) { mSystemListView->goToSystem(system, animate); } void ViewController::goToNextGamelist() { assert(mState.viewing == GAMELIST); SystemData* system = getState().getSystem(); assert(system); NavigationSounds::getInstance().playThemeNavigationSound(QUICKSYSSELECTSOUND); mNextSystem = true; goToGamelist(system->getNext()); } void ViewController::goToPrevGamelist() { assert(mState.viewing == GAMELIST); SystemData* system = getState().getSystem(); assert(system); NavigationSounds::getInstance().playThemeNavigationSound(QUICKSYSSELECTSOUND); mNextSystem = false; goToGamelist(system->getPrev()); } void ViewController::goToGamelist(SystemData* system) { bool wrapFirstToLast {false}; bool wrapLastToFirst {false}; bool slideTransitions {false}; if (mCurrentView != nullptr) mCurrentView->onTransition(); if (Settings::getInstance()->getString("TransitionStyle") == "slide") slideTransitions = true; // Restore the X position for the view, if it was previously moved. if (mWrappedViews) restoreViewPosition(); if (mPreviousView && Settings::getInstance()->getString("TransitionStyle") == "fade" && isAnimationPlaying(0)) { mPreviousView->onHide(); } if (mPreviousView) { mSkipView = mPreviousView; mPreviousView.reset(); mPreviousView = nullptr; } else if (!mPreviousView && mState.viewing == GAMELIST) { // This is needed as otherwise the static image would not get rendered during the // first Slide transition when coming from the System view. mSkipView = getGamelistView(system); } if (mState.viewing != SYSTEM_SELECT) { mPreviousView = mCurrentView; mSystemViewTransition = false; } else { mSystemViewTransition = true; } // Find if we're wrapping around the first and last systems, which requires the gamelist // to be moved in order to avoid weird camera movements. This is only needed for the // slide transition style. if (mState.viewing == GAMELIST && SystemData::sSystemVector.size() > 1 && slideTransitions) { if (SystemData::sSystemVector.front() == mState.getSystem()) { if (SystemData::sSystemVector.back() == system) wrapFirstToLast = true; } else if (SystemData::sSystemVector.back() == mState.getSystem()) { if (SystemData::sSystemVector.front() == system) wrapLastToFirst = true; } } // Stop any scrolling, animations and camera movements. if (mState.viewing == SYSTEM_SELECT) { mSystemListView->stopScrolling(); if (mSystemListView->isSystemAnimationPlaying(0)) mSystemListView->finishSystemAnimation(0); } if (slideTransitions) cancelViewTransitions(); if (mState.viewing == SYSTEM_SELECT) { // Move the system list. auto sysList = getSystemListView(); float offsetX = sysList->getPosition().x; int sysId = getSystemId(system); sysList->setPosition(sysId * Renderer::getScreenWidth(), sysList->getPosition().y); offsetX = sysList->getPosition().x - offsetX; mCamera[3].x -= offsetX; } // If we are wrapping around, either from the first to last system, or the other way // around, we need to temporarily move the gamelist view location so that the camera // movements will be correct. This is accomplished by simply offsetting the X position // with the position of the first or last system plus the screen width. if (wrapFirstToLast) { glm::vec3 currentPosition {mCurrentView->getPosition()}; mWrapPreviousPositionX = currentPosition.x; float offsetX {getGamelistView(system)->getPosition().x}; // This is needed to move the camera in the correct direction if there are only two systems. if (SystemData::sSystemVector.size() == 2 && mNextSystem) offsetX -= Renderer::getScreenWidth(); else offsetX += Renderer::getScreenWidth(); currentPosition.x = offsetX; mCurrentView->setPosition(currentPosition); mCamera[3].x -= offsetX; mWrappedViews = true; } else if (wrapLastToFirst) { glm::vec3 currentPosition {mCurrentView->getPosition()}; mWrapPreviousPositionX = currentPosition.x; float offsetX {getGamelistView(system)->getPosition().x}; if (SystemData::sSystemVector.size() == 2 && !mNextSystem) offsetX += Renderer::getScreenWidth(); else offsetX -= Renderer::getScreenWidth(); currentPosition.x = offsetX; mCurrentView->setPosition(currentPosition); mCamera[3].x = -offsetX; mWrappedViews = true; } mCurrentView = getGamelistView(system); mCurrentView->finishAnimation(0); // Application startup animation, if starting in a gamelist rather than in the system view. if (mState.viewing == NOTHING) { mCamera = glm::translate(mCamera, -mCurrentView->getPosition()); if (Settings::getInstance()->getString("TransitionStyle") == "slide") { mCamera[3].y -= Renderer::getScreenHeight(); updateHelpPrompts(); } else if (Settings::getInstance()->getString("TransitionStyle") == "fade") { mCamera[3].y += Renderer::getScreenHeight() * 2.0f; } else { updateHelpPrompts(); } } mState.viewing = GAMELIST; mState.system = system; auto it = mGamelistViews.find(system); if (it != mGamelistViews.cend()) { std::string viewStyle = it->second->getName(); if (viewStyle == "basic") mState.viewstyle = BASIC; else if (viewStyle == "detailed") mState.viewstyle = DETAILED; else if (viewStyle == "video") mState.viewstyle = VIDEO; } if (mCurrentView) mCurrentView->onShow(); playViewTransition(); } void ViewController::playViewTransition(bool instant) { mCancelledTransition = false; glm::vec3 target {}; if (mCurrentView) target = mCurrentView->getPosition(); // No need to animate, we're not going anywhere (probably due to goToNextGamelist() // or goToPrevGamelist() being called when there's only 1 system). if (target == static_cast(-mCamera[3]) && !isAnimationPlaying(0)) return; std::string transition_style {Settings::getInstance()->getString("TransitionStyle")}; if (instant || transition_style == "instant") { setAnimation(new LambdaAnimation( [this, target](float /*t*/) { this->mCamera[3].x = -target.x; this->mCamera[3].y = -target.y; this->mCamera[3].z = -target.z; if (mPreviousView) mPreviousView->onHide(); }, 1)); updateHelpPrompts(); } else if (transition_style == "fade") { // Stop whatever's currently playing, leaving mFadeOpacity wherever it is. cancelAnimation(0); auto fadeFunc = [this](float t) { // The flag mCancelledTransition is required only when cancelViewTransitions() // cancels the animation, and it's only needed for the Fade transitions. // Without this, a (much shorter) fade transition would still play as // finishedCallback is calling this function. if (!mCancelledTransition) mFadeOpacity = glm::mix(0.0f, 1.0f, t); }; auto fadeCallback = [this]() { if (mPreviousView) mPreviousView->onHide(); }; const static int FADE_DURATION = 120; // Fade in/out time. const static int FADE_WAIT = 200; // Time to wait between in/out. setAnimation(new LambdaAnimation(fadeFunc, FADE_DURATION), 0, [this, fadeFunc, fadeCallback, target] { this->mCamera[3].x = -target.x; this->mCamera[3].y = -target.y; this->mCamera[3].z = -target.z; updateHelpPrompts(); setAnimation(new LambdaAnimation(fadeFunc, FADE_DURATION), FADE_WAIT, fadeCallback, true); }); // Fast-forward animation if we're partially faded. if (target == static_cast(-mCamera[3])) { // Not changing screens, so cancel the first half entirely. advanceAnimation(0, FADE_DURATION); advanceAnimation(0, FADE_WAIT); advanceAnimation(0, FADE_DURATION - static_cast(mFadeOpacity * FADE_DURATION)); } else { advanceAnimation(0, static_cast(mFadeOpacity * FADE_DURATION)); } } else if (transition_style == "slide") { auto slideCallback = [this]() { if (mSkipView) { mSkipView->onHide(); mSkipView.reset(); mSkipView = nullptr; } else if (mPreviousView) { mPreviousView->onHide(); } }; setAnimation(new MoveCameraAnimation(mCamera, target), 0, slideCallback); updateHelpPrompts(); // Update help prompts immediately. } } void ViewController::onFileChanged(FileData* file, bool reloadGamelist) { auto it = mGamelistViews.find(file->getSystem()); if (it != mGamelistViews.cend()) it->second->onFileChanged(file, reloadGamelist); } void ViewController::launch(FileData* game) { if (game->getType() != GAME) { LOG(LogError) << "Tried to launch something that isn't a game"; return; } // Disable text scrolling and stop any Lottie animations. These will be enabled again in // FileData upon returning from the game. mWindow->setAllowTextScrolling(false); mWindow->setAllowFileAnimation(false); stopAnimation(1); // Make sure the fade in isn't still playing. mWindow->stopInfoPopup(); // Make sure we disable any existing info popup. int duration = 0; std::string durationString = Settings::getInstance()->getString("LaunchScreenDuration"); if (durationString == "disabled") { // If the game launch screen has been set as disabled, show a simple info popup // notification instead. mWindow->queueInfoPopup( "LAUNCHING GAME '" + Utils::String::toUpper(game->metadata.get("name") + "'"), 10000); duration = 1700; } else if (durationString == "brief") { duration = 1700; } else if (durationString == "long") { duration = 4500; } else { // Normal duration. duration = 3000; } if (durationString != "disabled") mWindow->displayLaunchScreen(game->getSourceFileData()); NavigationSounds::getInstance().playThemeNavigationSound(LAUNCHSOUND); // This is just a dummy animation in order for the launch screen or notification popup // to be displayed briefly, and for the navigation sound playing to be able to complete. // During this time period, all user input is blocked. setAnimation(new LambdaAnimation([](float t) {}, duration), 0, [this, game] { game->launchGame(); // If the launch screen is disabled then this will do nothing. mWindow->closeLaunchScreen(); onFileChanged(game, true); // This is a workaround so that any keys or button presses used for exiting the emulator // are not captured upon returning. setAnimation(new LambdaAnimation([](float t) {}, 1), 0, [this] { mLockInput = false; }); }); } void ViewController::removeGamelistView(SystemData* system) { auto exists = mGamelistViews.find(system); if (exists != mGamelistViews.cend()) { exists->second.reset(); mGamelistViews.erase(system); } } std::shared_ptr ViewController::getGamelistView(SystemData* system) { // If we have already created an entry for this system, then return that one. auto exists = mGamelistViews.find(system); if (exists != mGamelistViews.cend()) return exists->second; system->getIndex()->setKidModeFilters(); // If there's no entry, then create it and return it. std::shared_ptr view; bool themeHasVideoView {system->getTheme()->hasView("video")}; // Decide which view style to use. GamelistViewStyle selectedViewStyle = AUTOMATIC; std::string viewPreference {Settings::getInstance()->getString("GamelistViewStyle")}; if (viewPreference == "basic") selectedViewStyle = BASIC; else if (viewPreference == "detailed") selectedViewStyle = DETAILED; else if (viewPreference == "video") selectedViewStyle = VIDEO; if (selectedViewStyle == AUTOMATIC) { std::vector files {system->getRootFolder()->getFilesRecursive(GAME | FOLDER)}; for (auto it = files.cbegin(); it != files.cend(); ++it) { if (themeHasVideoView && !(*it)->getVideoPath().empty()) { selectedViewStyle = VIDEO; break; } else if (!(*it)->getImagePath().empty()) { selectedViewStyle = DETAILED; // Don't break out in case any subsequent files have videos. } } } // Create the view. switch (selectedViewStyle) { case VIDEO: { mState.viewstyle = VIDEO; break; } case DETAILED: { mState.viewstyle = DETAILED; break; } case BASIC: { } default: { mState.viewstyle = BASIC; break; } } view = std::shared_ptr(new GamelistView(system->getRootFolder())); view->setTheme(system->getTheme()); std::vector& sysVec = SystemData::sSystemVector; int id {static_cast(std::find(sysVec.cbegin(), sysVec.cend(), system) - sysVec.cbegin())}; view->setPosition(id * Renderer::getScreenWidth(), Renderer::getScreenHeight() * 2.0f); addChild(view.get()); mGamelistViews[system] = view; return view; } std::shared_ptr ViewController::getSystemListView() { // If we have already created a system view entry, then return it. if (mSystemListView) return mSystemListView; mSystemListView = std::shared_ptr(new SystemView); addChild(mSystemListView.get()); mSystemListView->setPosition(0, Renderer::getScreenHeight()); return mSystemListView; } bool ViewController::input(InputConfig* config, Input input) { if (mLockInput) return true; // If using the %RUNINBACKGROUND% variable in a launch command or if enabling the // RunInBackground setting, ES-DE will run in the background while a game is launched. // If we're in this state and then register some input, it means that the user is back in ES-DE. // Therefore unset the game launch flag and update all the GUI components. This will re-enable // the video player and scrolling of game names and game descriptions as well as letting the // screensaver start on schedule. if (mWindow->getGameLaunchedState()) { mWindow->setAllowTextScrolling(true); mWindow->setAllowFileAnimation(true); mWindow->setLaunchedGame(false); // Filter out the "a" button so the game is not restarted if there was such a button press // queued when leaving the game. if (config->isMappedTo("a", input) && input.value != 0) return true; } // Open the main menu. if (!(UIModeController::getInstance()->isUIModeKid() && !Settings::getInstance()->getBool("EnableMenuKidMode")) && config->isMappedTo("start", input) && input.value != 0) { // If we don't stop the scrolling here, it will continue to // run after closing the menu. if (mSystemListView->isScrolling()) mSystemListView->stopScrolling(); // Finish the animation too, so that it doesn't continue // to play when we've closed the menu. if (mSystemListView->isSystemAnimationPlaying(0)) mSystemListView->finishSystemAnimation(0); // Stop the gamelist scrolling as well as it would otherwise continue to run after // closing the menu. mCurrentView->stopListScrolling(); // Pause all videos as they would otherwise continue to play beneath the menu. mCurrentView->pauseViewVideos(); mCurrentView->stopScrollFadeIn(); // Finally, if the camera is currently moving, reset its position. cancelViewTransitions(); mWindow->pushGui(new GuiMenu); return true; } // Check if UI mode has changed due to passphrase completion. if (UIModeController::getInstance()->listen(config, input)) return true; if (mCurrentView) return mCurrentView->input(config, input); return false; } void ViewController::update(int deltaTime) { if (mWindow->getChangedThemeSet()) cancelViewTransitions(); if (mCurrentView) mCurrentView->update(deltaTime); updateSelf(deltaTime); if (mGameToLaunch) { launch(mGameToLaunch); mGameToLaunch = nullptr; } } void ViewController::render(const glm::mat4& parentTrans) { glm::mat4 trans {mCamera * parentTrans}; glm::mat4 transInverse {glm::inverse(trans)}; // Camera position, position + size. glm::vec3 viewStart {transInverse[3]}; glm::vec3 viewEnd {std::fabs(trans[3].x) + Renderer::getScreenWidth(), std::fabs(trans[3].y) + Renderer::getScreenHeight(), 0.0f}; // Keep track of UI mode changes. UIModeController::getInstance()->monitorUIMode(); // Render the system view if it's the currently displayed view, or if we're in the progress // of transitioning to or from this view. if (mSystemListView == mCurrentView || (mSystemViewTransition && isCameraMoving())) getSystemListView()->render(trans); // Draw the gamelists. for (auto it = mGamelistViews.cbegin(); it != mGamelistViews.cend(); ++it) { // Same thing as for the system view, limit the rendering only to what needs to be drawn. if (it->second == mCurrentView || (it->second == mPreviousView && isCameraMoving())) { // Clipping. glm::vec3 guiStart {it->second->getPosition()}; glm::vec3 guiEnd {it->second->getPosition() + glm::vec3 {it->second->getSize().x, it->second->getSize().y, 0.0f}}; if (guiEnd.x >= viewStart.x && guiEnd.y >= viewStart.y && guiStart.x <= viewEnd.x && guiStart.y <= viewEnd.y) it->second->render(trans); } } if (mWindow->peekGui() == this) mWindow->renderHelpPromptsEarly(); // Fade out. if (mFadeOpacity) { unsigned int fadeColor = 0x00000000 | static_cast(mFadeOpacity * 255.0f); mRenderer->setMatrix(parentTrans); mRenderer->drawRect(0.0f, 0.0f, Renderer::getScreenWidth(), Renderer::getScreenHeight(), fadeColor, fadeColor); } } void ViewController::preload() { unsigned int systemCount = static_cast(SystemData::sSystemVector.size()); // This reduces the amount of texture pop-in when loading theme extras. if (!SystemData::sSystemVector.empty()) getSystemListView(); for (auto it = SystemData::sSystemVector.cbegin(); it != SystemData::sSystemVector.cend(); ++it) { if (Settings::getInstance()->getBool("SplashScreen") && Settings::getInstance()->getBool("SplashScreenProgress")) { mWindow->renderLoadingScreen( "Loading '" + (*it)->getFullName() + "' (" + std::to_string(std::distance(SystemData::sSystemVector.cbegin(), it) + 1) + "/" + std::to_string(systemCount) + ")"); } (*it)->getIndex()->resetFilters(); if (Settings::getInstance()->getBool("PreloadGamelists")) getGamelistView(*it)->preloadGamelist(); else getGamelistView(*it); } // Load navigation sounds, either from the theme if it supports it, or otherwise from // the bundled fallback sound files. bool themeSoundSupport = false; for (SystemData* system : SystemData::sSystemVector) { if (system->getTheme()->hasView("all")) { NavigationSounds::getInstance().loadThemeNavigationSounds(system->getTheme().get()); themeSoundSupport = true; break; } } if (!SystemData::sSystemVector.empty() && !themeSoundSupport) NavigationSounds::getInstance().loadThemeNavigationSounds(nullptr); } void ViewController::reloadGamelistView(GamelistView* view, bool reloadTheme) { for (auto it = mGamelistViews.cbegin(); it != mGamelistViews.cend(); ++it) { if (it->second.get() == view) { bool isCurrent {(mCurrentView == it->second)}; SystemData* system {it->first}; FileData* cursor {view->getCursor()}; // Retain the cursor history for the view. std::vector cursorHistoryTemp; it->second->copyCursorHistory(cursorHistoryTemp); mGamelistViews.erase(it); if (isCurrent) mCurrentView = nullptr; if (reloadTheme) system->loadTheme(); system->getIndex()->setKidModeFilters(); std::shared_ptr newView = getGamelistView(system); // To counter having come from a placeholder. if (!cursor->isPlaceHolder()) { newView->setCursor(cursor); } if (isCurrent) mCurrentView = newView; newView->populateCursorHistory(cursorHistoryTemp); updateHelpPrompts(); break; } } // If using the %RUNINBACKGROUND% variable in a launch command or if enabling the // RunInBackground setting, ES-DE will run in the background while a game is launched. // If this flag has been set, then update all the GUI components. This will block the // video player, prevent scrolling of game names and game descriptions and prevent the // screensaver from starting on schedule. if (mWindow->getGameLaunchedState()) mWindow->setLaunchedGame(true); // Redisplay the current view. if (mCurrentView) mCurrentView->onShow(); } void ViewController::reloadAll() { if (mRenderer->getSDLWindow() == nullptr) return; cancelViewTransitions(); // Clear all GamelistViews. std::map cursorMap; for (auto it = mGamelistViews.cbegin(); it != mGamelistViews.cend(); ++it) cursorMap[it->first] = it->second->getCursor(); mGamelistViews.clear(); mCurrentView = nullptr; // Load themes, create GamelistViews and reset filters. for (auto it = cursorMap.cbegin(); it != cursorMap.cend(); ++it) { it->first->loadTheme(); it->first->getIndex()->resetFilters(); getGamelistView(it->first)->setCursor(it->second); } // Rebuild SystemListView. mSystemListView.reset(); getSystemListView(); // Update mCurrentView since the pointers changed. if (mState.viewing == GAMELIST) { mCurrentView = getGamelistView(mState.getSystem()); } else if (mState.viewing == SYSTEM_SELECT) { SystemData* system = mState.getSystem(); mSystemListView->goToSystem(system, false); mCurrentView = mSystemListView; mCamera[3].x = 0.0f; } else { goToSystemView(SystemData::sSystemVector.front(), false); } // Load navigation sounds, either from the theme if it supports it, or otherwise from // the bundled fallback sound files. NavigationSounds::getInstance().deinit(); bool themeSoundSupport = false; for (SystemData* system : SystemData::sSystemVector) { if (system->getTheme()->hasView("all")) { NavigationSounds::getInstance().loadThemeNavigationSounds(system->getTheme().get()); themeSoundSupport = true; break; } } if (!SystemData::sSystemVector.empty() && !themeSoundSupport) NavigationSounds::getInstance().loadThemeNavigationSounds(nullptr); mCurrentView->onShow(); updateHelpPrompts(); } std::vector ViewController::getHelpPrompts() { std::vector prompts; if (!mCurrentView) return prompts; prompts = mCurrentView->getHelpPrompts(); if (!(UIModeController::getInstance()->isUIModeKid() && !Settings::getInstance()->getBool("EnableMenuKidMode"))) prompts.push_back(HelpPrompt("start", "menu")); return prompts; } HelpStyle ViewController::getHelpStyle() { if (!mCurrentView) return GuiComponent::getHelpStyle(); return mCurrentView->getHelpStyle(); } HelpStyle ViewController::getViewHelpStyle() { if (mState.getSystem()->getTheme()->isLegacyTheme()) { // For backward compatibility with legacy theme sets, read the helpsystem theme config // from the system view entry. return getSystemListView()->getHelpStyle(); } else { if (mState.viewing == ViewMode::GAMELIST) return getGamelistView(mState.getSystem())->getHelpStyle(); else return getSystemListView()->getHelpStyle(); } }