274 lines
6.7 KiB
C++
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
|