/*
* The Mana Client
* Copyright (C) 2004-2009 The Mana World Development Team
* Copyright (C) 2009-2012 The Mana Developers
*
* This file is part of The Mana 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 .
*/
#include "gui/widgets/textfield.h"
#include "graphics.h"
#include "gui/gui.h"
#include "gui/sdlinput.h"
#include "resources/theme.h"
#include "utils/copynpaste.h"
#include "utils/stringutils.h"
#include
#include
TextField::TextField(const std::string &text, bool loseFocusOnTab)
: gcn::TextField(text)
, mLoseFocusOnTab(loseFocusOnTab)
{
auto &skin = gui->getTheme()->getSkin(SkinType::TextField);
setFrameSize(skin.frameSize);
mPadding = skin.padding;
setWidth(getFont()->getWidth(mText) + 2 * mPadding);
setHeight(getFont()->getHeight() + 2 * mPadding);
fixScroll();
}
void TextField::draw(gcn::Graphics *graphics)
{
if (getFrameSize() == 0)
drawFrame(graphics);
auto g = static_cast(graphics);
g->pushClipRect(gcn::Rectangle(0, 0, getWidth(), getHeight()));
if (isFocused())
{
drawCaret(graphics,
getFont()->getWidth(mText.substr(0, mCaretPosition)) - mXScroll);
}
graphics->setColor(Theme::getThemeColor(Theme::TEXT));
graphics->setFont(getFont());
graphics->drawText(mText, mPadding - mXScroll, mPadding);
g->popClipRect();
}
void TextField::drawFrame(gcn::Graphics *graphics)
{
const int bs = getFrameSize();
WidgetState state(this);
state.width += bs * 2;
state.height += bs * 2;
gui->getTheme()->drawSkin(static_cast(graphics), SkinType::TextField, state);
}
void TextField::setNumeric(bool numeric)
{
mNumeric = numeric;
if (!numeric)
return;
const char *text = mText.c_str();
for (const char *textPtr = text; *textPtr; ++textPtr)
{
if (*textPtr < '0' || *textPtr > '9')
{
setText(mText.substr(0, textPtr - text));
return;
}
}
}
int TextField::getValue() const
{
if (!mNumeric)
return 0;
int value = atoi(mText.c_str());
if (value < mMinimum)
return mMinimum;
if (value > mMaximum)
return mMaximum;
return value;
}
void TextField::drawCaret(gcn::Graphics *graphics, int x)
{
graphics->setColor(Theme::getThemeColor(Theme::CARET));
graphics->drawLine(mPadding + x, mPadding, mPadding + x, getHeight() - mPadding);
}
void TextField::keyPressed(gcn::KeyEvent &keyEvent)
{
switch (keyEvent.getKey().getValue())
{
case Key::LEFT:
{
while (mCaretPosition > 0)
{
--mCaretPosition;
if ((mText[mCaretPosition] & 192) != 128)
break;
}
} break;
case Key::RIGHT:
{
unsigned sz = mText.size();
while (mCaretPosition < sz)
{
++mCaretPosition;
if (mCaretPosition == sz ||
(mText[mCaretPosition] & 192) != 128)
break;
}
} break;
case Key::UP:
{
if (mHistory && !mHistory->atBegining() && !mHistory->empty())
{
// Move backward through the history
mHistory->current--;
setText(*mHistory->current);
setCaretPosition(getText().length());
}
} break;
case Key::DOWN:
{
if (mHistory && !mHistory->atEnd())
{
// Move forward through the history
auto prevHist = mHistory->current++;
if (!mHistory->atEnd())
{
setText(*mHistory->current);
setCaretPosition(getText().length());
}
else
{
setText(std::string());
mHistory->current = prevHist;
}
}
else if (!getText().empty())
{
// Always clear (easy access to useful function)
setText(std::string());
}
} break;
case Key::DELETE_KEY:
{
unsigned sz = mText.size();
while (mCaretPosition < sz)
{
--sz;
mText.erase(mCaretPosition, 1);
if (mCaretPosition == sz ||
(mText[mCaretPosition] & 192) != 128)
break;
}
} break;
case Key::BACKSPACE:
{
while (mCaretPosition > 0)
{
--mCaretPosition;
int v = mText[mCaretPosition];
mText.erase(mCaretPosition, 1);
if ((v & 192) != 128) break;
}
} break;
case Key::ENTER:
if (mHistory)
{
// If the input is different from previous, put it in the history
if (!getText().empty() && (mHistory->empty() ||
!mHistory->matchesLastEntry(getText())))
{
mHistory->addEntry(getText());
}
mHistory->toEnd();
}
distributeActionEvent();
break;
case Key::HOME:
mCaretPosition = 0;
break;
case Key::END:
mCaretPosition = mText.size();
break;
case Key::TAB:
autoComplete();
if (mLoseFocusOnTab)
return;
break;
case SDLK_v:
if (keyEvent.isControlPressed())
handlePaste();
break;
}
keyEvent.consume();
fixScroll();
}
void TextField::textInput(const TextInput &textInput)
{
mText.insert(mCaretPosition, textInput.getText());
mCaretPosition += textInput.getText().length();
}
void TextField::autoComplete()
{
if (mAutoComplete && !mText.empty())
{
const int caretPos = getCaretPosition();
int startName = 0;
const std::string inputText = getText();
std::string name = inputText.substr(0, caretPos);
std::string newName;
for (int f = caretPos - 1; f > -1; f--)
{
if (isWordSeparator(inputText[f]))
{
startName = f + 1;
name = inputText.substr(f + 1, caretPos - startName);
break;
}
}
if (caretPos == startName)
return;
std::vector nameList;
mAutoComplete->getAutoCompleteList(nameList);
newName = autocomplete(nameList, name);
if (newName.empty() && mHistory)
{
auto i = mHistory->history.begin();
std::vector nameList;
while (i != mHistory->history.end())
{
std::string line = *i;
unsigned int f = 0;
while (f < line.length() && !isWordSeparator(line.at(f)))
{
f++;
}
line = line.substr(0, f);
if (!line.empty())
{
nameList.push_back(line);
}
++i;
}
newName = autocomplete(nameList, name);
}
if (!newName.empty())
{
if(inputText[0] == '@' || inputText[0] == '/')
newName = "\"" + newName + "\"";
setText(inputText.substr(0, startName) + newName
+ inputText.substr(caretPos, inputText.length()
- caretPos));
setCaretPosition(caretPos - name.length() + newName.length());
}
}
}
void TextField::handlePaste()
{
std::string text = getText();
std::string::size_type caretPos = getCaretPosition();
if (insertFromClipboard(text, caretPos)) {
setText(text);
setCaretPosition(caretPos);
}
}