Blame view

QChatClient/chatroom.cpp 10.7 KB
1
2
3
#include "chatroom.h"
#include "ui_chatroom.h"
4
5
6
list<QString> sendQueue;
std::mutex myMutex;
std::mutex msgMutex;
Imanol-Mikel Barba Sabariego authored
7
std::mutex printMutex;
8
9
10
11
unique_lock<std::mutex>* msgLock;
std::condition_variable msgListNotEmpty;

void sendThread(Socket* s, Chatroom* chat)
12
13
{
    string send;
14
    while(chat->getConnected())
15
    {
16
        while(sendQueue.empty())
17
        {
18
19
            msgLock->lock();
            msgListNotEmpty.wait(*msgLock);
Imanol-Mikel Barba Sabariego authored
20
            msgLock->unlock();
21
22
23
24
            if(!chat->getConnected())
            {
                return;
            }
25
        }
26
27
28
        msgLock->lock();
        send = sendQueue.front().toStdString();
        sendQueue.pop_front();
29
30
        try
        {
31
32
            if(send[0] == '@')
            {
Imanol-Mikel Barba Sabariego authored
33
                chat->relayMsg(send);
34
35
36
37
38
                send = '1' + send;
            }
            else
            {
                send = '0' + send;
Imanol-Mikel Barba Sabariego authored
39
40
                if(send.substr(1) != "/exit" && send.substr(1) != "/disconnect")
                {
41
                    chat->putMsgToPrintQueue(chat->getNickname().toStdString().append(": ").append(send.substr(1)),MSG_USER);
Imanol-Mikel Barba Sabariego authored
42
                }
43
44
            }
            msgLock->unlock();
Imanol-Mikel Barba Sabariego authored
45
            *s << send;
46
            send = send.substr(1);
47
48
49
50
51
52
53
54
55
56
57
58
            if(send == "/disconnect" || send == "/exit")
            {
                break;
            }
        }
        catch(SocketException& e)
        {
            cout << e.description() << endl;
        }
    }
}
59
void recvThread(Socket* s, Chatroom* chat)
60
61
62
63
64
65
{
    string recv;
    while(true)
    {
        try
        {
66
67
68
            *s >> recv;
            if(recv[0] == '1')
            {
Imanol-Mikel Barba Sabariego authored
69
                chat->relayMsg(recv.substr(1));
70
71
72
73
74
75
                continue;
            }
            else if(recv[0] == '0')
            {
                recv = recv.substr(1);
            }
76
77
78
        }
        catch(SocketException &e)
        {
79
            chat->toggleConnected(false);
80
            cout << e.description() << endl;
81
            msgListNotEmpty.notify_all();
82
83
            string msg = "Connection to server lost";
            chat->putMsgToPrintQueue(msg,MSG_STATUS);
84
85
86
87
88
            break;
        }
        if(recv == "DISC_OK")
        {
            cout << "Disconnecting" << endl;
89
            chat->toggleConnected(false);
90
            msgListNotEmpty.notify_all();
91
92
93
94
95
            break;
        }
        else if(recv == "EXIT_OK")
        {
            cout << "Exiting" << endl;
96
            chat->toggleConnected(false);
97
            msgListNotEmpty.notify_all();
Imanol-Mikel Barba Sabariego authored
98
            chat->close();
99
100
            break;
        }
Imanol-Mikel Barba Sabariego authored
101
102
103
104
105
        else if(recv.substr(0,8) == "USERLIST")
        {
            QString userlist = QString::fromStdString(recv.substr(8));
            chat->updateUserList(userlist);
        }
106
107
        else
        {
108
            chat->putMsgToPrintQueue(recv,MSG_USER);
109
110
        }
    }
Imanol-Mikel Barba Sabariego authored
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
}

