/**************************************************************** * RetroShare is distributed under the following license: * * Copyright (C) 2006 - 2011 RetroShare Team * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. ****************************************************************/ #include #include #include #include #include #include #include #include #include #include "retroshare/rsinit.h" #include "retroshare/rsnotify.h" #include "retroshare/rspeers.h" #include #include "channels/CreateChannel.h" #include "common/Emoticons.h" #include "common/vmessagebox.h" #include "chat/ChatDialog.h" #include "connect/ConfCertDialog.h" #include "connect/ConnectFriendWizard.h" #include "forums/CreateForum.h" #include "gui/common/AvatarDefs.h" #include "gui/mainpagestack.h" #include "groups/CreateGroup.h" #include "im_history/ImHistoryBrowser.h" #include "MainWindow.h" #include "NewsFeed.h" #include "ChatLobbyWidget.h" #include "notifyqt.h" #include "profile/ProfileWidget.h" #include "profile/StatusMessage.h" #include "RetroShareLink.h" #include "settings/rsharesettings.h" #include "util/misc.h" #include "chat/CreateLobbyDialog.h" #include "FriendRecommendDialog.h" #include "FriendsDialog.h" /* Images for Newsfeed icons */ #define IMAGE_NEWSFEED "" #define IMAGE_NEWSFEED_NEW ":/images/message-state-new.png" /****** * #define FRIENDS_DEBUG 1 *****/ static FriendsDialog *instance = NULL; /** Constructor */ FriendsDialog::FriendsDialog(QWidget *parent) : RsAutoUpdatePage(1500,parent) { /* Invoke the Qt Designer generated object setup routine */ ui.setupUi(this); if (instance == NULL) { instance = this; } last_status_send_time = 0 ; connect( ui.mypersonalstatuslabel, SIGNAL(clicked()), SLOT(statusmessage())); connect( ui.actionSet_your_Avatar, SIGNAL(triggered()), this, SLOT(getAvatar())); connect( ui.actionSet_your_Personal_Message, SIGNAL(triggered()), this, SLOT(statusmessage())); connect( ui.addfileButton, SIGNAL(clicked() ), this , SLOT(addExtraFile())); connect( ui.actionAdd_Friend, SIGNAL(triggered()), this, SLOT(addFriend())); connect( ui.actionCreate_new_Chat_lobby, SIGNAL(triggered()), this, SLOT(createChatLobby())); connect( ui.actionFriendRecommendations, SIGNAL(triggered()), this, SLOT(recommendFriends())); ui.avatar->setFrameType(AvatarWidget::STATUS_FRAME); ui.avatar->setOwnId(); ui.tabWidget->setTabPosition(QTabWidget::North); ui.tabWidget->addTab(new ChatLobbyWidget(), tr("Chat lobbies")); ui.tabWidget->addTab(new ProfileWidget(), tr("Profile")); newsFeed = new NewsFeed(); int newsFeedTabIndex = ui.tabWidget->insertTab(0, newsFeed, tr("News Feed")); ui.tabWidget->setCurrentIndex(newsFeedTabIndex); ui.tabWidget->hideCloseButton(0); ui.tabWidget->hideCloseButton(1); ui.tabWidget->hideCloseButton(2); ui.tabWidget->hideCloseButton(3); /* get the current text and text color of the tab bar */ newsFeedTabColor = ui.tabWidget->tabBar()->tabTextColor(newsFeedTabIndex); newsFeedText = ui.tabWidget->tabBar()->tabText(newsFeedTabIndex); connect(newsFeed, SIGNAL(newsFeedChanged(int)), this, SLOT(newsFeedChanged(int))); connect(ui.Sendbtn, SIGNAL(clicked()), this, SLOT(sendMsg())); connect(ui.emoticonBtn, SIGNAL(clicked()), this, SLOT(smileyWidgetgroupchat())); connect(ui.lineEdit,SIGNAL(customContextMenuRequested(QPoint)),this,SLOT(contextMenu(QPoint))); connect(ui.textboldChatButton, SIGNAL(clicked()), this, SLOT(setFont())); connect(ui.textunderlineChatButton, SIGNAL(clicked()), this, SLOT(setFont())); connect(ui.textitalicChatButton, SIGNAL(clicked()), this, SLOT(setFont())); connect(ui.fontsButton, SIGNAL(clicked()), this, SLOT(chooseFont())); connect(ui.colorChatButton, SIGNAL(clicked()), this, SLOT(chooseColor())); connect(ui.actionSave_History, SIGNAL(triggered()), this, SLOT(fileSaveAs())); connect(ui.hashBox, SIGNAL(fileHashingFinished(QList)), this, SLOT(fileHashingFinished(QList))); ui.fontsButton->setIcon(QIcon(QString(":/images/fonts.png"))); mCurrentColor = Qt::black; mCurrentFont.fromString(Settings->getChatScreenFont()); colorChanged(); fontChanged(); setColorAndFont(); style.setStyleFromSettings(ChatStyle::TYPE_PUBLIC); setChatInfo(tr("Welcome to RetroShare's group chat."), QString::fromUtf8("blue")); if (rsHistory->getEnable(true)) { int messageCount = Settings->getPublicChatHistoryCount(); if (messageCount > 0) { std::list historyMsgs; rsHistory->getMessages("", historyMsgs, messageCount); std::list::iterator it; for (it = historyMsgs.begin(); it != historyMsgs.end(); it++) { addChatMsg(it->incoming, true, QString::fromUtf8(it->peerName.c_str()), QDateTime::fromTime_t(it->sendTime), QDateTime::fromTime_t(it->recvTime), QString::fromUtf8(it->message.c_str())); } } } QMenu *menu = new QMenu(); menu->addAction(ui.actionClear_Chat_History); menu->addAction(ui.actionDelete_Chat_History); menu->addAction(ui.actionSave_History); menu->addAction(ui.actionMessageHistory); ui.menuButton->setMenu(menu); menu = new QMenu(); menu->addAction(ui.actionAdd_Friend); menu->addAction(ui.actionAdd_Group); menu->addAction(ui.actionCreate_new_Chat_lobby); menu->addAction(ui.actionFriendRecommendations); menu->addSeparator(); menu->addAction(ui.actionCreate_New_Forum); #ifndef RS_RELEASE_VERSION menu->addAction(ui.actionCreate_New_Channel); #endif menu->addAction(ui.actionSet_your_Avatar); menu->addAction(ui.actionSet_your_Personal_Message); ui.menupushButton->setMenu(menu); setAcceptDrops(true); ui.lineEdit->setAcceptDrops(false); ui.hashBox->setDropWidget(this); ui.hashBox->setAutoHide(true); /* Set initial size the splitter */ QList sizes; sizes << height() << 100; // Qt calculates the right sizes ui.splitter_2->setSizes(sizes); loadmypersonalstatus(); ui.displayButton->setMenu(ui.friendList->createDisplayMenu()); // load settings RsAutoUpdatePage::lockAllEvents(); ui.friendList->setShowStatusColumn(true); ui.friendList->setShowLastContactColumn(false); ui.friendList->setShowAvatarColumn(false); ui.friendList->setRootIsDecorated(true); ui.friendList->setShowGroups(true); processSettings(true); RsAutoUpdatePage::unlockAllEvents(); ui.lineEdit->installEventFilter(this); // add self nick and Avatar to Friends. RsPeerDetails pd ; if (rsPeers->getPeerDetails(rsPeers->getOwnId(),pd)) { QString titleStr("%1"); ui.nicklabel->setText(titleStr.arg(QString::fromUtf8(pd.name.c_str()) + " (" + tr("me") + ") " + QString::fromUtf8(pd.location.c_str()))); } /* Hide platform specific features */ #ifdef Q_WS_WIN #endif } FriendsDialog::~FriendsDialog () { // save settings processSettings(false); if (this == instance) { instance = NULL; } } void FriendsDialog::processSettings(bool bLoad) { Settings->beginGroup(QString("FriendsDialog")); if (bLoad) { // load settings // state of splitter ui.splitter->restoreState(Settings->value("Splitter").toByteArray()); ui.splitter_2->restoreState(Settings->value("GroupChatSplitter").toByteArray()); } else { // save settings // state of splitter Settings->setValue("Splitter", ui.splitter->saveState()); Settings->setValue("GroupChatSplitter", ui.splitter_2->saveState()); } ui.friendList->processSettings(bLoad); Settings->endGroup(); } void FriendsDialog::showEvent(QShowEvent *event) { static bool first = true; if (first) { // Workaround: now the scroll position is correct calculated first = false; QScrollBar *scrollbar = ui.msgText->verticalScrollBar(); scrollbar->setValue(scrollbar->maximum()); } RsAutoUpdatePage::showEvent(event); } void FriendsDialog::pasteLink() { ui.lineEdit->insertHtml(RSLinkClipboard::toHtml()) ; } void FriendsDialog::contextMenu(QPoint /*point*/) { QMenu *contextMnu = ui.lineEdit->createStandardContextMenu(); contextMnu->addSeparator(); QAction *action = contextMnu->addAction(QIcon(":/images/pasterslink.png"), tr("Paste RetroShare Link"), this, SLOT(pasteLink())); action->setDisabled(RSLinkClipboard::empty()); contextMnu->exec(QCursor::pos()); delete(contextMnu); } void FriendsDialog::updateDisplay() { } void FriendsDialog::addFriend() { std::string groupId = ui.friendList->getSelectedGroupId(); ConnectFriendWizard connwiz (this); if (groupId.empty() == false) { connwiz.setGroup(groupId); } connwiz.exec (); } void FriendsDialog::resetStatusBar() { #ifdef FRIENDS_DEBUG std::cerr << "FriendsDialog: reseting status bar." << std::endl ; #endif ui.statusStringLabel->setText(QString("")) ; } void FriendsDialog::updateStatusTyping() { if(time(NULL) - last_status_send_time > 5) // limit 'peer is typing' packets to at most every 10 sec { #ifdef FRIENDS_DEBUG std::cerr << "FriendsDialog: sending group chat typing info." << std::endl ; #endif #ifdef ONLY_FOR_LINGUIST tr("is typing..."); #endif rsMsgs->sendGroupChatStatusString("is typing..."); last_status_send_time = time(NULL) ; } } // Called by libretroshare through notifyQt to display the peer's status // void FriendsDialog::updateStatusString(const QString& peer_id, const QString& status_string) { #ifdef FRIENDS_DEBUG std::cerr << "FriendsDialog: received group chat typing info. updating gui." << std::endl ; #endif QString status = QString::fromUtf8(rsPeers->getPeerName(peer_id.toStdString()).c_str()) + " " + tr(status_string.toAscii()); ui.statusStringLabel->setText(status) ; // displays info for 5 secs. QTimer::singleShot(5000,this,SLOT(resetStatusBar())) ; } void FriendsDialog::updatePeerStatusString(const QString& peer_id,const QString& status_string,bool is_private_chat) { if (!is_private_chat) { #ifdef FRIENDS_DEBUG std::cerr << "Updating public chat msg from peer " << rsPeers->getPeerName(peer_id.toStdString()) << ": " << status_string.toStdString() << std::endl ; #endif updateStatusString(peer_id, status_string); } } void FriendsDialog::publicChatChanged(int type) { if (type == NOTIFY_TYPE_ADD) { insertChat(); } } void FriendsDialog::addChatMsg(bool incoming, bool history, const QString &name, const QDateTime &sendTime, const QDateTime &recvTime, const QString &message) { unsigned int formatFlag = CHAT_FORMATMSG_EMBED_LINKS; // embed smileys ? if (Settings->valueFromGroup("Chat", "Emoteicons_GroupChat", true).toBool()) { formatFlag |= CHAT_FORMATMSG_EMBED_SMILEYS; } ChatStyle::enumFormatMessage type; if (incoming) { if (history) { type = ChatStyle::FORMATMSG_HINCOMING; } else { type = ChatStyle::FORMATMSG_INCOMING; } } else { if (history) { type = ChatStyle::FORMATMSG_HOUTGOING; } else { type = ChatStyle::FORMATMSG_OUTGOING; } } // Remove

