Files
scummvm-cursorfix/engines/ultima/ultima8/gumps/widgets/edit_widget.cpp
2026-02-02 04:50:13 +01:00

274 lines
6.7 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "ultima/ultima8/gumps/widgets/edit_widget.h"
#include "ultima/ultima8/gfx/fonts/rendered_text.h"
#include "ultima/ultima8/gfx/render_surface.h"
#include "ultima/ultima8/gfx/fonts/font_manager.h"
#include "common/system.h"
#include "common/events.h"
namespace Ultima {
namespace Ultima8 {
DEFINE_RUNTIME_CLASSTYPE_CODE(EditWidget)
EditWidget::EditWidget(int x, int y, Std::string txt, bool gamefont, int font,
int w, int h, unsigned int maxlength, bool multiline)
: Gump(x, y, w, h), _text(txt), _gameFont(gamefont), _fontNum(font),
_maxLength(maxlength), _multiLine(multiline),
_cursorChanged(0), _cursorVisible(true), _cachedText(nullptr) {
_cursor = _text.size();
}
EditWidget::~EditWidget(void) {
delete _cachedText;
}
// Init the gump, call after construction
void EditWidget::InitGump(Gump *newparent, bool take_focus) {
Gump::InitGump(newparent, take_focus);
Font *font = getFont();
// Y offset is always baseline
_dims.moveTo(0, -font->getBaseline());
if (_gameFont && getFont()->isHighRes()) {
Common::Rect32 rect(_dims);
ScreenSpaceToGumpRect(rect, ROUND_OUTSIDE);
_dims.moveTo(0, rect.top);
}
}
Font *EditWidget::getFont() const {
if (_gameFont)
return FontManager::get_instance()->getGameFont(_fontNum, true);
else
return FontManager::get_instance()->getTTFont(_fontNum);
}
void EditWidget::setText(const Std::string &t) {
_text = t;
_cursor = _text.size();
delete _cachedText;
_cachedText = nullptr;
}
void EditWidget::ensureCursorVisible() {
_cursorVisible = true;
_cursorChanged = g_system->getMillis();
}
bool EditWidget::textFits(Std::string &t) {
Font *font = getFont();
unsigned int remaining;
int32 width, height;
int32 max_width = _multiLine ? _dims.width() : 0;
int32 max_height = _dims.height();
if (_gameFont && font->isHighRes()) {
Common::Rect32 rect(0, 0, max_width, max_height);
GumpRectToScreenSpace(rect, ROUND_INSIDE);
max_width = rect.width();
max_height = rect.height();
}
font->getTextSize(t, width, height, remaining,
max_width, max_height,
Font::TEXT_LEFT, false);
if (_gameFont && font->isHighRes()) {
Common::Rect32 rect(0, 0, width, height);
ScreenSpaceToGumpRect(rect, ROUND_OUTSIDE);
width = rect.width();
height = rect.height();
}
if (_multiLine)
return (remaining >= t.size());
else
return (width <= _dims.width());
}
void EditWidget::renderText() {
bool cv = _cursorVisible;
if (!IsFocus()) {
cv = false;
} else {
uint32 now = g_system->getMillis();
if (now > _cursorChanged + 750) {
cv = !_cursorVisible;
_cursorChanged = now;
}
}
if (cv != _cursorVisible) {
delete _cachedText;
_cachedText = nullptr;
_cursorVisible = cv;
}
if (!_cachedText) {
Font *font = getFont();
int32 max_width = _multiLine ? _dims.width() : 0;
int32 max_height = _dims.height();
if (_gameFont && font->isHighRes()) {
Common::Rect32 rect(0, 0, max_width, max_height);
GumpRectToScreenSpace(rect, ROUND_INSIDE);
max_width = rect.width();
max_height = rect.height();
}
unsigned int remaining;
_cachedText = font->renderText(_text, remaining,
max_width, max_height,
Font::TEXT_LEFT,
false, false,
cv ? _cursor : Std::string::npos);
// Trim text to fit
if (remaining < _text.size()) {
_text.erase(remaining);
_cursor = _text.size();
}
}
}
// Overloadable method to Paint just this Gump (RenderSurface is relative to this)
void EditWidget::PaintThis(RenderSurface *surf, int32 lerp_factor, bool scaled) {
Gump::PaintThis(surf, lerp_factor, scaled);
renderText();
if (scaled && _gameFont && getFont()->isHighRes()) {
return;
}
_cachedText->draw(surf, 0, 0);
}
// Overloadable method to Paint just this gumps unscaled components that require compositing (RenderSurface is relative to parent).
void EditWidget::PaintComposited(RenderSurface *surf, int32 lerp_factor, int32 sx, int32 sy) {
Font *font = getFont();
if (!_gameFont || !font->isHighRes()) return;
int32 x = 0, y = 0;
GumpToScreenSpace(x, y, ROUND_BOTTOMRIGHT);
_cachedText->draw(surf, x, y, true);
Common::Rect32 rect(_dims);
GumpRectToScreenSpace(rect, ROUND_OUTSIDE);
}
// don't handle any mouse motion events, so let parent handle them for us.
Gump *EditWidget::onMouseMotion(int32 mx, int32 my) {
return nullptr;
}
bool EditWidget::OnKeyDown(int key, int mod) {
switch (key) {
case Common::KEYCODE_RETURN:
case Common::KEYCODE_KP_ENTER:
_parent->ChildNotify(this, EDIT_ENTER);
break;
case Common::KEYCODE_ESCAPE:
_parent->ChildNotify(this, EDIT_ESCAPE);
break;
case Common::KEYCODE_BACKSPACE:
if (_cursor > 0) {
_text.erase(--_cursor, 1);
delete _cachedText;
_cachedText = nullptr;
ensureCursorVisible();
}
break;
case Common::KEYCODE_DELETE:
if (_cursor != _text.size()) {
_text.erase(_cursor, 1);
delete _cachedText;
_cachedText = nullptr;
}
break;
case Common::KEYCODE_LEFT:
if (_cursor > 0) {
_cursor--;
delete _cachedText;
_cachedText = nullptr;
ensureCursorVisible();
}
break;
case Common::KEYCODE_RIGHT:
if (_cursor < _text.size()) {
_cursor++;
delete _cachedText;
_cachedText = nullptr;
ensureCursorVisible();
}
break;
default:
break;
}
return true;
}
bool EditWidget::OnKeyUp(int key) {
return true;
}
bool EditWidget::OnTextInput(int unicode) {
if (_maxLength > 0 && _text.size() >= _maxLength)
return true;
char c = 0;
if (unicode >= 0 && unicode < 256)
c = reverse_encoding[unicode];
if (!c) return true;
Std::string newtext = _text;
newtext.insertChar(c, _cursor);
if (textFits(newtext)) {
_text = newtext;
_cursor++;
delete _cachedText;
_cachedText = nullptr;
}
return true;
}
void EditWidget::OnFocus(bool gain) {
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, gain);
}
} // End of namespace Ultima8
} // End of namespace Ultima