/**************************************************************************** Copyright (c) 2013 cocos2d-x.org Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd. Copyright (c) 2021 Bytedance Inc. https://axmolengine.github.io/ 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 "ui/UIRichText.h" #include #include #include #include #include "platform/CCFileUtils.h" #include "platform/CCApplication.h" #include "base/CCEventListenerTouch.h" #include "base/CCEventDispatcher.h" #include "base/CCDirector.h" #include "2d/CCLabel.h" #include "2d/CCSprite.h" #include "base/ccUTF8.h" #include "ui/UIHelper.h" #include "fmt/compile.h" #include "platform/CCSAXParser.h" USING_NS_AX; using namespace ax::ui; class ListenerComponent : public Component { public: static const std::string COMPONENT_NAME; /*!< component name */ static ListenerComponent* create(Node* parent, std::string_view url, const RichText::OpenUrlHandler handleOpenUrl = nullptr) { auto component = new ListenerComponent(parent, url, std::move(handleOpenUrl)); component->autorelease(); return component; } explicit ListenerComponent(Node* parent, std::string_view url, const RichText::OpenUrlHandler handleOpenUrl) : _parent(parent), _url(url), _handleOpenUrl(std::move(handleOpenUrl)) { setName(ListenerComponent::COMPONENT_NAME); _touchListener = ax::EventListenerTouchAllAtOnce::create(); _touchListener->onTouchesEnded = AX_CALLBACK_2(ListenerComponent::onTouchesEnded, this); Director::getInstance()->getEventDispatcher()->addEventListenerWithSceneGraphPriority(_touchListener, _parent); _touchListener->retain(); } virtual ~ListenerComponent() { Director::getInstance()->getEventDispatcher()->removeEventListener(_touchListener); _touchListener->release(); } void onTouchesEnded(const std::vector& touches, Event* /*event*/) { for (const auto& touch : touches) { // FIXME: Node::getBoundBox() doesn't return it in local coordinates... so create one manually. Rect localRect = Rect(Vec2::ZERO, _parent->getContentSize()); if (localRect.containsPoint(_parent->convertTouchToNodeSpace(touch))) { if (_handleOpenUrl) { _handleOpenUrl(_url); } } } } void setOpenUrlHandler(const RichText::OpenUrlHandler& handleOpenUrl) { _handleOpenUrl = handleOpenUrl; } private: Node* _parent; // weak ref. std::string _url; RichText::OpenUrlHandler _handleOpenUrl; EventListenerTouchAllAtOnce* _touchListener; // strong ref. }; const std::string ListenerComponent::COMPONENT_NAME("cocos2d_ui_UIRichText_ListenerComponent"); bool RichElement::init(int tag, const Color3B& color, uint8_t opacity) { _tag = tag; _color = color; _opacity = opacity; return true; } bool RichElement::equalType(Type type) { return (_type == type); } void RichElement::setColor(const Color3B& color) { _color = color; } RichElementText* RichElementText::create(int tag, const Color3B& color, uint8_t opacity, std::string_view text, std::string_view fontName, float fontSize, uint32_t flags, std::string_view url, const Color3B& outlineColor, int outlineSize, const Color3B& shadowColor, const Vec2& shadowOffset, int shadowBlurRadius, const Color3B& glowColor) { RichElementText* element = new RichElementText(); if (element->init(tag, color, opacity, text, fontName, fontSize, flags, url, outlineColor, outlineSize, shadowColor, shadowOffset, shadowBlurRadius, glowColor)) { element->autorelease(); return element; } AX_SAFE_DELETE(element); return nullptr; } bool RichElementText::init(int tag, const Color3B& color, uint8_t opacity, std::string_view text, std::string_view fontName, float fontSize, uint32_t flags, std::string_view url, const Color3B& outlineColor, int outlineSize, const Color3B& shadowColor, const Vec2& shadowOffset, int shadowBlurRadius, const Color3B& glowColor) { if (RichElement::init(tag, color, opacity)) { _text = text; _fontName = fontName; _fontSize = fontSize; _flags = flags; _url = url; _outlineColor = outlineColor; _outlineSize = outlineSize; _shadowColor = shadowColor; _shadowOffset = shadowOffset; _shadowBlurRadius = shadowBlurRadius; _glowColor = glowColor; return true; } return false; } RichElementImage* RichElementImage::create(int tag, const Color3B& color, uint8_t opacity, std::string_view filePath, std::string_view url, Widget::TextureResType texType) { RichElementImage* element = new RichElementImage(); if (element->init(tag, color, opacity, filePath, url, texType)) { element->autorelease(); return element; } AX_SAFE_DELETE(element); return nullptr; } bool RichElementImage::init(int tag, const Color3B& color, uint8_t opacity, std::string_view filePath, std::string_view url, Widget::TextureResType texType) { if (RichElement::init(tag, color, opacity)) { _filePath = filePath; _width = -1; _height = -1; _url = url; _textureType = texType; return true; } return false; } void RichElementImage::setWidth(int width) { _width = width; } void RichElementImage::setHeight(int height) { _height = height; } void RichElementImage::setUrl(std::string_view url) { _url = url; } RichElementCustomNode* RichElementCustomNode::create(int tag, const Color3B& color, uint8_t opacity, ax::Node* customNode) { RichElementCustomNode* element = new RichElementCustomNode(); if (element->init(tag, color, opacity, customNode)) { element->autorelease(); return element; } AX_SAFE_DELETE(element); return nullptr; } bool RichElementCustomNode::init(int tag, const Color3B& color, uint8_t opacity, ax::Node* customNode) { if (RichElement::init(tag, color, opacity)) { _customNode = customNode; _customNode->retain(); return true; } return false; } RichElementNewLine* RichElementNewLine::create(int tag, const Color3B& color, uint8_t opacity) { RichElementNewLine* element = new RichElementNewLine(); if (element->init(tag, color, opacity)) { element->autorelease(); return element; } AX_SAFE_DELETE(element); return nullptr; } /** @brief parse a XML. */ class MyXMLVisitor : public SAXDelegator { public: /** @brief underline or strikethrough */ enum class StyleLine { NONE, UNDERLINE, /*!< underline */ STRIKETHROUGH /*!< a typographical presentation of words with a horizontal line through their center */ }; /** @brief outline, shadow or glow */ enum class StyleEffect { NONE, OUTLINE, /*!< outline effect enabled */ SHADOW, /*!< shadow effect enabled */ GLOW /*!< glow effect enabled @discussion Limiting use to only when the Label created with true type font. */ }; /** @brief the attributes of text tag */ struct Attributes { std::string face; /*!< font name */ std::string url; /*!< url is a attribute of a anchor tag */ float fontSize; /*!< font size */ Color3B color; /*!< font color */ bool hasColor; /*!< or color is specified? */ bool bold; /*!< bold text */ bool italics; /*!< italic text */ StyleLine line; /*!< underline or strikethrough */ StyleEffect effect; /*!< outline, shadow or glow */ Color3B outlineColor; /*!< the color of the outline */ int outlineSize; /*!< the outline effect size value */ Color3B shadowColor; /*!< the shadow effect color value */ Vec2 shadowOffset; /*!< shadow effect offset value */ int shadowBlurRadius; /*!< the shadow effect blur radius */ Color3B glowColor; /*!< the glow effect color value */ void setColor(const Color3B& acolor) { color = acolor; hasColor = true; } Attributes() : fontSize(-1) , hasColor(false) , bold(false) , italics(false) , line(StyleLine::NONE) , effect(StyleEffect::NONE) {} }; private: std::vector _fontElements; RichText* _richText; struct TagBehavior { bool isFontElement; RichText::VisitEnterHandler handleVisitEnter; }; typedef hlookup::string_map TagTables; static TagTables _tagTables; public: explicit MyXMLVisitor(RichText* richText); virtual ~MyXMLVisitor(); Color3B getColor() const; float getFontSize() const; std::string getFace() const; std::string getURL() const; bool getBold() const; bool getItalics() const; bool getUnderline() const; bool getStrikethrough() const; std::tuple getOutline() const; std::tuple getShadow() const; std::tuple getGlow() const; void startElement(void* ctx, const char* name, const char** atts) override; void endElement(void* ctx, const char* name) override; void textHandler(void* ctx, const char* s, size_t len) override; void pushBackFontElement(const Attributes& attribs); void popBackFontElement(); void pushBackElement(RichElement* element); static void setTagDescription(std::string_view tag, bool isFontElement, RichText::VisitEnterHandler&& handleVisitEnter); static void removeTagDescription(std::string_view tag); private: ValueMap tagAttrMapWithXMLElement(const char** attrs); }; MyXMLVisitor::TagTables MyXMLVisitor::_tagTables; MyXMLVisitor::MyXMLVisitor(RichText* richText) : _fontElements(20), _richText(richText) { MyXMLVisitor::setTagDescription("font", true, [](const ValueMap& tagAttrValueMap) { // supported attributes: // size, color, align, face ValueMap attrValueMap; if (tagAttrValueMap.find("size") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_FONT_SIZE] = tagAttrValueMap.at("size").asString(); } if (tagAttrValueMap.find("color") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_FONT_COLOR_STRING] = tagAttrValueMap.at("color").asString(); } if (tagAttrValueMap.find("face") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_FONT_FACE] = tagAttrValueMap.at("face").asString(); } return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("b", true, [](const ValueMap& /*tagAttrValueMap*/) { // no supported attributes ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_BOLD] = true; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("i", true, [](const ValueMap& /*tagAttrValueMap*/) { // no supported attributes ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_ITALIC] = true; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("del", true, [](const ValueMap& /*tagAttrValueMap*/) { // no supported attributes ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_LINE] = RichText::VALUE_TEXT_LINE_DEL; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("u", true, [](const ValueMap& /*tagAttrValueMap*/) { // no supported attributes ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_LINE] = RichText::VALUE_TEXT_LINE_UNDER; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("small", true, [](const ValueMap& /*tagAttrValueMap*/) { ValueMap attrValueMap; attrValueMap[RichText::KEY_FONT_SMALL] = true; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("big", true, [](const ValueMap& /*tagAttrValueMap*/) { ValueMap attrValueMap; attrValueMap[RichText::KEY_FONT_BIG] = true; return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("img", false, [](const ValueMap& tagAttrValueMap) { // supported attributes: // src, height, width, scaleX, scaleY, scale std::string src; int height = -1; int width = -1; float scaleX = 1.f; float scaleY = 1.f; Widget::TextureResType resType = Widget::TextureResType::LOCAL; auto it = tagAttrValueMap.find("src"); if (it != tagAttrValueMap.end()) { src = it->second.asString(); } it = tagAttrValueMap.find("height"); if (it != tagAttrValueMap.end()) { auto str = it->second.asStringRef(); if (!str.empty() && str[str.length() - 1] == '%') { scaleY = std::atoi(str.data()) / 100.f; } else { height = it->second.asInt(); } } it = tagAttrValueMap.find("width"); if (it != tagAttrValueMap.end()) { auto str = it->second.asStringRef(); if (!str.empty() && str[str.length() - 1] == '%') { scaleX = std::atoi(str.data()) / 100.f; } else { width = it->second.asInt(); } } it = tagAttrValueMap.find("scaleX"); if (it != tagAttrValueMap.end()) { scaleX = it->second.asFloat(); } it = tagAttrValueMap.find("scaleY"); if (it != tagAttrValueMap.end()) { scaleY = it->second.asFloat(); } it = tagAttrValueMap.find("scale"); if (it != tagAttrValueMap.end()) { scaleX = scaleY = it->second.asFloat(); } it = tagAttrValueMap.find("type"); if (it != tagAttrValueMap.end()) { // texture type // 0: normal file path // 1: sprite frame name int type = it->second.asInt(); resType = type == 0 ? Widget::TextureResType::LOCAL : Widget::TextureResType::PLIST; } RichElementImage* elementImg = nullptr; if (src.length()) { elementImg = RichElementImage::create(0, Color3B::WHITE, 255, src, "", resType); if (height >= 0) elementImg->setHeight(height); if (width >= 0) elementImg->setWidth(width); elementImg->setScaleX(scaleX); elementImg->setScaleY(scaleY); } return make_pair(ValueMap(), elementImg); }); MyXMLVisitor::setTagDescription("a", true, [](const ValueMap& tagAttrValueMap) { // supported attributes: ValueMap attrValueMap; if (tagAttrValueMap.find("href") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_URL] = tagAttrValueMap.at("href").asString(); } return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("br", false, [](const ValueMap& /*tagAttrValueMap*/) { RichElementNewLine* richElement = RichElementNewLine::create(0, Color3B::WHITE, 255); return make_pair(ValueMap(), richElement); }); MyXMLVisitor::setTagDescription("outline", true, [](const ValueMap& tagAttrValueMap) { // supported attributes: // color, size ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_STYLE] = RichText::VALUE_TEXT_STYLE_OUTLINE; if (tagAttrValueMap.find("color") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_OUTLINE_COLOR] = tagAttrValueMap.at("color").asString(); } if (tagAttrValueMap.find("size") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_OUTLINE_SIZE] = tagAttrValueMap.at("size").asString(); } return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("shadow", true, [](const ValueMap& tagAttrValueMap) { // supported attributes: // color, offsetWidth, offsetHeight, blurRadius ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_STYLE] = RichText::VALUE_TEXT_STYLE_SHADOW; if (tagAttrValueMap.find("color") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_SHADOW_COLOR] = tagAttrValueMap.at("color").asString(); } if (tagAttrValueMap.find("offsetWidth") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_SHADOW_OFFSET_WIDTH] = tagAttrValueMap.at("offsetWidth").asString(); } if (tagAttrValueMap.find("offsetHeight") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_SHADOW_OFFSET_HEIGHT] = tagAttrValueMap.at("offsetHeight").asString(); } if (tagAttrValueMap.find("blurRadius") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_SHADOW_BLUR_RADIUS] = tagAttrValueMap.at("blurRadius").asString(); } return make_pair(attrValueMap, nullptr); }); MyXMLVisitor::setTagDescription("glow", true, [](const ValueMap& tagAttrValueMap) { // supported attributes: // color ValueMap attrValueMap; attrValueMap[RichText::KEY_TEXT_STYLE] = RichText::VALUE_TEXT_STYLE_GLOW; if (tagAttrValueMap.find("color") != tagAttrValueMap.end()) { attrValueMap[RichText::KEY_TEXT_GLOW_COLOR] = tagAttrValueMap.at("color").asString(); } return make_pair(attrValueMap, nullptr); }); } MyXMLVisitor::~MyXMLVisitor() {} Color3B MyXMLVisitor::getColor() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->hasColor) return i->color; } return Color3B::WHITE; } float MyXMLVisitor::getFontSize() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->fontSize != -1) return i->fontSize; } return 12; } std::string MyXMLVisitor::getFace() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (!i->face.empty()) return i->face; } return "fonts/Marker Felt.ttf"; } std::string MyXMLVisitor::getURL() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (!i->url.empty()) return i->url; } return ""; } bool MyXMLVisitor::getBold() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->bold) return true; } return false; } bool MyXMLVisitor::getItalics() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->italics) return true; } return false; } bool MyXMLVisitor::getUnderline() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->line == StyleLine::UNDERLINE) return true; } return false; } bool MyXMLVisitor::getStrikethrough() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->line == StyleLine::STRIKETHROUGH) return true; } return false; } std::tuple MyXMLVisitor::getOutline() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->effect == StyleEffect::OUTLINE) return std::make_tuple(true, i->outlineColor, i->outlineSize); } return std::make_tuple(false, Color3B::WHITE, -1); } std::tuple MyXMLVisitor::getShadow() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->effect == StyleEffect::SHADOW) return std::make_tuple(true, i->shadowColor, i->shadowOffset, i->shadowBlurRadius); } return std::make_tuple(false, Color3B::BLACK, Vec2(2.0, -2.0), 0); } std::tuple MyXMLVisitor::getGlow() const { for (auto i = _fontElements.rbegin(), iRend = _fontElements.rend(); i != iRend; ++i) { if (i->effect == StyleEffect::GLOW) return std::make_tuple(true, i->glowColor); } return std::make_tuple(false, Color3B::WHITE); } void MyXMLVisitor::startElement(void* /*ctx*/, const char* elementName, const char** atts) { auto it = _tagTables.find(elementName); if (it != _tagTables.end()) { auto tagBehavior = it->second; if (tagBehavior.handleVisitEnter != nullptr) { ValueMap&& tagAttrValueMap = tagAttrMapWithXMLElement(atts); auto result = tagBehavior.handleVisitEnter(tagAttrValueMap); ValueMap& attrValueMap = result.first; RichElement* richElement = result.second; if (tagBehavior.isFontElement) { Attributes attributes; if (attrValueMap.find(RichText::KEY_FONT_SIZE) != attrValueMap.end()) { attributes.fontSize = attrValueMap.at(RichText::KEY_FONT_SIZE).asFloat(); } if (attrValueMap.find(RichText::KEY_FONT_SMALL) != attrValueMap.end()) { attributes.fontSize = getFontSize() * 0.8f; } if (attrValueMap.find(RichText::KEY_FONT_BIG) != attrValueMap.end()) { attributes.fontSize = getFontSize() * 1.25f; } if (attrValueMap.find(RichText::KEY_FONT_COLOR_STRING) != attrValueMap.end()) { attributes.setColor( _richText->color3BWithString(attrValueMap.at(RichText::KEY_FONT_COLOR_STRING).asString())); } if (attrValueMap.find(RichText::KEY_FONT_FACE) != attrValueMap.end()) { attributes.face = attrValueMap.at(RichText::KEY_FONT_FACE).asString(); } if (attrValueMap.find(RichText::KEY_TEXT_BOLD) != attrValueMap.end()) { attributes.bold = true; } if (attrValueMap.find(RichText::KEY_TEXT_ITALIC) != attrValueMap.end()) { attributes.italics = true; } if (attrValueMap.find(RichText::KEY_TEXT_LINE) != attrValueMap.end()) { auto keyTextLine = attrValueMap.at(RichText::KEY_TEXT_LINE).asString(); if (keyTextLine == RichText::VALUE_TEXT_LINE_DEL) { attributes.line = StyleLine::STRIKETHROUGH; } else if (keyTextLine == RichText::VALUE_TEXT_LINE_UNDER) { attributes.line = StyleLine::UNDERLINE; } } if (attrValueMap.find(RichText::KEY_URL) != attrValueMap.end()) { attributes.url = attrValueMap.at(RichText::KEY_URL).asString(); attributes.setColor(_richText->getAnchorFontColor3B()); if (_richText->isAnchorTextBoldEnabled()) { attributes.bold = true; } if (_richText->isAnchorTextItalicEnabled()) { attributes.italics = true; } if (_richText->isAnchorTextUnderlineEnabled()) { attributes.line = StyleLine::UNDERLINE; } if (_richText->isAnchorTextDelEnabled()) { attributes.line = StyleLine::STRIKETHROUGH; } if (_richText->isAnchorTextOutlineEnabled()) { attributes.effect = StyleEffect::OUTLINE; attributes.outlineColor = _richText->getAnchorTextOutlineColor3B(); attributes.outlineSize = _richText->getAnchorTextOutlineSize(); } if (_richText->isAnchorTextShadowEnabled()) { attributes.effect = StyleEffect::SHADOW; attributes.shadowColor = _richText->getAnchorTextShadowColor3B(); attributes.shadowOffset = _richText->getAnchorTextShadowOffset(); attributes.shadowBlurRadius = _richText->getAnchorTextShadowBlurRadius(); } if (_richText->isAnchorTextGlowEnabled()) { attributes.effect = StyleEffect::GLOW; attributes.glowColor = _richText->getAnchorTextGlowColor3B(); } } if (attrValueMap.find(RichText::KEY_TEXT_STYLE) != attrValueMap.end()) { auto keyTextStyle = attrValueMap.at(RichText::KEY_TEXT_STYLE).asString(); if (keyTextStyle == RichText::VALUE_TEXT_STYLE_OUTLINE) { attributes.effect = StyleEffect::OUTLINE; if (attrValueMap.find(RichText::KEY_TEXT_OUTLINE_COLOR) != attrValueMap.end()) { attributes.outlineColor = _richText->color3BWithString( attrValueMap.at(RichText::KEY_TEXT_OUTLINE_COLOR).asString()); } if (attrValueMap.find(RichText::KEY_TEXT_OUTLINE_SIZE) != attrValueMap.end()) { attributes.outlineSize = attrValueMap.at(RichText::KEY_TEXT_OUTLINE_SIZE).asInt(); } } else if (keyTextStyle == RichText::VALUE_TEXT_STYLE_SHADOW) { attributes.effect = StyleEffect::SHADOW; if (attrValueMap.find(RichText::KEY_TEXT_SHADOW_COLOR) != attrValueMap.end()) { attributes.shadowColor = _richText->color3BWithString( attrValueMap.at(RichText::KEY_TEXT_SHADOW_COLOR).asString()); } if ((attrValueMap.find(RichText::KEY_TEXT_SHADOW_OFFSET_WIDTH) != attrValueMap.end()) && (attrValueMap.find(RichText::KEY_TEXT_SHADOW_OFFSET_HEIGHT) != attrValueMap.end())) { attributes.shadowOffset = Vec2(attrValueMap.at(RichText::KEY_TEXT_SHADOW_OFFSET_WIDTH).asFloat(), attrValueMap.at(RichText::KEY_TEXT_SHADOW_OFFSET_HEIGHT).asFloat()); } if (attrValueMap.find(RichText::KEY_TEXT_SHADOW_BLUR_RADIUS) != attrValueMap.end()) { attributes.shadowBlurRadius = attrValueMap.at(RichText::KEY_TEXT_SHADOW_BLUR_RADIUS).asInt(); } } else if (keyTextStyle == RichText::VALUE_TEXT_STYLE_GLOW) { attributes.effect = StyleEffect::GLOW; if (attrValueMap.find(RichText::KEY_TEXT_GLOW_COLOR) != attrValueMap.end()) { attributes.glowColor = _richText->color3BWithString(attrValueMap.at(RichText::KEY_TEXT_GLOW_COLOR).asString()); } } } pushBackFontElement(attributes); } if (richElement) { if (richElement->equalType(RichElement::Type::IMAGE)) { richElement->setColor(getColor()); auto* richElementImage = static_cast(richElement); richElementImage->setUrl(getURL()); } else if (richElement->equalType(RichElement::Type::NEWLINE)) { richElement->setColor(getColor()); } pushBackElement(richElement); } } } } void MyXMLVisitor::endElement(void* /*ctx*/, const char* elementName) { auto it = _tagTables.find(elementName); if (it != _tagTables.end()) { auto tagBehavior = it->second; if (tagBehavior.isFontElement) { popBackFontElement(); } } } void MyXMLVisitor::textHandler(void* /*ctx*/, const char* str, size_t len) { std::string text(str, len); auto color = getColor(); auto face = getFace(); auto fontSize = getFontSize(); auto italics = getItalics(); auto underline = getUnderline(); auto strikethrough = getStrikethrough(); auto bold = getBold(); auto url = getURL(); auto outline = getOutline(); auto shadow = getShadow(); auto glow = getGlow(); uint32_t flags = 0; if (italics) flags |= RichElementText::ITALICS_FLAG; if (bold) flags |= RichElementText::BOLD_FLAG; if (underline) flags |= RichElementText::UNDERLINE_FLAG; if (strikethrough) flags |= RichElementText::STRIKETHROUGH_FLAG; if (!url.empty()) flags |= RichElementText::URL_FLAG; if (std::get<0>(outline)) flags |= RichElementText::OUTLINE_FLAG; if (std::get<0>(shadow)) flags |= RichElementText::SHADOW_FLAG; if (std::get<0>(glow)) flags |= RichElementText::GLOW_FLAG; auto element = RichElementText::create(0, color, 255, text, face, fontSize, flags, url, std::get<1>(outline), std::get<2>(outline), std::get<1>(shadow), std::get<2>(shadow), std::get<3>(shadow), std::get<1>(glow)); _richText->pushBackElement(element); } void MyXMLVisitor::pushBackFontElement(const MyXMLVisitor::Attributes& attribs) { _fontElements.emplace_back(attribs); } void MyXMLVisitor::popBackFontElement() { _fontElements.pop_back(); } void MyXMLVisitor::pushBackElement(RichElement* element) { _richText->pushBackElement(element); } void MyXMLVisitor::setTagDescription(std::string_view tag, bool isFontElement, RichText::VisitEnterHandler&& handleVisitEnter) { hlookup::set_item( MyXMLVisitor::_tagTables, tag, TagBehavior{ isFontElement, std::move( handleVisitEnter)}); // MyXMLVisitor::_tagTables[tag] = {isFontElement, std::move(handleVisitEnter)}; } void MyXMLVisitor::removeTagDescription(std::string_view tag) { MyXMLVisitor::_tagTables.erase(tag); } ValueMap MyXMLVisitor::tagAttrMapWithXMLElement(const char** attrs) { ValueMap tagAttrValueMap; for (const char** attr = attrs; *attr != nullptr; attr = (attrs += 2)) { if (attr[0] && attr[1]) { tagAttrValueMap[attr[0]] = attr[1]; } } return tagAttrValueMap; } const std::string RichText::KEY_VERTICAL_SPACE("KEY_VERTICAL_SPACE"); const std::string RichText::KEY_WRAP_MODE("KEY_WRAP_MODE"); const std::string RichText::KEY_HORIZONTAL_ALIGNMENT("KEY_HORIZONTAL_ALIGNMENT"); const std::string RichText::KEY_FONT_COLOR_STRING("KEY_FONT_COLOR_STRING"); const std::string RichText::KEY_FONT_SIZE("KEY_FONT_SIZE"); const std::string RichText::KEY_FONT_SMALL("KEY_FONT_SMALL"); const std::string RichText::KEY_FONT_BIG("KEY_FONT_BIG"); const std::string RichText::KEY_FONT_FACE("KEY_FONT_FACE"); const std::string RichText::KEY_TEXT_BOLD("KEY_TEXT_BOLD"); const std::string RichText::KEY_TEXT_ITALIC("KEY_TEXT_ITALIC"); const std::string RichText::KEY_TEXT_LINE("KEY_TEXT_LINE"); const std::string RichText::VALUE_TEXT_LINE_NONE("VALUE_TEXT_LINE_NONE"); const std::string RichText::VALUE_TEXT_LINE_DEL("VALUE_TEXT_LINE_DEL"); const std::string RichText::VALUE_TEXT_LINE_UNDER("VALUE_TEXT_LINE_UNDER"); const std::string RichText::KEY_TEXT_STYLE("KEY_TEXT_STYLE"); const std::string RichText::VALUE_TEXT_STYLE_NONE("VALUE_TEXT_STYLE_NONE"); const std::string RichText::VALUE_TEXT_STYLE_OUTLINE("VALUE_TEXT_STYLE_OUTLINE"); const std::string RichText::VALUE_TEXT_STYLE_SHADOW("VALUE_TEXT_STYLE_SHADOW"); const std::string RichText::VALUE_TEXT_STYLE_GLOW("VALUE_TEXT_STYLE_GLOW"); const std::string RichText::KEY_TEXT_OUTLINE_COLOR("KEY_TEXT_OUTLINE_COLOR"); const std::string RichText::KEY_TEXT_OUTLINE_SIZE("KEY_TEXT_OUTLINE_SIZE"); const std::string RichText::KEY_TEXT_SHADOW_COLOR("KEY_TEXT_SHADOW_COLOR"); const std::string RichText::KEY_TEXT_SHADOW_OFFSET_WIDTH("KEY_TEXT_SHADOW_OFFSET_WIDTH"); const std::string RichText::KEY_TEXT_SHADOW_OFFSET_HEIGHT("KEY_TEXT_SHADOW_OFFSET_HEIGHT"); const std::string RichText::KEY_TEXT_SHADOW_BLUR_RADIUS("KEY_TEXT_SHADOW_BLUR_RADIUS"); const std::string RichText::KEY_TEXT_GLOW_COLOR("KEY_TEXT_GLOW_COLOR"); const std::string RichText::KEY_URL("KEY_URL"); const std::string RichText::KEY_ANCHOR_FONT_COLOR_STRING("KEY_ANCHOR_FONT_COLOR_STRING"); const std::string RichText::KEY_ANCHOR_TEXT_BOLD("KEY_ANCHOR_TEXT_BOLD"); const std::string RichText::KEY_ANCHOR_TEXT_ITALIC("KEY_ANCHOR_TEXT_ITALIC"); const std::string RichText::KEY_ANCHOR_TEXT_LINE("KEY_ANCHOR_TEXT_LINE"); const std::string RichText::KEY_ANCHOR_TEXT_STYLE("KEY_ANCHOR_TEXT_STYLE"); const std::string RichText::KEY_ANCHOR_TEXT_OUTLINE_COLOR("KEY_ANCHOR_TEXT_OUTLINE_COLOR"); const std::string RichText::KEY_ANCHOR_TEXT_OUTLINE_SIZE("KEY_ANCHOR_TEXT_OUTLINE_SIZE"); const std::string RichText::KEY_ANCHOR_TEXT_SHADOW_COLOR("KEY_ANCHOR_TEXT_SHADOW_COLOR"); const std::string RichText::KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH("KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH"); const std::string RichText::KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT("KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT"); const std::string RichText::KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS("KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS"); const std::string RichText::KEY_ANCHOR_TEXT_GLOW_COLOR("KEY_ANCHOR_TEXT_GLOW_COLOR"); RichText::RichText() : _formatTextDirty(true), _leftSpaceWidth(0.0f) { _defaults[KEY_VERTICAL_SPACE] = 0.0f; _defaults[KEY_WRAP_MODE] = static_cast(WrapMode::WRAP_PER_WORD); _defaults[KEY_HORIZONTAL_ALIGNMENT] = static_cast(HorizontalAlignment::LEFT); _defaults[KEY_FONT_COLOR_STRING] = "#ffffff"; _defaults[KEY_FONT_SIZE] = 12.0f; _defaults[KEY_FONT_FACE] = "Verdana"; _defaults[KEY_ANCHOR_FONT_COLOR_STRING] = "#0000FF"; _defaults[KEY_ANCHOR_TEXT_BOLD] = false; _defaults[KEY_ANCHOR_TEXT_ITALIC] = false; _defaults[KEY_ANCHOR_TEXT_LINE] = VALUE_TEXT_LINE_NONE; _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_NONE; } RichText::~RichText() { _richElements.clear(); } RichText* RichText::create() { RichText* widget = new RichText(); if (widget->init()) { widget->autorelease(); return widget; } AX_SAFE_DELETE(widget); return nullptr; } RichText* RichText::createWithXML(std::string_view xml, const ValueMap& defaults, const OpenUrlHandler& handleOpenUrl) { RichText* widget = new RichText(); if (widget->initWithXML(xml, defaults, handleOpenUrl)) { widget->autorelease(); return widget; } AX_SAFE_DELETE(widget); return nullptr; } bool RichText::init() { if (Widget::init()) { return true; } return false; } bool RichText::initWithXML(std::string_view origxml, const ValueMap& defaults, const OpenUrlHandler& handleOpenUrl) { if (Widget::init()) { setDefaults(defaults); setOpenUrlHandler(handleOpenUrl); return this->setString(origxml); } return false; } bool RichText::setString(std::string_view text) { if (_text != text) { _formatTextDirty = true; _richElements.clear(); _text = text; // solves to issues: // - creates defaults values // - makes sure that the xml well formed and starts with an element _xmlText.clear(); fmt::format_to(std::back_inserter(_xmlText), FMT_COMPILE(R"({})"), this->getFontFace(), this->getFontSize(), this->getFontColor(), _text); MyXMLVisitor visitor(this); SAXParser parser; parser.setDelegator(&visitor); return parser.parseIntrusive(&_xmlText.front(), _xmlText.length(), SAXParser::ParseOption::HTML); } return true; } void RichText::initRenderer() {} void RichText::insertElement(RichElement* element, int index) { _richElements.insert(index, element); _formatTextDirty = true; } void RichText::pushBackElement(RichElement* element) { _richElements.pushBack(element); _formatTextDirty = true; } void RichText::removeElement(int index) { _richElements.erase(index); _formatTextDirty = true; } void RichText::removeElement(RichElement* element) { _richElements.eraseObject(element); _formatTextDirty = true; } RichText::WrapMode RichText::getWrapMode() const { return static_cast(_defaults.at(KEY_WRAP_MODE).asInt()); } void RichText::setWrapMode(RichText::WrapMode wrapMode) { if (static_cast(_defaults.at(KEY_WRAP_MODE).asInt()) != wrapMode) { _defaults[KEY_WRAP_MODE] = static_cast(wrapMode); _formatTextDirty = true; } } RichText::HorizontalAlignment RichText::getHorizontalAlignment() const { return static_cast(_defaults.at(KEY_HORIZONTAL_ALIGNMENT).asInt()); } void RichText::setHorizontalAlignment(ax::ui::RichText::HorizontalAlignment a) { if (static_cast(_defaults.at(KEY_HORIZONTAL_ALIGNMENT).asInt()) != a) { _defaults[KEY_HORIZONTAL_ALIGNMENT] = static_cast(a); _formatTextDirty = true; } } void RichText::setFontColor(std::string_view color) { _defaults[KEY_FONT_COLOR_STRING] = color; } std::string RichText::getFontColor() { return _defaults.at(KEY_FONT_COLOR_STRING).asString(); } ax::Color3B RichText::getFontColor3B() { return color3BWithString(getFontColor()); } void RichText::setFontSize(float size) { _defaults[KEY_FONT_SIZE] = size; } float RichText::getFontSize() { return _defaults.at(KEY_FONT_SIZE).asFloat(); } void RichText::setFontFace(std::string_view face) { _defaults[KEY_FONT_FACE] = face; } std::string RichText::getFontFace() { return _defaults.at(KEY_FONT_FACE).asString(); } void RichText::setAnchorFontColor(std::string_view color) { _defaults[KEY_ANCHOR_FONT_COLOR_STRING] = color; } std::string RichText::getAnchorFontColor() { return _defaults.at(KEY_ANCHOR_FONT_COLOR_STRING).asString(); } ax::Color3B RichText::getAnchorFontColor3B() { return color3BWithString(getAnchorFontColor()); } void RichText::setAnchorTextBold(bool enable) { _defaults[KEY_ANCHOR_TEXT_BOLD] = enable; } bool RichText::isAnchorTextBoldEnabled() { return _defaults[KEY_ANCHOR_TEXT_BOLD].asBool(); } void RichText::setAnchorTextItalic(bool enable) { _defaults[KEY_ANCHOR_TEXT_ITALIC] = enable; } bool RichText::isAnchorTextItalicEnabled() { return _defaults[KEY_ANCHOR_TEXT_ITALIC].asBool(); } void RichText::setAnchorTextDel(bool enable) { if (enable) _defaults[KEY_ANCHOR_TEXT_LINE] = VALUE_TEXT_LINE_DEL; else if (_defaults[KEY_ANCHOR_TEXT_LINE].asString() == VALUE_TEXT_LINE_DEL) _defaults[KEY_ANCHOR_TEXT_LINE] = VALUE_TEXT_LINE_NONE; } bool RichText::isAnchorTextDelEnabled() { return (_defaults[KEY_ANCHOR_TEXT_LINE].asString() == VALUE_TEXT_LINE_DEL); } void RichText::setAnchorTextUnderline(bool enable) { if (enable) _defaults[KEY_ANCHOR_TEXT_LINE] = VALUE_TEXT_LINE_UNDER; else if (_defaults[KEY_ANCHOR_TEXT_LINE].asString() == VALUE_TEXT_LINE_UNDER) _defaults[KEY_ANCHOR_TEXT_LINE] = VALUE_TEXT_LINE_NONE; } bool RichText::isAnchorTextUnderlineEnabled() { return (_defaults[KEY_ANCHOR_TEXT_LINE].asString() == VALUE_TEXT_LINE_UNDER); } void RichText::setAnchorTextOutline(bool enable, const Color3B& outlineColor, int outlineSize) { if (enable) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_OUTLINE; else if (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_OUTLINE) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_NONE; _defaults[KEY_ANCHOR_TEXT_OUTLINE_COLOR] = stringWithColor3B(outlineColor); _defaults[KEY_ANCHOR_TEXT_OUTLINE_SIZE] = outlineSize; } bool RichText::isAnchorTextOutlineEnabled() { return (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_OUTLINE); } Color3B RichText::getAnchorTextOutlineColor3B() { if (_defaults.find(KEY_ANCHOR_TEXT_OUTLINE_COLOR) != _defaults.end()) { return color3BWithString(_defaults.at(KEY_ANCHOR_TEXT_OUTLINE_COLOR).asString()); } return Color3B(); } int RichText::getAnchorTextOutlineSize() { if (_defaults.find(KEY_ANCHOR_TEXT_OUTLINE_SIZE) != _defaults.end()) { return _defaults.at(KEY_ANCHOR_TEXT_OUTLINE_SIZE).asInt(); } return -1; } void RichText::setAnchorTextShadow(bool enable, const Color3B& shadowColor, const Vec2& offset, int blurRadius) { if (enable) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_SHADOW; else if (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_SHADOW) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_NONE; _defaults[KEY_ANCHOR_TEXT_SHADOW_COLOR] = stringWithColor3B(shadowColor); _defaults[KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH] = offset.width; _defaults[KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT] = offset.height; _defaults[KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS] = blurRadius; } bool RichText::isAnchorTextShadowEnabled() { return (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_SHADOW); } Color3B RichText::getAnchorTextShadowColor3B() { if (_defaults.find(KEY_ANCHOR_TEXT_SHADOW_COLOR) != _defaults.end()) { return color3BWithString(_defaults.at(KEY_ANCHOR_TEXT_SHADOW_COLOR).asString()); } return Color3B(); } Vec2 RichText::getAnchorTextShadowOffset() { float width = 2.0f; float height = -2.0f; if (_defaults.find(KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH) != _defaults.end()) { width = _defaults.at(KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH).asFloat(); } if (_defaults.find(KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT) != _defaults.end()) { height = _defaults.at(KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT).asFloat(); } return Vec2(width, height); } int RichText::getAnchorTextShadowBlurRadius() { if (_defaults.find(KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS) != _defaults.end()) { return _defaults.at(KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS).asInt(); } return 0; } void RichText::setAnchorTextGlow(bool enable, const Color3B& glowColor) { if (enable) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_GLOW; else if (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_GLOW) _defaults[KEY_ANCHOR_TEXT_STYLE] = VALUE_TEXT_STYLE_NONE; _defaults[KEY_ANCHOR_TEXT_GLOW_COLOR] = stringWithColor3B(glowColor); } bool RichText::isAnchorTextGlowEnabled() { return (_defaults[KEY_ANCHOR_TEXT_STYLE].asString() == VALUE_TEXT_STYLE_GLOW); } Color3B RichText::getAnchorTextGlowColor3B() { if (_defaults.find(KEY_ANCHOR_TEXT_GLOW_COLOR) != _defaults.end()) { return color3BWithString(_defaults.at(KEY_ANCHOR_TEXT_GLOW_COLOR).asString()); } return Color3B(); } void RichText::setDefaults(const ValueMap& defaults) { if (defaults.find(KEY_VERTICAL_SPACE) != defaults.end()) { _defaults[KEY_VERTICAL_SPACE] = defaults.at(KEY_VERTICAL_SPACE).asFloat(); } if (defaults.find(KEY_WRAP_MODE) != defaults.end()) { _defaults[KEY_WRAP_MODE] = defaults.at(KEY_WRAP_MODE).asInt(); } if (defaults.find(KEY_HORIZONTAL_ALIGNMENT) != defaults.end()) { _defaults[KEY_HORIZONTAL_ALIGNMENT] = defaults.at(KEY_HORIZONTAL_ALIGNMENT).asInt(); } if (defaults.find(KEY_FONT_COLOR_STRING) != defaults.end()) { _defaults[KEY_FONT_COLOR_STRING] = defaults.at(KEY_FONT_COLOR_STRING).asString(); } if (defaults.find(KEY_FONT_SIZE) != defaults.end()) { _defaults[KEY_FONT_SIZE] = defaults.at(KEY_FONT_SIZE).asFloat(); } if (defaults.find(KEY_FONT_FACE) != defaults.end()) { _defaults[KEY_FONT_FACE] = defaults.at(KEY_FONT_FACE).asString(); } if (defaults.find(KEY_ANCHOR_FONT_COLOR_STRING) != defaults.end()) { _defaults[KEY_ANCHOR_FONT_COLOR_STRING] = defaults.at(KEY_ANCHOR_FONT_COLOR_STRING).asString(); } if (defaults.find(KEY_ANCHOR_TEXT_BOLD) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_BOLD] = defaults.at(KEY_ANCHOR_TEXT_BOLD).asBool(); } if (defaults.find(KEY_ANCHOR_TEXT_ITALIC) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_ITALIC] = defaults.at(KEY_ANCHOR_TEXT_ITALIC).asBool(); } if (defaults.find(KEY_ANCHOR_TEXT_LINE) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_LINE] = defaults.at(KEY_ANCHOR_TEXT_LINE).asString(); } if (defaults.find(KEY_ANCHOR_TEXT_STYLE) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_STYLE] = defaults.at(KEY_ANCHOR_TEXT_STYLE).asString(); } if (defaults.find(KEY_ANCHOR_TEXT_OUTLINE_COLOR) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_OUTLINE_COLOR] = defaults.at(KEY_ANCHOR_TEXT_OUTLINE_COLOR).asString(); } if (defaults.find(KEY_ANCHOR_TEXT_OUTLINE_SIZE) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_OUTLINE_SIZE] = defaults.at(KEY_ANCHOR_TEXT_OUTLINE_SIZE).asInt(); } if (defaults.find(KEY_ANCHOR_TEXT_SHADOW_COLOR) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_SHADOW_COLOR] = defaults.at(KEY_ANCHOR_TEXT_SHADOW_COLOR).asString(); } if (defaults.find(KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH] = defaults.at(KEY_ANCHOR_TEXT_SHADOW_OFFSET_WIDTH).asFloat(); } if (defaults.find(KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT] = defaults.at(KEY_ANCHOR_TEXT_SHADOW_OFFSET_HEIGHT).asFloat(); } if (defaults.find(KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS] = defaults.at(KEY_ANCHOR_TEXT_SHADOW_BLUR_RADIUS).asInt(); } if (defaults.find(KEY_ANCHOR_TEXT_GLOW_COLOR) != defaults.end()) { _defaults[KEY_ANCHOR_TEXT_GLOW_COLOR] = defaults.at(KEY_ANCHOR_TEXT_GLOW_COLOR).asString(); } } ValueMap RichText::getDefaults() const { ValueMap defaults; return defaults; } ax::Color3B RichText::color3BWithString(std::string_view color) { if (color.length() == 4) { unsigned int r, g, b; sscanf(color.data(), "%*c%1x%1x%1x", &r, &g, &b); r += r * 16; g += g * 16; b += b * 16; return Color3B(r, g, b); } else if (color.length() == 7) { unsigned int r, g, b; sscanf(color.data(), "%*c%2x%2x%2x", &r, &g, &b); return Color3B(r, g, b); } else if (color.length() == 9) { unsigned int r, g, b, a; sscanf(color.data(), "%*c%2x%2x%2x%2x", &r, &g, &b, &a); return Color3B(r, g, b); } return Color3B::WHITE; } std::string RichText::stringWithColor3B(const ax::Color3B& color3b) { int r = color3b.r; int g = color3b.g; int b = color3b.b; char buf[8]; snprintf(buf, sizeof(buf), "#%02x%02x%02x", r, g, b); return std::string(buf, 7); } std::string RichText::stringWithColor4B(const ax::Color4B& color4b) { int r = color4b.r; int g = color4b.g; int b = color4b.b; int a = color4b.a; char buf[10]; snprintf(buf, sizeof(buf), "#%02x%02x%02x%02x", r, g, b, a); return std::string(buf, 9); } void RichText::setTagDescription(std::string_view tag, bool isFontElement, VisitEnterHandler handleVisitEnter) { MyXMLVisitor::setTagDescription(tag, isFontElement, std::move(handleVisitEnter)); } void RichText::removeTagDescription(std::string_view tag) { MyXMLVisitor::removeTagDescription(tag); } void RichText::openUrl(std::string_view url) { if (_handleOpenUrl) { _handleOpenUrl(url); } else if (!url.empty()) { Application::getInstance()->openURL(url); } } void RichText::setOpenUrlHandler(const OpenUrlHandler& handleOpenUrl) { _handleOpenUrl = handleOpenUrl; } void RichText::formatText(bool force) { _formatTextDirty |= force; if (_formatTextDirty) { this->removeAllProtectedChildren(); _elementRenders.clear(); _lineHeights.clear(); if (_ignoreSize) { addNewLine(); for (ssize_t i = 0, size = _richElements.size(); i < size; ++i) { RichElement* element = _richElements.at(i); Node* elementRenderer = nullptr; switch (element->_type) { case RichElement::Type::TEXT: { RichElementText* elmtText = static_cast(element); Label* label; if (FileUtils::getInstance()->isFileExist(elmtText->_fontName)) { label = Label::createWithTTF(elmtText->_text, elmtText->_fontName, elmtText->_fontSize); } else { label = Label::createWithSystemFont(elmtText->_text, elmtText->_fontName, elmtText->_fontSize); } if (elmtText->_flags & RichElementText::ITALICS_FLAG) label->enableItalics(); if (elmtText->_flags & RichElementText::BOLD_FLAG) label->enableBold(); if (elmtText->_flags & RichElementText::UNDERLINE_FLAG) label->enableUnderline(); if (elmtText->_flags & RichElementText::STRIKETHROUGH_FLAG) label->enableStrikethrough(); if (elmtText->_flags & RichElementText::URL_FLAG) label->addComponent(ListenerComponent::create( label, elmtText->_url, std::bind(&RichText::openUrl, this, std::placeholders::_1))); if (elmtText->_flags & RichElementText::OUTLINE_FLAG) { label->enableOutline(Color4B(elmtText->_outlineColor), elmtText->_outlineSize); } if (elmtText->_flags & RichElementText::SHADOW_FLAG) { label->enableShadow(Color4B(elmtText->_shadowColor), elmtText->_shadowOffset, elmtText->_shadowBlurRadius); } if (elmtText->_flags & RichElementText::GLOW_FLAG) { label->enableGlow(Color4B(elmtText->_glowColor)); } label->setTextColor(Color4B(elmtText->_color)); elementRenderer = label; break; } case RichElement::Type::IMAGE: { RichElementImage* elmtImage = static_cast(element); if (elmtImage->_textureType == Widget::TextureResType::LOCAL) elementRenderer = Sprite::create(elmtImage->_filePath); else elementRenderer = Sprite::createWithSpriteFrameName(elmtImage->_filePath); if (elementRenderer && (elmtImage->_height != -1 || elmtImage->_width != -1)) { auto currentSize = elementRenderer->getContentSize(); if (elmtImage->_width != -1) elementRenderer->setScaleX((elmtImage->_width / currentSize.width) * elmtImage->_scaleX); else elementRenderer->setScaleX(elmtImage->_scaleX); if (elmtImage->_height != -1) elementRenderer->setScaleY((elmtImage->_height / currentSize.height) * elmtImage->_scaleY); else elementRenderer->setScaleY(elmtImage->_scaleY); elementRenderer->setContentSize(Vec2(currentSize.width * elementRenderer->getScaleX(), currentSize.height * elementRenderer->getScaleY())); elementRenderer->addComponent( ListenerComponent::create(elementRenderer, elmtImage->_url, std::bind(&RichText::openUrl, this, std::placeholders::_1))); elementRenderer->setColor(element->_color); } break; } case RichElement::Type::CUSTOM: { RichElementCustomNode* elmtCustom = static_cast(element); elementRenderer = elmtCustom->_customNode; elementRenderer->setColor(element->_color); break; } case RichElement::Type::NEWLINE: { addNewLine(); break; } default: break; } if (elementRenderer) { elementRenderer->setOpacity(element->_opacity); pushToContainer(elementRenderer); } } } else { addNewLine(); for (ssize_t i = 0, size = _richElements.size(); i < size; ++i) { RichElement* element = static_cast(_richElements.at(i)); switch (element->_type) { case RichElement::Type::TEXT: { RichElementText* elmtText = static_cast(element); handleTextRenderer(elmtText->_text, elmtText->_fontName, elmtText->_fontSize, elmtText->_color, elmtText->_opacity, elmtText->_flags, elmtText->_url, elmtText->_outlineColor, elmtText->_outlineSize, elmtText->_shadowColor, elmtText->_shadowOffset, elmtText->_shadowBlurRadius, elmtText->_glowColor); break; } case RichElement::Type::IMAGE: { RichElementImage* elmtImage = static_cast(element); handleImageRenderer(elmtImage->_filePath, elmtImage->_textureType, elmtImage->_color, elmtImage->_opacity, elmtImage->_width, elmtImage->_height, elmtImage->_url, elmtImage->_scaleX, elmtImage->_scaleY); break; } case RichElement::Type::CUSTOM: { RichElementCustomNode* elmtCustom = static_cast(element); handleCustomRenderer(elmtCustom->_customNode); break; } case RichElement::Type::NEWLINE: { addNewLine(); break; } default: break; } } } formatRenderers(); _formatTextDirty = false; } } namespace { inline bool isUTF8CharWrappable(const StringUtils::StringUTF8::CharUTF8& ch) { return (!ch.isASCII() || !std::isgraph(ch._char[0], std::locale())); } int getPrevWordPos(const StringUtils::StringUTF8& text, int idx) { if (idx <= 0) return -1; // start from idx-1 const StringUtils::StringUTF8::CharUTF8Store& str = text.getString(); const auto it = std::find_if(str.rbegin() + (str.size() - idx + 1), str.rend(), isUTF8CharWrappable); if (it == str.rend()) return -1; return static_cast(it.base() - str.begin()); } int getNextWordPos(const StringUtils::StringUTF8& text, int idx) { const StringUtils::StringUTF8::CharUTF8Store& str = text.getString(); if (idx + 1 >= static_cast(str.size())) return static_cast(str.size()); const auto it = std::find_if(str.begin() + idx + 1, str.end(), isUTF8CharWrappable); return static_cast(it - str.begin()); } bool isWrappable(const StringUtils::StringUTF8& text) { const StringUtils::StringUTF8::CharUTF8Store& str = text.getString(); return std::any_of(str.begin(), str.end(), isUTF8CharWrappable); } int findSplitPositionForWord(Label* label, const StringUtils::StringUTF8& text, int estimatedIdx, float originalLeftSpaceWidth, float newLineWidth) { const bool startingNewLine = (newLineWidth == originalLeftSpaceWidth); if (!isWrappable(text)) return (startingNewLine ? static_cast(text.length()) : 0); // The adjustment of the new line position int idx = getNextWordPos(text, estimatedIdx); std::string leftStr = text.getAsCharSequence(0, idx); label->setString(leftStr); float textRendererWidth = label->getContentSize().width; if (originalLeftSpaceWidth < textRendererWidth) // Have protruding { while (1) { // try to erase a word int newidx = getPrevWordPos(text, idx); if (newidx >= 0) { leftStr = text.getAsCharSequence(0, newidx); label->setString(leftStr); textRendererWidth = label->getContentSize().width; if (textRendererWidth <= originalLeftSpaceWidth) // is fitted return newidx; idx = newidx; continue; } // newidx < 0 means no prev word return (startingNewLine ? idx : 0); } } else if (textRendererWidth < originalLeftSpaceWidth) // A wide margin { while (1) { // try to append a word int newidx = getNextWordPos(text, idx); leftStr = text.getAsCharSequence(0, newidx); label->setString(leftStr); textRendererWidth = label->getContentSize().width; if (textRendererWidth < originalLeftSpaceWidth) { // the whole string is tested if (newidx == static_cast(text.length())) return newidx; idx = newidx; continue; } // protruded ? undo add, or quite fit return (textRendererWidth > originalLeftSpaceWidth ? idx : newidx); } } return idx; } int findSplitPositionForChar(Label* label, const StringUtils::StringUTF8& text, int estimatedIdx, float originalLeftSpaceWidth, float newLineWidth) { bool startingNewLine = (newLineWidth == originalLeftSpaceWidth); int stringLength = static_cast(text.length()); int leftLength = estimatedIdx; // The adjustment of the new line position std::string leftStr = text.getAsCharSequence(0, leftLength); label->setString(leftStr); float textRendererWidth = label->getContentSize().width; if (originalLeftSpaceWidth < textRendererWidth) // Have protruding { while (leftLength-- > 0) { // try to erase a char auto& ch = text.getString().at(leftLength); leftStr.erase(leftStr.end() - ch._char.length(), leftStr.end()); label->setString(leftStr); textRendererWidth = label->getContentSize().width; if (textRendererWidth <= originalLeftSpaceWidth) // is fitted break; } } else if (textRendererWidth < originalLeftSpaceWidth) // A wide margin { while (leftLength < stringLength) { // try to append a char auto& ch = text.getString().at(leftLength); ++leftLength; leftStr.append(ch._char); label->setString(leftStr); textRendererWidth = label->getContentSize().width; if (originalLeftSpaceWidth < textRendererWidth) // protruded, undo add { --leftLength; break; } else if (originalLeftSpaceWidth == textRendererWidth) // quite fit { break; } } } if (leftLength <= 0) return (startingNewLine) ? 1 : 0; return leftLength; } } // namespace void RichText::handleTextRenderer(std::string_view text, std::string_view fontName, float fontSize, const Color3B& color, uint8_t opacity, uint32_t flags, std::string_view url, const Color3B& outlineColor, int outlineSize, const Color3B& shadowColor, const Vec2& shadowOffset, int shadowBlurRadius, const Color3B& glowColor) { bool fileExist = FileUtils::getInstance()->isFileExist(fontName); RichText::WrapMode wrapMode = static_cast(_defaults.at(KEY_WRAP_MODE).asInt()); // split text by \n std::stringstream ss; ss << text; std::string currentText; size_t realLines = 0; while (std::getline(ss, currentText, '\n')) { if (realLines > 0) { addNewLine(); _lineHeights.back() = fontSize; } ++realLines; size_t splitParts = 0; StringUtils::StringUTF8 utf8Text(currentText); while (!currentText.empty()) { if (splitParts > 0) { addNewLine(); _lineHeights.back() = fontSize; } ++splitParts; Label* textRenderer = fileExist ? Label::createWithTTF(currentText, fontName, fontSize) : Label::createWithSystemFont(currentText, fontName, fontSize); if (flags & RichElementText::ITALICS_FLAG) textRenderer->enableItalics(); if (flags & RichElementText::BOLD_FLAG) textRenderer->enableBold(); if (flags & RichElementText::UNDERLINE_FLAG) textRenderer->enableUnderline(); if (flags & RichElementText::STRIKETHROUGH_FLAG) textRenderer->enableStrikethrough(); if (flags & RichElementText::URL_FLAG) textRenderer->addComponent(ListenerComponent::create( textRenderer, url, std::bind(&RichText::openUrl, this, std::placeholders::_1))); if (flags & RichElementText::OUTLINE_FLAG) textRenderer->enableOutline(Color4B(outlineColor), outlineSize); if (flags & RichElementText::SHADOW_FLAG) textRenderer->enableShadow(Color4B(shadowColor), shadowOffset, shadowBlurRadius); if (flags & RichElementText::GLOW_FLAG) textRenderer->enableGlow(Color4B(glowColor)); textRenderer->setTextColor(Color4B(color)); textRenderer->setOpacity(opacity); // textRendererWidth will get 0.0f, when we've got glError: 0x0501 in Label::getContentSize // It happens when currentText is very very long so that can't generate a texture const float textRendererWidth = textRenderer->getContentSize().width; // no splitting if (textRendererWidth > 0.0f && _leftSpaceWidth >= textRendererWidth) { _leftSpaceWidth -= textRendererWidth; pushToContainer(textRenderer); break; } // rough estimate // when textRendererWidth == 0.0f, use fontSize as the rough estimate of width for each char, // (_leftSpaceWidth / fontSize) means how many chars can be aligned in leftSpaceWidth. int estimatedIdx = 0; if (textRendererWidth > 0.0f) estimatedIdx = static_cast(_leftSpaceWidth / textRendererWidth * utf8Text.length()); else estimatedIdx = static_cast(_leftSpaceWidth / fontSize); int leftLength = 0; if (wrapMode == WRAP_PER_WORD) leftLength = findSplitPositionForWord(textRenderer, utf8Text, estimatedIdx, _leftSpaceWidth, _customSize.width); else leftLength = findSplitPositionForChar(textRenderer, utf8Text, estimatedIdx, _leftSpaceWidth, _customSize.width); // split string if (leftLength > 0) { textRenderer->setString(utf8Text.getAsCharSequence(0, leftLength)); pushToContainer(textRenderer); } StringUtils::StringUTF8::CharUTF8Store& str = utf8Text.getString(); // after the first line, skip any spaces to the left const auto startOfWordItr = std::find_if( str.begin() + leftLength, str.end(), [](const StringUtils::StringUTF8::CharUTF8& ch) { return !std::isspace(ch._char[0], std::locale()); }); if (startOfWordItr != str.end()) leftLength = static_cast(startOfWordItr - str.begin()); // erase the chars which are processed str.erase(str.begin(), str.begin() + leftLength); currentText = utf8Text.getAsCharSequence(); } } // std::getline discards the delimiter, so if it exists at the end of the text, then // a new line entry should be added if (!text.empty() && (text.back() == '\n')) { addNewLine(); _lineHeights.back() = fontSize; } } void RichText::handleImageRenderer(std::string_view filePath, Widget::TextureResType textureType, const Color3B& /*color*/, uint8_t /*opacity*/, int width, int height, std::string_view url, float scaleX, float scaleY) { Sprite* imageRenderer; if (textureType == Widget::TextureResType::LOCAL) imageRenderer = Sprite::create(filePath); else imageRenderer = Sprite::createWithSpriteFrameName(filePath); if (imageRenderer) { auto currentSize = imageRenderer->getContentSize(); if (width != -1) imageRenderer->setScaleX(width / currentSize.width); if (height != -1) imageRenderer->setScaleY(height / currentSize.height); imageRenderer->setScaleX(imageRenderer->getScaleX() * scaleX); imageRenderer->setScaleY(imageRenderer->getScaleY() * scaleY); imageRenderer->setContentSize( Vec2(currentSize.width * imageRenderer->getScaleX(), currentSize.height * imageRenderer->getScaleY())); imageRenderer->setScale(1.f, 1.f); handleCustomRenderer(imageRenderer); imageRenderer->addComponent( ListenerComponent::create(imageRenderer, url, std::bind(&RichText::openUrl, this, std::placeholders::_1))); } } void RichText::handleCustomRenderer(ax::Node* renderer) { Vec2 imgSize = renderer->getContentSize(); _leftSpaceWidth -= imgSize.width; if (_leftSpaceWidth < 0.0f) { addNewLine(); pushToContainer(renderer); _leftSpaceWidth -= imgSize.width; } else { pushToContainer(renderer); } } void RichText::addNewLine() { _leftSpaceWidth = _customSize.width; _elementRenders.emplace_back(); _lineHeights.emplace_back(); } void RichText::formatRenderers() { float verticalSpace = _defaults[KEY_VERTICAL_SPACE].asFloat(); float fontSize = _defaults[KEY_FONT_SIZE].asFloat(); if (_ignoreSize) { float newContentSizeWidth = 0.0f; float nextPosY = 0.0f; std::vector*, float>> rowWidthPairs; rowWidthPairs.reserve(_elementRenders.size()); for (auto&& element : _elementRenders) { float nextPosX = 0.0f; float maxY = 0.0f; for (auto&& iter : element) { iter->setAnchorPoint(Vec2::ZERO); iter->setPosition(nextPosX, nextPosY); this->addProtectedChild(iter, 1); Vec2 iSize = iter->getContentSize(); newContentSizeWidth += iSize.width; nextPosX += iSize.width; maxY = std::max(maxY, iSize.height); } nextPosY -= maxY; rowWidthPairs.emplace_back(&element, nextPosX); } this->setContentSize(Vec2(newContentSizeWidth, -nextPosY)); for (auto&& row : rowWidthPairs) doHorizontalAlignment(*row.first, row.second); } else { // calculate real height float newContentSizeHeight = 0.0f; std::vector maxHeights(_elementRenders.size()); for (size_t i = 0, size = _elementRenders.size(); i < size; i++) { Vector& row = _elementRenders[i]; float maxHeight = 0.0f; for (auto&& iter : row) { maxHeight = std::max(iter->getContentSize().height, maxHeight); } // gap for empty line, if _lineHeights[i] == 0, use current RichText's fontSize if (row.empty()) { maxHeight = (_lineHeights[i] != 0.0f ? _lineHeights[i] : fontSize); } maxHeights[i] = maxHeight; // vertical space except for first line newContentSizeHeight += (i != 0 ? maxHeight + verticalSpace : maxHeight); } _customSize.height = newContentSizeHeight; // align renders float nextPosY = _customSize.height; for (size_t i = 0, size = _elementRenders.size(); i < size; i++) { Vector& row = _elementRenders[i]; float nextPosX = 0.0f; nextPosY -= (i != 0 ? maxHeights[i] + verticalSpace : maxHeights[i]); for (auto&& iter : row) { iter->setAnchorPoint(Vec2::ZERO); iter->setPosition(nextPosX, nextPosY); this->addProtectedChild(iter, 1); nextPosX += iter->getContentSize().width; } doHorizontalAlignment(row, nextPosX); } } _elementRenders.clear(); _lineHeights.clear(); if (_ignoreSize) { Vec2 s = getVirtualRendererSize(); this->setContentSize(s); } else { this->setContentSize(_customSize); } updateContentSizeWithTextureSize(_contentSize); } namespace { float getPaddingAmount(const RichText::HorizontalAlignment alignment, const float leftOver) { switch (alignment) { case RichText::HorizontalAlignment::CENTER: return leftOver / 2.f; case RichText::HorizontalAlignment::RIGHT: return leftOver; default: AXASSERT(false, "invalid horizontal alignment!"); return 0.f; } } } // namespace void RichText::doHorizontalAlignment(const Vector& row, float rowWidth) { const auto alignment = static_cast(_defaults.at(KEY_HORIZONTAL_ALIGNMENT).asInt()); if (alignment != HorizontalAlignment::LEFT) { const auto diff = stripTrailingWhitespace(row); const auto leftOver = getContentSize().width - (rowWidth + diff); const float leftPadding = getPaddingAmount(alignment, leftOver); const Vec2 offset(leftPadding, 0.f); for (auto&& node : row) { node->setPosition(node->getPosition() + offset); } } } namespace { bool isWhitespace(char c) { return std::isspace(c, std::locale()); } void rtrim(std::string& s) { s.erase(std::find_if_not(s.rbegin(), s.rend(), isWhitespace).base(), s.end()); } } // namespace float RichText::stripTrailingWhitespace(const Vector& row) { if (!row.empty()) { if (auto label = dynamic_cast(row.back())) { const auto width = label->getContentSize().width; std::string trimmedString{label->getString()}; rtrim(trimmedString); if (label->getString() != trimmedString) { label->setString(trimmedString); return label->getContentSize().width - width; } } } return 0.0f; } void RichText::adaptRenderers() { this->formatText(); } void RichText::pushToContainer(ax::Node* renderer) { if (_elementRenders.empty()) { return; } _elementRenders[_elementRenders.size() - 1].pushBack(renderer); } void RichText::setVerticalSpace(float space) { _defaults[KEY_VERTICAL_SPACE] = space; } void RichText::ignoreContentAdaptWithSize(bool ignore) { if (_ignoreSize != ignore) { _formatTextDirty = true; Widget::ignoreContentAdaptWithSize(ignore); } } std::string RichText::getDescription() const { return "RichText"; }