Qt 遍历不规则树的节点
在使用Qt的GraphicsScene作图时,遇到类似这样的需求:在scene中创建节点类似下图,

现在我要把每个节点的txt保存到xml文件中,结构为
<?xml version='1.0' encoding='UTF-8'?>
<root>
<A>
<B>
</B>
<C>
<E>
...
</E>
</C>
<D>
<F>
...
</F>
</D>
</A>
</root>
并且,还要能根据xml文件将这个结构还原出来(由于没有保存坐标信息,当然不能恢复原状)。
我将过程分为:遍历,保存xml;读取xml,还原图形。
我使用Qt的示例项目DiagramScene来改造。
首先,起始节点要与其他节点有所不同,便于我们区分。diagramitem类中,DiagramType { Step, Conditional, StartEnd, Io };,我添加一个virStep类型作为开始节点,其他节点类型为Step。其实virStep和Step的图形一模一样,只是类型不同。
在DiagramItem类中增加一些成员
QList<Arrow *> getArrow(){return arrows;}
void setText(QString text){ mText = text;}
void setItemId(QString pid){ ItemId = pid;}//唯一标识,在创建时设置
QString getItemId(){return ItemId;}
QString getText(){ return mText; }
重写其paint函数,在其中添加:
painter->drawText(boundingRect() , Qt::AlignCenter ,mText);
保存:
bool OperateItemFile::SaveRelationship(QList<DiagramItem *> &listItem, QList<Arrow *> &listArrow)
{
QFileDialog dlg;
dlg.setAcceptMode(QFileDialog::AcceptSave);
QString strName = dlg.getSaveFileName(getMainWindow() ,QObject::tr("save xml file"), "" ,QObject::tr("XML文件(*.XML)"));
if(strName.isEmpty())
return false; QDomDocument doc;
doc.appendChild( doc.createProcessingInstruction("xml", XML_HEAD_INFO));
QFile file(strName);
if (!file.open(QIODevice::WriteOnly)){
return false;
}
QDomElement root = doc.createElement("Root");
doc.appendChild(root); DiagramItem* startItem = NULL ;
QDomElement startEle;
for (int i = 0; i < listItem.size(); i++) {
if(listItem.at(i)->diagramType() == DiagramItem::virStep){
//开始节点
startEle = doc.createElement(listItem.at(i)->getText());
root.appendChild(startEle);
startItem = listItem.at(i);
//listItem.removeAt(i);
break;
}
} if(!startItem) return false; TravelAllItem(startItem, doc, startEle, listItem); QString xml = doc.toString();
QTextStream txtOutput(&file);
txtOutput.setCodec("UTF-8");
txtOutput<<xml;
file.close(); return true;
} void OperateItemFile::GetSubItems(DiagramItem *item, QList<DiagramItem*>& listNodeItem)
{
if(!item) return; listNodeItem.clear();
QList<Arrow*> listArrow = item->getArrow();
for (int i = 0; i < listArrow.size(); i++) {
if(listArrow.at(i)->startItem() == item){
listNodeItem.push_back(listArrow.at(i)->endItem());
}
}
} void OperateItemFile::TravelAllItem(DiagramItem *rootItem, QDomDocument &doc, QDomElement &rootEle, QList<DiagramItem *> &listItem)
{
if(listItem.size() < 1)
return;
QList<DiagramItem*> listNodeItem;
GetSubItems(rootItem, listNodeItem);
for (int i = 0; i < listNodeItem.size(); i++) {
QDomElement ele = doc.createElement(listNodeItem.at(i)->getText());
rootEle.appendChild(ele);
listItem.removeOne(rootItem);
TravelAllItem(listNodeItem.at(i), doc, ele, listItem); } }
使用GraphicsScene的selectItems函数,从中分别获得diagramitem和arrow。代码如果编译不通,可根据情况修改。
读入xml:
void OperateItemFile::ImportTargetFile(DiagramScene* pScene)
{
QString strName;
strName = QFileDialog::getOpenFileName(nullptr, QObject::tr("open xml file"), p_globalObject->ProjectPath() + "/" + p_globalObject->getProjectName() ,QObject::tr("XML文件(*XML*)"));
if(strName.isEmpty()){
return;
} QDomDocument doc;
QFile file(strName);
if (!file.open(QIODevice::ReadOnly)){
return ;
}
if (!doc.setContent(&file))
{
file.close();
return ;
} QDomNode domNodeStart = doc.documentElement();
QList<DiagramItem*> listItem;
QList<Arrow*> listArrow;
m_bStartNode = true;
SetItemPos(pScene->sceneRect().center());
parsePackage(domNodeStart, listItem, listArrow); pScene->ItemClear(); int i=0;
for(i=0 ;i<listItem.size() ;i++){
pScene->addItem(listItem.at(i));
}
pScene->setMode(DiagramScene::Mode::MoveItem);
////////////////////////////////////////////////////////////////////////////////////
//设置箭头起始节点
int j=0;
for(i=0 ;i<listArrow.size(); i++){
for(j=0 ;j<listItem.size() ;j++){
if(listArrow.at(i)->getStartId() == listItem.at(j)->getItemId()){
listArrow.at(i)->setStartItem(listItem.at(j));
break;
}
}
}
//设置箭头结尾节点
for(i=0 ;i<listArrow.size(); i++){
for(j=0 ;j<listItem.size() ;j++){
if(listArrow.at(i)->getEndId() == listItem.at(j)->getItemId()){
listArrow.at(i)->setEndItem(listItem.at(j));
break;
}
}
} for(i=0 ;i<listArrow.size() ;i++){
if(listArrow.at(i)->startItem()!=NULL)
listArrow.at(i)->startItem()->addArrow(listArrow.at(i)); if(listArrow.at(i)->endItem()!=NULL)
listArrow.at(i)->endItem()->addArrow(listArrow.at(i)); listArrow.at(i)->setColor(pScene->lineColor());
listArrow.at(i)->setZValue(-1000.0);
pScene->addItem(listArrow.at(i));
if(p_globalObject->getCurStatus() == QGlobalObject::devStatus)
listArrow.at(i)->setArrowFlag(true);
else
listArrow.at(i)->setArrowFlag(true);
listArrow.at(i)->updatePosition();
} file.close();
} bool OperateItemFile::parsePackage(QDomNode pNode, QList<DiagramItem*>& listItem, QList<Arrow*>& listArrow)
{
QDomNode myNode = pNode.firstChild();
while(!myNode.isNull()){
QDomElement domE = myNode.toElement();
QString qstrTagName = domE.tagName();
myDebug(qstrTagName);
if(qstrTagName.compare(PackageRoot) != 0 && !qstrTagName.isEmpty()) {
DiagramItem* pItem = NULL;
if(m_bStartNode){
pItem = new DiagramItem(DiagramItem::virStep ,NULL);
domE.setAttribute("StartItem", "true");
m_bStartNode = false;
}
else {
pItem = new DiagramItem(DiagramItem::Step ,NULL);
}
pItem->setPos(m_itemPos);
m_itemPos = GetNextItemPos(m_itemPos);
//pItem->setZValue(z.toFloat());
pItem->setBkColor(QColor(4294967295));
pItem->setItemId(p_globalObject->getGUID());
pItem->setText(qstrTagName);
pItem->setBrush(QBrush(4294967295, (Qt::BrushStyle)1));
domE.setAttribute("ItemId", pItem->getItemId());
if(!domE.hasAttribute("StartItem")){
Arrow* pArrow = new Arrow(NULL, NULL);
QString startItemId = pNode.toElement().attribute("ItemId");
QString endItemId = pItem->getItemId();
pArrow->setItemId(startItemId, endItemId);
listArrow.push_back(pArrow);
}
listItem.push_back(pItem);
} parsePackage(myNode, listItem, listArrow); myNode = myNode.nextSibling();
}
return true;
} QPointF OperateItemFile::GetNextItemPos(QPointF preItemPos)
{
float rat = 1;
qreal x = preItemPos.x()+80;
qreal y = x*rat;
QPointF pt(x,y);
return pt;
}
先写到这里,下一篇将写一下,如何完全还原图形。
Qt 遍历不规则树的节点的更多相关文章
- SAX解析xml (遍历DOM树各节点)
本文参考 http://yangjunfeng.iteye.com/blog/401377 1. books.xml <?xml version="1.0" encoding ...
- Qt实现表格树控件-自绘树节点虚线
目录 一.开心一刻 二.自绘树节点? 三.效果展示 四.实现思路 1.可扩展接口 2.函数重写 3.同步左侧表头 五.相关文章 原文链接:Qt实现表格树控件-自绘树节点虚线 一.开心一刻 一程序员第一 ...
- POJ 3321 Apple Tree(后根遍历将树转化成序列,用树状数组维护)
题意:一棵树,有很多分叉,每个分叉上最多有1个苹果. 给出n,接下来n-1行,每行u,v,表示分叉u,v之间有树枝相连.这里数据中u相当于树中的父节点,v相当于子节点. 给出两个操作: 1.C x ...
- 根据 中序遍历 和 后序遍历构造树(Presentation)(C++)
好不容易又到周五了,周末终于可以休息休息了.写这一篇随笔只是心血来潮,下午问了一位朋友PAT考的如何,顺便看一下他考的试题,里面有最后一道题,是关于给出中序遍历和后序遍历然后求一个层次遍历.等等,我找 ...
- 函数遍历DOM树
//获取页面中的根节点--根标签 var root=document.documentElement;//html //函数遍历DOM树 //根据根节点,调用fn的函数,显示的是根节点的名 ...
- 探索未知种族之osg类生物---状态树与渲染树以及节点树之间的关系
节点树 首先我们来看一个场景构建的实例,并通过它来了解一下“状态节点”StateGraph 和“渲染叶”RenderLeaf 所构成的状态树,“渲染台”RenderStage 和“渲染元”Render ...
- 先序遍历DOM树的5种方法
DOM树由文档中的所有节点(元素节点.文本节点.注释节点等)所构成的一个树结构,DOM树的解析和构建是浏览器要实现的关键功能.既然DOM树是一个树结构,那么我们就可以使用遍历树结构的相关方法来对DOM ...
- 利用层序遍历(含空节点)和中序遍历重建二叉树 python
给定一颗二叉树的层序遍历(不含None的形式)和中序遍历序列,利用两个序列完成对二叉树的重建. 还是通过一个例子来说明整个过程,下图所示的二叉树,层序遍历结果为[a,b,c,d,e],中序遍历结果为[ ...
- 九度OJ 1359:大魏树遍历 (树)
时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:299 解决:29 题目描述: 大魏是JOBDU技术组里最喜欢折腾的一个了,单反.骑车.改九度页面,当然还有YY prado.我们姑且先把这些 ...
随机推荐
- JAVA笔记15-线程同步
一.概念 1.多个线程之间访问同一资源,进行协调的过程是线程同步.例如两个人同时操作同一银行账户.解决方法:加锁 2.Java种引入对象互斥锁的概念,保证共享数据操作的完整性.每个对象都对应于一个可称 ...
- EXCL单元格公式——组装SQL用
="'"&F3&"'"
- N皇后问题 --使用位运算解决
关键位运算 x & (-x) 取得最低位1 x & (x-1) 去掉最低位1 class Solution(object): def totalNQueens(self, n): &q ...
- linux-包管理器-4
安装 升级 查询 导入公钥 rpm -K|checksig rpmfile 检查包的完整性和签名 rpm --import /etc/pki/rpm-gpg/RPM-GPG-KEY-CentOS-7 ...
- 创建一个简单的 Springboot web项目
1.点击Project 2.点击 Next 3.项目名 4.web 项目 4.确认 5.pom.xml <?xml version="1.0" encoding=" ...
- UVa 1600 Patrol Robot (BFS最短路 && 略不一样的vis标记)
题意 : 机器人要从一个m * n 网格的左上角(1,1) 走到右下角(m, n).网格中的一些格子是空地(用0表示),其他格子是障碍(用1表示).机器人每次可以往4个方向走一格,但不能连续地穿越k( ...
- Codeforces Round #287 (Div. 2) E. Breaking Good 路径记录!!!+最短路+堆优化
E. Breaking Good time limit per test 2 seconds memory limit per test 256 megabytes input standard in ...
- Java数据库之数据库的连接操作
这里面我们所连接的数据库是mysql数据库,Oracle数据库暂且先不讨论,并且mysql中的基本语法,这里面也不在一一表述了,但是看这篇文章之前,最好先仔细的连接mysql的基本语法,看起来方便~ ...
- CDH安装时,部分节点不受管控
解决方案: /opt/cm-5.12.0/etc/init.d/cloudera-scm-agent stop cd /opt/cm-5.12.0/lib/cloudera-scm-agent/ rm ...
- @清晰掉 string.h之基础堵漏
一个标准的strcpy函数: 原本以为自己对strcpy还算比较了解,结果面试时还是悲剧了. 下面给出网上strcpy的得分版本: 2分 void strcpy( char *strDest, cha ...