2020-09-21 17:17:34 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-05-26 16:34:33 +00:00
|
|
|
//
|
2020-09-21 17:17:34 +00:00
|
|
|
// EmulationStation Desktop Edition
|
2022-01-16 10:10:32 +00:00
|
|
|
// GuiScraperSingle.cpp
|
2020-05-26 16:34:33 +00:00
|
|
|
//
|
2020-06-21 12:25:28 +00:00
|
|
|
// Single game scraping user interface.
|
|
|
|
// This interface is triggered from GuiMetaDataEd.
|
|
|
|
// GuiScraperSearch is called from here.
|
2020-05-26 16:34:33 +00:00
|
|
|
//
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
#include "guis/GuiScraperSingle.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
|
2021-07-07 18:03:42 +00:00
|
|
|
#include "FileData.h"
|
|
|
|
#include "MameNames.h"
|
|
|
|
#include "SystemData.h"
|
2014-06-25 16:29:58 +00:00
|
|
|
#include "components/ButtonComponent.h"
|
|
|
|
#include "components/MenuComponent.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
#include "components/TextComponent.h"
|
2014-06-25 16:29:58 +00:00
|
|
|
|
2022-01-19 17:01:54 +00:00
|
|
|
GuiScraperSingle::GuiScraperSingle(ScraperSearchParams& params,
|
2022-01-16 10:10:32 +00:00
|
|
|
std::function<void(const ScraperSearchResult&)> doneFunc,
|
|
|
|
bool& savedMediaAndAborted)
|
2022-01-19 17:01:54 +00:00
|
|
|
: mClose {false}
|
2023-02-07 17:51:04 +00:00
|
|
|
, mRenderer {Renderer::getInstance()}
|
2022-01-19 17:01:54 +00:00
|
|
|
, mBackground {":/graphics/frame.svg"}
|
2023-02-07 17:51:04 +00:00
|
|
|
, mGrid {glm::ivec2 {2, 6}}
|
2022-01-16 17:18:28 +00:00
|
|
|
, mSearchParams {params}
|
|
|
|
, mSavedMediaAndAborted {savedMediaAndAborted}
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2022-01-19 17:01:54 +00:00
|
|
|
addChild(&mBackground);
|
2020-06-21 12:25:28 +00:00
|
|
|
addChild(&mGrid);
|
|
|
|
|
2021-01-06 20:29:23 +00:00
|
|
|
std::string scrapeName;
|
|
|
|
|
|
|
|
if (Settings::getInstance()->getBool("ScraperSearchMetadataName")) {
|
|
|
|
scrapeName = mSearchParams.game->getName();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (params.game->isArcadeGame() &&
|
2021-07-07 18:03:42 +00:00
|
|
|
Settings::getInstance()->getString("Scraper") == "thegamesdb")
|
2021-11-17 20:32:40 +00:00
|
|
|
scrapeName = Utils::FileSystem::getFileName(mSearchParams.game->getPath()) + " (" +
|
|
|
|
MameNames::getInstance().getCleanName(mSearchParams.game->getCleanName()) +
|
|
|
|
")";
|
2021-01-06 20:29:23 +00:00
|
|
|
else
|
|
|
|
scrapeName = Utils::FileSystem::getFileName(mSearchParams.game->getPath());
|
|
|
|
}
|
|
|
|
|
2021-07-07 18:03:42 +00:00
|
|
|
mGameName = std::make_shared<TextComponent>(
|
|
|
|
scrapeName +
|
2020-12-25 14:28:19 +00:00
|
|
|
((mSearchParams.game->getType() == FOLDER) ? " " + ViewController::FOLDER_CHAR : ""),
|
2021-10-15 18:58:40 +00:00
|
|
|
Font::get(FONT_SIZE_LARGE), 0x777777FF, ALIGN_CENTER);
|
|
|
|
mGameName->setColor(0x555555FF);
|
2022-01-16 11:09:55 +00:00
|
|
|
mGrid.setEntry(mGameName, glm::ivec2 {0, 0}, false, true, glm::ivec2 {2, 2});
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2022-01-19 17:01:54 +00:00
|
|
|
mSystemName =
|
|
|
|
std::make_shared<TextComponent>(Utils::String::toUpper(mSearchParams.system->getFullName()),
|
|
|
|
Font::get(FONT_SIZE_SMALL), 0x888888FF, ALIGN_CENTER);
|
2022-01-16 11:09:55 +00:00
|
|
|
mGrid.setEntry(mSystemName, glm::ivec2 {0, 2}, false, true, glm::ivec2 {2, 1});
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2021-10-15 18:58:40 +00:00
|
|
|
// Row 3 is a spacer.
|
2020-06-21 12:25:28 +00:00
|
|
|
|
|
|
|
// GuiScraperSearch.
|
2022-01-19 17:01:54 +00:00
|
|
|
mSearch = std::make_shared<GuiScraperSearch>(GuiScraperSearch::NEVER_AUTO_ACCEPT, 1);
|
2022-01-16 11:09:55 +00:00
|
|
|
mGrid.setEntry(mSearch, glm::ivec2 {0, 4}, true, true, glm::ivec2 {2, 1});
|
2021-10-15 18:58:40 +00:00
|
|
|
|
|
|
|
mResultList = mSearch->getResultList();
|
|
|
|
|
|
|
|
// Set up scroll indicators.
|
2022-01-19 17:01:54 +00:00
|
|
|
mScrollUp = std::make_shared<ImageComponent>();
|
|
|
|
mScrollDown = std::make_shared<ImageComponent>();
|
2021-10-15 18:58:40 +00:00
|
|
|
mScrollIndicator =
|
|
|
|
std::make_shared<ScrollIndicatorComponent>(mResultList, mScrollUp, mScrollDown);
|
|
|
|
|
|
|
|
mScrollUp->setResize(0.0f, mGameName->getFont()->getLetterHeight() / 2.0f);
|
|
|
|
mScrollUp->setOrigin(0.0f, -0.35f);
|
|
|
|
|
|
|
|
mScrollDown->setResize(0.0f, mGameName->getFont()->getLetterHeight() / 2.0f);
|
|
|
|
mScrollDown->setOrigin(0.0f, 0.35f);
|
|
|
|
|
2022-01-16 11:09:55 +00:00
|
|
|
mGrid.setEntry(mScrollUp, glm::ivec2 {1, 0}, false, false, glm::ivec2 {1, 1});
|
|
|
|
mGrid.setEntry(mScrollDown, glm::ivec2 {1, 1}, false, false, glm::ivec2 {1, 1});
|
2020-06-21 12:25:28 +00:00
|
|
|
|
|
|
|
// Buttons
|
2021-03-27 09:26:13 +00:00
|
|
|
std::vector<std::shared_ptr<ButtonComponent>> buttons;
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2022-01-19 17:01:54 +00:00
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>("REFINE SEARCH", "refine search", [&] {
|
|
|
|
// Refine the search, unless the result has already been accepted.
|
|
|
|
if (!mSearch->getAcceptedResult()) {
|
|
|
|
// Copy any search refine that may have been previously entered by opening
|
|
|
|
// the input screen using the "Y" button shortcut.
|
|
|
|
mSearchParams.nameOverride = mSearch->getNameOverride();
|
|
|
|
mSearch->openInputScreen(mSearchParams);
|
|
|
|
mGrid.resetCursor();
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>("CANCEL", "cancel", [&] {
|
2021-07-07 18:03:42 +00:00
|
|
|
if (mSearch->getSavedNewMedia()) {
|
|
|
|
// If the user aborted the scraping but there was still some media downloaded,
|
2021-12-02 17:35:02 +00:00
|
|
|
// then flag to GuiMetaDataEd that the image and marquee textures need to be
|
|
|
|
// manually unloaded and that the gamelist needs to be reloaded. Otherwise the
|
|
|
|
// images would not get updated until the user scrolls up and down the gamelist.
|
|
|
|
mSavedMediaAndAborted = true;
|
2021-07-07 18:03:42 +00:00
|
|
|
}
|
|
|
|
delete this;
|
2020-06-21 12:25:28 +00:00
|
|
|
}));
|
2022-01-19 17:01:54 +00:00
|
|
|
mButtonGrid = makeButtonGrid(buttons);
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2022-01-16 11:09:55 +00:00
|
|
|
mGrid.setEntry(mButtonGrid, glm::ivec2 {0, 5}, true, false, glm::ivec2 {2, 1});
|
2020-06-21 12:25:28 +00:00
|
|
|
|
|
|
|
mSearch->setAcceptCallback([this, doneFunc](const ScraperSearchResult& result) {
|
2021-07-07 18:03:42 +00:00
|
|
|
doneFunc(result);
|
|
|
|
close();
|
|
|
|
});
|
2020-06-21 12:25:28 +00:00
|
|
|
mSearch->setCancelCallback([&] { delete this; });
|
2021-10-15 18:58:40 +00:00
|
|
|
mSearch->setRefineCallback([&] {
|
2022-02-11 21:10:25 +00:00
|
|
|
mScrollUp->setOpacity(0.0f);
|
|
|
|
mScrollDown->setOpacity(0.0f);
|
2021-10-15 18:58:40 +00:00
|
|
|
mResultList->resetScrollIndicatorStatus();
|
|
|
|
});
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2021-07-02 15:51:07 +00:00
|
|
|
// Limit the width of the GUI on ultrawide monitors. The 1.778 aspect ratio value is
|
|
|
|
// the 16:9 reference.
|
2023-02-07 17:51:04 +00:00
|
|
|
const float aspectValue {1.778f / Renderer::getScreenAspectRatio()};
|
|
|
|
const float width {glm::clamp(0.95f * aspectValue, 0.70f, 0.95f) * mRenderer->getScreenWidth()};
|
|
|
|
const float screenSize {mRenderer->getIsVerticalOrientation() ? mRenderer->getScreenWidth() :
|
|
|
|
mRenderer->getScreenHeight()};
|
2022-02-11 22:38:23 +00:00
|
|
|
|
2023-02-07 17:51:04 +00:00
|
|
|
const float height {(mGameName->getFont()->getLetterHeight() + screenSize * 0.0637f) +
|
|
|
|
mSystemName->getFont()->getLetterHeight() + screenSize * 0.04f +
|
|
|
|
mButtonGrid->getSize().y + Font::get(FONT_SIZE_MEDIUM)->getHeight() * 8.0f};
|
2021-10-15 18:58:40 +00:00
|
|
|
|
|
|
|
setSize(width, height);
|
2023-02-07 17:51:04 +00:00
|
|
|
setPosition((mRenderer->getScreenWidth() - mSize.x) / 2.0f,
|
|
|
|
(mRenderer->getScreenHeight() - mSize.y) / 2.0f);
|
2020-06-21 12:25:28 +00:00
|
|
|
|
|
|
|
mGrid.resetCursor();
|
|
|
|
mSearch->search(params); // Start the search.
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
void GuiScraperSingle::onSizeChanged()
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2023-02-07 17:51:04 +00:00
|
|
|
const float gameNameHeight {mRenderer->getIsVerticalOrientation() ?
|
|
|
|
mRenderer->getScreenWidth() * 0.0637f :
|
|
|
|
mRenderer->getScreenHeight() * 0.0637f};
|
|
|
|
|
|
|
|
mGrid.setRowHeightPerc(0, (mGameName->getFont()->getLetterHeight() + gameNameHeight) / mSize.y /
|
|
|
|
2.0f);
|
|
|
|
mGrid.setRowHeightPerc(1, (mGameName->getFont()->getLetterHeight() + gameNameHeight) / mSize.y /
|
|
|
|
2.0f);
|
2021-10-15 18:58:40 +00:00
|
|
|
mGrid.setRowHeightPerc(2, mSystemName->getFont()->getLetterHeight() / mSize.y, false);
|
|
|
|
mGrid.setRowHeightPerc(3, 0.04f, false);
|
2021-11-05 19:31:24 +00:00
|
|
|
mGrid.setRowHeightPerc(4, (Font::get(FONT_SIZE_MEDIUM)->getHeight() * 8.0f) / mSize.y, false);
|
2021-10-15 18:58:40 +00:00
|
|
|
|
2023-02-07 17:51:04 +00:00
|
|
|
if (mRenderer->getIsVerticalOrientation())
|
|
|
|
mGrid.setColWidthPerc(1, 0.05f);
|
|
|
|
else
|
|
|
|
mGrid.setColWidthPerc(1, 0.04f);
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2021-11-05 19:31:24 +00:00
|
|
|
mGrid.setSize(glm::round(mSize));
|
2022-10-24 22:19:02 +00:00
|
|
|
mBackground.fitTo(mSize, glm::vec3 {0.0f, 0.0f, 0.0f}, glm::vec2 {-32.0f, -32.0f});
|
2021-10-15 18:58:40 +00:00
|
|
|
|
|
|
|
// Add some extra margins to the game name.
|
2022-09-03 10:44:49 +00:00
|
|
|
const float newSizeX {mSize.x * 0.96f};
|
2021-10-15 18:58:40 +00:00
|
|
|
mGameName->setSize(newSizeX, mGameName->getSize().y);
|
|
|
|
mGameName->setPosition((mSize.x - newSizeX) / 2.0f, 0.0f);
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
bool GuiScraperSingle::input(InputConfig* config, Input input)
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2020-07-13 18:58:25 +00:00
|
|
|
if (config->isMappedTo("b", input) && input.value) {
|
2021-02-01 17:52:49 +00:00
|
|
|
if (mSearch->getSavedNewMedia()) {
|
|
|
|
// If the user aborted the scraping but there was still some media downloaded,
|
2021-12-02 16:34:30 +00:00
|
|
|
// then flag to GuiMetaDataEd that the image and marquee textures need to be
|
|
|
|
// manually unloaded and that the gamelist needs to be reloaded. Otherwise the
|
|
|
|
// images would not get updated until the user scrolls up and down the gamelist.
|
|
|
|
mSavedMediaAndAborted = true;
|
2021-02-01 17:52:49 +00:00
|
|
|
}
|
2020-06-21 12:25:28 +00:00
|
|
|
delete this;
|
|
|
|
return true;
|
|
|
|
}
|
2014-06-25 16:29:58 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
return GuiComponent::input(config, input);
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
void GuiScraperSingle::update(int deltaTime)
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
GuiComponent::update(deltaTime);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
2020-07-13 18:58:25 +00:00
|
|
|
if (mClose)
|
2020-06-21 12:25:28 +00:00
|
|
|
delete this;
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
std::vector<HelpPrompt> GuiScraperSingle::getHelpPrompts()
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2022-09-03 10:44:49 +00:00
|
|
|
std::vector<HelpPrompt> prompts {mGrid.getHelpPrompts()};
|
2021-07-02 16:02:39 +00:00
|
|
|
prompts.push_back(HelpPrompt("b", "back (cancel)"));
|
|
|
|
|
|
|
|
return prompts;
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2022-01-16 10:10:32 +00:00
|
|
|
void GuiScraperSingle::close()
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2021-07-07 18:03:42 +00:00
|
|
|
// This will cause update() to close the GUI.
|
2020-06-21 12:25:28 +00:00
|
|
|
mClose = true;
|
2017-08-11 17:03:12 +00:00
|
|
|
}
|