2020-09-18 16:16:12 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-06-07 08:57:49 +00:00
|
|
|
//
|
2020-09-18 16:16:12 +00:00
|
|
|
// EmulationStation Desktop Edition
|
2020-06-21 12:25:28 +00:00
|
|
|
// GuiTextEditPopup.cpp
|
2020-06-07 08:57:49 +00:00
|
|
|
//
|
2021-09-17 20:23:41 +00:00
|
|
|
// Text edit popup.
|
|
|
|
// Has a default mode and a complex mode, both with various options passed as arguments.
|
2020-06-07 08:57:49 +00:00
|
|
|
//
|
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
#define DELETE_REPEAT_START_DELAY 600
|
|
|
|
#define DELETE_REPEAT_SPEED 90 // Lower is faster.
|
|
|
|
|
2014-06-20 01:30:09 +00:00
|
|
|
#include "guis/GuiTextEditPopup.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
|
2014-06-20 01:30:09 +00:00
|
|
|
#include "components/MenuComponent.h"
|
2020-09-18 16:16:12 +00:00
|
|
|
#include "guis/GuiMsgBox.h"
|
2020-06-07 08:57:49 +00:00
|
|
|
|
2021-07-07 18:31:46 +00:00
|
|
|
GuiTextEditPopup::GuiTextEditPopup(Window* window,
|
|
|
|
const HelpStyle& helpstyle,
|
|
|
|
const std::string& title,
|
|
|
|
const std::string& initValue,
|
|
|
|
const std::function<void(const std::string&)>& okCallback,
|
|
|
|
bool multiLine,
|
|
|
|
const std::string& acceptBtnText,
|
2021-09-17 20:23:41 +00:00
|
|
|
const std::string& saveConfirmationText,
|
|
|
|
const std::string& infoString,
|
|
|
|
const std::string& defaultValue,
|
|
|
|
const std::string& loadBtnHelpText,
|
|
|
|
const std::string& clearBtnHelpText,
|
|
|
|
const std::string& cancelBtnHelpText)
|
|
|
|
: GuiComponent{window}
|
|
|
|
, mHelpStyle{helpstyle}
|
|
|
|
, mInitValue{initValue}
|
|
|
|
, mOkCallback{okCallback}
|
|
|
|
, mMultiLine{multiLine}
|
|
|
|
, mAcceptBtnText{acceptBtnText}
|
|
|
|
, mSaveConfirmationText{saveConfirmationText}
|
|
|
|
, mLoadBtnHelpText{loadBtnHelpText}
|
|
|
|
, mClearBtnHelpText{clearBtnHelpText}
|
|
|
|
, mCancelBtnHelpText{cancelBtnHelpText}
|
|
|
|
, mBackground{window, ":/graphics/frame.svg"}
|
|
|
|
, mGrid{window, glm::ivec2{1, (infoString != "" && defaultValue != "" ? 5 : 3)}}
|
|
|
|
, mComplexMode{(infoString != "" && defaultValue != "")}
|
|
|
|
, mDeleteRepeat{false}
|
|
|
|
, mDeleteRepeatTimer{0}
|
2014-03-21 16:10:19 +00:00
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
addChild(&mBackground);
|
|
|
|
addChild(&mGrid);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
mTitle = std::make_shared<TextComponent>(mWindow, Utils::String::toUpper(title),
|
2021-07-07 18:31:46 +00:00
|
|
|
Font::get(FONT_SIZE_MEDIUM), 0x555555FF, ALIGN_CENTER);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
if (mComplexMode) {
|
|
|
|
mInfoString = std::make_shared<TextComponent>(
|
|
|
|
mWindow, infoString, Font::get(FONT_SIZE_SMALL), 0x555555FF, ALIGN_CENTER);
|
|
|
|
mDefaultValue = std::make_shared<TextComponent>(
|
|
|
|
mWindow, defaultValue, Font::get(FONT_SIZE_SMALL), 0x555555FF, ALIGN_CENTER);
|
|
|
|
}
|
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
mText = std::make_shared<TextEditComponent>(mWindow);
|
|
|
|
mText->setValue(initValue);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2021-03-27 09:26:13 +00:00
|
|
|
std::vector<std::shared_ptr<ButtonComponent>> buttons;
|
2020-06-21 12:25:28 +00:00
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>(mWindow, acceptBtnText, acceptBtnText,
|
2021-07-07 18:31:46 +00:00
|
|
|
[this, okCallback] {
|
|
|
|
okCallback(mText->getValue());
|
|
|
|
delete this;
|
|
|
|
}));
|
2021-09-17 20:23:41 +00:00
|
|
|
if (mComplexMode) {
|
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>(
|
|
|
|
mWindow, "load", loadBtnHelpText, [this, defaultValue] {
|
|
|
|
mText->setValue(defaultValue);
|
|
|
|
mText->setCursor(0);
|
|
|
|
mText->setCursor(defaultValue.size());
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>(mWindow, "clear", clearBtnHelpText,
|
2021-07-07 18:31:46 +00:00
|
|
|
[this] { mText->setValue(""); }));
|
2021-09-17 20:23:41 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
buttons.push_back(std::make_shared<ButtonComponent>(mWindow, "CANCEL", "discard changes",
|
2021-07-07 18:31:46 +00:00
|
|
|
[this] { delete this; }));
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
mButtonGrid = makeButtonGrid(mWindow, buttons);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2021-08-17 16:41:45 +00:00
|
|
|
mGrid.setEntry(mTitle, glm::ivec2{0, 0}, false, true);
|
2021-09-17 20:23:41 +00:00
|
|
|
|
|
|
|
int yPos = 1;
|
|
|
|
|
|
|
|
if (mComplexMode) {
|
|
|
|
mGrid.setEntry(mInfoString, glm::ivec2{0, yPos}, false, true);
|
|
|
|
mGrid.setEntry(mDefaultValue, glm::ivec2{0, yPos + 1}, false, false);
|
|
|
|
yPos += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
mGrid.setEntry(mText, glm::ivec2{0, yPos}, true, false, glm::ivec2{1, 1},
|
2021-07-07 18:31:46 +00:00
|
|
|
GridFlags::BORDER_TOP | GridFlags::BORDER_BOTTOM);
|
2021-09-17 20:23:41 +00:00
|
|
|
mGrid.setEntry(mButtonGrid, glm::ivec2{0, yPos + 1}, true, false);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
float textHeight = mText->getFont()->getHeight();
|
2020-06-07 08:57:49 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
if (multiLine)
|
2021-07-07 18:31:46 +00:00
|
|
|
textHeight *= 6.0f;
|
2021-09-17 20:23:41 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
mText->setSize(0, textHeight);
|
2014-03-21 16:10:19 +00:00
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
// Adapt width to the geometry of the display. The 1.778 aspect ratio is the 16:9 reference.
|
2021-07-02 15:57:52 +00:00
|
|
|
float aspectValue = 1.778f / Renderer::getScreenAspectRatio();
|
2021-03-09 16:17:33 +00:00
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
if (mComplexMode) {
|
|
|
|
float infoWidth =
|
|
|
|
glm::clamp(0.70f * aspectValue, 0.34f, 0.85f) * Renderer::getScreenWidth();
|
|
|
|
float windowWidth =
|
|
|
|
glm::clamp(0.75f * aspectValue, 0.40f, 0.90f) * Renderer::getScreenWidth();
|
|
|
|
|
|
|
|
mDefaultValue->setSize(infoWidth, mDefaultValue->getFont()->getHeight());
|
|
|
|
|
|
|
|
setSize(windowWidth, mTitle->getFont()->getHeight() + textHeight +
|
|
|
|
mButtonGrid->getSize().y + mButtonGrid->getSize().y * 1.85f);
|
|
|
|
setPosition((Renderer::getScreenWidth() - mSize.x) / 2.0f,
|
|
|
|
(Renderer::getScreenHeight() - mSize.y) / 2.0f);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
float width = glm::clamp(0.54f * aspectValue, 0.20f, 0.70f) * Renderer::getScreenWidth();
|
|
|
|
|
|
|
|
setSize(width, mTitle->getFont()->getHeight() + textHeight + mButtonGrid->getSize().y +
|
|
|
|
mButtonGrid->getSize().y / 2.0f);
|
|
|
|
setPosition((Renderer::getScreenWidth() - mSize.x) / 2.0f,
|
|
|
|
(Renderer::getScreenHeight() - mSize.y) / 2.0f);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!multiLine)
|
|
|
|
mText->setCursor(initValue.size());
|
|
|
|
|
2020-11-07 11:45:57 +00:00
|
|
|
mText->startEditing();
|
2014-03-21 16:10:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GuiTextEditPopup::onSizeChanged()
|
|
|
|
{
|
2021-08-17 16:41:45 +00:00
|
|
|
mBackground.fitTo(mSize, glm::vec3{}, glm::vec2{-32.0f, -32.0f});
|
2021-09-17 20:23:41 +00:00
|
|
|
mText->setSize(mSize.x - 40.0f * Renderer::getScreenHeightModifier(), mText->getSize().y);
|
2014-03-25 23:41:50 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
// Update grid.
|
2021-08-16 16:25:01 +00:00
|
|
|
mGrid.setRowHeightPerc(0, mTitle->getFont()->getHeight() / mSize.y);
|
2021-09-17 20:23:41 +00:00
|
|
|
|
|
|
|
if (mComplexMode)
|
|
|
|
mGrid.setRowHeightPerc(1, 0.15f);
|
|
|
|
|
2021-08-16 16:25:01 +00:00
|
|
|
mGrid.setRowHeightPerc(2, mButtonGrid->getSize().y / mSize.y);
|
2020-06-21 12:25:28 +00:00
|
|
|
mGrid.setSize(mSize);
|
2014-03-21 16:10:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GuiTextEditPopup::input(InputConfig* config, Input input)
|
|
|
|
{
|
2021-09-17 20:23:41 +00:00
|
|
|
// Enter key (main key or via numpad) accepts the changes.
|
|
|
|
if (config->getDeviceId() == DEVICE_KEYBOARD && mText->isEditing() && !mMultiLine &&
|
|
|
|
input.value && (input.id == SDLK_RETURN || input.id == SDLK_KP_ENTER)) {
|
|
|
|
this->mOkCallback(mText->getValue());
|
|
|
|
delete this;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Dito for the A button if using a controller.
|
|
|
|
else if (config->getDeviceId() != DEVICE_KEYBOARD && mText->isEditing() &&
|
|
|
|
config->isMappedTo("a", input) && input.value) {
|
|
|
|
this->mOkCallback(mText->getValue());
|
|
|
|
delete this;
|
2020-06-21 12:25:28 +00:00
|
|
|
return true;
|
2021-09-17 20:23:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the keyboard has been configured with backspace as the back button (which is the default
|
|
|
|
// configuration) then ignore this key if we're currently editing or otherwise it would be
|
|
|
|
// impossible to erase characters using this key.
|
|
|
|
bool keyboardBackspace = (config->getDeviceId() == DEVICE_KEYBOARD && mText->isEditing() &&
|
|
|
|
input.id == SDLK_BACKSPACE);
|
2020-06-21 12:25:28 +00:00
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
// Pressing back (or the escape key if using keyboard input) closes us.
|
|
|
|
if ((config->getDeviceId() == DEVICE_KEYBOARD && input.value && input.id == SDLK_ESCAPE) ||
|
|
|
|
(!keyboardBackspace && config->isMappedTo("b", input)) && input.value) {
|
2020-06-21 12:25:28 +00:00
|
|
|
if (mText->getValue() != mInitValue) {
|
|
|
|
// Changes were made, ask if the user wants to save them.
|
2021-07-07 18:31:46 +00:00
|
|
|
mWindow->pushGui(new GuiMsgBox(
|
|
|
|
mWindow, mHelpStyle, mSaveConfirmationText, "YES",
|
|
|
|
[this] {
|
|
|
|
this->mOkCallback(mText->getValue());
|
|
|
|
delete this;
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
"NO",
|
|
|
|
[this] {
|
|
|
|
delete this;
|
|
|
|
return false;
|
|
|
|
}));
|
2020-06-21 12:25:28 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
}
|
2021-09-17 20:23:41 +00:00
|
|
|
|
|
|
|
if (mText->isEditing() && config->isMappedLike("down", input) && input.value) {
|
|
|
|
mText->stopEditing();
|
|
|
|
mGrid.setCursorTo(mGrid.getSelectedComponent());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Left shoulder button deletes a character (backspace).
|
|
|
|
if (config->isMappedTo("leftshoulder", input)) {
|
|
|
|
if (input.value) {
|
|
|
|
mDeleteRepeat = true;
|
|
|
|
mDeleteRepeatTimer = -(DELETE_REPEAT_START_DELAY - DELETE_REPEAT_SPEED);
|
|
|
|
|
|
|
|
bool editing = mText->isEditing();
|
|
|
|
if (!editing)
|
|
|
|
mText->startEditing();
|
|
|
|
|
|
|
|
mText->textInput("\b");
|
|
|
|
|
|
|
|
if (!editing)
|
|
|
|
mText->stopEditing();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mDeleteRepeat = false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Right shoulder button inserts a blank space.
|
|
|
|
if (config->isMappedTo("rightshoulder", input) && input.value) {
|
|
|
|
bool editing = mText->isEditing();
|
|
|
|
if (!editing)
|
|
|
|
mText->startEditing();
|
|
|
|
|
|
|
|
mText->textInput(" ");
|
|
|
|
|
|
|
|
if (!editing)
|
|
|
|
mText->stopEditing();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GuiComponent::input(config, input))
|
|
|
|
return true;
|
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
return false;
|
2014-03-24 01:33:27 +00:00
|
|
|
}
|
|
|
|
|
2021-09-17 20:23:41 +00:00
|
|
|
void GuiTextEditPopup::update(int deltaTime)
|
|
|
|
{
|
|
|
|
updateDeleteRepeat(deltaTime);
|
|
|
|
GuiComponent::update(deltaTime);
|
|
|
|
}
|
|
|
|
|
2014-03-24 01:33:27 +00:00
|
|
|
std::vector<HelpPrompt> GuiTextEditPopup::getHelpPrompts()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
std::vector<HelpPrompt> prompts = mGrid.getHelpPrompts();
|
2021-09-17 20:23:41 +00:00
|
|
|
|
|
|
|
if (mText->isEditing())
|
|
|
|
prompts.push_back(HelpPrompt("a", mAcceptBtnText));
|
|
|
|
|
|
|
|
prompts.push_back(HelpPrompt("l", "backspace"));
|
|
|
|
prompts.push_back(HelpPrompt("r", "space"));
|
2020-06-21 12:25:28 +00:00
|
|
|
prompts.push_back(HelpPrompt("b", "back"));
|
|
|
|
return prompts;
|
2014-03-24 01:33:27 +00:00
|
|
|
}
|
2021-09-17 20:23:41 +00:00
|
|
|
|
|
|
|
void GuiTextEditPopup::updateDeleteRepeat(int deltaTime)
|
|
|
|
{
|
|
|
|
if (!mDeleteRepeat)
|
|
|
|
return;
|
|
|
|
|
|
|
|
mDeleteRepeatTimer += deltaTime;
|
|
|
|
|
|
|
|
while (mDeleteRepeatTimer >= DELETE_REPEAT_SPEED) {
|
|
|
|
bool editing = mText->isEditing();
|
|
|
|
if (!editing)
|
|
|
|
mText->startEditing();
|
|
|
|
|
|
|
|
mText->textInput("\b");
|
|
|
|
|
|
|
|
if (!editing)
|
|
|
|
mText->stopEditing();
|
|
|
|
|
|
|
|
mDeleteRepeatTimer -= DELETE_REPEAT_SPEED;
|
|
|
|
}
|
|
|
|
}
|