/*************************************************************************** parsercommon.cpp - description ------------------- begin : Wed Feb 11 2004 copyright : (C) 2004 Andras Mantia <amantia@kde.org> ***************************************************************************/ /*************************************************************************** * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * ***************************************************************************/ //qt includes #include <tqstring.h> //kde includes #include <kdebug.h> #include <klocale.h> #include <ktexteditor/editinterface.h> //own includes #include "parsercommon.h" #include "node.h" #include "document.h" #include "qtag.h" #include "quantacommon.h" #include "resource.h" #include "dtds.h" class KDirWatch; int nodeNum; //for memory debugging - remove if not needed namespace ParserCommon { TQStringList includedFiles; TQPtrList<const DTDStruct> includedFilesDTD; KDirWatch *includeWatch; //common methods. TQString getLine(Document *write, int line, int endLine, int endCol) { TQString textLine = write->editIf->textLine(line); if (line == endLine) { if (endCol >0) textLine.truncate(endCol + 1); else textLine = ""; } return textLine; } void appendAreaToTextNode(Document *write, const AreaStruct &area, Node *node) { TQString tagStr = write->text(area); TQString cleanStr = node->tag->cleanStr; node->tag->setStr(node->tag->tagStr() + tagStr); if (node->tag->type == Tag::Empty) { TQString s = tagStr; if (s.simplifyWhiteSpace().isEmpty()) { node->tag->type = Tag::Empty; } else { node->tag->type = Tag::Text; } } TQString cleanedTagStr = tagStr; QuantaCommon::removeCommentsAndQuotes(cleanedTagStr, node->tag->dtd()); node->tag->cleanStr = cleanStr + cleanedTagStr; int bLine, bCol; node->tag->beginPos(bLine, bCol); node->tag->setTagPosition(bLine, bCol, area.eLine, area.eCol); } Node* createTextNode(Document *write, Node *node, int eLine, int eCol, Node *parentNode) { Tag *textTag = 0L; Node *textNode = 0L; int bLine = 0; int bCol = 0; const DTDStruct *dtd = write->defaultDTD(); if (node) { node->tag->endPos(bLine, bCol); } else if (parentNode) parentNode->tag->endPos(bLine, bCol); if (parentNode) dtd = parentNode->tag->dtd(); eCol--; if (bLine == 0 && bCol == 0) bCol = -1; if ( !(bLine == eLine && bCol == eCol) ) { AreaStruct area(bLine, bCol + 1, eLine, eCol); textTag = new Tag(area, write, dtd); TQString s = textTag->tagStr(); textTag->single = true; if (s.simplifyWhiteSpace().isEmpty()) { textTag->type = Tag::Empty; } else { textTag->type = Tag::Text; } if (parentNode && parentNode->tag->single) { textNode = new Node(parentNode->parent); nodeNum++; textNode->prev = parentNode; parentNode->next = textNode; parentNode = parentNode->parent; } else { if ( node && (node->tag->type == Tag::Empty || node->tag->type == Tag::Text) ) //merge two consquent text or empty nodes { AreaStruct area(bLine, bCol, eLine, eCol); appendAreaToTextNode(write, area, node); delete textTag; textTag = 0L; } else { textNode = new Node(parentNode); nodeNum++; if (node && node != parentNode) { node->next = textNode; textNode->prev = node; } else { if (parentNode) { Node *n = parentNode->child; while (n && n->next) n = n->next; if (!n) parentNode->child = textNode; else { n->next = textNode; textNode->prev = n; } } } } } if (textTag) { textNode->tag = textTag; node = textNode; } } return node; } Node* createScriptTagNode(Document *write, const AreaStruct &area, const TQString &areaName, const DTDStruct *dtd, Node *parentNode, Node *currentNode) { Tag *tag = new Tag(); tag->setTagPosition(area); tag->setStr(areaName); tag->setWrite(write); const DTDStruct *d = DTDs::ref()->find(dtd->specialAreaNames[areaName]); if (d) tag->setDtd(d); else tag->setDtd(dtd); tag->name = i18n("%1 block").tqarg(dtd->specialAreaNames[areaName].upper()); tag->type = Tag::ScriptTag; tag->validXMLTag = false; Node *node = new Node(parentNode); nodeNum++; node->tag = tag; node->insideSpecial = true; if (parentNode) { if (!parentNode->child) parentNode->child = node; else { Node *n = parentNode->child; while (n->next) n = n->next; n->next = node; node->prev = n; } } else if (currentNode) { node->prev = currentNode; currentNode->next = node; } return node; } void coutTree(Node *node, int indent) { TQString output; int bLine, bCol, eLine, eCol; if (!node) kdDebug(24000)<< "undoRedo::coutTree() - bad node!" << endl; while (node) { output = ""; output.fill('.', indent); node->tag->beginPos(bLine, bCol); node->tag->endPos(eLine, eCol); if (node->tag->type != Tag::Text) output += node->tag->name.replace('\n'," "); else output+= node->tag->tagStr().replace('\n'," "); kdDebug(24000) << output <<" (" << node->tag->type << ") at pos " << bLine << ":" << bCol << " - " << eLine << ":" << eCol << " This: "<< node << " Parent: " << node->parent << " Prev: " << node->prev << " Next: " << node->next << " Child: " << node->child << " Tag:" << node->tag << endl; /* for(j = 0; j < node->tag->attrCount(); j++) { kdDebug(24000)<< " attr" << j << " " << node->tag->getAttribute(j).nameLine << ":" << node->tag->getAttribute(j).nameCol << " - " << node->tag->getAttribute(j).valueLine << ":" << node->tag->getAttribute(j).valueCol << endl; } */ if (node->child) coutTree(node->child, indent + 4); node = node->next; } } void verifyTree(Node *node) { TQString output; int bLine, bCol, eLine, eCol; while (node) { if (!node->tag) { kdDebug(24000) << "Bad node: " << node << endl; kdDebug(24000) << "Parent: " << node->parent << " " << node->parent->tag->tagStr() << endl; } if (node->child) verifyTree(node->child); node = node->next; } } }