[906] | 1 | #include "xmltools.h"
|
---|
| 2 |
|
---|
[920] | 3 | // Filters constructor
|
---|
[923] | 4 | XmlTools::XmlTools(QString filesWildcard, XmlFilter filter, bool noBackups, bool noVerbose)
|
---|
[906] | 5 | {
|
---|
| 6 | this->filesToProcess=UtilXmlTools::getAllXmlFilesByWildcard(filesWildcard);
|
---|
| 7 | this->filters=filter;
|
---|
| 8 | this->backupsEnabled=!noBackups;
|
---|
[923] | 9 | this->verboseEnabled=!noVerbose;
|
---|
[906] | 10 |
|
---|
[910] | 11 | if(this->filesToProcess.isEmpty()){
|
---|
[906] | 12 | UtilXmlTools::displayErrorMessage("Loading xml files","No XML files were found for the wildcard: "+filesWildcard);
|
---|
| 13 | }
|
---|
| 14 | }
|
---|
| 15 |
|
---|
[920] | 16 | // XPath constructor
|
---|
[923] | 17 | XmlTools::XmlTools(QString filesWildcard, QString xPathExpression, bool noBackups, bool noVerbose)
|
---|
[920] | 18 | {
|
---|
| 19 | this->filesToProcess=UtilXmlTools::getAllXmlFilesByWildcard(filesWildcard);
|
---|
| 20 | this->xPathExpression=xPathExpression;
|
---|
| 21 | this->backupsEnabled=!noBackups;
|
---|
[923] | 22 | this->verboseEnabled=!noVerbose;
|
---|
[920] | 23 | }
|
---|
| 24 |
|
---|
[906] | 25 | // Adds new values to an element
|
---|
| 26 | void XmlTools::addValues(QString newValues){
|
---|
| 27 |
|
---|
| 28 | // Process all XmlFiles
|
---|
| 29 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
| 30 |
|
---|
| 31 | QStringList newValuesList, currValuesList;
|
---|
| 32 | QList<pugi::xml_node> elements;
|
---|
| 33 |
|
---|
[923] | 34 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled,this->verboseEnabled,"add-values");
|
---|
[906] | 35 |
|
---|
| 36 | newValuesList=Util::qStringListFromSpacedString(newValues);
|
---|
| 37 |
|
---|
[920] | 38 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 39 | if(this->xPathExpression==""){
|
---|
| 40 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 41 | }
|
---|
| 42 | else{
|
---|
| 43 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 44 | }
|
---|
| 45 |
|
---|
[906] | 46 | for(int j=0; j<elements.size(); j++){
|
---|
| 47 |
|
---|
| 48 | currValuesList=Util::qStringListFromSpacedString(Util::toQString(elements[j].text().as_string())); // convert each element in a list (uses space as separator)
|
---|
| 49 |
|
---|
| 50 | for(int k=0; k<newValuesList.size(); k++){
|
---|
| 51 | if(currValuesList.contains(newValuesList[k])){ // If the current element already contains this value proceed to the next
|
---|
| 52 | continue;
|
---|
| 53 | }
|
---|
| 54 |
|
---|
[935] | 55 | elements[j].text()=QString(Util::toQString(elements[j].text().as_string()) + " " + newValuesList[k]).toUtf8().constData(); // If it doesn't exists yet let's add it
|
---|
[906] | 56 | }
|
---|
| 57 | }
|
---|
| 58 |
|
---|
| 59 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document,"add-values");
|
---|
| 60 | }
|
---|
| 61 |
|
---|
| 62 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(), "add-values");
|
---|
| 63 |
|
---|
| 64 | }
|
---|
| 65 |
|
---|
| 66 | void XmlTools::removeValues(QString valuesToRemove){
|
---|
| 67 |
|
---|
| 68 | // Process all XmlFiles
|
---|
| 69 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
| 70 |
|
---|
| 71 | QList<pugi::xml_node> elements;
|
---|
| 72 | QStringList valuesToRemoveList, currValuesList;
|
---|
| 73 | bool elementChanged=false;
|
---|
| 74 |
|
---|
[923] | 75 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled, this->verboseEnabled, "remove-values");
|
---|
[906] | 76 |
|
---|
[920] | 77 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 78 | if(this->xPathExpression==""){
|
---|
| 79 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 80 | }
|
---|
| 81 | else{
|
---|
| 82 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 83 | }
|
---|
[906] | 84 |
|
---|
| 85 | valuesToRemoveList=Util::qStringListFromSpacedString(valuesToRemove);
|
---|
| 86 |
|
---|
| 87 | for(int j=0; j<elements.size(); j++){ // O(3)... Optimization may be necessary.
|
---|
| 88 | currValuesList=Util::qStringListFromSpacedString(Util::toQString(elements[j].text().as_string())); // convert each element in a list (uses space as separator)
|
---|
| 89 |
|
---|
| 90 | for(int k=0; k<currValuesList.size(); k++){
|
---|
| 91 | for(int m=0; m<valuesToRemoveList.size(); m++){
|
---|
| 92 | if(currValuesList[k]==valuesToRemoveList[m]){
|
---|
[910] | 93 | currValuesList[k]=""; // flag to deletion
|
---|
[906] | 94 | elementChanged=true;
|
---|
| 95 | }
|
---|
| 96 | }
|
---|
| 97 | }
|
---|
| 98 |
|
---|
| 99 | if(elementChanged){ // If curr element changed update the XML
|
---|
[910] | 100 | currValuesList.removeAll(""); // remove all files flagged to deletion
|
---|
[935] | 101 | elements[j].text()=currValuesList.join(' ').toUtf8().constData();
|
---|
[906] | 102 | elementChanged=false;
|
---|
| 103 | }
|
---|
| 104 | }
|
---|
| 105 |
|
---|
| 106 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document, "remove-values");
|
---|
| 107 | }
|
---|
| 108 |
|
---|
| 109 |
|
---|
| 110 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(),"remove-values");
|
---|
| 111 | }
|
---|
| 112 |
|
---|
| 113 | void XmlTools::replaceValue(QString oldValue, QString newValue){
|
---|
| 114 |
|
---|
| 115 | // Process all XmlFiles
|
---|
| 116 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
| 117 |
|
---|
| 118 | QList<pugi::xml_node> elements;
|
---|
| 119 | QStringList currValuesList;
|
---|
| 120 | bool elementChanged=false;
|
---|
| 121 |
|
---|
[923] | 122 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled, this->verboseEnabled, "replace-value");
|
---|
[906] | 123 |
|
---|
[920] | 124 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 125 | if(this->xPathExpression==""){
|
---|
| 126 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 127 | }
|
---|
| 128 | else{
|
---|
| 129 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 130 | }
|
---|
[906] | 131 |
|
---|
| 132 | for(int j=0; j<elements.size(); j++){
|
---|
| 133 | currValuesList=Util::qStringListFromSpacedString(Util::toQString(elements[j].text().as_string())); // convert each element in a list (uses space as separator)
|
---|
| 134 |
|
---|
| 135 | for(int k=0; k<currValuesList.size(); k++){
|
---|
| 136 | if(currValuesList[k]==oldValue){ // Found a match with the old value?
|
---|
| 137 | currValuesList[k]=newValue; // If found replace it with the new value
|
---|
| 138 | elementChanged=true;
|
---|
| 139 | }
|
---|
| 140 | }
|
---|
| 141 |
|
---|
| 142 | if(elementChanged){ // If curr element changed update the XML
|
---|
[935] | 143 | elements[j].text()=currValuesList.join(" ").toUtf8().constData();
|
---|
[906] | 144 | }
|
---|
| 145 | elementChanged=false;
|
---|
| 146 | }
|
---|
| 147 |
|
---|
| 148 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document, "replace-value");
|
---|
| 149 | }
|
---|
| 150 |
|
---|
| 151 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(), "replace-value");
|
---|
| 152 | }
|
---|
| 153 |
|
---|
| 154 | // Replaces all current values of an element by another (can be specified only specific positions)
|
---|
| 155 | void XmlTools::replaceAll(QString value, QString valuePositions){
|
---|
| 156 |
|
---|
| 157 | // Process all XmlFiles
|
---|
| 158 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
| 159 |
|
---|
| 160 | QList<pugi::xml_node> elements;
|
---|
| 161 |
|
---|
[923] | 162 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled, this->verboseEnabled, "replace-all");
|
---|
[906] | 163 |
|
---|
[920] | 164 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 165 | if(this->xPathExpression==""){
|
---|
| 166 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 167 | }
|
---|
| 168 | else{
|
---|
| 169 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 170 | }
|
---|
[906] | 171 |
|
---|
| 172 |
|
---|
| 173 | // Let's start the override
|
---|
| 174 | for(int j=0; j<elements.size(); j++){
|
---|
| 175 | if(valuePositions!=""){
|
---|
[935] | 176 | elements[j].text()=replaceSpecificPositions(value, Util::toQString(elements[j].text().as_string()),valuePositions).toUtf8().constData();
|
---|
[906] | 177 | }
|
---|
| 178 | else{
|
---|
[935] | 179 | elements[j].text()=value.toUtf8().constData();
|
---|
[906] | 180 | }
|
---|
| 181 | }
|
---|
| 182 |
|
---|
| 183 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document, "replace-all");
|
---|
| 184 | }
|
---|
| 185 |
|
---|
| 186 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(), "replace-all");
|
---|
| 187 | }
|
---|
| 188 |
|
---|
| 189 | // Update a set of XML elements values based in the difference between the old and new value
|
---|
| 190 | // This can be used in multiple files if the difference between one file and other are the same e.g. increment to all current object positions 100 in y (A difference of -100).
|
---|
| 191 | // E.g. oldValue=5 , newValue=7; diffBetweenOldAndNewValue=-2
|
---|
| 192 | void XmlTools::updateElements(QString diffBetweenOldAndNewValue){
|
---|
| 193 |
|
---|
| 194 | // Process all XmlFiles
|
---|
| 195 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
| 196 |
|
---|
| 197 | QList<pugi::xml_node> elements;
|
---|
| 198 | MultiDimVar lastXmlValue(0); // inicialize with any value or dimension
|
---|
| 199 | MultiDimVar currXmlValue(0);
|
---|
| 200 | MultiDimVar newXmlValue(0); // value that will update currValue
|
---|
| 201 |
|
---|
[923] | 202 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled,this->verboseEnabled, "update-elements");
|
---|
[906] | 203 |
|
---|
[920] | 204 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 205 | if(this->xPathExpression==""){
|
---|
| 206 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 207 | }
|
---|
| 208 | else{
|
---|
| 209 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 210 | }
|
---|
[906] | 211 |
|
---|
| 212 |
|
---|
| 213 | if(elements.size()>1){
|
---|
| 214 | lastXmlValue=MultiDimVar(Util::toQString(elements[0].text().as_string())); // the lastXmlValue will begin to be the first one of the node
|
---|
| 215 | currXmlValue=MultiDimVar(Util::toQString(elements[1].text().as_string())); // the currXmlValue will begin to be the second one of the node
|
---|
| 216 | newXmlValue=MultiDimVar::sub(lastXmlValue, MultiDimVar(diffBetweenOldAndNewValue));
|
---|
[935] | 217 | elements[0].text() = newXmlValue.toString().toUtf8().constData(); // update the first eblement with the new one already
|
---|
[906] | 218 | }
|
---|
| 219 |
|
---|
| 220 | // Let's start the node update
|
---|
| 221 | for(int j=1; j<elements.size()-1; j++){ // We start in 1 because the 0 is already saved in lastXmlValue // -1 because we will also work with the next one in the current
|
---|
| 222 |
|
---|
| 223 | newXmlValue=MultiDimVar::sum(newXmlValue,MultiDimVar::sub(currXmlValue,lastXmlValue));
|
---|
[935] | 224 | elements[j].text() = newXmlValue.toString().toUtf8().constData(); // update element with the new value
|
---|
[906] | 225 | lastXmlValue=currXmlValue;
|
---|
| 226 | currXmlValue=MultiDimVar(Util::toQString(elements[j+1].text().as_string()));
|
---|
| 227 |
|
---|
| 228 | }
|
---|
| 229 |
|
---|
| 230 | // To update too last element (avoid out of bound because i+1)
|
---|
| 231 | newXmlValue=MultiDimVar::sum(newXmlValue,MultiDimVar::sub(currXmlValue,lastXmlValue));
|
---|
[935] | 232 | elements[elements.size()-1].text() = newXmlValue.toString().toUtf8().constData(); // update element with the new value
|
---|
[906] | 233 |
|
---|
| 234 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document, "update-elements");
|
---|
| 235 | }
|
---|
| 236 |
|
---|
| 237 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(),"updateNode");
|
---|
| 238 |
|
---|
| 239 | }
|
---|
| 240 |
|
---|
| 241 | // Invert a set of XML elements with specified name (and optionally a parent name)
|
---|
| 242 | void XmlTools::invertElements(){
|
---|
| 243 |
|
---|
| 244 | // Process all XmlFiles
|
---|
| 245 | for(int i=0; i<this->filesToProcess.size(); i++){
|
---|
[923] | 246 | UtilXmlTools::loadXmlFile(this->filesToProcess[i],this->document,this->rootNode,this->backupsEnabled, this->verboseEnabled, "invert-elements");
|
---|
[906] | 247 |
|
---|
| 248 | QList<pugi::xml_node> elements;
|
---|
| 249 | QStringList invertedElements; //Inverting the element order
|
---|
| 250 |
|
---|
[920] | 251 | // Check how the elements will be fetched via element name or xpath expression
|
---|
| 252 | if(this->xPathExpression==""){
|
---|
| 253 | UtilXmlTools::getAllNamedElements(this->rootNode,elements,this->filters);
|
---|
| 254 | }
|
---|
| 255 | else{
|
---|
| 256 | UtilXmlTools::getAllXpathElements(this->xPathExpression,this->document,elements);
|
---|
| 257 | }
|
---|
[906] | 258 |
|
---|
| 259 | // Read all elements and save to the list
|
---|
| 260 | for(int j=elements.size()-1; j>=0; j--){
|
---|
| 261 | invertedElements << Util::toQString(elements[j].text().as_string());
|
---|
| 262 | }
|
---|
| 263 |
|
---|
| 264 | // Override the tree with the inverted order
|
---|
| 265 | for(int j=0; j<elements.size(); j++){
|
---|
[935] | 266 | elements[j].text()= invertedElements[j].toUtf8().constData();
|
---|
[906] | 267 | }
|
---|
| 268 |
|
---|
| 269 | UtilXmlTools::saveXmlFile(this->filesToProcess[i],this->document, "invert-elements");
|
---|
| 270 | }
|
---|
| 271 |
|
---|
| 272 | UtilXmlTools::displaySuccessMessage(this->filesToProcess.size(),"invert-elements");
|
---|
| 273 |
|
---|
| 274 | }
|
---|
| 275 |
|
---|
| 276 | // Replaces specific values (given the position) for a new value
|
---|
| 277 | // [ currValues / positionsToReplaced are strings with composed by another strings space separated ]
|
---|
| 278 | // Returns a new string (space separated) will values replaced
|
---|
| 279 | QString XmlTools::replaceSpecificPositions(const QString &newValue, const QString &currValues, const QString &positionsToReplace){
|
---|
| 280 |
|
---|
| 281 | QList<int> positionsToReplaceList;
|
---|
| 282 | QStringList currValuesList;
|
---|
| 283 |
|
---|
| 284 | positionsToReplaceList=Util::qListIntFromSpacedString(positionsToReplace);
|
---|
| 285 | currValuesList=Util::qStringListFromSpacedString(currValues);
|
---|
| 286 |
|
---|
| 287 | // Make some validation before the replacing
|
---|
| 288 | if(currValuesList.size()<positionsToReplaceList.size()){
|
---|
| 289 | UtilXmlTools::displayErrorMessage("replaceSpecificPositions","There are more positions to replace than the current xml values.");
|
---|
| 290 | }
|
---|
| 291 |
|
---|
| 292 | foreach(int pos, positionsToReplaceList){
|
---|
| 293 | if(pos>currValuesList.size()-1 || pos < 0){ //Are positions valid for the current values? //-1 because starts at 0
|
---|
| 294 | UtilXmlTools::displayErrorMessage("replaceSpecificPositions","One or more of the specified positions to replace are out of range.");
|
---|
| 295 | }
|
---|
| 296 | }
|
---|
| 297 | //
|
---|
| 298 |
|
---|
| 299 | // Finally replace the specified values
|
---|
| 300 | foreach(int pos, positionsToReplaceList){
|
---|
| 301 | currValuesList[pos]=newValue;
|
---|
| 302 | }
|
---|
| 303 |
|
---|
| 304 | return currValuesList.join(" ");
|
---|
| 305 |
|
---|
| 306 | }
|
---|