Blob Blame Raw


#include "tproperty.h"
#include "tstream.h"
#include "texception.h"
//#include "tconvert.h"

void TProperty::addListener(Listener *listener)
{
	if (std::find(m_listeners.begin(), m_listeners.end(), listener) == m_listeners.end())
		m_listeners.push_back(listener);
}

void TProperty::removeListener(Listener *listener)
{
	m_listeners.erase(std::remove(m_listeners.begin(), m_listeners.end(), listener), m_listeners.end());
}

void TProperty::notifyListeners() const
{
	std::vector<Listener *>::const_iterator it;
	for (it = m_listeners.begin(); it != m_listeners.end(); ++it)
		(*it)->onPropertyChanged();
}

//=============================================================================

TPropertyGroup::TPropertyGroup()
{
}

TPropertyGroup::~TPropertyGroup()
{
	for (PropertyVector::iterator it = m_properties.begin();
		 it != m_properties.end(); ++it)
		if (it->second)
			delete it->first;
}

void TPropertyGroup::clear()
{
	m_properties.clear();
	m_table.clear();
}

TPropertyGroup *TPropertyGroup::clone() const
{
	TPropertyGroup *g = new TPropertyGroup();
	for (PropertyVector::const_iterator i = m_properties.begin();
		 i != m_properties.end(); ++i)
		g->add(i->first->clone());
	return g;
}

void TPropertyGroup::add(TProperty *p)
{
	std::string name = p->getName();
	assert(m_table.find(name) == m_table.end());
	m_properties.push_back(std::make_pair(p, true));
	m_table[name] = p;
}

void TPropertyGroup::bind(TProperty &p)
{
	std::string name = p.getName();
	assert(m_table.find(name) == m_table.end());
	m_properties.push_back(std::make_pair(&p, false));
	m_table[name] = &p;
}

TProperty *TPropertyGroup::getProperty(std::string name)
{
	PropertyTable::iterator i = m_table.find(name);
	if (i == m_table.end())
		return 0;
	else
		return i->second;
}

template <class Property>
void assign(Property *dst, TProperty *src)
{
	Property *s = dynamic_cast<Property *>(src);
	if (!s)
		throw TProperty::TypeError();
	dst->setValue(s->getValue());
}

class Setter : public TProperty::Visitor
{
	TProperty *m_src;

public:
	Setter(TProperty *src) : m_src(src) {}

	void visit(TDoubleProperty *dst) { assign(dst, m_src); }
	void visit(TIntProperty *dst) { assign(dst, m_src); }
	void visit(TStringProperty *dst) { assign(dst, m_src); }
	void visit(TBoolProperty *dst) { assign(dst, m_src); }
	void visit(TEnumProperty *dst) { assign(dst, m_src); }
	void visit(TDoublePairProperty *dst) { assign(dst, m_src); }
	void visit(TIntPairProperty *dst) { assign(dst, m_src); }
	void visit(TStyleIndexProperty *dst) { assign(dst, m_src); }
	void visit(TPointerProperty *dst) { assign(dst, m_src); }
};

void TPropertyGroup::setProperties(TPropertyGroup *g)
{
	for (PropertyVector::const_iterator i = g->m_properties.begin();
		 i != g->m_properties.end(); ++i) {
		TProperty *src = i->first;
		TProperty *dst = getProperty(src->getName());
		if (dst) {
			Setter setter(src);
			TProperty::Visitor *visitor = &setter;
			dst->accept(*visitor);
		}
	}
}

void TPropertyGroup::accept(TProperty::Visitor &v)
{
	for (PropertyVector::const_iterator i = m_properties.begin();
		 i != m_properties.end(); ++i)
		i->first->accept(v);
}

class PropertyWriter : public TProperty::Visitor
{
	TOStream &m_os;

public:
	PropertyWriter(TOStream &os) : m_os(os) {}

	void visit(TDoubleProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "double";
		attr["name"] = p->getName();
		attr["min"] = std::to_string(p->getRange().first);
		attr["max"] = std::to_string(p->getRange().second);
		attr["value"] = std::to_string(p->getValue());
		m_os.openCloseChild("property", attr);
	}
	void visit(TDoublePairProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "pair";
		attr["name"] = p->getName();
		attr["min"] = std::to_string(p->getRange().first);
		attr["max"] = std::to_string(p->getRange().second);
		TDoublePairProperty::Value value = p->getValue();
		attr["value"] = std::to_string(value.first) + " " + std::to_string(value.second);
		m_os.openCloseChild("property", attr);
	}
	void visit(TIntPairProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "pair";
		attr["name"] = p->getName();
		attr["min"] = std::to_string(p->getRange().first);
		attr["max"] = std::to_string(p->getRange().second);
		TIntPairProperty::Value value = p->getValue();
		attr["value"] = std::to_string(value.first) + " " + std::to_string(value.second);
		m_os.openCloseChild("property", attr);
	}
	void visit(TIntProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "int";
		attr["name"] = p->getName();
		attr["min"] = std::to_string(p->getRange().first);
		attr["max"] = std::to_string(p->getRange().second);
		attr["value"] = std::to_string(p->getValue());
		m_os.openCloseChild("property", attr);
	}
	void visit(TBoolProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "bool";
		attr["name"] = p->getName();
		attr["value"] = p->getValue() ? "true" : "false";
		m_os.openCloseChild("property", attr);
	}
	void visit(TStringProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "string";
		attr["name"] = p->getName();
		attr["value"] = ::to_string(p->getValue());
		m_os.openCloseChild("property", attr);
	}