Chatroom::Chatroom(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::Chatroom)
{
    ui->setupUi(this);
    QList<int> chatSizes;
    QList<int> splitSizes;
    chatSizes.push_front(ui->chatText->height());
    chatSizes.push_front(100);
    splitSizes.push_front(ui->chatWindow->width());
    splitSizes.push_front(200);
    ui->chatSplitter->setSizes(chatSizes);
    ui->windowSplitter->setSizes(splitSizes);

    connected = false;
    msgLock = new unique_lock<std::mutex>(msgMutex,std::defer_lock);
    chatLock = new unique_lock<std::mutex>(chatMutex,std::defer_lock);
    send = NULL;
    recv = NULL;

    connect(ui->actionConnect,SIGNAL(triggered()),this,SLOT(startSession()));
    connect(ui->actionExit,SIGNAL(triggered()),this,SLOT(close()));
    connect(ui->actionDisconnect,SIGNAL(triggered()),this,SLOT(disconnectChatroom()));
    connect(ui->actionSave_chat,SIGNAL(triggered()),ui->chatText,SLOT(saveChatToFile()));
    connect(ui->inputText,SIGNAL(msgReady()),this,SLOT(sendMsg()));
    connect(ui->sendButton,SIGNAL(clicked()),this,SLOT(sendMsg()));
    connect(this,SIGNAL(messagesToPrint()),this,SLOT(printMsg()));
140
141
    connect(this,SIGNAL(serverMessagesToPrint()),this,SLOT(printServerMsg()));
    connect(this,SIGNAL(statusMessagesToPrint()),this,SLOT(printStatusMsg()));
Imanol-Mikel Barba Sabariego authored
142
143
    connect(ui->userList,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(launchChatWindow(QModelIndex)));
    connect(this,SIGNAL(createChat(QString)),this,SLOT(newChat(QString)));
144
    connect(this,SIGNAL(toggleConnected(bool)),this,SLOT(setConnected(bool)));
Imanol-Mikel Barba Sabariego authored
145
146
147
148
149
150
151
152
153
154
155
156
157
158
}

void Chatroom::closeEvent(QCloseEvent *event)
{
    if(recv == NULL && send == NULL)
    {
        event->accept();
        return;
    }
    sendQueue.clear();
    QString exitMsg("/exit");
    putMsgToSendQueue(exitMsg);
    recv->join();
    send->join();
159
160
    freeChats(true);
    freeChats(false);
Imanol-Mikel Barba Sabariego authored
161
    event->accept();
162
163
}
164
bool Chatroom::getConnected()
165
{
166
167
168
169
170
171
172
173
174
175
    myMutex.lock();
    bool val = connected;
    myMutex.unlock();
    return val;
}

void Chatroom::setConnected(bool status)
{
    myMutex.lock();
    connected = status;
176
177
178
179
180
181
    if(status == false)
    {
        ui->chatText->printServerMsg("Disconnected");
        this->ui->inputText->setReadOnly(true);
        this->ui->sendButton->setDisabled(true);
        this->ui->actionDisconnect->setDisabled(true);
182
        this->ui->userList->clear();
183
184
185
186
187
188
189
190
    }
    else if(status == true)
    {
        this->ui->inputText->setReadOnly(false);
        this->ui->sendButton->setDisabled(false);
        this->ui->actionDisconnect->setDisabled(false);
        ui->chatText->printServerMsg("Connected to chatroom");
    }
191
192
193
    myMutex.unlock();
}
Imanol-Mikel Barba Sabariego authored
194
void Chatroom::setNickname(QString nick)
195
{
Imanol-Mikel Barba Sabariego authored
196
    nickname = nick;
197
198
}
Imanol-Mikel Barba Sabariego authored
199
QString Chatroom::getNickname()
200
{
Imanol-Mikel Barba Sabariego authored
201
    return nickname;
202
203
}
Imanol-Mikel Barba Sabariego authored
204
void Chatroom::updateUserList(QString userlist)
205
{
Imanol-Mikel Barba Sabariego authored
206
207
208
    ui->userList->clear();
    QStringList users = userlist.split('\n',QString::SkipEmptyParts);
    for(int i = 0; i < users.size(); i++)
209
    {
Imanol-Mikel Barba Sabariego authored
210
        ui->userList->addItem(users[i]);
211
    }
Imanol-Mikel Barba Sabariego authored
212
213
214
215
216
}

