// SPDX-License-Identifier: MIT // // EmulationStation Desktop Edition // GuiLaunchScreen.cpp // // Screen shown when launching a game. // #include "guis/GuiLaunchScreen.h" #include "FileData.h" #include "SystemData.h" #include "components/ComponentGrid.h" #include "components/TextComponent.h" #include "utils/StringUtil.h" GuiLaunchScreen::GuiLaunchScreen() : mRenderer {Renderer::getInstance()} , mBackground {":/graphics/frame.svg"} , mGrid {nullptr} , mMarquee {nullptr} { addChild(&mBackground); mWindow->setLaunchScreen(this); } GuiLaunchScreen::~GuiLaunchScreen() { // This only executes when exiting the application. closeLaunchScreen(); } void GuiLaunchScreen::displayLaunchScreen(FileData* game) { mGrid = new ComponentGrid(glm::ivec2 {3, 8}); addChild(mGrid); mImagePath = game->getMarqueePath(); // We need to unload the image first as it may be cached at a modified resolution // which would lead to the wrong size when using the image here. if (mImagePath != "") { TextureResource::manualUnload(mImagePath, false); mMarquee = new ImageComponent; } mScaleUp = 0.5f; const float titleFontSize {0.060f}; const float gameNameFontSize {0.073f}; // Spacer row. mGrid->setEntry(std::make_shared(), glm::ivec2 {1, 0}, false, false, glm::ivec2 {1, 1}); // Title. mTitle = std::make_shared( "LAUNCHING GAME", Font::get(titleFontSize * std::min(Renderer::getScreenHeight(), Renderer::getScreenWidth())), 0x666666FF, ALIGN_CENTER); mGrid->setEntry(mTitle, glm::ivec2 {1, 1}, false, true, glm::ivec2 {1, 1}); // Spacer row. mGrid->setEntry(std::make_shared(), glm::ivec2 {1, 2}, false, false, glm::ivec2 {1, 1}); // Row for the marquee. mGrid->setEntry(std::make_shared(), glm::ivec2 {1, 3}, false, false, glm::ivec2 {1, 1}); // Spacer row. mGrid->setEntry(std::make_shared(), glm::ivec2 {1, 4}, false, false, glm::ivec2 {1, 1}); // Game name. mGameName = std::make_shared( "GAME NAME", Font::get(gameNameFontSize * std::min(Renderer::getScreenHeight(), Renderer::getScreenWidth())), 0x444444FF, ALIGN_CENTER); mGrid->setEntry(mGameName, glm::ivec2 {1, 5}, false, true, glm::ivec2 {1, 1}); // System name. mSystemName = std::make_shared("SYSTEM NAME", Font::get(FONT_SIZE_MEDIUM), 0x666666FF, ALIGN_CENTER); mGrid->setEntry(mSystemName, glm::ivec2 {1, 6}, false, true, glm::ivec2 {1, 1}); // Spacer row. mGrid->setEntry(std::make_shared(), glm::ivec2 {1, 7}, false, false, glm::ivec2 {1, 1}); // Left spacer. mGrid->setEntry(std::make_shared(), glm::ivec2 {0, 0}, false, false, glm::ivec2 {1, 8}); // Right spacer. mGrid->setEntry(std::make_shared(), glm::ivec2 {2, 0}, false, false, glm::ivec2 {1, 8}); // Adjust the width depending on the aspect ratio of the screen, to make the screen look // somewhat coherent regardless of screen type. The 1.778 aspect ratio value is the 16:9 // reference. float aspectValue {1.778f / Renderer::getScreenAspectRatio()}; float maxWidthModifier {glm::clamp(0.78f * aspectValue, 0.78f, 0.90f)}; float minWidthModifier {glm::clamp(0.50f * aspectValue, 0.50f, (mRenderer->getIsVerticalOrientation() ? 0.80f : 0.65f))}; float maxWidth {mRenderer->getScreenWidth() * maxWidthModifier}; float minWidth {mRenderer->getScreenWidth() * minWidthModifier}; float fontWidth {Font::get(gameNameFontSize * std::min(Renderer::getScreenHeight(), Renderer::getScreenWidth())) ->sizeText(Utils::String::toUpper(game->getName())) .x}; // Add a bit of width to compensate for the left and right spacers. fontWidth += Renderer::getScreenWidth() * 0.05f; float width {glm::clamp(fontWidth, minWidth, maxWidth)}; if (mImagePath != "") { if (mRenderer->getIsVerticalOrientation()) setSize(width, mRenderer->getScreenWidth() * 0.60f); else setSize(width, mRenderer->getScreenHeight() * 0.60f); } else { if (mRenderer->getIsVerticalOrientation()) setSize(width, mRenderer->getScreenWidth() * 0.38f); else setSize(width, mRenderer->getScreenHeight() * 0.38f); } // Set row heights. if (mImagePath != "") mGrid->setRowHeightPerc(0, 0.09f, false); else mGrid->setRowHeightPerc(0, 0.15f, false); mGrid->setRowHeightPerc(1, mTitle->getFont()->getLetterHeight() * 1.70f / mSize.y, false); mGrid->setRowHeightPerc(2, 0.05f, false); if (mImagePath != "") mGrid->setRowHeightPerc(3, 0.35f, false); else mGrid->setRowHeightPerc(3, 0.01f, false); mGrid->setRowHeightPerc(4, 0.05f, false); mGrid->setRowHeightPerc(5, mGameName->getFont()->getHeight() * 0.80f / mSize.y, false); mGrid->setRowHeightPerc(6, mSystemName->getFont()->getHeight() * 0.90f / mSize.y, false); // Set left and right spacers column widths. mGrid->setColWidthPerc(0, 0.025f); mGrid->setColWidthPerc(2, 0.025f); mGrid->setSize(mSize); float totalRowHeight {0.0f}; // Hack to adjust the window height to the row boundary. for (int i = 0; i < 7; ++i) totalRowHeight += mGrid->getRowHeight(i); setSize(mSize.x, totalRowHeight); mGameName->setText(Utils::String::toUpper(game->getName())); mSystemName->setText(Utils::String::toUpper(game->getSystem()->getFullName())); // For the marquee we strip away any transparent padding around the actual image. // When doing this, we restrict the scale-up to a certain percentage of the screen // width so that the sizes look somewhat consistent regardless of the aspect ratio // of the images. if (mImagePath != "") { const float multiplier {mRenderer->getIsVerticalOrientation() ? 0.20f : 0.25f}; mMarquee->setLinearInterpolation(true); mMarquee->setImage(game->getMarqueePath(), false); mMarquee->cropTransparentPadding( mRenderer->getScreenWidth() * (multiplier * (1.778f / mRenderer->getScreenAspectRatio())), mGrid->getRowHeight(3)); mMarquee->setOrigin(0.5f, 0.5f); glm::vec3 currentPos {mMarquee->getPosition()}; // Position the image in the middle of row four. currentPos.x = mSize.x / 2.0f; currentPos.y = mGrid->getRowHeight(0) + mGrid->getRowHeight(1) + mGrid->getRowHeight(2) + mGrid->getRowHeight(3) / 2.0f; mMarquee->setPosition(currentPos); } setOrigin({0.5f, 0.5f}); // Center on the X axis and keep slightly off-center on the Y axis. setPosition(Renderer::getScreenWidth() / 2.0f, Renderer::getScreenHeight() / 2.25f); mBackground.fitTo(mSize, glm::vec3 {}, glm::vec2 {-32.0f, -32.0f}); mBackground.setEdgeColor(0xEEEEEEFF); } void GuiLaunchScreen::closeLaunchScreen() { if (mGrid) { delete mGrid; mGrid = nullptr; } if (mMarquee) { delete mMarquee; mMarquee = nullptr; } // An extra precaution. if (mImagePath != "") { TextureResource::manualUnload(mImagePath, false); mImagePath = ""; } } void GuiLaunchScreen::onSizeChanged() { // Update mGrid size. mGrid->setSize(mSize); } void GuiLaunchScreen::update(int deltaTime) { if (Settings::getInstance()->getString("MenuOpeningEffect") == "none") mScaleUp = 1.0f; else if (mScaleUp < 1.0f) mScaleUp = glm::clamp(mScaleUp + 0.07f, 0.0f, 1.0f); } void GuiLaunchScreen::render(const glm::mat4& /*parentTrans*/) { // Scale up animation. setScale(mScaleUp); glm::mat4 trans {Renderer::getIdentity() * getTransform()}; mRenderer->setMatrix(trans); GuiComponent::renderChildren(trans); if (mMarquee) mMarquee->render(trans); }