blob: 35df07198f49b795b740fca0da45fb0367b29d91 [file] [log] [blame]
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -07001#include <QtGui>
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -07002#include <vector>
3#include <iostream>
4#include <assert.h>
Zhenkai Zhu56a88592012-06-04 09:42:53 -07005#include <boost/lexical_cast.hpp>
Zhenkai Zhu88380c12012-06-04 21:23:34 -07006#include <memory>
7#include "digesttreescene.h"
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -07008
9static const double Pi = 3.14159265358979323846264338327950288419717;
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -070010
Zhenkai Zhue5660932012-06-04 15:25:20 -070011
12DigestTreeScene::DisplayUserPtr DigestTreeScene::DisplayUserNullPtr;
13
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -070014DigestTreeScene::DigestTreeScene(QWidget *parent)
15 : QGraphicsScene(parent)
16{
Zhenkai Zhue5660932012-06-04 15:25:20 -070017 previouslyUpdatedUser = DisplayUserNullPtr;
Zhenkai Zhueeff9662012-05-30 17:02:49 -070018}
19
20void
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070021DigestTreeScene::processUpdate(const std::vector<Sync::MissingDataInfo> &v, QString digest)
Zhenkai Zhu56a88592012-06-04 09:42:53 -070022{
23 int n = v.size();
24 bool rePlot = false;
25 for (int i = 0; i < n; i++)
26 {
27 Roster_iterator it = m_roster.find(v[i].prefix.c_str());
28 if (it == m_roster.end()) {
29 rePlot = true;
30 DisplayUserPtr p(new DisplayUser());
31 p->setPrefix(v[i].prefix.c_str());
32 p->setSeq(v[i].high);
33 m_roster.insert(p->getPrefix(), p);
34 }
35 }
36
37 if (rePlot)
38 {
39 plot(digest);
40 }
41 else
42 {
43 for (int i = 0; i < n; i++)
44 {
45 Roster_iterator it = m_roster.find(v[i].prefix.c_str());
46 if (it != m_roster.end()) {
47 DisplayUserPtr p = it.value();
48 QGraphicsTextItem *item = p->getSeqTextItem();
Zhenkai Zhue5660932012-06-04 15:25:20 -070049 QGraphicsRectItem *rectItem = p->getInnerRectItem();
Zhenkai Zhu56a88592012-06-04 09:42:53 -070050 std::string s = boost::lexical_cast<std::string>(p->getSeqNo().getSeq());
51 item->setPlainText(s.c_str());
Zhenkai Zhue5660932012-06-04 15:25:20 -070052 QRectF textBR = item->boundingRect();
53 QRectF rectBR = rectItem->boundingRect();
54 item->setPos(rectBR.x() + (rectBR.width() - textBR.width())/2, rectBR.y() + (rectBR.height() - textBR.height())/2);
Zhenkai Zhu56a88592012-06-04 09:42:53 -070055 }
56 }
57 m_rootDigest->setPlainText(digest);
58 }
59}
60
61void
62DigestTreeScene::msgReceived(QString prefix, QString nick)
63{
64 Roster_iterator it = m_roster.find(prefix);
65 if (it != m_roster.end())
66 {
67 DisplayUserPtr p = it.value();
68 if (nick != p->getNick()) {
69 p->setNick(nick);
70 QGraphicsTextItem *nickItem = p->getNickTextItem();
Zhenkai Zhue5660932012-06-04 15:25:20 -070071 QGraphicsRectItem *nickRectItem = p->getNickRectItem();
Zhenkai Zhu56a88592012-06-04 09:42:53 -070072 nickItem->setPlainText(p->getNick());
Zhenkai Zhue5660932012-06-04 15:25:20 -070073 QRectF rectBR = nickRectItem->boundingRect();
74 QRectF nickBR = nickItem->boundingRect();
75 nickItem->setPos(rectBR.x() + (rectBR.width() - nickBR.width())/2, rectBR.y() + 5);
Zhenkai Zhu56a88592012-06-04 09:42:53 -070076 }
Zhenkai Zhue5660932012-06-04 15:25:20 -070077
78 reDrawNode(p, Qt::red);
79
80 if (previouslyUpdatedUser != DisplayUserNullPtr)
81 {
82 reDrawNode(previouslyUpdatedUser, Qt::darkBlue);
83 }
84
85 previouslyUpdatedUser = p;
Zhenkai Zhu56a88592012-06-04 09:42:53 -070086 }
87}
88
89void
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070090DigestTreeScene::clearAll()
91{
92 clear();
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070093 m_roster.clear();
94}
95
96void
Zhenkai Zhu56a88592012-06-04 09:42:53 -070097DigestTreeScene::plot(QString digest)
98{
Zhenkai Zhueeff9662012-05-30 17:02:49 -070099 clear();
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700100
Zhenkai Zhue5660932012-06-04 15:25:20 -0700101 int nodeSize = 40;
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700102
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700103 int siblingDistance = 100, levelDistance = 100;
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700104 std::auto_ptr<TreeLayout> layout(new OneLevelTreeLayout());
105 layout->setSiblingDistance(siblingDistance);
106 layout->setLevelDistance(levelDistance);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700107
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700108 int n = m_roster.size();
109 std::vector<TreeLayout::Coordinate> childNodesCo(n);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700110
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700111 layout->setOneLevelLayout(childNodesCo);
112
113 plotEdge(childNodesCo, nodeSize);
114 plotNode(childNodesCo, digest, nodeSize);
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700115
Zhenkai Zhue5660932012-06-04 15:25:20 -0700116 previouslyUpdatedUser = DisplayUserNullPtr;
117
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700118}
119
120void
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700121DigestTreeScene::plotEdge(const std::vector<TreeLayout::Coordinate> &childNodesCo, int nodeSize)
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700122{
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700123 int n = childNodesCo.size();
124 for (int i = 0; i < n; i++) {
125 double x1 = 0.0, y1 = 0.0;
126 double x2 = childNodesCo[i].x, y2 = childNodesCo[i].y;
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700127 QPointF src(x1 + nodeSize/2, y1 + nodeSize/2);
128 QPointF dest(x2 + nodeSize/2, y2 + nodeSize/2);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700129 QLineF line(src, dest);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700130 double angle = ::acos(line.dx() / line.length());
131
132 double arrowSize = 10;
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700133 QPointF sourceArrowP0 = src + QPointF((nodeSize/2 + 10) * line.dx() / line.length(), (nodeSize/2 +10) * line.dy() / line.length());
134 QPointF sourceArrowP1 = sourceArrowP0 + QPointF(cos(angle + Pi / 3 - Pi/2) * arrowSize,
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700135 sin(angle + Pi / 3 - Pi/2) * arrowSize);
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700136 QPointF sourceArrowP2 = sourceArrowP0 + QPointF(cos(angle + Pi - Pi / 3 - Pi/2) * arrowSize,
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700137 sin(angle + Pi - Pi / 3 - Pi/2) * arrowSize);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700138
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700139 addLine(QLineF(sourceArrowP0, dest), QPen(Qt::black));
140 addPolygon(QPolygonF() << sourceArrowP0<< sourceArrowP1 << sourceArrowP2, QPen(Qt::black), QBrush(Qt::black));
141 }
142}
143
144void
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700145DigestTreeScene::plotNode(const std::vector<TreeLayout::Coordinate> &childNodesCo, QString digest, int nodeSize)
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700146{
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700147 RosterIterator it(m_roster);
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700148 int n = childNodesCo.size();
149 int rim = 3;
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700150
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700151 // plot root node
152 QRectF rootBoundingRect(0, 0, nodeSize, nodeSize);
153 QRectF rootInnerBoundingRect(rim, rim, nodeSize - rim * 2, nodeSize - rim * 2);
154 addRect(rootBoundingRect, QPen(Qt::black), QBrush(Qt::darkRed));
155 addRect(rootInnerBoundingRect, QPen(Qt::black), QBrush(Qt::lightGray));
Zhenkai Zhu3e6d4792012-06-05 12:28:59 -0700156 QRectF digestRect(- 5.5 * nodeSize , - nodeSize, 12 * nodeSize, 30);
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700157 addRect(digestRect, QPen(Qt::darkCyan), QBrush(Qt::darkCyan));
Zhenkai Zhu43b98fe2012-06-05 12:04:04 -0700158
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700159 QGraphicsTextItem *digestItem = addText(digest);
160 QRectF digestBoundingRect = digestItem->boundingRect();
Zhenkai Zhu43b98fe2012-06-05 12:04:04 -0700161 digestItem->setDefaultTextColor(Qt::black);
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700162 digestItem->setFont(QFont("Cursive", 12, QFont::Bold));
Zhenkai Zhu3e6d4792012-06-05 12:28:59 -0700163 digestItem->setPos(- 4.5 * nodeSize + (12 * nodeSize - digestBoundingRect.width()) / 2, - nodeSize + 5);
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700164 m_rootDigest = digestItem;
165
166 // plot child nodes
167 for (int i = 0; i < n; i++)
168 {
169 if (it.hasNext())
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700170 {
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700171 it.next();
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700172 }
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700173 else
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700174 {
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700175 abort();
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700176 }
Zhenkai Zhue5660932012-06-04 15:25:20 -0700177
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700178 double x = childNodesCo[i].x;
179 double y = childNodesCo[i].y;
180 QRectF boundingRect(x, y, nodeSize, nodeSize);
181 QRectF innerBoundingRect(x + rim, y + rim, nodeSize - rim * 2, nodeSize - rim * 2);
182 DisplayUserPtr p = it.value();
183 QGraphicsRectItem *rectItem = addRect(boundingRect, QPen(Qt::black), QBrush(Qt::darkBlue));
184 p->setRimRectItem(rectItem);
185
186 QGraphicsRectItem *innerRectItem = addRect(innerBoundingRect, QPen(Qt::black), QBrush(Qt::lightGray));
187 p->setInnerRectItem(innerRectItem);
188
189 std::string s = boost::lexical_cast<std::string>(p->getSeqNo().getSeq());
190 QGraphicsTextItem *seqItem = addText(s.c_str());
191 seqItem->setFont(QFont("Cursive", 12, QFont::Bold));
192 QRectF seqBoundingRect = seqItem->boundingRect();
193 seqItem->setPos(x + nodeSize / 2 - seqBoundingRect.width() / 2, y + nodeSize / 2 - seqBoundingRect.height() / 2);
194 p->setSeqTextItem(seqItem);
195
196 QRectF textRect(x - nodeSize / 2, y + nodeSize, 2 * nodeSize, 30);
197 QGraphicsRectItem *nickRectItem = addRect(textRect, QPen(Qt::darkCyan), QBrush(Qt::darkCyan));
198 p->setNickRectItem(nickRectItem);
199 QGraphicsTextItem *nickItem = addText(p->getNick());
200 QRectF textBoundingRect = nickItem->boundingRect();
201 nickItem->setDefaultTextColor(Qt::white);
202 nickItem->setFont(QFont("Cursive", 12, QFont::Bold));
203 nickItem->setPos(x + nodeSize / 2 - textBoundingRect.width() / 2, y + nodeSize + 5);
204 p->setNickTextItem(nickItem);
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700205 }
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700206
Zhenkai Zhuc5c79b62012-05-30 15:06:29 -0700207}
208
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700209void
210DigestTreeScene::reDrawNode(DisplayUserPtr p, QColor rimColor)
211{
212 QGraphicsRectItem *rimItem = p->getRimRectItem();
213 rimItem->setBrush(QBrush(rimColor));
214 QGraphicsRectItem *innerItem = p->getInnerRectItem();
215 innerItem->setBrush(QBrush(Qt::lightGray));
216 QGraphicsTextItem *seqTextItem = p->getSeqTextItem();
217 std::string s = boost::lexical_cast<std::string>(p->getSeqNo().getSeq());
218 seqTextItem->setPlainText(s.c_str());
219 QRectF textBR = seqTextItem->boundingRect();
220 QRectF innerBR = innerItem->boundingRect();
221 seqTextItem->setPos(innerBR.x() + (innerBR.width() - textBR.width())/2, innerBR.y() + (innerBR.height() - textBR.height())/2);
222}
223