void Chatroom::relayMsg(string msg)
{
    string sender = getSender(msg);
217
    if(!ui->userList->findItems(QString::fromStdString(sender),Qt::MatchExactly | Qt::MatchCaseSensitive).isEmpty())
Imanol-Mikel Barba Sabariego authored
218
    {
219
        map<string,void*>::iterator it = activeChats.begin();
Imanol-Mikel Barba Sabariego authored
220
        it = activeChats.find(sender);
221
222
223
224
225
226
227
228
229
        if(it == activeChats.end())
        {
            chatLock->lock();
            emit createChat(QString::fromStdString(sender));
            chatCreated.wait(*chatLock);
            chatLock->unlock();
            it = activeChats.find(sender);
        }
        ((ChatWindow*)it->second)->notifyPrint(msg);
Imanol-Mikel Barba Sabariego authored
230
231
232
233
234
    }
}

void Chatroom::newChat(QString peerNick)
{
235
    freeChats(false);
Imanol-Mikel Barba Sabariego authored
236
237
238
239
240
241
242
243
    ChatWindow* newchat = new ChatWindow(peerNick,this);
    activeChats.insert(std::pair<string,ChatWindow*>(peerNick.toStdString(),newchat));
    newchat->show();
    chatCreated.notify_all();
}

void Chatroom::removeChat(QString &nickname)
{
244
    chatMutex.lock();
245
246
    ChatWindow* chat = (ChatWindow*) activeChats.find(nickname.toStdString())->second;
    inactiveChats.insert(std::pair<string,ChatWindow*>(nickname.toStdString(),chat));
Imanol-Mikel Barba Sabariego authored
247
    activeChats.erase(nickname.toStdString());
248
    chatMutex.unlock();
Imanol-Mikel Barba Sabariego authored
249
250
251
252
253
254
255
}

void Chatroom::launchChatWindow(QModelIndex index)
{
    QString peerNick = ui->userList->model()->data(index).toString();
    if(activeChats.find(peerNick.toStdString()) == activeChats.end())
    {
256
        emit createChat(peerNick);
Imanol-Mikel Barba Sabariego authored
257
258
259
    }
}
260
void Chatroom::putMsgToPrintQueue(string &msg, int type)
Imanol-Mikel Barba Sabariego authored
261
262
{
    printMutex.lock();
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
    if(type == MSG_USER)
    {
        printQueue.push_back(msg);
        emit messagesToPrint();
        return;
    }
    else if(type == MSG_SERVER)
    {
        printServerQueue.push_back(msg);
        emit serverMessagesToPrint();
        return;
    }
    else if(type == MSG_STATUS)
    {
        printStatusQueue.push_back(msg);
        emit statusMessagesToPrint();
        return;
    }
281
}
282
283
284
285
286
287
288
289
290
void Chatroom::printMsg()
{
    while(!printQueue.empty())
    {
        string msg = printQueue.front();
        ui->chatText->printMsg(msg);
        printQueue.pop_front();
    }
Imanol-Mikel Barba Sabariego authored
291
    printMutex.unlock();
292
293
}
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
void Chatroom::printServerMsg()
{
    while(!printServerQueue.empty())
    {
        string msg = printServerQueue.front();
        ui->chatText->printServerMsg(msg);
        printServerQueue.pop_front();
    }
    printMutex.unlock();
}

