Blame view

QChatClient/chatroom.cpp 10.8 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(true);
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
159
160
161
}

void Chatroom::closeEvent(QCloseEvent *event)
{
    if(recv == NULL && send == NULL)
    {
        ui->inputText->setReadOnly(true);
        ui->chatText->printServerMsg("Disconnected");
        event->accept();
        return;
    }
    sendQueue.clear();
    QString exitMsg("/exit");
    putMsgToSendQueue(exitMsg);
    ui->inputText->setReadOnly(true);
    recv->join();
    send->join();
162
163
164
165
166
167
    map<string,void*>::iterator it;
    for(it = activeChats.begin(); it != activeChats.end(); it++)
    {
        ChatWindow* chat = (ChatWindow*) it->second;
        chat->close();
    }
Imanol-Mikel Barba Sabariego authored
168
169
170
171
    ui->chatText->printServerMsg("Disconnected");
    send = NULL;
    recv = NULL;
    event->accept();
172
173
}
174
bool Chatroom::getConnected()
175
{
176
177
178
179
180
181
182
183
184
185
    myMutex.lock();
    bool val = connected;
    myMutex.unlock();
    return val;
}

void Chatroom::setConnected(bool status)
{
    myMutex.lock();
    connected = status;
186
187
188
189
190
191
192
193
194
195
196
197
198
199
    if(status == false)
    {
        ui->chatText->printServerMsg("Disconnected");
        this->ui->inputText->setReadOnly(true);
        this->ui->sendButton->setDisabled(true);
        this->ui->actionDisconnect->setDisabled(true);
    }
    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");
    }
200
201
202
    myMutex.unlock();
}
Imanol-Mikel Barba Sabariego authored
203
void Chatroom::setNickname(QString nick)
204
{
Imanol-Mikel Barba Sabariego authored
205
    nickname = nick;
206
207
}
Imanol-Mikel Barba Sabariego authored
208
QString Chatroom::getNickname()
209
{
Imanol-Mikel Barba Sabariego authored
210
    return nickname;
211
212
}
Imanol-Mikel Barba Sabariego authored
213
void Chatroom::updateUserList(QString userlist)
214
{
Imanol-Mikel Barba Sabariego authored
215
216
217
    ui->userList->clear();
    QStringList users = userlist.split('\n',QString::SkipEmptyParts);
    for(int i = 0; i < users.size(); i++)
218
    {
Imanol-Mikel Barba Sabariego authored
219
        ui->userList->addItem(users[i]);
220
    }
Imanol-Mikel Barba Sabariego authored
221
222
223
224
225
}

void Chatroom::relayMsg(string msg)
{
    string sender = getSender(msg);
226
    if(!ui->userList->findItems(QString::fromStdString(sender),Qt::MatchExactly | Qt::MatchCaseSensitive).isEmpty())
Imanol-Mikel Barba Sabariego authored
227
    {
228
        map<string,void*>::iterator it = activeChats.begin();
Imanol-Mikel Barba Sabariego authored
229
        it = activeChats.find(sender);
230
231
232
233
234
235
236
237
238
        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
239
240
241
242
243
    }
}

void Chatroom::newChat(QString peerNick)
{
244
    freeChats(false);
Imanol-Mikel Barba Sabariego authored
245
246
247
248
249
250
251
252
    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)
{
253
254
    ChatWindow* chat = (ChatWindow*) activeChats.find(nickname.toStdString())->second;
    inactiveChats.insert(std::pair<string,ChatWindow*>(nickname.toStdString(),chat));
Imanol-Mikel Barba Sabariego authored
255
256
257
258
259
260
261
262
    activeChats.erase(nickname.toStdString());
}

void Chatroom::launchChatWindow(QModelIndex index)
{
    QString peerNick = ui->userList->model()->data(index).toString();
    if(activeChats.find(peerNick.toStdString()) == activeChats.end())
    {
263
        emit createChat(peerNick);
Imanol-Mikel Barba Sabariego authored
264
265
266
    }
}
267
void Chatroom::putMsgToPrintQueue(string &msg, int type)
Imanol-Mikel Barba Sabariego authored
268
269
{
    printMutex.lock();
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
    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;
    }
288
}
289
290
291
292
293
294
295
296
297
void Chatroom::printMsg()
{
    while(!printQueue.empty())
    {
        string msg = printQueue.front();
        ui->chatText->printMsg(msg);
        printQueue.pop_front();
    }
Imanol-Mikel Barba Sabariego authored
298
    printMutex.unlock();
299
300
}
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
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();
}
323
324
325
326
327
328
329
330
void Chatroom::putMsgToSendQueue(QString& msg)
{
    msgMutex.lock();
    sendQueue.push_back(msg);
    msgMutex.unlock();
    msgListNotEmpty.notify_all();
}
Imanol-Mikel Barba Sabariego authored
331
void Chatroom::sendMsg()
332
{
Imanol-Mikel Barba Sabariego authored
333
334
335
    QString msg = ui->inputText->toPlainText();
    ui->inputText->clear();
    if(msg == "/disconnect")
336
    {
Imanol-Mikel Barba Sabariego authored
337
        this->disconnectChatroom();
338
    }
Imanol-Mikel Barba Sabariego authored
339
    else if(msg == "/exit")
340
    {
Imanol-Mikel Barba Sabariego authored
341
342
343
344
345
        this->close();
    }
    else
    {
        putMsgToSendQueue(msg);
346
347
348
    }
}
Imanol-Mikel Barba Sabariego authored
349
string Chatroom::getSender(string msg)
350
{
Imanol-Mikel Barba Sabariego authored
351
352
    cout << msg << endl;
    if(msg[0] == '@')
353
    {
Imanol-Mikel Barba Sabariego authored
354
355
356
357
358
        return msg.substr(1,msg.find(" ")-1);
    }
    else
    {
        return msg.substr(0,msg.find(":"));
359
360
361
362
363
364
365
366
367
368
369
370
    }
}

void Chatroom::disconnectChatroom()
{
    msgMutex.lock();
    sendQueue.clear();
    sendQueue.push_back("/disconnect");
    msgMutex.unlock();
    msgListNotEmpty.notify_all();
}
Imanol-Mikel Barba Sabariego authored
371
void Chatroom::startSession()
372
{
Imanol-Mikel Barba Sabariego authored
373
374
375
376
    LoginScreen login(&s,this);
    login.exec();
    int result = login.result();
    if(result == QDialog::Accepted)
377
    {
Imanol-Mikel Barba Sabariego authored
378
379
380
381
382
        ui->inputText->setReadOnly(false);
    }
    else if(result == QDialog::Rejected)
    {
        return;
383
    }
384
    setConnected(true);
Imanol-Mikel Barba Sabariego authored
385
386
    recv = new std::thread(recvThread,&s,this);
    send = new std::thread(sendThread,&s,this);
387
388
}
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
void Chatroom::freeChats(bool active)
{
    int count = 0;
    if(active)
    {
        cout << "FREE ACTIVE" << endl;
        map<string,void*>::iterator it;
        for(it = activeChats.begin(); it != activeChats.end(); it++)
        {
            count++;
            ChatWindow* chat = (ChatWindow*) it->second;
            activeChats.erase(it);
            delete chat;
        }
    }
    else
    {
        cout << "FREE INACTIVE" << endl;
        map<string,void*>::iterator it;
        for(it = inactiveChats.begin(); it != inactiveChats.end(); it++)
        {
            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
    freeChats(false);
    freeChats(true);
427
428
    delete ui;
}
429
430