/* * Copyright (c) 2010-2014 OTClient * * 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. */ #ifndef OTMLNODE_H #define OTMLNODE_H #include "declarations.h" class OTMLNode : public stdext::shared_object { public: virtual ~OTMLNode() { } static OTMLNodePtr create(std::string tag = "", bool unique = false); static OTMLNodePtr create(std::string tag, std::string value); std::string tag() { return m_tag; } int size() { return m_children.size(); } std::string source() { return m_source; } std::string rawValue() { return m_value; } bool isUnique() { return m_unique; } bool isNull() { return m_null; } bool hasTag() { return !m_tag.empty(); } bool hasValue() { return !m_value.empty(); } bool hasChildren(); bool hasChildAt(const std::string& childTag) { return !!get(childTag); } bool hasChildAtIndex(int childIndex) { return !!getIndex(childIndex); } void setTag(const std::string& tag) { m_tag = tag; } void setValue(const std::string& value) { m_value = value; } void setNull(bool null) { m_null = null; } void setUnique(bool unique) { m_unique = unique; } void setSource(const std::string& source) { m_source = source; } OTMLNodePtr get(const std::string& childTag); OTMLNodePtr getIndex(int childIndex); OTMLNodePtr at(const std::string& childTag); OTMLNodePtr atIndex(int childIndex); void addChild(const OTMLNodePtr& newChild); bool removeChild(const OTMLNodePtr& oldChild); bool replaceChild(const OTMLNodePtr& oldChild, const OTMLNodePtr& newChild); void copy(const OTMLNodePtr& node); void merge(const OTMLNodePtr& node); void clear(); OTMLNodeList children(); OTMLNodePtr clone(); template T value(); template T valueAt(const std::string& childTag); template T valueAtIndex(int childIndex); template T valueAt(const std::string& childTag, const T& def); template T valueAtIndex(int childIndex, const T& def); template void write(const T& v); template void writeAt(const std::string& childTag, const T& v); template void writeIn(const T& v); virtual std::string emit(); OTMLNodePtr asOTMLNode() { return static_self_cast(); } protected: OTMLNode() : m_unique(false), m_null(false) { } OTMLNodeList m_children; std::string m_tag; std::string m_value; std::string m_source; bool m_unique; bool m_null; }; #include "otmlexception.h" template<> inline std::string OTMLNode::value() { std::string value = m_value; if(stdext::starts_with(value, "\"") && stdext::ends_with(value, "\"")) { value = value.substr(1, value.length()-2); stdext::replace_all(value, "\\\\", "\\"); stdext::replace_all(value, "\\\"", "\""); stdext::replace_all(value, "\\t", "\t"); stdext::replace_all(value, "\\n", "\n"); stdext::replace_all(value, "\\'", "\'"); } return value; } template T OTMLNode::value() { T ret; if(!stdext::cast(m_value, ret)) throw OTMLException(asOTMLNode(), stdext::format("failed to cast node value '%s' to type '%s'", m_value, stdext::demangle_type())); return ret; } template T OTMLNode::valueAt(const std::string& childTag) { OTMLNodePtr node = at(childTag); return node->value(); } template T OTMLNode::valueAtIndex(int childIndex) { OTMLNodePtr node = atIndex(childIndex); return node->value(); } template T OTMLNode::valueAt(const std::string& childTag, const T& def) { if(OTMLNodePtr node = get(childTag)) if(!node->isNull()) return node->value(); return def; } template T OTMLNode::valueAtIndex(int childIndex, const T& def) { if(OTMLNodePtr node = getIndex(childIndex)) return node->value(); return def; } template void OTMLNode::write(const T& v) { m_value = stdext::safe_cast(v); } template void OTMLNode::writeAt(const std::string& childTag, const T& v) { OTMLNodePtr child = OTMLNode::create(childTag); child->setUnique(true); child->write(v); addChild(child); } template void OTMLNode::writeIn(const T& v) { OTMLNodePtr child = OTMLNode::create(); child->write(v); addChild(child); } #endif