#include "XML.h" using namespace Framework; using namespace XML; // Erstellt ein XML Element // string: entweder der name des Elements oder ein XML Text der geparsed werden // soll Element::Element(Text string) : Element(string, 0) {} // Erstellt ein XML Element // string: entweder der name des Elements oder ein XML Text der geparsed werden // soll zParent: Ein Zeiger auf das eltern element (ohne erh�hten reference // Counter) Element::Element(Text string, Element* zParent) : ReferenceCounter() { children = new RCArray<Element>(); attributes = new RCArray<Text>(); attributeValues = new RCArray<Text>(); text = new Text(); name = new Text(); string.removeWhitespaceAfter(0); string.removeWhitespaceBefore(string.getLength()); setText(string); if (string[0] == '<' && string[string.getLength() - 1] == '>') { string.removeWhitespaceAfter(1); string.removeWhitespaceBefore(string.getLength() - 1); int nameEnd = 0; for (int i = 1; i < string.getLength(); i++) { if ((string[i] < 'a' || string[i] > 'z') && (string[i] < 'A' || string[i] > 'Z') && (string[i] < '0' || string[i] > '9') && string[i] != '-' && string[i] != '_' && string[i] != '.') { nameEnd = i; break; } } name->setText(string.getTeilText(1, nameEnd)); if (string.hatAt( string.getLength() - 1 - name->getLength(), name->getText()) || string[string.getLength() - 2] == '/') { string.removeWhitespaceAfter(nameEnd); // parse attributes int start = nameEnd; while (string[nameEnd] != '>' && string[nameEnd] != '/') { for (int i = nameEnd + 1; i < string.getLength(); i++) { if ((string[i] < 'a' || string[i] > 'z') && (string[i] < 'A' || string[i] > 'Z') && (string[i] < '0' || string[i] > '9') && string[i] != '-' && string[i] != '_' && string[i] != '.') { nameEnd = i; break; } } Text* attrName = string.getTeilText(start, nameEnd); string.removeWhitespaceAfter(nameEnd); if (string[nameEnd] == '=') { string.removeWhitespaceAfter(nameEnd + 1); Text value = ""; if (string[nameEnd + 1] == '"') { bool esc = 0; start = nameEnd + 2; for (int i = nameEnd + 2; string[i]; i++) { if (string[i] == '\\') esc = !esc; else { if (string[i] == '"' && !esc) { nameEnd = i + 1; break; } esc = 0; } } value.setText(string.getTeilText(start, nameEnd - 1)); value.ersetzen("\\\"", "\""); } if (string[nameEnd + 1] == '\'') { bool esc = 0; start = nameEnd + 2; for (int i = nameEnd + 2; string[i]; i++) { if (string[i] == '\\') esc = !esc; else { if (string[i] == '\'' && !esc) { nameEnd = i + 1; break; } esc = 0; } } value.setText(string.getTeilText(start, nameEnd - 1)); value.ersetzen("\\'", "'"); } setAttribute(attrName->getText(), value); } else setAttribute(attrName->getText(), ""); attrName->release(); string.removeWhitespaceAfter(nameEnd); start = nameEnd; } if (string[string.getLength() - 2] != '/') { string.removeWhitespaceBefore( string.getLength() - 1 - name->getLength()); if (string[string.getLength() - 2 - name->getLength()] == '/') { string.removeWhitespaceBefore( string.getLength() - 2 - name->getLength()); if (string[string.getLength() - 3 - name->getLength()] == '<') { text->setText(string.getTeilText(nameEnd + 1, string.getLength() - 3 - name->getLength())); // parse children text->removeWhitespaceAfter(0); text->removeWhitespaceBefore(text->getLength()); if (text->getText()[0] == '<' && text->getText()[text->getLength() - 1] == '>') { int start = 0; int lastStart = -1; while (start < text->getLength()) { if (lastStart == start) break; lastStart = start; bool esc = 0; bool inString1 = 0; bool inString2 = 0; int poc = 0; bool lastSlash = 0; bool lastOpen = 0; bool openSlash = 0; for (int i = 0; text->getText()[i]; i++) { switch (text->getText()[i]) { case '\\': esc = !esc; lastSlash = 0; lastOpen = 0; break; case '"': if (!esc && !inString2) inString1 = !inString1; esc = 0; lastSlash = 0; lastOpen = 0; break; case '\'': if (!esc && !inString1) inString2 = !inString2; esc = 0; lastSlash = 0; lastOpen = 0; break; case '<': if (!inString1 && !inString2) lastOpen = 1; esc = 0; lastSlash = 0; break; case '/': lastSlash = 0; if (!inString1 && !inString2) { lastSlash = 1; if (lastOpen) openSlash = 1; } esc = 0; lastOpen = 0; break; case '>': if (!inString1 && !inString2) { if (openSlash) poc--; else if (!lastSlash) poc++; if (poc == 0) { Text* str = text->getTeilText( start, i + 1); addChild(new Element( str->getText(), this)); str->release(); start = i + 1; } } esc = 0; lastSlash = 0; openSlash = 0; break; default: esc = 0; if (text->getText()[i] != ' ' && text->getText()[i] != '\t' && text->getText()[i] != '\r' && text->getText()[i] != '\n') { lastSlash = 0; lastOpen = 0; } } } } } } } } else text->setText(""); } } parent = zParent; } Element::~Element() { children->release(); attributes->release(); attributeValues->release(); text->release(); name->release(); } // �ndert ein attribut oder f�gt eines hinzu // attribut: Der Name des Attributes // value: Der Wert des Attributes void Element::setAttribute(Text attribut, Text value) { for (auto i = attributes->begin(), j = attributeValues->begin(); i && j; i++, j++) { if (i->istGleich(attribut)) { j->setText(value); return; } } attributes->add(new Text(attribut)); attributeValues->add(new Text(value)); } // entfernt ein attribut // attribut: Der Name des Attributes void Element::removeAttribute(Text attribut) { for (int i = 0; i < attributes->getEintragAnzahl(); i++) { if (attributes->z(i)->istGleich(attribut)) { attributes->remove(i); attributeValues->remove(i); i--; } } } // f�gt ein child hinzu // child: Das neue Child Element void Element::addChild(Element* child) { child->parent = this; children->add(child); } // f�gt ein child hinzu // child: Das neue Child Element void Element::addChildAtFront(Element* child) { child->parent = this; children->add(child, 0); } // entfernt ein child // zChild: das zu entfernende Child void Element::removeChild(Element* child) { for (int i = 0; i < children->getEintragAnzahl(); i++) { if (children->z(i) == child) { children->remove(i); i--; } } child->release(); } // entfernt das i-te child // i: der Index des childs (bei 0 beginnend) void Element::removeChild(int i) { children->remove(i); } // entfernt alle childs void Element::removeAllChilds() { children->leeren(); } // entfernt eine Liste mit childs // childs: alle Childs die entfernt werden sollen void Element::removeChilds(RCArray<Element>* childs) { for (auto i : *childs) removeChild(dynamic_cast<XML::Element*>(i->getThis())); childs->release(); } // entfernt dieses Element vom Eltern element void Element::remove() { if (parent) parent->removeChild(dynamic_cast<XML::Element*>(getThis())); } // setzt den Text in dem Element falls es keine childs gibt // text: dert Text void Element::setText(Text text) { this->text->setText(text); } // gibt den Text im Element zur�ck Text Element::getText() const { return text->getText(); } // gibt die Anzahl der Childs zur�ck int Element::getChildCount() const { return children->getEintragAnzahl(); } int Framework::XML::Element::getChildIndex(Element* zChild) const { return children->indexOf(zChild); } // gibt das i-te child zur�ck Element* Element::getChild(int i) const { return children->get(i); } // gibt das i-te child zur�ck (ohne erh�hten reference Counter) Element* Element::zChild(int i) const { return children->z(i); } // gibt das parent element zur�ck Element* Element::getParent() const { return parent ? dynamic_cast<Element*>(parent->getThis()) : 0; } // gibt das parent element zur�ck (ohne erh�hten reference Counter) Element* Element::zParent() const { return parent; } // gibt einen iterator zur�ck mit dem durch alle childs iteriert werden kann ArrayIterator<Element*> Element::getChilds() const { return children->begin(); } //! gibt einen Editor f�r dieses Element zur�ck Editor Element::select() { RCArray<Element>* tmp = new RCArray<Element>(); tmp->add(dynamic_cast<XML::Element*>(getThis())); return Editor(tmp); } // gibt einen selector zur�ck der alle childs beinhaltet Editor Element::selectChildren() const { return Editor(dynamic_cast<RCArray<XML::Element>*>(children->getThis())); } // gibt eine Liste mit childs zur�ck, die einen bestimmten Namen haben // name: der name der Childs Editor Element::selectChildsByName(Text name) const { RCArray<Element>* tmp = new RCArray<Element>(); for (auto i : *children) { if (i->getName().istGleich(name)) tmp->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(tmp); } // gibt eine Liste mit childs zur�ck, die ein bestimmtes Attribut haben // attribute: der name des Attributes Editor Element::selectChildsByAttribute(Text attribute) const { RCArray<Element>* tmp = new RCArray<Element>(); for (auto i : *children) { if (i->hasAttribute(attribute)) tmp->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(tmp); } // gibt eine Liste mit childs zur�ck, die ein bestimmtes Attribut mit einem // bestimmten wert haben // attribute: der name des Attributes // value: der Wert des Attributes Editor Element::selectChildsByAttribute(Text attribute, Text value) const { RCArray<Element>* tmp = new RCArray<Element>(); for (auto i : *children) { if (i->hasAttribute(attribute) && i->getAttributeValue(attribute).istGleich(value)) tmp->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(tmp); } // gibt 1 zur�ck, falls ein Attribut Name existiert, 0 sonnst bool Element::hasAttribute(Text name) const { for (auto i : *attributes) { if (i->istGleich(name)) return 1; } return 0; } // gibt die Anzahl der Attribute zur�ck int Element::getAttributeCount() const { return attributes->getEintragAnzahl(); } // gibt den Namen des i-ten Attributes zur�ck Text Element::getAttributeName(int i) const { return attributes->z(i)->getText(); } // gibt den Wert des i-ten Attributes zur�ck Text Element::getAttributeValue(int i) const { return attributeValues->z(i)->getText(); } // gibt den Wert eines Attributes zur�ck // attribut: Der Name des Attributes Text Element::getAttributeValue(Text attribut) const { for (auto i = attributes->begin(), j = attributeValues->begin(); i && j; i++, j++) { if (i->istGleich(attribut)) return j->getText(); } return ""; } // gibt einen iterator zur�ck mit dem durch alle Attribut Namen iteriert werden // kann ArrayIterator<Text*> Element::getAttributeNames() const { return attributes->begin(); } // gibt einen iterator zur�ck mit dem durch alle Attribut Werte iteriert werden // kann ArrayIterator<Text*> Element::getAttributeValues() const { return attributeValues->begin(); } // gibt den Namen des Elementes zur�ck zur�ck Text Element::getName() const { return name->getText(); } // erzeugt einen XML Text der dieses Element und alle childs beinhaltet Text Element::toString() const { Text ret = "<"; ret += name->getText(); if (attributes->getEintragAnzahl()) ret += " "; for (auto i = attributes->begin(), j = attributeValues->begin(); i && j; i++, j++) { ret += i->getText(); if (j->hat('"')) { ret += "='"; Text txt = j->getText(); txt.ersetzen("'", "\\'"); ret += txt; ret += "'"; } else { ret += "=\""; Text txt = j->getText(); txt.ersetzen("\"", "\\\""); ret += txt; ret += "\""; } if (i.hasNext()) ret += " "; } if (children->getEintragAnzahl() || text->getLength()) { ret += ">"; if (children->getEintragAnzahl()) { for (auto i : *children) ret += i->toString(); } else ret += text->getText(); ret += "</"; ret += name->getText(); ret += ">"; } else ret += "/>"; return ret; } // Erzeugt eine Kopie ohne referenzen auf dieses objekt Element* Element::dublicate() const { return new Element(toString()); } // Erzeugt einen neuen XML Editor mit einer Liste von Objekten die editiert // werden sollen Editor::Editor(RCArray<Element>* elements) : ReferenceCounter() { this->elements = new RCArray<Element>(); for (auto i : *elements) this->elements->add(dynamic_cast<XML::Element*>(i->getThis())); elements->release(); } Editor::Editor(const Editor& e) : Editor(dynamic_cast<RCArray<XML::Element>*>(e.elements->getThis())) {} Editor::~Editor() { elements->release(); } Maybe<RCPointer<Element>> Framework::XML::Editor::getFirstElement() const { if (this->elements->getEintragAnzahl() > 0) { return Maybe<RCPointer<Element>>::of( RCPointer<Element>::of(this->elements->get(0))); } return Maybe<RCPointer<Element>>::empty(); } // �ndert ein attribut oder f�gt eines hinzu (auf allen elementen in der Liste) // attribut: Der Name des Attributes // value: Der Wert des Attributes void Editor::setAttribute(Text attribut, Text value) { for (auto i : *elements) i->setAttribute(attribut, value); } // entfernt ein attribut (auf allen elementen in der Liste) // attribut: Der Name des Attributes void Editor::removeAttribute(Text attribut) { for (auto i : *elements) i->removeAttribute(attribut); } // f�gt ein child hinzu (auf allen elementen in der Liste) // child: Das neue Child Element void Editor::addChild(Element* child) { for (auto i : *elements) i->addChild(child->dublicate()); child->release(); } // entfernt ein child (auf allen elementen in der Liste) // zChild: das zu entfernende Child void Editor::removeChild(Element* child) { for (auto i : *elements) i->removeChild(dynamic_cast<XML::Element*>(child->getThis())); child->release(); } // entfernt das i-te child (auf allen elementen in der Liste) // i: der Index des childs (bei 0 beginnend) void Editor::removeChild(int i) { for (auto j : *elements) j->removeChild(i); } // entfernt alle childs (auf allen elementen in der Liste) void Editor::removeAllChilds() { for (auto i : *elements) i->removeAllChilds(); } // entfernt eine Liste mit childs (auf allen elementen in der Liste) // childs: alle Childs die entfernt werden sollen void Editor::removeChilds(RCArray<Element>* childs) { for (auto i : *elements) i->removeChilds( dynamic_cast<RCArray<XML::Element>*>(childs->getThis())); childs->release(); } // entfernt dieses Element vom Eltern element (auf allen elementen in der Liste) void Editor::remove() { for (auto i : *elements) i->remove(); } // setzt den Text in dem Element falls es keine childs gibt (auf allen elementen // in der Liste) // text: dert Text void Editor::setText(Text text) { for (auto i : *elements) i->setText(text); } // Gibt ein Iterator durch alle Elemente zur�ck ArrayIterator<Element*> Editor::begin() { return elements->begin(); } //! Gibt das ende des iterators zur�ck ArrayIterator<Element*> Editor::end() { return elements->end(); } //! Gibt einen selector zur�ck der alle elemente beinhaltet die in diesem //! selector vorkommen und rekursiv alle Kinder der elemente Enth�lt Editor Editor::selectAllElements() { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { list->add(dynamic_cast<XML::Element*>(i->getThis())); for (Element* j : i->selectChildren().selectAllElements()) { list->add(dynamic_cast<XML::Element*>(j->getThis())); } } return Editor(list); } // gibt einen selector zur�ck der alle childs beinhaltet Editor Editor::selectChildren() const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { for (Element* j : i->selectChildren()) { list->add(dynamic_cast<XML::Element*>(j->getThis())); } } return Editor(list); } // gibt einen selector zur�ck der alle parents beinhaltet Editor Editor::selectParents() const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->parent) list->add(dynamic_cast<XML::Element*>(i->parent->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die einen bestimmten Namen haben // name: der name der Childs Editor Editor::whereNameEquals(Text name) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->getName().istGleich(name)) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die ein bestimmtes child haben // name: der name des childs Editor Editor::whereChildWithNameExists(Text name) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->selectChildsByName(name).elements->getEintragAnzahl()) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die ein bestimmtes child haben // attribute: der name des attributes Editor Editor::whereChildWithAttributeExists(Text attribute) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->selectChildsByAttribute(attribute).elements->getEintragAnzahl()) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die ein bestimmtes child haben // attribute: der name des attributes // value: der Wert des Attributes Editor Editor::whereChildWithAttributeExists(Text attribute, Text value) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->selectChildsByAttribute(attribute, value) .elements->getEintragAnzahl()) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die ein bestimmtes Attribut haben // attribute: der name des Attributes Editor Editor::whereAttributeExists(Text attribute) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->hasAttribute(attribute)) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // gibt eine Liste mit elementen zur�ck, die ein bestimmtes Attribut mit einem // bestimmten wert haben // attribute: der name des Attributes // value: der Wert des Attributes Editor Editor::whereAttributeEquals(Text attribute, Text value) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { if (i->hasAttribute(attribute) && i->getAttributeValue(attribute).istGleich(value)) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // Gibt einen Editor zur�ck welcher nurnoch die Elemente enth�lt die nicht in e // sind // e: Ein Editor mit elementen die nicht enthalten sein sollen Editor Editor::without(Editor e) const { RCArray<Element>* list = new RCArray<Element>(); for (auto i : *elements) { bool found = 0; for (auto j : *e.elements) found |= i == j; if (!found) list->add(dynamic_cast<XML::Element*>(i->getThis())); } return Editor(list); } // Ruft eine funktion f�r jedes Element auf // f: die funktion (nimmt als argument ein Element objekt ohne erh�hten // reference Counter) void Editor::forEach(std::function<void(Element*)> f) const { for (auto i : *elements) f(i); } //! gibt 1 zur�ck, wenn mindestens ein Element gefunden wurde bool Editor::exists() const { return elements->getEintragAnzahl() > 0; } //! gibt die anzahl der ausgew�hlten elemente zur�ck int Editor::getSize() const { return elements->getEintragAnzahl(); } DLLEXPORT Editor& Framework::XML::Editor::operator=(const Editor& e) { if (this != &e) { this->elements->leeren(); for (auto i : *e.elements) this->elements->add(dynamic_cast<XML::Element*>(i->getThis())); } return *this; }