/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2014 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * 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 2 of the License, or * 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 "gui/widgets/textbox.h" #include "input/keydata.h" #include "input/keyevent.h" #include "gui/base/font.hpp" #include <sstream> #include "debug.h" TextBox::TextBox(const Widget2 *const widget) : gcn::TextBox(), Widget2(widget), mMinWidth(getWidth()) { mForegroundColor = getThemeColor(Theme::TEXTBOX); setOpaque(false); setFrameSize(0); } TextBox::~TextBox() { if (gui) gui->removeDragged(this); } void TextBox::setTextWrapped(const std::string &text, const int minDimension) { // Make sure parent scroll area sets width of this widget if (getParent()) getParent()->logic(); // Take the supplied minimum dimension as a starting // point and try to beat it mMinWidth = minDimension; const size_t textSize = text.size(); size_t spacePos = text.rfind(" ", textSize); if (spacePos != std::string::npos) { const std::string word = text.substr(spacePos + 1); const int length = getFont()->getWidth(word); if (length > mMinWidth) mMinWidth = length; } std::stringstream wrappedStream; size_t newlinePos; size_t lastNewlinePos = 0; int minWidth = 0; int xpos; do { // Determine next piece of string to wrap newlinePos = text.find("\n", lastNewlinePos); if (newlinePos == std::string::npos) newlinePos = textSize; std::string line = text.substr(lastNewlinePos, newlinePos - lastNewlinePos); size_t lastSpacePos = 0; xpos = 0; const gcn::Font *const font = getFont(); const int spaceWidth = font->getWidth(" "); size_t sz = line.size(); do { spacePos = line.find(" ", lastSpacePos); if (spacePos == std::string::npos) spacePos = sz; const std::string word = line.substr(lastSpacePos, spacePos - lastSpacePos); const int width = font->getWidth(word); if (xpos == 0 && width > mMinWidth) { mMinWidth = width; xpos = width; wrappedStream << word; } else if (xpos != 0 && xpos + spaceWidth + width <= mMinWidth) { xpos += spaceWidth + width; wrappedStream << " " << word; } else if (lastSpacePos == 0) { xpos += width; wrappedStream << word; } else { if (xpos > minWidth) minWidth = xpos; // The window wasn't big enough. Resize it and try again. if (minWidth > mMinWidth) { mMinWidth = minWidth; wrappedStream.clear(); wrappedStream.str(""); lastNewlinePos = 0; newlinePos = text.find("\n", lastNewlinePos); if (newlinePos == std::string::npos) newlinePos = textSize; line = text.substr(lastNewlinePos, newlinePos - lastNewlinePos); sz = line.size(); break; } else { wrappedStream << "\n" << word; } xpos = width; } lastSpacePos = spacePos + 1; } while (spacePos != sz); if (text.find("\n", lastNewlinePos) != std::string::npos) wrappedStream << "\n"; lastNewlinePos = newlinePos + 1; } while (newlinePos != textSize); if (xpos > minWidth) minWidth = xpos; mMinWidth = minWidth; gcn::TextBox::setText(wrappedStream.str()); } void TextBox::keyPressed(gcn::KeyEvent& keyEvent) { const gcn::Key &key = keyEvent.getKey(); const int action = static_cast<KeyEvent*>(&keyEvent)->getActionId(); switch (action) { case Input::KEY_GUI_LEFT: { --mCaretColumn; if (mCaretColumn < 0) { --mCaretRow; if (mCaretRow < 0) { mCaretRow = 0; mCaretColumn = 0; } else { mCaretColumn = static_cast<int>( mTextRows[mCaretRow].size()); } } break; } case Input::KEY_GUI_RIGHT: { ++mCaretColumn; if (mCaretColumn > static_cast<int>(mTextRows[mCaretRow].size())) { ++ mCaretRow; const int sz = static_cast<int>(mTextRows.size()); if (mCaretRow >= sz) { mCaretRow = sz - 1; if (mCaretRow < 0) mCaretRow = 0; mCaretColumn = static_cast<int>( mTextRows[mCaretRow].size()); } else { mCaretColumn = 0; } } break; } case Input::KEY_GUI_DOWN: { setCaretRow(mCaretRow + 1); break; } case Input::KEY_GUI_UP: { setCaretRow(mCaretRow - 1); break; } case Input::KEY_GUI_HOME: { mCaretColumn = 0; break; } case Input::KEY_GUI_END: { mCaretColumn = static_cast<int>(mTextRows[mCaretRow].size()); break; } case Input::KEY_GUI_SELECT2: { if (mEditable) { mTextRows.insert(mTextRows.begin() + mCaretRow + 1, mTextRows[mCaretRow].substr(mCaretColumn, mTextRows[mCaretRow].size() - mCaretColumn)); mTextRows[mCaretRow].resize(mCaretColumn); ++mCaretRow; mCaretColumn = 0; } break; } case Input::KEY_GUI_BACKSPACE: { if (mCaretColumn != 0 && mEditable) { mTextRows[mCaretRow].erase(mCaretColumn - 1, 1); --mCaretColumn; } else if (mCaretColumn == 0 && mCaretRow != 0 && mEditable) { mCaretColumn = static_cast<int>( mTextRows[mCaretRow - 1].size()); mTextRows[mCaretRow - 1] += mTextRows[mCaretRow]; mTextRows.erase(mTextRows.begin() + mCaretRow); --mCaretRow; } break; } case Input::KEY_GUI_DELETE: { if (mCaretColumn < static_cast<int>( mTextRows[mCaretRow].size()) && mEditable) { mTextRows[mCaretRow].erase(mCaretColumn, 1); } else if (mCaretColumn == static_cast<int>( mTextRows[mCaretRow].size()) && mCaretRow < (static_cast<int>(mTextRows.size()) - 1) && mEditable) { mTextRows[mCaretRow] += mTextRows[mCaretRow + 1]; mTextRows.erase(mTextRows.begin() + mCaretRow + 1); } break; } case Input::KEY_GUI_PAGE_UP: { gcn::Widget *const par = getParent(); if (par) { const int rowsPerPage = par->getChildrenArea().height / getFont()->getHeight(); mCaretRow -= rowsPerPage; if (mCaretRow < 0) mCaretRow = 0; } break; } case Input::KEY_GUI_PAGE_DOWN: { gcn::Widget *const par = getParent(); if (par) { const int rowsPerPage = par->getChildrenArea().height / getFont()->getHeight(); mCaretRow += rowsPerPage; const int sz = static_cast<int>(mTextRows.size()); if (mCaretRow >= sz) mCaretRow = sz - 1; } break; } case Input::KEY_GUI_TAB: { if (mEditable) { mTextRows[mCaretRow].insert(mCaretColumn, std::string(" ")); mCaretColumn += 4; } break; } default: { if (key.isCharacter() && mEditable) { mTextRows[mCaretRow].insert(mCaretColumn, std::string(1, static_cast<signed char>(key.getValue()))); ++ mCaretColumn; } break; } } adjustSize(); scrollToCaret(); keyEvent.consume(); } void TextBox::draw(gcn::Graphics* graphics) { BLOCK_START("TextBox::draw") if (mOpaque) { graphics->setColor(mBackgroundColor); graphics->fillRectangle(gcn::Rectangle(0, 0, getWidth(), getHeight())); } gcn::Font *const font = getFont(); if (isFocused() && isEditable()) { drawCaret(graphics, font->getWidth( mTextRows[mCaretRow].substr(0, mCaretColumn)), mCaretRow * font->getHeight()); } static_cast<Graphics*>(graphics)->setColorAll( mForegroundColor, mForegroundColor2); const int fontHeight = font->getHeight(); for (size_t i = 0, sz = mTextRows.size(); i < sz; i++) { // Move the text one pixel so we can have a caret before a letter. font->drawString(graphics, mTextRows[i], 1, i * fontHeight); } BLOCK_END("TextBox::draw") } void TextBox::setForegroundColor(const gcn::Color &color) { mForegroundColor = color; mForegroundColor2 = color; } void TextBox::setForegroundColorAll(const gcn::Color &color1, const gcn::Color &color2) { mForegroundColor = color1; mForegroundColor2 = color2; }