	void visit(TStyleIndexProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "string";
		attr["name"] = p->getName();
		attr["value"] = p->getValueAsString();
		m_os.openCloseChild("property", attr);
	}

	void visit(TEnumProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "enum";
		attr["name"] = p->getName();
		attr["value"] = ::to_string(p->getValue());
		if (TEnumProperty::isRangeSavingEnabled()) {
			m_os.openChild("property", attr);
			std::vector<std::wstring> range = p->getRange();
			for (int i = 0; i < (int)range.size(); i++) {
				attr.clear();
				attr["value"] = ::to_string(range[i]);
				m_os.openCloseChild("item", attr);
			}
			m_os.closeChild();
		} else
			m_os.openCloseChild("property", attr);
	}
	void visit(TPointerProperty *p)
	{
		std::map<std::string, std::string> attr;
		attr["type"] = "pointer";
		attr["name"] = p->getName();
		attr["value"] = p->getValueAsString();
		m_os.openCloseChild("property", attr);
	}
};

void TPropertyGroup::loadData(TIStream &is)
{
	for (PropertyVector::iterator it = m_properties.begin();
		 it != m_properties.end(); ++it)
		if (it->second)
			delete it->first;
	m_properties.clear();
	m_table.clear();
	std::string tagName;
	while (is.matchTag(tagName)) {
		if (tagName == "property") {
			std::string name = is.getTagAttribute("name");
			std::string type = is.getTagAttribute("type");
			std::string svalue = is.getTagAttribute("value");
			if (name == "")
				throw TException("missing property name");
			if (type == "")
				throw TException("missing property type");
			if (type != "string" && svalue == "")
				throw TException("missing property value");
			if (type == "double") {
				double min = std::stod(is.getTagAttribute("min"));
				double max = std::stod(is.getTagAttribute("max"));
				add(new TDoubleProperty(name, min, max, std::stod(svalue)));
			}
			if (type == "pair") {
				double min = std::stod(is.getTagAttribute("min"));
				double max = std::stod(is.getTagAttribute("max"));
				TDoublePairProperty::Value v(0, 0);
				int i = svalue.find(' ');
				if (i != (int)std::string::npos) {
					v.first = std::stod(svalue.substr(0, i));
					v.second = std::stod(svalue.substr(i + 1));
				}
				add(new TDoublePairProperty(name, min, max, v.first, v.second));
			} else if (type == "int") {
				int min = std::stoi(is.getTagAttribute("min"));
				int max = std::stoi(is.getTagAttribute("max"));
				add(new TIntProperty(name, min, max, std::stoi(svalue)));
			} else if (type == "bool") {
				if (svalue != "true" && svalue != "false")
					throw TException("bad boolean property value");
				add(new TBoolProperty(name, svalue == "true" ? true : false));
			} else if (type == "string") {
				add(new TStringProperty(name, ::to_wstring(svalue)));
			} else if (type == "enum") {
				TEnumProperty *p = new TEnumProperty(name);
				if (is.isBeginEndTag())
					p->addValue(::to_wstring(svalue));
				else {
					while (is.matchTag(tagName)) {
						if (tagName == "item") {
							std::string item = is.getTagAttribute("value");
							p->addValue(::to_wstring(item));
						} else
							throw TException("expected range property <item>");
					}
					is.closeChild();
				}
				p->setValue(::to_wstring(svalue));
				add(p);
			} else
				throw TException("unrecognized property type : " + type);
		} else
			throw TException("expected <property>");
		// is.closeChild();
	}
}

void TPropertyGroup::saveData(TOStream &os) const
{
	PropertyWriter writer(os);
	const_cast<TPropertyGroup *>(this)->accept(writer);
}

namespace
{
bool EnumRangeSavingEnabled = true;
}

void TEnumProperty::enableRangeSaving(bool on)
{
	EnumRangeSavingEnabled = on;
}

bool TEnumProperty::isRangeSavingEnabled()
{
	return EnumRangeSavingEnabled;
}