blob: c31a0bbfe1a8abe1570e5195e61a01b5c977a361 [file] [log] [blame]
Zhenkai Zhufd52ab72012-05-29 17:34:35 -07001#include <QtGui>
2#include "chatdialog.h"
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -07003#include "settingdialog.h"
Zhenkai Zhuc71da772012-05-30 21:25:23 -07004#include <ctime>
Zhenkai Zhu2c55b382012-05-31 13:27:24 -07005#include <iostream>
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -07006#include <QTimer>
Zhenkai Zhue8b9e8e2012-06-04 21:55:14 -07007#include <QMetaType>
Zhenkai Zhufd52ab72012-05-29 17:34:35 -07008
Zhenkai Zhu36c6b782012-06-04 17:11:04 -07009#define BROADCAST_PREFIX_FOR_SYNC_DEMO "/ndn/broadcast/sync-demo"
10
Zhenkai Zhu88380c12012-06-04 21:23:34 -070011void
12ChatDialog::testDraw()
13{
14 std::string prefix[5] = {"/ndn/1", "/ndn/2", "/ndn/3", "/ndn/4", "/ndn/5"};
15 std::string nick[5] = {"tom", "jerry", "jason", "michael", "hurry"};
16 std::vector<Sync::MissingDataInfo> v;
17 for (int i = 0; i < 5; i++)
18 {
19 Sync::MissingDataInfo mdi = {prefix[i], Sync::SeqNo(0), Sync::SeqNo(i * (2 << i) )};
20 v.push_back(mdi);
21 }
22
23 m_scene->processUpdate(v, "12341234@!#%!@");
24
25 for (int i = 0; i < 5; i++)
26 {
27 m_scene-> msgReceived(prefix[i].c_str(), nick[i].c_str());
28 }
29
30 fitView();
31}
32
Zhenkai Zhufd52ab72012-05-29 17:34:35 -070033ChatDialog::ChatDialog(QWidget *parent)
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070034 : QDialog(parent), m_sock(NULL)
Zhenkai Zhufd52ab72012-05-29 17:34:35 -070035{
Zhenkai Zhue8b9e8e2012-06-04 21:55:14 -070036 // have to register this, otherwise
37 // the signal-slot system won't recognize this type
Zhenkai Zhub285a412012-06-05 11:32:00 -070038 qRegisterMetaType<std::vector<Sync::MissingDataInfo> >("std::vector<Sync::MissingDataInfo>");
39 qRegisterMetaType<size_t>("size_t");
Zhenkai Zhufd52ab72012-05-29 17:34:35 -070040 setupUi(this);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070041 m_session = time(NULL);
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -070042
43 readSettings();
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070044
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -070045 updateLabels();
46
Zhenkai Zhufd52ab72012-05-29 17:34:35 -070047 lineEdit->setFocusPolicy(Qt::StrongFocus);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070048 m_scene = new DigestTreeScene(this);
Zhenkai Zhu9ec8f412012-06-01 15:44:36 -070049
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070050 treeViewer->setScene(m_scene);
51 m_scene->plot("Empty");
52 QRectF rect = m_scene->itemsBoundingRect();
53 m_scene->setSceneRect(rect);
54
55 // create sync socket
56 if(!m_user.getChatroom().isEmpty()) {
57 std::string syncPrefix = BROADCAST_PREFIX_FOR_SYNC_DEMO;
58 syncPrefix += "/";
59 syncPrefix += m_user.getChatroom().toStdString();
Zhenkai Zhub285a412012-06-05 11:32:00 -070060 m_sock = new Sync::SyncAppSocket(syncPrefix, bind(&ChatDialog::processTreeUpdateWrapper, this, _1, _2), bind(&ChatDialog::processRemove, this, _1));
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070061 }
Zhenkai Zhu9ec8f412012-06-01 15:44:36 -070062
Zhenkai Zhuc71da772012-05-30 21:25:23 -070063 connect(lineEdit, SIGNAL(returnPressed()), this, SLOT(returnPressed()));
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -070064 connect(setButton, SIGNAL(pressed()), this, SLOT(buttonPressed()));
Zhenkai Zhub285a412012-06-05 11:32:00 -070065 connect(this, SIGNAL(dataReceived(QString, const char *, size_t)), this, SLOT(processData(QString, const char *, size_t)));
66 connect(this, SIGNAL(treeUpdated(const std::vector<Sync::MissingDataInfo>)), this, SLOT(processTreeUpdate(const std::vector<Sync::MissingDataInfo>)));
Zhenkai Zhue8b9e8e2012-06-04 21:55:14 -070067 //testDraw();
Zhenkai Zhuc71da772012-05-30 21:25:23 -070068}
69
Zhenkai Zhu36c6b782012-06-04 17:11:04 -070070ChatDialog::~ChatDialog()
71{
72 if (m_sock != NULL)
73 {
74 delete m_sock;
75 m_sock = NULL;
76 }
77}
78
Zhenkai Zhuc71da772012-05-30 21:25:23 -070079void
Zhenkai Zhue8b9e8e2012-06-04 21:55:14 -070080ChatDialog::appendMessage(const SyncDemo::ChatMessage msg)
Zhenkai Zhuc71da772012-05-30 21:25:23 -070081{
Zhenkai Zhu2c55b382012-05-31 13:27:24 -070082
83 if (msg.type() != SyncDemo::ChatMessage::CHAT) {
Zhenkai Zhuc71da772012-05-30 21:25:23 -070084 return;
Zhenkai Zhu2c55b382012-05-31 13:27:24 -070085 }
86
87 if (!msg.has_data()) {
88 return;
89 }
90
91 if (msg.from().empty() || msg.data().empty()) {
92 return;
93 }
Zhenkai Zhuc71da772012-05-30 21:25:23 -070094
95 QTextCursor cursor(textEdit->textCursor());
96 cursor.movePosition(QTextCursor::End);
97 QTextTableFormat tableFormat;
98 tableFormat.setBorder(0);
99 QTextTable *table = cursor.insertTable(1, 2, tableFormat);
Zhenkai Zhu2c55b382012-05-31 13:27:24 -0700100 QString from = QString("<%1>: ").arg(msg.from().c_str());
101 table->cellAt(0, 0).firstCursorPosition().insertText(from);
102 table->cellAt(0, 1).firstCursorPosition().insertText(msg.data().c_str());
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700103 QScrollBar *bar = textEdit->verticalScrollBar();
104 bar->setValue(bar->maximum());
105}
106
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -0700107void
Zhenkai Zhub285a412012-06-05 11:32:00 -0700108ChatDialog::processTreeUpdateWrapper(const std::vector<Sync::MissingDataInfo> v, Sync::SyncAppSocket *sock)
109{
110 emit treeUpdated(v);
Zhenkai Zhu68f04d52012-06-05 14:07:41 -0700111#ifdef __DEBUG
112 std::cout << "<<< Tree update signal emitted" << std::endl;
113#endif
Zhenkai Zhub285a412012-06-05 11:32:00 -0700114}
115
116void
117ChatDialog::processTreeUpdate(const std::vector<Sync::MissingDataInfo> v)
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700118{
Zhenkai Zhu68f04d52012-06-05 14:07:41 -0700119#ifdef __DEBUG
120 std::cout << "<<< processing Tree Update" << std::endl;
121#endif
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700122 if (v.empty())
123 {
124 return;
125 }
126
127 // reflect the changes on digest tree
128 m_scene->processUpdate(v, m_sock->getRootDigest().c_str());
129
130 int n = v.size();
131 int totalMissingPackets = 0;
132 for (int i = 0; i < n; i++)
133 {
134 totalMissingPackets += v[i].high.getSeq() - v[i].low.getSeq() + 1;
135 }
136
137 if (totalMissingPackets < 10) {
138 for (int i = 0; i < n; i++)
139 {
140 for (Sync::SeqNo seq = v[i].low; seq <= v[i].high; ++seq)
141 {
Zhenkai Zhub285a412012-06-05 11:32:00 -0700142 m_sock->fetchRaw(v[i].prefix, seq, bind(&ChatDialog::processDataWrapper, this, _1, _2, _3), 2);
Zhenkai Zhu68f04d52012-06-05 14:07:41 -0700143#ifdef __DEBUG
144 std::cout << "<<< Fetching " << v[i].prefix << "/" <<seq.getSession() <<"/" << seq.getSeq() << std::endl;
145#endif
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700146 }
147 }
148 }
149 else
150 {
151 // too bad; too many missing packets
152 // we may just join a new chatroom
153 // or some network patition just healed
154 // we don't try to fetch any data in this case (for now)
155 }
156
157 // adjust the view
158 fitView();
159
160}
161
162void
Zhenkai Zhub285a412012-06-05 11:32:00 -0700163ChatDialog::processDataWrapper(std::string name, const char *buf, size_t len)
164{
165 emit dataReceived(name.c_str(), buf, len);
Zhenkai Zhu68f04d52012-06-05 14:07:41 -0700166#ifdef __DEBUG
167 std::cout <<"<<< " << name << " fetched" << std::endl;
168#endif
Zhenkai Zhub285a412012-06-05 11:32:00 -0700169}
170
171void
172ChatDialog::processData(QString name, const char *buf, size_t len)
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700173{
174 SyncDemo::ChatMessage msg;
175 if (!msg.ParseFromArray(buf, len))
176 {
177 std::cerr << "Errrrr.. Can not parse msg at "<<__FILE__ <<":"<<__LINE__<<". what is happening?" << std::endl;
178 abort();
179 }
180
Zhenkai Zhue8b9e8e2012-06-04 21:55:14 -0700181 // display msg received from network
182 // we have to do so; this function is called by ccnd thread
183 // so if we call appendMsg directly
184 // Qt crash as "QObject: Cannot create children for a parent that is in a different thread"
185 // the "cannonical" way to is use signal-slot
Zhenkai Zhub285a412012-06-05 11:32:00 -0700186 appendMessage(msg);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700187
188 // update the tree view
Zhenkai Zhu1c7d40a2012-06-05 14:30:17 -0700189 std::string stdStrName = name.toStdString();
190 std::string stdStrNameWithoutSeq = stdStrName.substr(0, stdStrName.find_last_of('/'));
191 std::string prefix = stdStrNameWithoutSeq.substr(0, stdStrNameWithoutSeq.find_last_of('/'));
192#ifdef __DEBUG
193 std::cout <<"<<< updating scene for" << prefix << ": " << msg.from() << std::endl;
194#endif
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700195 m_scene->msgReceived(prefix.c_str(), msg.from().c_str());
Zhenkai Zhu3e6d4792012-06-05 12:28:59 -0700196 fitView();
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700197}
198
199void
200ChatDialog::processRemove(std::string prefix)
201{
202}
203
204void
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -0700205ChatDialog::formChatMessage(const QString &text, SyncDemo::ChatMessage &msg) {
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700206 msg.set_from(m_user.getNick().toStdString());
207 msg.set_to(m_user.getChatroom().toStdString());
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -0700208 msg.set_data(text.toStdString());
209 time_t seconds = time(NULL);
210 msg.set_timestamp(seconds);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700211 msg.set_type(SyncDemo::ChatMessage::CHAT);
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700212}
213
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700214bool
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700215ChatDialog::readSettings()
216{
Zhenkai Zhub285a412012-06-05 11:32:00 -0700217#ifndef __DEBUG
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700218 QSettings s(ORGANIZATION, APPLICATION);
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700219 QString nick = s.value("nick", "").toString();
220 QString chatroom = s.value("chatroom", "").toString();
221 QString prefix = s.value("prefix", "").toString();
222 if (nick == "" || chatroom == "" || prefix == "") {
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700223 QTimer::singleShot(500, this, SLOT(buttonPressed()));
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700224 return false;
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700225 }
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700226 else {
227 m_user.setNick(nick);
228 m_user.setChatroom(chatroom);
229 m_user.setPrefix(prefix);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700230 return true;
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700231 }
Zhenkai Zhub285a412012-06-05 11:32:00 -0700232#else
233 QTimer::singleShot(500, this, SLOT(buttonPressed()));
234 return false;
235#endif
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700236}
237
238void
239ChatDialog::writeSettings()
240{
Zhenkai Zhub285a412012-06-05 11:32:00 -0700241#ifndef __DEBUG
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700242 QSettings s(ORGANIZATION, APPLICATION);
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700243 s.setValue("nick", m_user.getNick());
244 s.setValue("chatroom", m_user.getChatroom());
245 s.setValue("prefix", m_user.getPrefix());
Zhenkai Zhub285a412012-06-05 11:32:00 -0700246#endif
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700247}
248
249void
250ChatDialog::updateLabels()
251{
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700252 QString settingDisp = QString("<User: %1>, <Chatroom: %2>").arg(m_user.getNick()).arg(m_user.getChatroom());
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700253 infoLabel->setText(settingDisp);
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700254 QString prefixDisp = QString("<Prefix: %1>").arg(m_user.getPrefix());
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700255 prefixLabel->setText(prefixDisp);
256}
257
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700258void
259ChatDialog::returnPressed()
260{
261 QString text = lineEdit->text();
262 if (text.isEmpty())
263 return;
264
Zhenkai Zhu2c55b382012-05-31 13:27:24 -0700265 lineEdit->clear();
266
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700267 SyncDemo::ChatMessage msg;
268 formChatMessage(text, msg);
269
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700270 appendMessage(msg);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700271
272 // send msg
273 size_t size = msg.ByteSize();
274 char *buf = new char[size];
275 msg.SerializeToArray(buf, size);
276 if (!msg.IsInitialized())
277 {
278 std::cerr << "Errrrr.. msg was not probally initialized "<<__FILE__ <<":"<<__LINE__<<". what is happening?" << std::endl;
279 abort();
280 }
281 m_sock->publishRaw(m_user.getPrefix().toStdString(), m_session, buf, size, 60);
Zhenkai Zhu3e6d4792012-06-05 12:28:59 -0700282
283 int nextSequence = m_sock->getNextSeq(m_user.getPrefix().toStdString(), m_session);
Zhenkai Zhu68f04d52012-06-05 14:07:41 -0700284 Sync::MissingDataInfo mdi = {m_user.getPrefix().toStdString(), Sync::SeqNo(0), Sync::SeqNo(nextSequence - 1)};
Zhenkai Zhu3e6d4792012-06-05 12:28:59 -0700285 std::vector<Sync::MissingDataInfo> v;
286 v.push_back(mdi);
287 m_scene->processUpdate(v, m_sock->getRootDigest().c_str());
288 m_scene->msgReceived(m_user.getPrefix(), m_user.getNick());
289 fitView();
Zhenkai Zhuc71da772012-05-30 21:25:23 -0700290}
291
292void
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -0700293ChatDialog::buttonPressed()
294{
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700295 SettingDialog dialog(this, m_user.getNick(), m_user.getChatroom(), m_user.getPrefix());
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700296 connect(&dialog, SIGNAL(updated(QString, QString, QString)), this, SLOT(settingUpdated(QString, QString, QString)));
Zhenkai Zhu275ba3a2012-06-01 23:10:43 -0700297 dialog.exec();
Zhenkai Zhufd52ab72012-05-29 17:34:35 -0700298}
Zhenkai Zhu80af0e02012-05-31 15:49:07 -0700299
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700300void
301ChatDialog::settingUpdated(QString nick, QString chatroom, QString prefix)
302{
303 bool needWrite = false;
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700304 if (!nick.isEmpty() && nick != m_user.getNick()) {
305 m_user.setNick(nick);
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700306 needWrite = true;
307 }
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700308 if (!prefix.isEmpty() && prefix != m_user.getPrefix()) {
309 m_user.setPrefix(prefix);
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700310 needWrite = true;
311 // TODO: set the previous prefix as left?
312 }
Zhenkai Zhu56a88592012-06-04 09:42:53 -0700313 if (!chatroom.isEmpty() && chatroom != m_user.getChatroom()) {
314 m_user.setChatroom(chatroom);
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700315 needWrite = true;
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700316
317 m_scene->clearAll();
318 m_scene->plot("Empty");
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700319 // TODO: perhaps need to do a lot. e.g. use a new SyncAppSokcet
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700320 if (m_sock != NULL)
321 {
322 delete m_sock;
323 m_sock = NULL;
324 }
325 std::string syncPrefix = BROADCAST_PREFIX_FOR_SYNC_DEMO;
326 syncPrefix += "/";
327 syncPrefix += m_user.getChatroom().toStdString();
Zhenkai Zhub285a412012-06-05 11:32:00 -0700328 m_sock = new Sync::SyncAppSocket(syncPrefix, bind(&ChatDialog::processTreeUpdateWrapper, this, _1, _2), bind(&ChatDialog::processRemove, this, _1));
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700329
330 fitView();
331
Zhenkai Zhu3472e3f2012-06-02 00:44:42 -0700332 }
333 if (needWrite) {
334 writeSettings();
335 updateLabels();
336 }
337}
Zhenkai Zhue5660932012-06-04 15:25:20 -0700338
339void
340ChatDialog::resizeEvent(QResizeEvent *e)
341{
342 fitView();
343}
344
345void
346ChatDialog::showEvent(QShowEvent *e)
347{
348 fitView();
349}
350
351void
352ChatDialog::fitView()
353{
Zhenkai Zhu88380c12012-06-04 21:23:34 -0700354 QRectF rect = m_scene->itemsBoundingRect();
355 m_scene->setSceneRect(rect);
Zhenkai Zhu36c6b782012-06-04 17:11:04 -0700356 treeViewer->fitInView(m_scene->itemsBoundingRect(), Qt::KeepAspectRatio);
Zhenkai Zhue5660932012-06-04 15:25:20 -0700357}