void Chatroom::printStatusMsg()
{
    while(!printStatusQueue.empty())
    {
        string msg = printStatusQueue.front();
        ui->chatText->printStatusMsg(msg);
        printStatusQueue.pop_front();
    }
    printMutex.unlock();
}
316
317
318
319
320
321
322
323
void Chatroom::putMsgToSendQueue(QString& msg)
{
    msgMutex.lock();
    sendQueue.push_back(msg);
    msgMutex.unlock();
    msgListNotEmpty.notify_all();
}
Imanol-Mikel Barba Sabariego authored
324
void Chatroom::sendMsg()
325
{
Imanol-Mikel Barba Sabariego authored
326
327
328
    QString msg = ui->inputText->toPlainText();
    ui->inputText->clear();
    if(msg == "/disconnect")
329
    {
Imanol-Mikel Barba Sabariego authored
330
        this->disconnectChatroom();
331
    }
Imanol-Mikel Barba Sabariego authored
332
    else if(msg == "/exit")
333
    {
Imanol-Mikel Barba Sabariego authored
334
335
336
337
338
        this->close();
    }
    else
    {
        putMsgToSendQueue(msg);
339
340
341
    }
}
Imanol-Mikel Barba Sabariego authored
342
string Chatroom::getSender(string msg)
343
{
Imanol-Mikel Barba Sabariego authored
344
345
    cout << msg << endl;
    if(msg[0] == '@')
346
    {
Imanol-Mikel Barba Sabariego authored
347
348
349
350
351
        return msg.substr(1,msg.find(" ")-1);
    }
    else
    {
        return msg.substr(0,msg.find(":"));
352
353
354
355
356
357
358
359
360
361
362
363
    }
}

void Chatroom::disconnectChatroom()
{
    msgMutex.lock();
    sendQueue.clear();
    sendQueue.push_back("/disconnect");
    msgMutex.unlock();
    msgListNotEmpty.notify_all();
}
Imanol-Mikel Barba Sabariego authored
364
void Chatroom::startSession()
365
{
Imanol-Mikel Barba Sabariego authored
366
367
368
    LoginScreen login(&s,this);
    login.exec();
    int result = login.result();
369
    if(result == QDialog::Rejected)
Imanol-Mikel Barba Sabariego authored
370
371
    {
        return;
372
    }
373
    setConnected(true);
Imanol-Mikel Barba Sabariego authored
374
375
    recv = new std::thread(recvThread,&s,this);
    send = new std::thread(sendThread,&s,this);
376
377
}
378
379
380
void Chatroom::freeChats(bool active)
{
    int count = 0;
381
    map<string,void*>::iterator it;
382
383
    if(active)
    {
384
385
        cout << "FREE ACTIVE (" << activeChats.size() << " remaining)" << endl;
        while(true)
386
        {
387
388
389
390
391
392
393
            chatMutex.lock();
            it = activeChats.begin();
            chatMutex.unlock();
            if(it == activeChats.end())
            {
                break;
            }
394
395
            count++;
            ChatWindow* chat = (ChatWindow*) it->second;
396
            chat->notifyClose();
397
398
399
400
        }
    }
    else
    {
401
        cout << "FREE INACTIVE (" << inactiveChats.size() << " remaining)" << endl;
402
        map<string,void*>::iterator it;
403
        while(true)
404
        {
405
406
407
408
409
            it = inactiveChats.begin();
            if(it == inactiveChats.end())
            {
                break;
            }
410
411
412
413
414
415
416
417
418
            count++;
            ChatWindow* chat = (ChatWindow*) it->second;
            inactiveChats.erase(it);
            delete chat;
        }
    }
    cout << "DELETED " << count << endl << "REMAINING" << endl << activeChats.size() << " ACTIVE" << endl << inactiveChats.size() << " INACTIVE" << endl;
}
419
420
Chatroom::~Chatroom()
{
421
    delete msgLock;
422
    delete chatLock;
423
424
    delete recv;
    delete send;
425
426
    delete ui;
}
427
428