在使用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 遍历不规则树的节点的更多相关文章

  1. SAX解析xml (遍历DOM树各节点)

    本文参考 http://yangjunfeng.iteye.com/blog/401377 1. books.xml <?xml version="1.0" encoding ...

  2. Qt实现表格树控件-自绘树节点虚线

    目录 一.开心一刻 二.自绘树节点? 三.效果展示 四.实现思路 1.可扩展接口 2.函数重写 3.同步左侧表头 五.相关文章 原文链接:Qt实现表格树控件-自绘树节点虚线 一.开心一刻 一程序员第一 ...

  3. POJ 3321 Apple Tree(后根遍历将树转化成序列,用树状数组维护)

    题意:一棵树,有很多分叉,每个分叉上最多有1个苹果. 给出n,接下来n-1行,每行u,v,表示分叉u,v之间有树枝相连.这里数据中u相当于树中的父节点,v相当于子节点. 给出两个操作: 1.C x  ...

  4. 根据 中序遍历 和 后序遍历构造树(Presentation)(C++)

    好不容易又到周五了,周末终于可以休息休息了.写这一篇随笔只是心血来潮,下午问了一位朋友PAT考的如何,顺便看一下他考的试题,里面有最后一道题,是关于给出中序遍历和后序遍历然后求一个层次遍历.等等,我找 ...

  5. 函数遍历DOM树

    //获取页面中的根节点--根标签   var root=document.documentElement;//html   //函数遍历DOM树   //根据根节点,调用fn的函数,显示的是根节点的名 ...

  6. 探索未知种族之osg类生物---状态树与渲染树以及节点树之间的关系

    节点树 首先我们来看一个场景构建的实例,并通过它来了解一下“状态节点”StateGraph 和“渲染叶”RenderLeaf 所构成的状态树,“渲染台”RenderStage 和“渲染元”Render ...

  7. 先序遍历DOM树的5种方法

    DOM树由文档中的所有节点(元素节点.文本节点.注释节点等)所构成的一个树结构,DOM树的解析和构建是浏览器要实现的关键功能.既然DOM树是一个树结构,那么我们就可以使用遍历树结构的相关方法来对DOM ...

  8. 利用层序遍历(含空节点)和中序遍历重建二叉树 python

    给定一颗二叉树的层序遍历(不含None的形式)和中序遍历序列,利用两个序列完成对二叉树的重建. 还是通过一个例子来说明整个过程,下图所示的二叉树,层序遍历结果为[a,b,c,d,e],中序遍历结果为[ ...

  9. 九度OJ 1359:大魏树遍历 (树)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:299 解决:29 题目描述: 大魏是JOBDU技术组里最喜欢折腾的一个了,单反.骑车.改九度页面,当然还有YY prado.我们姑且先把这些 ...

随机推荐

  1. 网络吞吐量(network)

    题目 分析 过一遍spfa,把从点1到其他每一个点的最短路求出来, 接着递归把所有最短路径上的路径保留,其他的删掉. 对于保留的路径作为网络流的边,流量为无穷大,对于每个点拆点两个点之间的流量为吞吐量 ...

  2. PIXI如何绘制离屏canvas到舞台上

    有个方法是toDataURL(),原生的,先转换成图片再绘制. 但是pixi提供了一个BaseTexture,其构造函数的参数可以是一个canvas 因此可以直接使用如下代码绘制canvas //微信 ...

  3. 【leetcode】1218. Longest Arithmetic Subsequence of Given Difference

    题目如下: Given an integer array arr and an integer difference, return the length of the longest subsequ ...

  4. css @keyframes属性 语法

    css @keyframes属性 语法 @keyframes是什么?直线电机生产厂家 @keyframes是CSS的一种规则,可以用来定义CSS动画的一个周期的行为,创建简单的动画. 作用:通过 @k ...

  5. CF 149E Martian Strings 后缀自动机

    这里给出来一个后缀自动机的题解. 考虑对 $s$ 的正串和反串分别建后缀自动机. 对于正串的每个节点维护 $endpos$ 的最小值. 对于反串的每个节点维护 $endpos$ 的最大值. 这两个东西 ...

  6. 2018百度之星初赛A轮 度度熊学队列

    注意:刚开始用数组存deque<int> qa[MAX]会爆内存 需要改用map<int, deque<int> > qa优化 不明觉厉 #include<b ...

  7. Centos7安装Redis3.X

    本文只是简单搭建Redis,为了整合ELK用,后面会详细写. Redis:REmote DIctionary Server(远程字典服务器) 是完全开源免费的,用C语言编写的,遵守BSD协议,是一个高 ...

  8. sh_10_体验模块

    sh_10_体验模块 import sh_10_分隔线模块 sh_10_分隔线模块.print_line("-", 50) print(sh_10_分隔线模块.name)

  9. BZOJ 2731 Luogu P3219 [HNOI2012]三角形覆盖问题 (扫描线)

    题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=2731 (luogu)https://www.luogu.org/probl ...

  10. mysql忘记root登录密码

    没有过忘记密码的程序员是不完美的,对于Oracle忘记密码可以设置orapwdfile文件, Mysql其中一种方法是通过修改文件免密然后再进行密码的修改: 1.忘记密码 [root@leader ~ ...