's from older RetroShare versions before 31.01.2012 (can be removed later) QString optimizedMessage = message; RsHtml::optimizeHtml(optimizedMessage); QString formatMsg = style.formatMessage(type, name, incoming ? recvTime : sendTime, optimizedMessage, formatFlag); ui.msgText->append(formatMsg); } void FriendsDialog::insertChat() { std::list newchat; if (!rsMsgs->getPublicChatQueue(newchat)) { #ifdef FRIENDS_DEBUG std::cerr << "no chat available." << std::endl ; #endif return; } #ifdef FRIENDS_DEBUG std::cerr << "got new chat." << std::endl; #endif std::list::iterator it; /* add in lines at the bottom */ for(it = newchat.begin(); it != newchat.end(); it++) { /* are they private? */ if (it->chatflags & RS_CHAT_PRIVATE) { /* this should not happen */ continue; } QDateTime sendTime = QDateTime::fromTime_t(it->sendTime); QDateTime recvTime = QDateTime::fromTime_t(it->recvTime); QString name = QString::fromUtf8(rsPeers->getPeerName(it->rsid).c_str()); QString msg = QString::fromStdWString(it->msg); #ifdef FRIENDS_DEBUG std::cerr << "FriendsDialog::insertChat(): " << msg.toStdString() << std::endl; #endif bool incoming = false; // notify with a systray icon msg if(it->rsid != rsPeers->getOwnId()) { incoming = true; // This is a trick to translate HTML into text. QTextEdit editor; editor.setHtml(msg); QString notifyMsg = name + ": " + editor.toPlainText(); if(notifyMsg.length() > 30) emit notifyGroupChat(tr("New group chat"), notifyMsg.left(30) + QString("...")); else emit notifyGroupChat(tr("New group chat"), notifyMsg); } addChatMsg(incoming, false, name, sendTime, recvTime, msg); } } bool FriendsDialog::eventFilter(QObject *obj, QEvent *event) { if (obj == ui.lineEdit) { if (event->type() == QEvent::KeyPress) { updateStatusTyping() ; QKeyEvent *keyEvent = static_cast(event); if (keyEvent && (keyEvent->key() == Qt::Key_Enter || keyEvent->key() == Qt::Key_Return)) { // Enter pressed if (Settings->getChatSendMessageWithCtrlReturn()) { if (keyEvent->modifiers() & Qt::ControlModifier) { // send message with Ctrl+Enter sendMsg(); return true; // eat event } } else { if (keyEvent->modifiers() & Qt::ControlModifier) { // insert return ui.lineEdit->textCursor().insertText("\n"); } else { // send message with Enter sendMsg(); } return true; // eat event } } } } // pass the event on to the parent class return RsAutoUpdatePage::eventFilter(obj, event); } void FriendsDialog::sendMsg() { QTextEdit *lineWidget = ui.lineEdit; if (lineWidget->toPlainText().isEmpty()) { // nothing to send return; } QString text; RsHtml::optimizeHtml(lineWidget, text); std::wstring message = text.toStdWString(); #ifdef FRIENDS_DEBUG std::string msg(message.begin(), message.end()); std::cerr << "FriendsDialog::sendMsg(): " << msg << std::endl; #endif rsMsgs->sendPublicChat(message); ui.lineEdit->clear(); // workaround for Qt bug - http://bugreports.qt.nokia.com/browse/QTBUG-2533 // QTextEdit::clear() does not reset the CharFormat if document contains hyperlinks that have been accessed. ui.lineEdit->setCurrentCharFormat(QTextCharFormat ()); setColorAndFont(); /* redraw send list */ insertSendList(); } void FriendsDialog::insertSendList() { #ifdef false std::list peers; std::list::iterator it; if (!rsPeers) { /* not ready yet! */ return; } rsPeers->getOnlineList(peers); /* get a link to the table */ //QTreeWidget *sendWidget = ui.msgSendList; QList items; for(it = peers.begin(); it != peers.end(); it++) { RsPeerDetails details; if (!rsPeers->getPeerDetails(*it, details)) { continue; /* BAD */ } /* make a widget per friend */ QTreeWidgetItem *item = new QTreeWidgetItem((QTreeWidget*)0); /* add all the labels */ /* (0) Person */ item -> setText(0, QString::fromStdString(details.name)); item -> setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled); //item -> setFlags(Qt::ItemIsUserCheckable); item -> setCheckState(0, Qt::Checked); if (rsicontrol->IsInChat(*it)) { item -> setCheckState(0, Qt::Checked); } else { item -> setCheckState(0, Qt::Unchecked); } /* disable for the moment */ item -> setFlags(Qt::ItemIsUserCheckable); item -> setCheckState(0, Qt::Checked); /* add to the list */ items.append(item); } /* remove old items */ //sendWidget->clear(); //sendWidget->setColumnCount(1); /* add the items in! */ //sendWidget->insertTopLevelItems(0, items); //sendWidget->update(); /* update display */ #endif } /* to toggle the state */ //void FriendsDialog::toggleSendItem( QTreeWidgetItem *item, int col ) //{ //#ifdef FRIENDS_DEBUG // std::cerr << "ToggleSendItem()" << std::endl; //#endif // // /* extract id */ // std::string id = (item -> text(4)).toStdString(); // // /* get state */ // bool inChat = (Qt::Checked == item -> checkState(0)); /* alway column 0 */ // // /* call control fns */ // // rsicontrol -> SetInChat(id, inChat); // return; //} //============================================================================ void FriendsDialog::chooseColor() { bool ok; QRgb color = QColorDialog::getRgba(ui.lineEdit->textColor().rgba(), &ok, this); if (ok) { mCurrentColor = QColor(color); colorChanged(); setColorAndFont(); } } void FriendsDialog::colorChanged() { QPixmap pxm(16,16); pxm.fill(mCurrentColor); ui.colorChatButton->setIcon(pxm); } void FriendsDialog::chooseFont() { bool ok; QFont font = QFontDialog::getFont(&ok, mCurrentFont, this); if (ok) { mCurrentFont = font; fontChanged(); setFont(); } } void FriendsDialog::fontChanged() { ui.textboldChatButton->setChecked(mCurrentFont.bold()); ui.textunderlineChatButton->setChecked(mCurrentFont.underline()); ui.textitalicChatButton->setChecked(mCurrentFont.italic()); } void FriendsDialog::setColorAndFont() { mCurrentFont.setBold(ui.textboldChatButton->isChecked()); mCurrentFont.setUnderline(ui.textunderlineChatButton->isChecked()); mCurrentFont.setItalic(ui.textitalicChatButton->isChecked()); ui.lineEdit->setFont(mCurrentFont); ui.lineEdit->setTextColor(mCurrentColor); ui.lineEdit->setFocus(); } void FriendsDialog::setFont() { setColorAndFont(); Settings->setChatScreenFont(mCurrentFont.toString()); } // Update Chat Info information void FriendsDialog::setChatInfo(QString info, QColor color) { static unsigned int nbLines = 0; ++nbLines; // Check log size, clear it if too big if(nbLines > 200) { ui.msgText->clear(); nbLines = 1; } ui.msgText->append(QString::fromUtf8("")+ QTime::currentTime().toString(QString::fromUtf8("hh:mm:ss")) + QString::fromUtf8(" - ") + info + QString::fromUtf8("")); } void FriendsDialog::on_actionClear_Chat_History_triggered() { ui.msgText->clear(); } void FriendsDialog::on_actionDelete_Chat_History_triggered() { if ((QMessageBox::question(this, "RetroShare", tr("Do you really want to physically delete the history?"), QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes)) == QMessageBox::Yes) { on_actionClear_Chat_History_triggered(); rsHistory->clear(""); } } void FriendsDialog::smileyWidgetgroupchat() { Emoticons::showSmileyWidget(this, ui.emoticonBtn, SLOT(addSmileys()), true); } void FriendsDialog::addSmileys() { ui.lineEdit->textCursor().insertText(qobject_cast(sender())->toolTip().split("|").first()); } void FriendsDialog::getAvatar() { QByteArray ba; if (misc::getOpenAvatarPicture(this, ba)) { #ifdef FRIENDS_DEBUG std::cerr << "Avatar image size = " << ba.size() << std::endl ; #endif rsMsgs->setOwnAvatarData((unsigned char *)(ba.data()), ba.size()) ; // last char 0 included. } } void FriendsDialog::on_actionCreate_New_Forum_activated() { MainWindow::activatePage (MainWindow::Forums); CreateForum cf (this); cf.exec(); } void FriendsDialog::on_actionCreate_New_Channel_activated() { #ifndef RS_RELEASE_VERSION MainWindow::activatePage (MainWindow::Channels); CreateChannel cf (this); cf.exec(); #endif } /** Loads own personal status */ void FriendsDialog::loadmypersonalstatus() { ui.mypersonalstatuslabel->setText(QString::fromUtf8(rsMsgs->getCustomStateString().c_str())); } void FriendsDialog::statusmessage() { StatusMessage statusmsgdialog (this); statusmsgdialog.exec(); } void FriendsDialog::addExtraFile() { QStringList files; if (misc::getOpenFileNames(this, RshareSettings::LASTDIR_EXTRAFILE, tr("Add Extra File"), "", files)) { ui.hashBox->addAttachments(files); } } void FriendsDialog::fileHashingFinished(QList hashedFiles) { std::cerr << "FriendsDialog::fileHashingFinished() started." << std::endl; QString mesgString; QList::iterator it; for (it = hashedFiles.begin(); it != hashedFiles.end(); ++it) { HashedFile& hashedFile = *it; RetroShareLink link; if (!link.createFile(hashedFile.filename, hashedFile.size, QString::fromStdString(hashedFile.hash))) { continue; } mesgString += link.toHtmlSize(); if (it!= hashedFiles.end()) { mesgString += "
"; } } #ifdef FRIENDS_DEBUG std::cerr << "FriendsDialog::fileHashingFinished mesgString : " << mesgString.toStdString() << std::endl; #endif if (!mesgString.isEmpty()) { /* convert to real html document */ QTextBrowser textBrowser; textBrowser.setHtml(mesgString); std::wstring msg = textBrowser.toHtml().toStdWString(); rsMsgs->sendPublicChat(msg); setFont(); } } bool FriendsDialog::fileSave() { if (fileName.isEmpty()) return fileSaveAs(); QFile file(fileName); if (!file.open(QFile::WriteOnly)) return false; QTextStream ts(&file); ts.setCodec(QTextCodec::codecForName("UTF-8")); ts << ui.msgText->document()->toPlainText(); ui.msgText->document()->setModified(false); return true; } bool FriendsDialog::fileSaveAs() { QString fn; if (misc::getSaveFileName(this, RshareSettings::LASTDIR_HISTORY, tr("Save as..."), tr("Text File (*.txt );;All Files (*)"), fn)) { setCurrentFileName(fn); return fileSave(); } return false; } void FriendsDialog::setCurrentFileName(const QString &fileName) { this->fileName = fileName; ui.msgText->document()->setModified(false); setWindowModified(false); } void FriendsDialog::on_actionMessageHistory_triggered() { ImHistoryBrowser imBrowser("", ui.lineEdit, this); imBrowser.exec(); } void FriendsDialog::on_actionAdd_Group_activated() { CreateGroup createGrpDialog ("", this); createGrpDialog.exec(); } void FriendsDialog::newsFeedChanged(int count) { int newsFeedTabIndex = ui.tabWidget->indexOf(newsFeed); if (newsFeedTabIndex < 0) { return; } if (count) { ui.tabWidget->tabBar()->setTabText(newsFeedTabIndex, QString("%1 (%2)").arg(newsFeedText).arg(count)); ui.tabWidget->tabBar()->setTabTextColor(newsFeedTabIndex, Qt::blue); ui.tabWidget->tabBar()->setTabIcon(newsFeedTabIndex, QIcon(IMAGE_NEWSFEED_NEW)); } else { ui.tabWidget->tabBar()->setTabText(newsFeedTabIndex, newsFeedText); ui.tabWidget->tabBar()->setTabTextColor(newsFeedTabIndex, newsFeedTabColor); ui.tabWidget->tabBar()->setTabIcon(newsFeedTabIndex, QIcon(IMAGE_NEWSFEED)); } } void FriendsDialog::createChatLobby() { std::list friends; CreateLobbyDialog(friends).exec(); } void FriendsDialog::recommendFriends() { FriendRecommendDialog::showYourself(); } /*static*/ ChatTabWidget *FriendsDialog::getTabWidget() { return instance ? instance->ui.tabWidget : NULL; }