// SPDX-License-Identifier: MIT // // EmulationStation Desktop Edition // GuiMetaDataEd.cpp // // Game metadata edit user interface. // This interface is triggered from the GuiGamelistOptions menu. // The scraping interface is handled by GuiGameScraper which calls // GuiScraperSearch. // #include "guis/GuiMetaDataEd.h" #include "components/ButtonComponent.h" #include "components/ComponentList.h" #include "components/DateTimeEditComponent.h" #include "components/MenuComponent.h" #include "components/RatingComponent.h" #include "components/SwitchComponent.h" #include "components/TextComponent.h" #include "guis/GuiGameScraper.h" #include "guis/GuiMsgBox.h" #include "guis/GuiTextEditPopup.h" #include "guis/GuiComplexTextEditPopup.h" #include "resources/Font.h" #include "utils/StringUtil.h" #include "views/ViewController.h" #include "CollectionSystemsManager.h" #include "FileData.h" #include "FileFilterIndex.h" #include "Gamelist.h" #include "SystemData.h" #include "Window.h" GuiMetaDataEd::GuiMetaDataEd( Window* window, MetaDataList* md, const std::vector& mdd, ScraperSearchParams scraperParams, const std::string& /*header*/, std::function saveCallback, std::function clearGameFunc, std::function deleteGameFunc) : GuiComponent(window), mScraperParams(scraperParams), mBackground(window, ":/graphics/frame.svg"), mGrid(window, Vector2i(1, 3)), mMetaDataDecl(mdd), mMetaData(md), mSavedCallback(saveCallback), mClearGameFunc(clearGameFunc), mDeleteGameFunc(deleteGameFunc), mMediaFilesUpdated(false) { addChild(&mBackground); addChild(&mGrid); mHeaderGrid = std::make_shared(mWindow, Vector2i(1, 5)); mTitle = std::make_shared(mWindow, "EDIT METADATA", Font::get(FONT_SIZE_LARGE), 0x555555FF, ALIGN_CENTER); // Extract possible subfolders from the path. std::string folderPath = Utils::String::replace( Utils::FileSystem::getParent(scraperParams.game->getPath()), scraperParams.system->getSystemEnvData()->mStartPath, ""); if (folderPath.size() >= 2) { folderPath.erase(0, 1); #if defined(_WIN64) folderPath.push_back('\\'); folderPath = Utils::String::replace(folderPath, "/", "\\"); #else folderPath.push_back('/'); #endif } mSubtitle = std::make_shared(mWindow, folderPath + Utils::FileSystem::getFileName(scraperParams.game->getPath()) + " [" + Utils::String::toUpper(scraperParams.system->getName()) + "]" + (scraperParams.game->getType() == FOLDER ? " " + ViewController::FOLDER_CHAR : ""), Font::get(FONT_SIZE_SMALL), 0x777777FF, ALIGN_CENTER, Vector3f(0.0f, 0.0f, 0.0f), Vector2f(0.0f, 0.0f), 0x00000000, 0.05f); mHeaderGrid->setEntry(mTitle, Vector2i(0, 1), false, true); mHeaderGrid->setEntry(mSubtitle, Vector2i(0, 3), false, true); mGrid.setEntry(mHeaderGrid, Vector2i(0, 0), false, true); mList = std::make_shared(mWindow); mGrid.setEntry(mList, Vector2i(0, 1), true, true); // Populate list. for (auto iter = mdd.cbegin(); iter != mdd.cend(); iter++) { std::shared_ptr ed; std::string currentKey = iter->key; std::string originalValue = mMetaData->get(iter->key); std::string gamePath; // Don't add statistics. if (iter->isStatistic) continue; // Don't show the launch command override entry if this option has been disabled. if (!Settings::getInstance()->getBool("LaunchCommandOverride") && iter->type == MD_LAUNCHCOMMAND) { ed = std::make_shared(window, "", Font::get(FONT_SIZE_SMALL, FONT_PATH_LIGHT), 0x777777FF, ALIGN_RIGHT); assert(ed); ed->setValue(mMetaData->get(iter->key)); mEditors.push_back(ed); continue; } // Create ed and add it (and any related components) to mMenu. // ed's value will be set below. // It's very important to put the element with the help prompt as the last row // entry instead of for instance the spacer. That is so because ComponentList // always looks for the help prompt at the back of the element stack. ComponentListRow row; auto lbl = std::make_shared(mWindow, Utils::String::toUpper(iter->displayName), Font::get(FONT_SIZE_SMALL), 0x777777FF); row.addElement(lbl, true); // Label. switch (iter->type) { case MD_BOOL: { ed = std::make_shared(window); // Make the switches slightly smaller. auto switchSize = ed->getSize() * 0.9f; ed->setResize(switchSize.x(), switchSize.y()); ed->setOrigin(-0.05f, -0.09f); ed->setChangedColor(ICONCOLOR_USERMARKED); row.addElement(ed, false, true); break; } case MD_RATING: { auto spacer = std::make_shared(mWindow); spacer->setSize(Renderer::getScreenWidth() * 0.0025f, 0); row.addElement(spacer, false); ed = std::make_shared(window, true); ed->setChangedColor(ICONCOLOR_USERMARKED); const float height = lbl->getSize().y() * 0.71f; ed->setSize(0, height); row.addElement(ed, false, true); // Pass input to the actual RatingComponent instead of the spacer. row.input_handler = std::bind(&GuiComponent::input, ed.get(), std::placeholders::_1, std::placeholders::_2); break; } case MD_DATE: { auto spacer = std::make_shared(mWindow); spacer->setSize(Renderer::getScreenWidth() * 0.0025f, 0); row.addElement(spacer, false); ed = std::make_shared(window); ed->setOriginalColor(DEFAULT_TEXTCOLOR); ed->setChangedColor(TEXTCOLOR_USERMARKED); row.addElement(ed, false); // Pass input to the actual DateTimeEditComponent instead of the spacer. row.input_handler = std::bind(&GuiComponent::input, ed.get(), std::placeholders::_1, std::placeholders::_2); break; } // Not in use as 'lastplayed' is flagged as statistics and these are skipped. // Let's still keep the code because it may be needed in the future. // case MD_TIME: { // ed = std::make_shared(window, // DateTimeEditComponent::DISP_RELATIVE_TO_NOW); // row.addElement(ed, false); // break; // } case MD_LAUNCHCOMMAND: { ed = std::make_shared(window, "", Font::get(FONT_SIZE_SMALL, FONT_PATH_LIGHT), 0x777777FF, ALIGN_RIGHT); row.addElement(ed, true); auto spacer = std::make_shared(mWindow); spacer->setSize(Renderer::getScreenWidth() * 0.005f, 0); row.addElement(spacer, false); auto bracket = std::make_shared(mWindow); bracket->setImage(":/graphics/arrow.svg"); bracket->setResize(Vector2f(0, lbl->getFont()->getLetterHeight())); row.addElement(bracket, false); bool multiLine = false; const std::string title = iter->displayPrompt; // OK callback (apply new value to ed). auto updateVal = [ed, originalValue](const std::string& newVal) { ed->setValue(newVal); if (newVal == originalValue) ed->setColor(DEFAULT_TEXTCOLOR); else ed->setColor(TEXTCOLOR_USERMARKED); }; std::string staticTextString = "Default value from es_systems.cfg:"; std::string defaultLaunchCommand = scraperParams.system-> getSystemEnvData()->mLaunchCommand; row.makeAcceptInputHandler([this, title, staticTextString, defaultLaunchCommand, ed, updateVal, multiLine] { mWindow->pushGui(new GuiComplexTextEditPopup(mWindow, getHelpStyle(), title, staticTextString, defaultLaunchCommand, ed->getValue(), updateVal, multiLine, "APPLY", "APPLY CHANGES?")); }); break; } case MD_MULTILINE_STRING: default: { // MD_STRING. ed = std::make_shared(window, "", Font::get(FONT_SIZE_SMALL, FONT_PATH_LIGHT), 0x777777FF, ALIGN_RIGHT); row.addElement(ed, true); auto spacer = std::make_shared(mWindow); spacer->setSize(Renderer::getScreenWidth() * 0.005f, 0); row.addElement(spacer, false); auto bracket = std::make_shared(mWindow); bracket->setImage(":/graphics/arrow.svg"); bracket->setResize(Vector2f(0, lbl->getFont()->getLetterHeight())); row.addElement(bracket, false); bool multiLine = iter->type == MD_MULTILINE_STRING; const std::string title = iter->displayPrompt; gamePath = Utils::FileSystem::getStem(mScraperParams.game->getPath()); // OK callback (apply new value to ed). auto updateVal = [ed, currentKey, originalValue, gamePath] (const std::string& newVal) { // If the user has entered a blank game name, then set the name to the ROM // filename (minus the extension). if (currentKey == "name" && newVal == "") { ed->setValue(gamePath); if (gamePath == originalValue) ed->setColor(DEFAULT_TEXTCOLOR); else ed->setColor(TEXTCOLOR_USERMARKED); } else if (newVal == "" && (currentKey == "developer" || currentKey == "publisher" || currentKey == "genre" || currentKey == "players")) { ed->setValue("unknown"); if (originalValue == "unknown") ed->setColor(DEFAULT_TEXTCOLOR); else ed->setColor(TEXTCOLOR_USERMARKED); } else { ed->setValue(newVal); if (newVal == originalValue) ed->setColor(DEFAULT_TEXTCOLOR); else ed->setColor(TEXTCOLOR_USERMARKED); } }; row.makeAcceptInputHandler([this, title, ed, updateVal, multiLine] { mWindow->pushGui(new GuiTextEditPopup(mWindow, getHelpStyle(), title, ed->getValue(), updateVal, multiLine, "APPLY", "APPLY CHANGES?")); }); break; } } assert(ed); mList->addRow(row); ed->setValue(mMetaData->get(iter->key)); mEditors.push_back(ed); } std::vector> buttons; if (!scraperParams.system->hasPlatformId(PlatformIds::PLATFORM_IGNORE)) buttons.push_back(std::make_shared(mWindow, "SCRAPE", "scrape", std::bind(&GuiMetaDataEd::fetch, this))); buttons.push_back(std::make_shared(mWindow, "SAVE", "save metadata", [&] { save(); ViewController::get()->onPauseVideo(); delete this; })); buttons.push_back(std::make_shared(mWindow, "CANCEL", "cancel changes", [&] { delete this; })); if (scraperParams.game->getType() == FOLDER) { if (mClearGameFunc) { auto clearSelf = [&] { mClearGameFunc(); delete this; }; auto clearSelfBtnFunc = [this, clearSelf] { mWindow->pushGui(new GuiMsgBox(mWindow, getHelpStyle(), "THIS WILL DELETE ANY MEDIA FILES AND\n" "THE GAMELIST.XML ENTRY FOR THIS FOLDER,\n" "BUT NEITHER THE FOLDER ITSELF OR ANY\n" "CONTENT INSIDE IT WILL BE REMOVED\n" "ARE YOU SURE?", "YES", clearSelf, "NO", nullptr)); }; buttons.push_back(std::make_shared(mWindow, "CLEAR", "clear folder", clearSelfBtnFunc)); } } else { if (mClearGameFunc) { auto clearSelf = [&] { mClearGameFunc(); delete this; }; auto clearSelfBtnFunc = [this, clearSelf] { mWindow->pushGui(new GuiMsgBox(mWindow, getHelpStyle(), "THIS WILL DELETE ANY MEDIA FILES\n" "AND THE GAMELIST.XML ENTRY FOR\n" "THIS GAME, BUT THE GAME FILE\n" "ITSELF WILL NOT BE REMOVED\n" "ARE YOU SURE?", "YES", clearSelf, "NO", nullptr)); }; buttons.push_back(std::make_shared(mWindow, "CLEAR", "clear file", clearSelfBtnFunc)); } if (mDeleteGameFunc) { auto deleteFilesAndSelf = [&] { mDeleteGameFunc(); delete this; }; auto deleteGameBtnFunc = [this, deleteFilesAndSelf] { mWindow->pushGui(new GuiMsgBox(mWindow, getHelpStyle(), "THIS WILL DELETE THE GAME\n" "FILE, ANY MEDIA FILES AND\n" "THE GAMELIST.XML ENTRY\n" "ARE YOU SURE?", "YES", deleteFilesAndSelf, "NO", nullptr)); }; buttons.push_back(std::make_shared(mWindow, "DELETE", "delete game", deleteGameBtnFunc)); } } mButtons = makeButtonGrid(mWindow, buttons); mGrid.setEntry(mButtons, Vector2i(0, 2), true, false); // Resize + center. float width = static_cast(std::min(static_cast(Renderer::getScreenHeight() * 1.05f), static_cast(Renderer::getScreenWidth() * 0.90f))); setSize(width, Renderer::getScreenHeight() * 0.83f); setPosition((Renderer::getScreenWidth() - mSize.x()) / 2, (Renderer::getScreenHeight() - mSize.y()) / 2); } void GuiMetaDataEd::onSizeChanged() { mGrid.setSize(mSize); const float titleHeight = mTitle->getFont()->getLetterHeight(); const float subtitleHeight = mSubtitle->getFont()->getLetterHeight(); const float titleSubtitleSpacing = mSize.y() * 0.03f; mGrid.setRowHeightPerc(0, (titleHeight + titleSubtitleSpacing + subtitleHeight + TITLE_VERT_PADDING) / mSize.y()); mGrid.setRowHeightPerc(2, mButtons->getSize().y() / mSize.y()); // Clamp list size to the row height to prevent a fraction of a row from being displayed. float listHeight = 0; float listSize = mList->getSize().y(); int i = 0; while (i < mList->size()) { // Add the separator height to the row height so that it also gets properly rendered. float rowHeight = mList->getRowHeight(i) + (1 * Renderer::getScreenHeightModifier()); if (listHeight + rowHeight < listSize) listHeight += rowHeight; else break; i++; } // Adjust the size of the list and window. float heightAdjustment = listSize - listHeight; mList->setSize(mList->getSize().x(), listHeight); Vector2f newWindowSize = mSize; newWindowSize.y() -= heightAdjustment; mBackground.fitTo(newWindowSize, Vector3f::Zero(), Vector2f(-32, -32)); // Move the buttons up as well to make the layout align correctly after the resize. Vector3f newButtonPos = mButtons->getPosition(); newButtonPos.y() -= heightAdjustment; mButtons->setPosition(newButtonPos); mHeaderGrid->setRowHeightPerc(1, titleHeight / mHeaderGrid->getSize().y()); mHeaderGrid->setRowHeightPerc(2, titleSubtitleSpacing / mHeaderGrid->getSize().y()); mHeaderGrid->setRowHeightPerc(3, subtitleHeight / mHeaderGrid->getSize().y()); } void GuiMetaDataEd::save() { // Remove game from index. mScraperParams.system->getIndex()->removeFromIndex(mScraperParams.game); // We need this to handle the special situation where the user sets a game to hidden while // ShowHiddenGames is set to false, meaning it will immediately disappear from the gamelist. bool showHiddenGames = Settings::getInstance()->getBool("ShowHiddenGames"); bool hideGameWhileHidden = false; bool setGameAsCounted = false; for (unsigned int i = 0; i < mEditors.size(); i++) { if (mMetaDataDecl.at(i).isStatistic) continue; if (!showHiddenGames && mMetaDataDecl.at(i).key == "hidden" && mEditors.at(i)->getValue() != mMetaData->get("hidden")) hideGameWhileHidden = true; // Check whether the flag to count the entry as a game was set to enabled. if (mMetaDataDecl.at(i).key == "nogamecount" && mEditors.at(i)->getValue() != mMetaData->get("nogamecount") && mMetaData->get("nogamecount") == "true") { setGameAsCounted = true; } mMetaData->set(mMetaDataDecl.at(i).key, mEditors.at(i)->getValue()); } // If hidden games are not shown and the hide flag was set for the entry, then write the // metadata immediately regardless of the SaveGamelistsMode setting. Otherwise the file // will never be written as the game will be filtered from the gamelist. This solution is not // really good as the gamelist will be written twice, but it's a very special and hopefully // rare situation. if (hideGameWhileHidden) updateGamelist(mScraperParams.system); // Enter game in index. mScraperParams.system->getIndex()->addToIndex(mScraperParams.game); // If it's a folder that has been updated, we need to manually sort the gamelist // as CollectionSystemsManager ignores folders. if (mScraperParams.game->getType() == FOLDER) mScraperParams.system->sortSystem(false); if (mSavedCallback) mSavedCallback(); // Update all collections where the game is present. if (mScraperParams.game->getType() == GAME) CollectionSystemsManager::get()->refreshCollectionSystems(mScraperParams.game); // If game counting was re-enabled for the game, then reactivate it in any custom collections // where it may exist. if (setGameAsCounted) CollectionSystemsManager::get()->reactivateCustomCollectionEntry(mScraperParams.game); mScraperParams.system->onMetaDataSavePoint(); // If hidden games are not shown and the hide flag was set for the entry, we also need // to re-sort the gamelist as we may need to remove the parent folder if all the entries // within this folder are now hidden. if (hideGameWhileHidden) mScraperParams.system->sortSystem(true); // Make sure that the cached background is updated to reflect any possible visible // changes to the gamelist (e.g. the game name). mWindow->invalidateCachedBackground(); } void GuiMetaDataEd::fetch() { GuiGameScraper* scr = new GuiGameScraper(mWindow, mScraperParams, std::bind(&GuiMetaDataEd::fetchDone, this, std::placeholders::_1)); mWindow->pushGui(scr); } void GuiMetaDataEd::fetchDone(const ScraperSearchResult& result) { // Clone the mMetaData object. MetaDataList* metadata = nullptr; metadata = new MetaDataList(*mMetaData); mMediaFilesUpdated = result.savedNewMedia; // Curently disabled as I'm not sure if this is more annoying than helpful. // // Select the Save button. // mButtons->moveCursor(Vector2i(1, 0)); // Check if any values were manually changed before starting the scraping. // If so, it's these values we should compare against when scraping, not // the values previously saved for the game. for (unsigned int i = 0; i < mEditors.size(); i++) { const std::string& key = mMetaDataDecl.at(i).key; if (metadata->get(key) != mEditors[i]->getValue()) metadata->set(key, mEditors[i]->getValue()); } GuiScraperSearch::saveMetadata(result, *metadata, mScraperParams.game); // Update the list with the scraped metadata values. for (unsigned int i = 0; i < mEditors.size(); i++) { const std::string& key = mMetaDataDecl.at(i).key; if (mEditors.at(i)->getValue() != metadata->get(key)) { if (key == "rating") mEditors.at(i)->setOriginalColor(ICONCOLOR_SCRAPERMARKED); else mEditors.at(i)->setColor(TEXTCOLOR_SCRAPERMARKED); } // Save all the keys that should be scraped. if (mMetaDataDecl.at(i).shouldScrape) mEditors.at(i)->setValue(metadata->get(key)); } delete metadata; } void GuiMetaDataEd::close() { // Find out if the user made any changes. bool metadataUpdated = false; for (unsigned int i = 0; i < mEditors.size(); i++) { const std::string& key = mMetaDataDecl.at(i).key; std::string mMetaDataValue = mMetaData->get(key); std::string mEditorsValue = mEditors.at(i)->getValue(); if (mMetaDataValue != mEditorsValue) { metadataUpdated = true; break; } } // Keep code for potential future use. // std::function closeFunc; // if (!closeAllWindows) { // closeFunc = [this] { delete this; }; // } // else { // Window* window = mWindow; // closeFunc = [window, this] { // while (window->peekGui() != ViewController::get()) // delete window->peekGui(); // }; // } std::function closeFunc; closeFunc = [this] { if (mMediaFilesUpdated) { // Always reload the gamelist if media files were updated, even if the user // chose to not save any metadata changes. Also manually unload the game image // and marquee, as otherwise they would not get updated until the user scrolls up // and down the gamelist. TextureResource::manualUnload(mScraperParams.game->getImagePath(), false); TextureResource::manualUnload(mScraperParams.game->getMarqueePath(), false); ViewController::get()->reloadGameListView(mScraperParams.system); mWindow->invalidateCachedBackground(); } ViewController::get()->onPauseVideo(); delete this; }; if (metadataUpdated) { // Changes were made, ask if the user wants to save them. mWindow->pushGui(new GuiMsgBox(mWindow, getHelpStyle(), "SAVE CHANGES?", "YES", [this, closeFunc] { save(); closeFunc(); }, "NO", closeFunc )); } else { // Always save if the media files have been changed (i.e. newly scraped images). if (mMediaFilesUpdated) save(); closeFunc(); } } bool GuiMetaDataEd::input(InputConfig* config, Input input) { if (GuiComponent::input(config, input)) return true; if (input.value != 0 && (config->isMappedTo("b", input))) { close(); return true; } if (input.value != 0 && (config->isMappedTo("y", input))) { fetch(); return true; } return false; } std::vector GuiMetaDataEd::getHelpPrompts() { std::vector prompts = mGrid.getHelpPrompts(); prompts.push_back(HelpPrompt("y", "scrape")); prompts.push_back(HelpPrompt("b", "back")); return prompts; } HelpStyle GuiMetaDataEd::getHelpStyle() { HelpStyle style = HelpStyle(); style.applyTheme(ViewController::get()->getState().getSystem()->getTheme(), "system"); return style; }