axmol/core/2d/TextFieldTTF.cpp

774 lines
20 KiB
C++

/****************************************************************************
Copyright (c) 2010-2012 cocos2d-x.org
Copyright (c) 2013-2016 Chukong Technologies Inc.
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
https://axmol.dev/
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
#include <algorithm>
#include "2d/TextFieldTTF.h"
#include "base/Director.h"
#include "platform/FileUtils.h"
#include "base/UTF8.h"
#include "2d/Sprite.h"
namespace ax
{
#define CURSOR_TIME_SHOW_HIDE 0.5f
#define CURSOR_DEFAULT_CHAR '|'
#define PASSWORD_STYLE_TEXT_DEFAULT "\xe2\x80\xa2"
static std::size_t _calcCharCount(const char* text)
{
int n = 0;
char ch = 0;
while ((ch = *text))
{
AX_BREAK_IF(!ch);
if (0x80 != (0xC0 & ch))
{
++n;
}
++text;
}
return n;
}
bool TextFieldDelegate::onTextFieldAttachWithIME(TextFieldTTF* /*sender*/)
{
return false;
}
bool TextFieldDelegate::onTextFieldDetachWithIME(TextFieldTTF* /*sender*/)
{
return false;
}
bool TextFieldDelegate::onTextFieldInsertText(TextFieldTTF* /*sender*/, const char* /*text*/, size_t /*nLen*/)
{
return false;
}
bool TextFieldDelegate::onTextFieldDeleteBackward(TextFieldTTF* /*sender*/, const char* /*delText*/, size_t /*nLen*/)
{
return false;
}
bool TextFieldDelegate::onVisit(TextFieldTTF* /*sender*/,
Renderer* /*renderer*/,
const Mat4& /*transform*/,
uint32_t /*flags*/)
{
return false;
}
//////////////////////////////////////////////////////////////////////////
// constructor and destructor
//////////////////////////////////////////////////////////////////////////
TextFieldTTF::TextFieldTTF()
: _delegate(0)
, _charCount(0)
, _inputText("")
, _placeHolder("") // prevent Label initWithString assertion
, _colorText(Color4B::WHITE)
, _secureTextEntry(false)
, _passwordStyleText(PASSWORD_STYLE_TEXT_DEFAULT)
, _cursorEnabled(false)
, _cursorPosition(0)
, _cursorChar(CURSOR_DEFAULT_CHAR)
, _cursorShowingTime(0.0f)
, _isAttachWithIME(false)
{
_colorSpaceHolder.r = _colorSpaceHolder.g = _colorSpaceHolder.b = 127;
_colorSpaceHolder.a = 255;
}
TextFieldTTF::~TextFieldTTF() {}
//////////////////////////////////////////////////////////////////////////
// static constructor
//////////////////////////////////////////////////////////////////////////
TextFieldTTF* TextFieldTTF::textFieldWithPlaceHolder(std::string_view placeholder,
const Vec2& dimensions,
TextHAlignment alignment,
std::string_view fontName,
float fontSize)
{
TextFieldTTF* ret = new TextFieldTTF();
if (ret->initWithPlaceHolder("", dimensions, alignment, fontName, fontSize))
{
ret->autorelease();
if (placeholder.size() > 0)
{
ret->setPlaceHolder(placeholder);
}
return ret;
}
AX_SAFE_DELETE(ret);
return nullptr;
}
TextFieldTTF* TextFieldTTF::textFieldWithPlaceHolder(std::string_view placeholder,
std::string_view fontName,
float fontSize)
{
TextFieldTTF* ret = new TextFieldTTF();
if (ret->initWithPlaceHolder("", fontName, fontSize))
{
ret->autorelease();
if (placeholder.size() > 0)
{
ret->setPlaceHolder(placeholder);
}
return ret;
}
AX_SAFE_DELETE(ret);
return nullptr;
}
//////////////////////////////////////////////////////////////////////////
// initialize
//////////////////////////////////////////////////////////////////////////
bool TextFieldTTF::initWithPlaceHolder(std::string_view placeholder,
const Vec2& dimensions,
TextHAlignment alignment,
std::string_view fontName,
float fontSize)
{
setDimensions(dimensions.width, dimensions.height);
setAlignment(alignment, TextVAlignment::CENTER);
return initWithPlaceHolder(placeholder, fontName, fontSize);
}
bool TextFieldTTF::initWithPlaceHolder(std::string_view placeholder, std::string_view fontName, float fontSize)
{
_placeHolder = placeholder;
do
{
// If fontName is ttf file and it corrected, use TTFConfig
if (FileUtils::getInstance()->isFileExist(fontName))
{
TTFConfig ttfConfig(fontName, fontSize, GlyphCollection::DYNAMIC);
if (setTTFConfig(ttfConfig))
{
break;
}
}
setSystemFontName(fontName);
setSystemFontSize(fontSize);
} while (false);
setTextColorInternally(_colorSpaceHolder);
Label::setString(_placeHolder);
#if (AX_TARGET_PLATFORM == AX_PLATFORM_MAC || AX_TARGET_PLATFORM == AX_PLATFORM_WIN32 || \
AX_TARGET_PLATFORM == AX_PLATFORM_LINUX)
// On desktop default enable cursor
if (_currentLabelType == LabelType::TTF)
{
setCursorEnabled(true);
}
#endif
return true;
}
//////////////////////////////////////////////////////////////////////////
// IMEDelegate
//////////////////////////////////////////////////////////////////////////
bool TextFieldTTF::attachWithIME()
{
bool ret = IMEDelegate::attachWithIME();
if (ret)
{
// open keyboard
auto pGlView = _director->getGLView();
if (pGlView)
{
pGlView->setIMEKeyboardState(true);
}
}
return ret;
}
bool TextFieldTTF::detachWithIME()
{
bool ret = IMEDelegate::detachWithIME();
if (ret)
{
// close keyboard
auto glView = _director->getGLView();
if (glView)
{
glView->setIMEKeyboardState(false);
}
}
return ret;
}
void TextFieldTTF::didAttachWithIME()
{
setAttachWithIME(true);
}
void TextFieldTTF::didDetachWithIME()
{
setAttachWithIME(false);
}
bool TextFieldTTF::canAttachWithIME()
{
return (_delegate) ? (!_delegate->onTextFieldAttachWithIME(this)) : true;
}
bool TextFieldTTF::canDetachWithIME()
{
return (_delegate) ? (!_delegate->onTextFieldDetachWithIME(this)) : true;
}
void TextFieldTTF::insertText(const char* text, size_t len)
{
std::string insert(text, len);
// insert \n means input end
int pos = static_cast<int>(insert.find(StringUtils::AsciiCharacters::NewLine));
if ((int)insert.npos != pos)
{
len = pos;
insert.erase(pos);
}
if (len > 0)
{
if (_delegate && _delegate->onTextFieldInsertText(this, insert.c_str(), len))
{
// delegate doesn't want to insert text
return;
}
std::size_t countInsertChar = _calcCharCount(insert.c_str());
_charCount += countInsertChar;
if (_cursorEnabled)
{
StringUtils::StringUTF8 stringUTF8;
stringUTF8.replace(_inputText);
stringUTF8.insert(_cursorPosition, insert);
setCursorPosition(_cursorPosition + countInsertChar);
setString(stringUTF8.getAsCharSequence());
}
else
{
std::string sText(_inputText);
sText.append(insert);
setString(sText);
}
}
if ((int)insert.npos == pos)
{
return;
}
// '\n' inserted, let delegate process first
if (_delegate && _delegate->onTextFieldInsertText(this, "\n", 1))
{
return;
}
// if delegate hasn't processed, detach from IME by default
detachWithIME();
}
void TextFieldTTF::deleteBackward(size_t numChars)
{
size_t len = _inputText.length();
if (!len)
{
// there is no string
return;
}
// Length of characters to delete is based on input editor, but the actual
// length of the displayed text may be less
numChars = std::min(numChars, len);
size_t totalDeleteLen = 0;
for (auto i = 0; i < numChars; ++i)
{
// get the delete byte number
size_t deleteLen = 1; // default, erase 1 byte
// Calculate the actual number of bytes to delete for a specific character
while (0x80 == (0xC0 & _inputText.at(len - totalDeleteLen - deleteLen)))
{
++deleteLen;
}
totalDeleteLen += deleteLen;
}
if (_delegate &&
_delegate->onTextFieldDeleteBackward(this, _inputText.c_str() + len - totalDeleteLen, static_cast<int>(totalDeleteLen)))
{
// delegate doesn't want to delete backwards
return;
}
// if all text deleted, show placeholder string
if (len <= totalDeleteLen)
{
_inputText = "";
_charCount = 0;
setCursorPosition(0);
setString(_inputText);
return;
}
// set new input text
if (_cursorEnabled)
{
if (_cursorPosition)
{
setCursorPosition(_cursorPosition - 1);
StringUtils::StringUTF8 stringUTF8;
stringUTF8.replace(_inputText);
stringUTF8.deleteChar(_cursorPosition);
_charCount = stringUTF8.length();
setString(stringUTF8.getAsCharSequence());
}
}
else
{
std::string text(_inputText.c_str(), len - totalDeleteLen);
setString(text);
}
}
std::string_view TextFieldTTF::getContentText()
{
return _inputText;
}
void TextFieldTTF::setCursorPosition(std::size_t cursorPosition)
{
if (_cursorEnabled && cursorPosition <= (std::size_t)_charCount)
{
_cursorPosition = cursorPosition;
_cursorShowingTime = CURSOR_TIME_SHOW_HIDE * 2.0f;
}
}
void TextFieldTTF::setCursorFromPoint(const Vec2& point, const Camera* camera)
{
if (_cursorEnabled)
{
// Reset Label, no cursor
bool oldIsAttachWithIME = _isAttachWithIME;
_isAttachWithIME = false;
updateCursorDisplayText();
Rect rect;
rect.size = getContentSize();
if (isScreenPointInRect(point, camera, getWorldToNodeTransform(), rect, nullptr))
{
int latterPosition = 0;
for (; latterPosition < _lengthOfString; ++latterPosition)
{
if (_lettersInfo[latterPosition].valid && _lettersInfo[latterPosition].atlasIndex >= 0)
{
auto sprite = getLetter(latterPosition);
if (sprite)
{
rect.size = Vec2(sprite->getContentSize().width, _lineHeight);
if (isScreenPointInRect(point, camera, sprite->getWorldToNodeTransform(), rect, nullptr))
{
setCursorPosition(latterPosition);
break;
}
}
}
}
if (latterPosition == _lengthOfString)
{
setCursorPosition(latterPosition);
}
}
// Set cursor
_isAttachWithIME = oldIsAttachWithIME;
updateCursorDisplayText();
}
}
void TextFieldTTF::setAttachWithIME(bool isAttachWithIME)
{
if (isAttachWithIME != _isAttachWithIME)
{
_isAttachWithIME = isAttachWithIME;
if (_isAttachWithIME)
{
setCursorPosition(_charCount);
}
updateCursorDisplayText();
}
}
void TextFieldTTF::setTextColorInternally(const Color4B& color)
{
if (_currentLabelType == LabelType::BMFONT)
{
Label::setColor(Color3B(color));
return;
}
Label::setTextColor(color);
}
void TextFieldTTF::setTextColor(const Color4B& color)
{
_colorText = color;
if (!_inputText.empty())
{
setTextColorInternally(color);
}
}
void TextFieldTTF::visit(Renderer* renderer, const Mat4& parentTransform, uint32_t parentFlags)
{
if (_delegate && _delegate->onVisit(this, renderer, parentTransform, parentFlags))
{
return;
}
Label::visit(renderer, parentTransform, parentFlags);
}
void TextFieldTTF::update(float delta)
{
if (_cursorEnabled && _isAttachWithIME)
{
_cursorShowingTime -= delta;
if (_cursorShowingTime < -CURSOR_TIME_SHOW_HIDE)
{
_cursorShowingTime = CURSOR_TIME_SHOW_HIDE;
}
// before cursor inserted '\b', need next letter
auto sprite = getLetter((int)_cursorPosition + 1);
if (sprite)
{
if (_currentLabelType == LabelType::BMFONT)
{
sprite->setColor(getColor());
}
if (_cursorShowingTime >= 0.0f)
{
sprite->setOpacity(255);
}
else
{
sprite->setOpacity(0);
}
sprite->setDirty(true);
}
}
}
const Color4B& TextFieldTTF::getColorSpaceHolder()
{
return _colorSpaceHolder;
}
void TextFieldTTF::setColorSpaceHolder(const Color3B& color)
{
setColorSpaceHolder(Color4B(color));
}
void TextFieldTTF::setColorSpaceHolder(const Color4B& color)
{
_colorSpaceHolder = color;
if (_inputText.empty())
{
setTextColorInternally(_colorSpaceHolder);
}
}
//////////////////////////////////////////////////////////////////////////
// properties
//////////////////////////////////////////////////////////////////////////
// input text property
void TextFieldTTF::setString(std::string_view text)
{
std::string displayText;
std::size_t charCount = 0;
if (!text.empty())
{
_inputText = text;
displayText = _inputText;
charCount = _calcCharCount(_inputText.c_str());
if (_secureTextEntry)
{
displayText = "";
size_t length = charCount;
while (length)
{
displayText.append(_passwordStyleText);
--length;
}
}
}
else
{
_inputText = "";
}
if (_cursorEnabled && charCount != _charCount)
{
_cursorPosition = charCount;
}
if (_cursorEnabled)
{
// Need for recreate all letters in Label
Label::removeAllChildrenWithCleanup(false);
}
// if there is no input text, display placeholder instead
if (_inputText.empty() && (!_cursorEnabled || !_isAttachWithIME))
{
setTextColorInternally(_colorSpaceHolder);
Label::setString(_placeHolder);
}
else
{
makeStringSupportCursor(displayText);
setTextColorInternally(_colorText);
Label::setString(displayText);
}
_charCount = charCount;
}
void TextFieldTTF::appendString(std::string_view text)
{
insertText(text.data(), text.length());
}
void TextFieldTTF::makeStringSupportCursor(std::string& displayText)
{
if (_cursorEnabled && _isAttachWithIME)
{
if (displayText.empty())
{
// \b - Next char not change x position
if (_currentLabelType == LabelType::TTF || _currentLabelType == LabelType::BMFONT)
displayText.push_back(StringUtils::AsciiCharacters::NextCharNoChangeX);
displayText.push_back(_cursorChar);
}
else
{
StringUtils::StringUTF8 stringUTF8;
stringUTF8.replace(displayText);
if (_cursorPosition > stringUTF8.length())
{
_cursorPosition = stringUTF8.length();
}
std::string cursorChar;
// \b - Next char not change x position
if (_currentLabelType == LabelType::TTF || _currentLabelType == LabelType::BMFONT)
cursorChar.push_back(StringUtils::AsciiCharacters::NextCharNoChangeX);
cursorChar.push_back(_cursorChar);
stringUTF8.insert(_cursorPosition, cursorChar);
displayText = stringUTF8.getAsCharSequence();
}
}
}
void TextFieldTTF::updateCursorDisplayText()
{
// Update Label content
setString(_inputText);
}
void TextFieldTTF::setCursorChar(char cursor)
{
if (_cursorChar != cursor)
{
_cursorChar = cursor;
updateCursorDisplayText();
}
}
void TextFieldTTF::controlKey(EventKeyboard::KeyCode keyCode)
{
if (_cursorEnabled)
{
switch (keyCode)
{
case EventKeyboard::KeyCode::KEY_HOME:
case EventKeyboard::KeyCode::KEY_KP_HOME:
setCursorPosition(0);
updateCursorDisplayText();
break;
case EventKeyboard::KeyCode::KEY_END:
setCursorPosition(_charCount);
updateCursorDisplayText();
break;
case EventKeyboard::KeyCode::KEY_DELETE:
case EventKeyboard::KeyCode::KEY_KP_DELETE:
if (_cursorPosition < (std::size_t)_charCount)
{
StringUtils::StringUTF8 stringUTF8;
stringUTF8.replace(_inputText);
stringUTF8.deleteChar(_cursorPosition);
setCursorPosition(_cursorPosition);
_charCount = stringUTF8.length();
setString(stringUTF8.getAsCharSequence());
}
break;
case EventKeyboard::KeyCode::KEY_LEFT_ARROW:
if (_cursorPosition)
{
setCursorPosition(_cursorPosition - 1);
updateCursorDisplayText();
}
break;
case EventKeyboard::KeyCode::KEY_RIGHT_ARROW:
if (_cursorPosition < (std::size_t)_charCount)
{
setCursorPosition(_cursorPosition + 1);
updateCursorDisplayText();
}
break;
case EventKeyboard::KeyCode::KEY_ESCAPE:
detachWithIME();
break;
default:
break;
}
}
}
std::string_view TextFieldTTF::getString() const
{
return _inputText;
}
// place holder text property
void TextFieldTTF::setPlaceHolder(std::string_view text)
{
_placeHolder = text;
if (_inputText.empty() && !_isAttachWithIME)
{
setTextColorInternally(_colorSpaceHolder);
Label::setString(_placeHolder);
}
}
std::string_view TextFieldTTF::getPlaceHolder() const
{
return _placeHolder;
}
void TextFieldTTF::setCursorEnabled(bool enabled)
{
if (_cursorEnabled == enabled)
{
return;
}
_cursorEnabled = enabled;
if (_cursorEnabled)
{
_cursorPosition = _charCount;
if (_currentLabelType == LabelType::TTF || _currentLabelType == LabelType::BMFONT)
{
scheduleUpdate();
}
return;
}
_cursorPosition = 0;
if (_currentLabelType == LabelType::TTF || _currentLabelType == LabelType::BMFONT)
{
unscheduleUpdate();
}
}
// secureTextEntry
void TextFieldTTF::setSecureTextEntry(bool value)
{
if (_secureTextEntry != value)
{
_secureTextEntry = value;
setString(_inputText);
}
}
void TextFieldTTF::setPasswordTextStyle(std::string_view text)
{
if (text.length() < 1)
{
return;
}
if (text != _passwordStyleText)
{
_passwordStyleText = text;
setString(_inputText);
}
}
std::string_view TextFieldTTF::getPasswordTextStyle() const
{
return _passwordStyleText;
}
bool TextFieldTTF::isSecureTextEntry() const
{
return _secureTextEntry;
}
}