From 22fcc3f7a0b854fa9c14090e7b529134046dcfeb Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 16 Jun 2015 14:25:19 -0400 Subject: [PATCH 01/24] Multi-window capabilities for all chats and menus --- qtox.pro | 4 +- src/mainwindow.ui | 551 +------------------- src/persistence/settings.cpp | 14 + src/persistence/settings.h | 4 + src/widget/contentlayout.cpp | 124 +++++ src/widget/contentlayout.h | 44 ++ src/widget/form/addfriendform.cpp | 19 +- src/widget/form/addfriendform.h | 5 +- src/widget/form/chatform.cpp | 4 +- src/widget/form/chatform.h | 3 +- src/widget/form/filesform.cpp | 18 +- src/widget/form/filesform.h | 5 +- src/widget/form/genericchatform.cpp | 10 +- src/widget/form/genericchatform.h | 7 +- src/widget/form/profileform.cpp | 8 +- src/widget/form/profileform.h | 4 +- src/widget/form/settings/generalform.cpp | 8 + src/widget/form/settings/generalform.h | 1 + src/widget/form/settings/generalsettings.ui | 7 + src/widget/form/settingswidget.cpp | 21 +- src/widget/form/settingswidget.h | 7 +- src/widget/friendwidget.cpp | 10 +- src/widget/friendwidget.h | 3 +- src/widget/genericchatroomwidget.h | 8 +- src/widget/groupwidget.cpp | 10 +- src/widget/groupwidget.h | 3 +- src/widget/widget.cpp | 205 ++++++-- src/widget/widget.h | 7 +- 28 files changed, 458 insertions(+), 656 deletions(-) create mode 100644 src/widget/contentlayout.cpp create mode 100644 src/widget/contentlayout.h diff --git a/qtox.pro b/qtox.pro index 3eec7904d..7b895882a 100644 --- a/qtox.pro +++ b/qtox.pro @@ -495,7 +495,8 @@ SOURCES += \ src/widget/friendlistlayout.cpp \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ - src/widget/tool/removefrienddialog.cpp + src/widget/tool/removefrienddialog.cpp \ + src/widget/contentlayout.cpp HEADERS += \ src/audio/audio.h \ @@ -539,4 +540,5 @@ HEADERS += \ src/widget/friendlistlayout.h \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ + src/widget/contentlayout.h \ src/widget/tool/removefrienddialog.h diff --git a/src/mainwindow.ui b/src/mainwindow.ui index b419431c6..32e6dc777 100644 --- a/src/mainwindow.ui +++ b/src/mainwindow.ui @@ -7,7 +7,7 @@ 0 0 775 - 537 + 420 @@ -1288,555 +1288,6 @@ QSplitter:handle{ - - - - 0 - 0 - - - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 127 - 127 - 127 - - - - - - - 170 - 170 - 170 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 220 - - - - - - - 0 - 0 - 0 - - - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 127 - 127 - 127 - - - - - - - 170 - 170 - 170 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 220 - - - - - - - 0 - 0 - 0 - - - - - - - - - 127 - 127 - 127 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 127 - 127 - 127 - - - - - - - 170 - 170 - 170 - - - - - - - 127 - 127 - 127 - - - - - - - 255 - 255 - 255 - - - - - - - 127 - 127 - 127 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 255 - - - - - - - 0 - 0 - 0 - - - - - - - 255 - 255 - 255 - - - - - - - 255 - 255 - 220 - - - - - - - 0 - 0 - 0 - - - - - - - - true - - - - 0 - - - 0 - - - 0 - - - 0 - - - 0 - - - - - - 0 - 0 - - - - - 0 - 0 - - - - - 16777215 - 16777215 - - - - - 0 - 57 - - - - true - - - - - - - - 0 - 1 - - - - - 16777215 - 1 - - - - - - - - - 193 - 193 - 193 - - - - - - - - - 193 - 193 - 193 - - - - - - - - - 127 - 127 - 127 - - - - - - - - QFrame::HLine - - - QFrame::Plain - - - - - - - - 0 - 0 - - - - - 375 - 0 - - - - - - diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index d480d92d4..81651b17b 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -174,6 +174,7 @@ void Settings::loadGlobal() globalAutoAcceptDir = s.value("globalAutoAcceptDir", QStandardPaths::locate(QStandardPaths::HomeLocation, QString(), QStandardPaths::LocateDirectory) ).toString(); + separateWindow = s.value("separateWindow", false).toBool(); groupchatPosition = s.value("groupchatPosition", true).toBool(); s.endGroup(); @@ -375,6 +376,7 @@ void Settings::saveGlobal() s.setValue("notifySound", notifySound); s.setValue("groupAlwaysNotify", groupAlwaysNotify); s.setValue("fauxOfflineMessaging", fauxOfflineMessaging); + s.setValue("separateWindow", separateWindow); s.setValue("groupchatPosition", groupchatPosition); s.setValue("autoSaveEnabled", autoSaveEnabled); s.setValue("globalAutoAcceptDir", globalAutoAcceptDir); @@ -1324,6 +1326,18 @@ void Settings::setCompactLayout(bool value) compactLayout = value; } +bool Settings::getSeparateWindow() const +{ + QMutexLocker locker{&bigLock}; + return separateWindow; +} + +void Settings::setSeparateWindow(bool value) +{ + QMutexLocker locker{&bigLock}; + separateWindow = value; +} + bool Settings::getGroupchatPosition() const { QMutexLocker locker{&bigLock}; diff --git a/src/persistence/settings.h b/src/persistence/settings.h index cd9ca5715..88e6f18e0 100644 --- a/src/persistence/settings.h +++ b/src/persistence/settings.h @@ -237,6 +237,9 @@ public: bool getCompactLayout() const; void setCompactLayout(bool compact); + bool getSeparateWindow() const; + void setSeparateWindow(bool value); + bool getGroupchatPosition() const; void setGroupchatPosition(bool value); @@ -295,6 +298,7 @@ private: bool fauxOfflineMessaging; bool compactLayout; bool groupchatPosition; + bool separateWindow; bool enableIPv6; QString translation; bool makeToxPortable; diff --git a/src/widget/contentlayout.cpp b/src/widget/contentlayout.cpp new file mode 100644 index 000000000..9012897b7 --- /dev/null +++ b/src/widget/contentlayout.cpp @@ -0,0 +1,124 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + +#include "contentlayout.h" +#include "src/persistence/settings.h" +#include "style.h" +#include +#include + +ContentLayout::ContentLayout() + : QVBoxLayout() +{ + init(); +} + +ContentLayout::ContentLayout(QWidget *parent) + : QVBoxLayout(parent) +{ + init(); + + QPalette palette = parent->palette(); + palette.setBrush(QPalette::WindowText, QColor(0, 0, 0)); + palette.setBrush(QPalette::Button, QColor(255, 255, 255)); + palette.setBrush(QPalette::Light, QColor(255, 255, 255)); + palette.setBrush(QPalette::Midlight, QColor(255, 255, 255)); + palette.setBrush(QPalette::Dark, QColor(127, 127, 127)); + palette.setBrush(QPalette::Mid, QColor(170, 170, 170)); + palette.setBrush(QPalette::Text, QColor(0, 0, 0)); + palette.setBrush(QPalette::BrightText, QColor(255, 255, 255)); + palette.setBrush(QPalette::ButtonText, QColor(0, 0, 0)); + palette.setBrush(QPalette::Base, QColor(255, 255, 255)); + palette.setBrush(QPalette::Window, QColor(255, 255, 255)); + palette.setBrush(QPalette::Shadow, QColor(0, 0, 0)); + palette.setBrush(QPalette::AlternateBase, QColor(255, 255, 255)); + palette.setBrush(QPalette::ToolTipBase, QColor(255, 255, 220)); + palette.setBrush(QPalette::ToolTipText, QColor(0, 0, 0)); + + palette.setBrush(QPalette::Disabled, QPalette::WindowText, QColor(127, 127, 127)); + palette.setBrush(QPalette::Disabled, QPalette::Text, QColor(127, 127, 127)); + palette.setBrush(QPalette::Disabled, QPalette::ButtonText, QColor(127, 127, 127)); + + parent->setPalette(palette); +} + +ContentLayout::~ContentLayout() +{ + clear(); + + mainHead->deleteLater(); + mainContent->deleteLater(); +} + +void ContentLayout::clear() +{ + QLayoutItem* item; + while ((item = mainHead->layout()->takeAt(0)) != 0) + { + item->widget()->hide(); + item->widget()->setParent(nullptr); + } + + while ((item = mainContent->layout()->takeAt(0)) != 0) + { + item->widget()->hide(); + item->widget()->setParent(nullptr); + } +} + +void ContentLayout::init() +{ + setMargin(0); + setSpacing(0); + + mainHead = new QWidget(); + mainHead->setLayout(new QVBoxLayout); + mainHead->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Fixed); + mainHead->layout()->setMargin(0); + mainHead->layout()->setSpacing(0); + mainHead->setMouseTracking(true); + + mainHLine = new QFrame(); + mainHLine->setFrameShape(QFrame::HLine); + mainHLine->setFrameShadow(QFrame::Plain); + QPalette palette = mainHLine->palette(); + palette.setBrush(QPalette::WindowText, QBrush(QColor(193, 193, 193))); + palette.setBrush(QPalette::WindowText, QBrush(QColor(193, 193, 193))); + mainHLine->setPalette(palette); + + mainContent = new QWidget(); + mainContent->setLayout(new QVBoxLayout); + mainContent->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); + + if (QStyleFactory::keys().contains(Settings::getInstance().getStyle()) + && Settings::getInstance().getStyle() != "None") + { + mainHead->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); + mainContent->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); + } + +#ifndef Q_OS_MAC + mainHead->setStyleSheet(Style::getStylesheet(":ui/settings/mainHead.css")); + mainContent->setStyleSheet(Style::getStylesheet(":ui/settings/mainContent.css")); +#endif + + addWidget(mainHead); + addWidget(mainHLine); + addWidget(mainContent); +} diff --git a/src/widget/contentlayout.h b/src/widget/contentlayout.h new file mode 100644 index 000000000..6955ca63e --- /dev/null +++ b/src/widget/contentlayout.h @@ -0,0 +1,44 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + +#ifndef CONTENTLAYOUT_H +#define CONTENTLAYOUT_H + +#include + +class QFrame; + +class ContentLayout : public QVBoxLayout +{ +public: + ContentLayout(); + ContentLayout(QWidget* parent); + ~ContentLayout(); + + void clear(); + + QWidget* mainContent; + QFrame* mainHLine; + QWidget* mainHead; + +private: + void init(); +}; + +#endif // CONTENTLAYOUT_H diff --git a/src/widget/form/addfriendform.cpp b/src/widget/form/addfriendform.cpp index 347903813..d46581202 100644 --- a/src/widget/form/addfriendform.cpp +++ b/src/widget/form/addfriendform.cpp @@ -22,9 +22,9 @@ #include #include #include +#include #include #include -#include "ui_mainwindow.h" #include "src/nexus.h" #include "src/core/core.h" #include "src/core/cdata.h" @@ -32,6 +32,7 @@ #include "src/persistence/settings.h" #include "src/widget/gui.h" #include "src/widget/translator.h" +#include "src/widget/contentlayout.h" AddFriendForm::AddFriendForm() { @@ -66,10 +67,20 @@ AddFriendForm::~AddFriendForm() main->deleteLater(); } -void AddFriendForm::show(Ui::MainWindow &ui) +bool AddFriendForm::isShown() const { - ui.mainContent->layout()->addWidget(main); - ui.mainHead->layout()->addWidget(head); + if (main->isVisible()) + { + return true; + } + + return false; +} + +void AddFriendForm::show(ContentLayout* contentLayout) +{ + contentLayout->mainContent->layout()->addWidget(main); + contentLayout->mainHead->layout()->addWidget(head); main->show(); head->show(); setIdFromClipboard(); diff --git a/src/widget/form/addfriendform.h b/src/widget/form/addfriendform.h index fb1441979..b45d408af 100644 --- a/src/widget/form/addfriendform.h +++ b/src/widget/form/addfriendform.h @@ -26,7 +26,7 @@ #include #include -namespace Ui {class MainWindow;} +class ContentLayout; class AddFriendForm : public QObject { @@ -37,7 +37,8 @@ public: AddFriendForm& operator=(const AddFriendForm&) = delete; ~AddFriendForm(); - void show(Ui::MainWindow &ui); + bool isShown() const; + void show(ContentLayout* contentLayout); QString getMessage() const; signals: diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index bc43b1674..b20e7480c 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -1009,9 +1009,9 @@ void ChatForm::setFriendTyping(bool isTyping) text->setText("
" + QString("%1 is typing").arg(f->getDisplayedName()) + "
"); } -void ChatForm::show(Ui::MainWindow &ui) +void ChatForm::show(ContentLayout* contentLayout) { - GenericChatForm::show(ui); + GenericChatForm::show(contentLayout); if (callConfirm) callConfirm->show(); diff --git a/src/widget/form/chatform.h b/src/widget/form/chatform.h index 2aaf72f04..e2bfd2c49 100644 --- a/src/widget/form/chatform.h +++ b/src/widget/form/chatform.h @@ -27,7 +27,6 @@ #include #include - class Friend; class FileTransferInstance; class NetCamView; @@ -50,7 +49,7 @@ public: void setFriendTyping(bool isTyping); OfflineMsgEngine* getOfflineMsgEngine(); - virtual void show(Ui::MainWindow &ui) final override; + virtual void show(ContentLayout* contentLayout) final override; signals: void sendFile(uint32_t friendId, QString, QString, long long); diff --git a/src/widget/form/filesform.cpp b/src/widget/form/filesform.cpp index 623b330e9..f6a6ea9fb 100644 --- a/src/widget/form/filesform.cpp +++ b/src/widget/form/filesform.cpp @@ -18,9 +18,9 @@ */ #include "filesform.h" -#include "ui_mainwindow.h" #include "src/widget/widget.h" #include "src/widget/translator.h" +#include "src/widget/contentlayout.h" #include FilesForm::FilesForm() @@ -54,10 +54,20 @@ FilesForm::~FilesForm() head->deleteLater(); } -void FilesForm::show(Ui::MainWindow& ui) +bool FilesForm::isShown() const { - ui.mainContent->layout()->addWidget(&main); - ui.mainHead->layout()->addWidget(head); + if (main.isVisible()) + { + return true; + } + + return false; +} + +void FilesForm::show(ContentLayout* contentLayout) +{ + contentLayout->mainContent->layout()->addWidget(&main); + contentLayout->mainHead->layout()->addWidget(head); main.show(); head->show(); } diff --git a/src/widget/form/filesform.h b/src/widget/form/filesform.h index cc523def2..bb18dc41b 100644 --- a/src/widget/form/filesform.h +++ b/src/widget/form/filesform.h @@ -26,7 +26,7 @@ #include #include -namespace Ui {class MainWindow;} +class ContentLayout; class QListWidget; class FilesForm : public QObject @@ -37,7 +37,8 @@ public: FilesForm(); ~FilesForm(); - void show(Ui::MainWindow &ui); + bool isShown() const; + void show(ContentLayout* contentLayout); public slots: void onFileDownloadComplete(const QString& path); diff --git a/src/widget/form/genericchatform.cpp b/src/widget/form/genericchatform.cpp index 4b2a4f47e..21c9f0c51 100644 --- a/src/widget/form/genericchatform.cpp +++ b/src/widget/form/genericchatform.cpp @@ -18,7 +18,6 @@ */ #include "genericchatform.h" -#include "ui_mainwindow.h" #include #include @@ -42,6 +41,9 @@ #include "src/chatlog/content/timestamp.h" #include "src/widget/tool/flyoutoverlaywidget.h" #include "src/widget/translator.h" +#include "src/widget/contentlayout.h" +#include "src/widget/tool/croppinglabel.h" +#include GenericChatForm::GenericChatForm(QWidget *parent) : QWidget(parent) @@ -255,10 +257,10 @@ void GenericChatForm::setName(const QString &newName) nameLabel->setToolTip(newName); // for overlength names } -void GenericChatForm::show(Ui::MainWindow &ui) +void GenericChatForm::show(ContentLayout* contentLayout) { - ui.mainContent->layout()->addWidget(this); - ui.mainHead->layout()->addWidget(headWidget); + contentLayout->mainContent->layout()->addWidget(this); + contentLayout->mainHead->layout()->addWidget(headWidget); headWidget->show(); QWidget::show(); } diff --git a/src/widget/form/genericchatform.h b/src/widget/form/genericchatform.h index 573d6757a..237afdc03 100644 --- a/src/widget/form/genericchatform.h +++ b/src/widget/form/genericchatform.h @@ -40,10 +40,7 @@ class ChatLog; class MaskablePixmapWidget; class Widget; class FlyoutOverlayWidget; - -namespace Ui { - class MainWindow; -} +class ContentLayout; class GenericChatForm : public QWidget { @@ -54,7 +51,7 @@ public: void setName(const QString &newName); virtual void show() final{} - virtual void show(Ui::MainWindow &ui); + virtual void show(ContentLayout* contentLayout); ChatMessage::Ptr addMessage(const ToxId& author, const QString &message, bool isAction, const QDateTime &datetime, bool isSent); ChatMessage::Ptr addSelfMessage(const QString &message, bool isAction, const QDateTime &datetime, bool isSent); diff --git a/src/widget/form/profileform.cpp b/src/widget/form/profileform.cpp index 051a1b8e7..ef8a7fa96 100644 --- a/src/widget/form/profileform.cpp +++ b/src/widget/form/profileform.cpp @@ -21,11 +21,11 @@ #include "src/nexus.h" #include "ui_profileform.h" #include "profileform.h" -#include "ui_mainwindow.h" #include "src/widget/form/settingswidget.h" #include "src/widget/maskablepixmapwidget.h" #include "src/widget/form/setpassworddialog.h" #include "src/persistence/settings.h" +#include "src/widget/contentlayout.h" #include "src/widget/tool/croppinglabel.h" #include "src/widget/widget.h" #include "src/widget/gui.h" @@ -134,10 +134,10 @@ ProfileForm::~ProfileForm() head->deleteLater(); } -void ProfileForm::show(Ui::MainWindow &ui) +void ProfileForm::show(ContentLayout* contentLayout) { - ui.mainHead->layout()->addWidget(head); - ui.mainContent->layout()->addWidget(this); + contentLayout->mainHead->layout()->addWidget(head); + contentLayout->mainContent->layout()->addWidget(this); head->show(); QWidget::show(); prFileLabelUpdate(); diff --git a/src/widget/form/profileform.h b/src/widget/form/profileform.h index db3c87068..2ef32429b 100644 --- a/src/widget/form/profileform.h +++ b/src/widget/form/profileform.h @@ -30,10 +30,10 @@ class CroppingLabel; class Core; class MaskablePixmapWidget; +class ContentLayout; namespace Ui { class IdentitySettings; -class MainWindow; } class ClickableTE : public QLineEdit @@ -54,7 +54,7 @@ public: ProfileForm(QWidget *parent = nullptr); ~ProfileForm(); virtual void show() final{} - void show(Ui::MainWindow &ui); + void show(ContentLayout* contentLayout); signals: void userNameChanged(QString); diff --git a/src/widget/form/settings/generalform.cpp b/src/widget/form/settings/generalform.cpp index 27dcf2b2e..d919f8555 100644 --- a/src/widget/form/settings/generalform.cpp +++ b/src/widget/form/settings/generalform.cpp @@ -87,6 +87,7 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : bodyUI->groupAlwaysNotify->setChecked(Settings::getInstance().getGroupAlwaysNotify()); bodyUI->cbFauxOfflineMessaging->setChecked(Settings::getInstance().getFauxOfflineMessaging()); bodyUI->cbCompactLayout->setChecked(Settings::getInstance().getCompactLayout()); + bodyUI->cbSeparateWindow->setChecked(Settings::getInstance().getSeparateWindow()); bodyUI->cbGroupchatPosition->setChecked(Settings::getInstance().getGroupchatPosition()); for (auto entry : SmileyPack::listSmileyPacks()) @@ -179,6 +180,7 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : connect(bodyUI->reconnectButton, &QPushButton::clicked, this, &GeneralForm::onReconnectClicked); connect(bodyUI->cbFauxOfflineMessaging, &QCheckBox::stateChanged, this, &GeneralForm::onFauxOfflineMessaging); connect(bodyUI->cbCompactLayout, &QCheckBox::stateChanged, this, &GeneralForm::onCompactLayout); + connect(bodyUI->cbSeparateWindow, &QCheckBox::stateChanged, this, &GeneralForm::onSeparateWindowChanged); connect(bodyUI->cbGroupchatPosition, &QCheckBox::stateChanged, this, &GeneralForm::onGroupchatPositionChanged); // prevent stealing mouse whell scroll @@ -430,6 +432,12 @@ void GeneralForm::onCompactLayout() emit parent->compactToggled(bodyUI->cbCompactLayout->isChecked()); } +void GeneralForm::onSeparateWindowChanged() +{ + Settings::getInstance().setSeparateWindow(bodyUI->cbSeparateWindow->isChecked()); + emit parent->separateWindowToggled(bodyUI->cbSeparateWindow->isChecked()); +} + void GeneralForm::onGroupchatPositionChanged() { Settings::getInstance().setGroupchatPosition(bodyUI->cbGroupchatPosition->isChecked()); diff --git a/src/widget/form/settings/generalform.h b/src/widget/form/settings/generalform.h index 4f53c70c1..31d24b8de 100644 --- a/src/widget/form/settings/generalform.h +++ b/src/widget/form/settings/generalform.h @@ -67,6 +67,7 @@ private slots: void onSetGroupAlwaysNotify(); void onFauxOfflineMessaging(); void onCompactLayout(); + void onSeparateWindowChanged(); void onGroupchatPositionChanged(); void onThemeColorChanged(int); diff --git a/src/widget/form/settings/generalsettings.ui b/src/widget/form/settings/generalsettings.ui index 81884fe90..54446a19b 100644 --- a/src/widget/form/settings/generalsettings.ui +++ b/src/widget/form/settings/generalsettings.ui @@ -374,6 +374,13 @@ instead of system taskbar.
+ + + + Use separate windows for friend list. + + + diff --git a/src/widget/form/settingswidget.cpp b/src/widget/form/settingswidget.cpp index ea9ce8b45..64ff40b74 100644 --- a/src/widget/form/settingswidget.cpp +++ b/src/widget/form/settingswidget.cpp @@ -19,7 +19,6 @@ #include "settingswidget.h" #include "src/widget/widget.h" -#include "ui_mainwindow.h" #include "src/video/camerasource.h" #include "src/widget/form/settings/generalform.h" #include "src/widget/form/settings/privacyform.h" @@ -27,12 +26,14 @@ #include "src/widget/form/settings/advancedform.h" #include "src/widget/form/settings/aboutform.h" #include "src/widget/translator.h" +#include "src/widget/contentlayout.h" #include +#include SettingsWidget::SettingsWidget(QWidget* parent) : QWidget(parent) { - body = new QWidget(this); + body = new QWidget(); QVBoxLayout* bodyLayout = new QVBoxLayout(); body->setLayout(bodyLayout); @@ -86,10 +87,20 @@ void SettingsWidget::showAbout() onTabChanged(settingsWidgets->count() - 1); } -void SettingsWidget::show(Ui::MainWindow& ui) +bool SettingsWidget::isShown() const { - ui.mainContent->layout()->addWidget(body); - ui.mainHead->layout()->addWidget(head); + if (body->isVisible()) + { + return true; + } + + return false; +} + +void SettingsWidget::show(ContentLayout* contentLayout) +{ + contentLayout->mainContent->layout()->addWidget(body); + contentLayout->mainHead->layout()->addWidget(head); body->show(); head->show(); onTabChanged(settingsWidgets->currentIndex()); diff --git a/src/widget/form/settingswidget.h b/src/widget/form/settingswidget.h index bdcba3a2e..8d696b069 100644 --- a/src/widget/form/settingswidget.h +++ b/src/widget/form/settingswidget.h @@ -32,8 +32,7 @@ class PrivacyForm; class AVForm; class QLabel; class QTabWidget; - -namespace Ui {class MainWindow;} +class ContentLayout; class SettingsWidget : public QWidget { @@ -42,7 +41,8 @@ public: SettingsWidget(QWidget* parent = nullptr); ~SettingsWidget(); - void show(Ui::MainWindow &ui); + bool isShown() const; + void show(ContentLayout* contentLayout); void setBodyHeadStyle(QString style); void showAbout(); @@ -50,6 +50,7 @@ public: signals: void setShowSystemTray(bool newValue); void compactToggled(bool compact); + void separateWindowToggled(bool separateWindow); void groupchatPositionToggled(bool groupchatPosition); private slots: diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index cf4a7ce8a..0f10e96ab 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -304,10 +304,16 @@ void FriendWidget::search(const QString &searchString, bool hide) circleWidget->search(searchString); } -void FriendWidget::setChatForm(Ui::MainWindow &ui) +bool FriendWidget::chatFormIsSet() const { Friend* f = FriendList::findFriend(friendId); - f->getChatForm()->show(ui); + return f->getChatForm()->isVisible(); +} + +void FriendWidget::setChatForm(ContentLayout* contentLayout) +{ + Friend* f = FriendList::findFriend(friendId); + f->getChatForm()->show(contentLayout); } void FriendWidget::resetEventFlags() diff --git a/src/widget/friendwidget.h b/src/widget/friendwidget.h index e1f6270f7..e319a9ebf 100644 --- a/src/widget/friendwidget.h +++ b/src/widget/friendwidget.h @@ -32,7 +32,8 @@ public: virtual void setAsActiveChatroom() override; virtual void setAsInactiveChatroom() override; virtual void updateStatusLight() override; - virtual void setChatForm(Ui::MainWindow &) override; + virtual bool chatFormIsSet() const override; + virtual void setChatForm(ContentLayout* contentLayout) override; virtual void resetEventFlags() override; virtual QString getStatusString() override; void search(const QString &searchString, bool hide = false); diff --git a/src/widget/genericchatroomwidget.h b/src/widget/genericchatroomwidget.h index dcccb2985..b74a0d9ac 100644 --- a/src/widget/genericchatroomwidget.h +++ b/src/widget/genericchatroomwidget.h @@ -26,10 +26,7 @@ class CroppingLabel; class MaskablePixmapWidget; class QVBoxLayout; class QHBoxLayout; - -namespace Ui { - class MainWindow; -} +class ContentLayout; class GenericChatroomWidget : public GenericChatItemWidget { @@ -40,7 +37,8 @@ public: virtual void setAsActiveChatroom() = 0; virtual void setAsInactiveChatroom() = 0; virtual void updateStatusLight() = 0; - virtual void setChatForm(Ui::MainWindow &) = 0; + virtual bool chatFormIsSet() const = 0; + virtual void setChatForm(ContentLayout* contentLayout) = 0; virtual void resetEventFlags() = 0; virtual QString getStatusString() = 0; diff --git a/src/widget/groupwidget.cpp b/src/widget/groupwidget.cpp index 4e78d71b5..36a32d174 100644 --- a/src/widget/groupwidget.cpp +++ b/src/widget/groupwidget.cpp @@ -137,10 +137,16 @@ void GroupWidget::editName() nameLabel->editBegin(); } -void GroupWidget::setChatForm(Ui::MainWindow &ui) +bool GroupWidget::chatFormIsSet() const { Group* g = GroupList::findGroup(groupId); - g->getChatForm()->show(ui); + return g->getChatForm()->isVisible(); +} + +void GroupWidget::setChatForm(ContentLayout* contentLayout) +{ + Group* g = GroupList::findGroup(groupId); + g->getChatForm()->show(contentLayout); } void GroupWidget::resetEventFlags() diff --git a/src/widget/groupwidget.h b/src/widget/groupwidget.h index 371a586de..5f7610f32 100644 --- a/src/widget/groupwidget.h +++ b/src/widget/groupwidget.h @@ -30,7 +30,8 @@ public: virtual void setAsInactiveChatroom() final override; virtual void setAsActiveChatroom() final override; virtual void updateStatusLight() final override; - virtual void setChatForm(Ui::MainWindow &) final override; + virtual bool chatFormIsSet() const final override; + virtual void setChatForm(ContentLayout* contentLayout) override; virtual void resetEventFlags() final override; virtual QString getStatusString() final override; void setName(const QString& name); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 0caaf1860..4cf5a8de5 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -18,6 +18,7 @@ */ #include "widget.h" +#include "contentlayout.h" #include "ui_mainwindow.h" #include "src/core/core.h" #include "src/persistence/settings.h" @@ -68,6 +69,7 @@ #include #include #include +#include #include #ifdef Q_OS_MAC @@ -113,11 +115,6 @@ void Widget::init() offlineMsgTimer = new QTimer(); offlineMsgTimer->start(15000); - //restore window state - restoreGeometry(Settings::getInstance().getWindowGeometry()); - restoreState(Settings::getInstance().getWindowState()); - ui->mainSplitter->restoreState(Settings::getInstance().getSplitterState()); - statusOnline = new QAction(this); statusOnline->setIcon(getStatusIcon(Status::Online, 10, 10)); connect(statusOnline, SIGNAL(triggered()), this, SLOT(setStatusOnline())); @@ -176,7 +173,7 @@ void Widget::init() ui->searchContactFilterBox->setMenu(filterMenu); - ui->mainContent->setLayout(new QVBoxLayout()); + /*ui->mainContent->setLayout(new QVBoxLayout()); ui->mainHead->setLayout(new QVBoxLayout()); ui->mainHead->layout()->setMargin(0); ui->mainHead->layout()->setSpacing(0); @@ -186,11 +183,11 @@ void Widget::init() { ui->mainHead->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); ui->mainContent->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); - } + }*/ #ifndef Q_OS_MAC - ui->mainHead->setStyleSheet(Style::getStylesheet(":ui/settings/mainHead.css")); - ui->mainContent->setStyleSheet(Style::getStylesheet(":ui/settings/mainContent.css")); + //ui->mainHead->setStyleSheet(Style::getStylesheet(":ui/settings/mainHead.css")); + //ui->mainContent->setStyleSheet(Style::getStylesheet(":ui/settings/mainContent.css")); ui->statusHead->setStyleSheet(Style::getStylesheet(":/ui/window/statusPanel.css")); ui->statusPanel->setStyleSheet(Style::getStylesheet(":/ui/window/statusPanel.css")); #endif @@ -331,15 +328,25 @@ void Widget::init() }); #endif - addFriendForm->show(*ui); - setWindowTitle(tr("Add friend")); + contentLayout = nullptr; + onSeparateWindowChanged(Settings::getInstance().getSeparateWindow(), false); + + if (contentLayout != nullptr) + onAddClicked(); + ui->addButton->setCheckable(true); ui->transferButton->setCheckable(true); ui->settingsButton->setCheckable(true); setActiveToolMenuButton(Widget::AddButton); + //restore window state + restoreGeometry(Settings::getInstance().getWindowGeometry()); + restoreState(Settings::getInstance().getWindowState()); + ui->mainSplitter->restoreState(Settings::getInstance().getSplitterState()); + connect(settingsWidget, &SettingsWidget::compactToggled, contactListWidget, &FriendListWidget::onCompactChanged); connect(settingsWidget, &SettingsWidget::groupchatPositionToggled, contactListWidget, &FriendListWidget::onGroupchatPositionChanged); + connect(settingsWidget, &SettingsWidget::separateWindowToggled, this, &Widget::onSeparateWindowClicked); #if (AUTOUPDATE_ENABLED) if (Settings::getInstance().getCheckUpdates()) AutoUpdater::checkUpdatesAsyncInteractive(); @@ -404,13 +411,13 @@ Widget::~Widget() if (icon) icon->hide(); - hideMainForms(); delete profileForm; delete settingsWidget; delete addFriendForm; delete filesForm; delete timer; delete offlineMsgTimer; + delete contentLayout; FriendList::clear(); GroupList::clear(); @@ -517,6 +524,51 @@ void Widget::onStatusSet(Status status) updateIcons(); } +void Widget::onSeparateWindowClicked(bool separate) +{ + onSeparateWindowChanged(separate, true); +} + +void Widget::onSeparateWindowChanged(bool separate, bool clicked) +{ + if (!separate) + { + QWindowList windowList = QGuiApplication::topLevelWindows(); + for (QWindow* window : windowList) + { + qDebug() << window->objectName(); + if (window->objectName() == "detachedWindow") + window->close(); + } + + QWidget* contentWidget = new QWidget(this); + contentWidget->setObjectName("yolo"); + contentLayout = new ContentLayout(contentWidget); + ui->mainSplitter->addWidget(contentWidget); + + clicked = true; + setMinimumWidth(775); + } + else + { + if (contentLayout != nullptr) + { + contentLayout->parentWidget()->hide(); + contentLayout->parentWidget()->deleteLater(); + contentLayout->deleteLater(); + contentLayout = nullptr; + } + + setMinimumWidth(ui->tooliconsZone->sizeHint().width()); + + if (clicked) + resize(ui->statusPanel->width(), height()); + } + + if (clicked) + onSettingsClicked(); +} + void Widget::setWindowTitle(const QString& title) { QString tmp = title; @@ -533,11 +585,21 @@ void Widget::forceShow() void Widget::onAddClicked() { - hideMainForms(); - addFriendForm->show(*ui); - setWindowTitle(tr("Add friend")); - setActiveToolMenuButton(Widget::AddButton); - activeChatroomWidget = nullptr; + if (Settings::getInstance().getSeparateWindow()) + { + if (!addFriendForm->isShown()) + { + addFriendForm->show(createContentDialog(tr("Add friend"))); + setActiveToolMenuButton(Widget::None); + } + } + else + { + hideMainForms(nullptr); + addFriendForm->show(contentLayout); + setWindowTitle(tr("Add friend")); + setActiveToolMenuButton(Widget::AddButton); + } } void Widget::onGroupClicked() @@ -547,11 +609,21 @@ void Widget::onGroupClicked() void Widget::onTransferClicked() { - hideMainForms(); - filesForm->show(*ui); - setWindowTitle(tr("File transfers")); - setActiveToolMenuButton(Widget::TransferButton); - activeChatroomWidget = nullptr; + if (Settings::getInstance().getSeparateWindow()) + { + if (!filesForm->isShown()) + { + filesForm->show(createContentDialog(tr("File transfers"))); + setActiveToolMenuButton(Widget::None); + } + } + else + { + hideMainForms(nullptr); + filesForm->show(contentLayout); + setWindowTitle(tr("File transfers")); + setActiveToolMenuButton(Widget::TransferButton); + } } void Widget::confirmExecutableOpen(const QFileInfo file) @@ -626,33 +698,39 @@ void Widget::onIconClick(QSystemTrayIcon::ActivationReason reason) void Widget::onSettingsClicked() { - hideMainForms(); - settingsWidget->show(*ui); - setWindowTitle(tr("Settings")); - setActiveToolMenuButton(Widget::SettingButton); - activeChatroomWidget = nullptr; + if (Settings::getInstance().getSeparateWindow()) + { + if (!settingsWidget->isShown()) + { + settingsWidget->show(createContentDialog(tr("Settings"))); + setActiveToolMenuButton(Widget::None); + } + } + else + { + hideMainForms(nullptr); + settingsWidget->show(contentLayout); + setWindowTitle(tr("Settings")); + setActiveToolMenuButton(Widget::SettingButton); + } } void Widget::showProfile() // onAvatarClicked, onUsernameClicked { - hideMainForms(); - profileForm->show(*ui); + hideMainForms(nullptr); + profileForm->show(contentLayout); setWindowTitle(tr("Profile")); - activeChatroomWidget = nullptr; } -void Widget::hideMainForms() +void Widget::hideMainForms(GenericChatroomWidget* chatroomWidget) { setActiveToolMenuButton(Widget::None); - QLayoutItem* item; - while ((item = ui->mainHead->layout()->takeAt(0)) != 0) - item->widget()->hide(); - - while ((item = ui->mainContent->layout()->takeAt(0)) != 0) - item->widget()->hide(); + contentLayout->clear(); if (activeChatroomWidget != nullptr) activeChatroomWidget->setAsInactiveChatroom(); + + activeChatroomWidget = chatroomWidget; } void Widget::onUsernameChanged(const QString& newUsername, const QString& oldUsername) @@ -860,21 +938,23 @@ void Widget::onFriendDisplayChanged(FriendWidget *friendWidget, Status s) void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget) { - hideMainForms(); - - if (activeChatroomWidget != nullptr) - activeChatroomWidget->setAsInactiveChatroom(); - activeChatroomWidget = widget; - - widget->setAsActiveChatroom(); - widget->setChatForm(*ui); - setWindowTitle(widget->getName()); - widget->resetEventFlags(); - widget->updateStatusLight(); - QString windowTitle = widget->getName(); - if (!widget->getStatusString().isNull()) - windowTitle += " (" + widget->getStatusString() + ")"; - setWindowTitle(windowTitle); + if (Settings::getInstance().getSeparateWindow()) + { + if (!widget->chatFormIsSet()) + widget->setChatForm(createContentDialog(widget->getName())); + } + else + { + widget->setChatForm(contentLayout); + setWindowTitle(widget->getName()); + widget->setAsActiveChatroom(); + widget->resetEventFlags(); + widget->updateStatusLight(); + QString windowTitle = widget->getName(); + if (!widget->getStatusString().isNull()) + windowTitle += " (" + widget->getStatusString() + ")"; + setWindowTitle(windowTitle); + } } void Widget::onFriendMessageReceived(int friendId, const QString& message, bool isAction) @@ -1017,7 +1097,7 @@ void Widget::removeFriend(Friend* f, bool fake) Nexus::getCore()->removeFriend(f->getFriendID(), fake); delete f; - if (ui->mainHead->layout()->isEmpty()) + if (contentLayout->mainHead->layout()->isEmpty()) onAddClicked(); contactListWidget->reDraw(); @@ -1041,6 +1121,23 @@ void Widget::clearContactsList() removeGroup(g, true); } +ContentLayout* Widget::createContentDialog(const QString &title) const +{ + QDialog* dialog = new QDialog(); + ContentLayout* contentLayout = new ContentLayout(dialog); + + dialog->setObjectName("detached"); + dialog->setLayout(contentLayout); + dialog->layout()->setMargin(0); + dialog->layout()->setSpacing(0); + dialog->setMinimumSize(775, 420); + dialog->setAttribute(Qt::WA_DeleteOnClose); + dialog->setWindowTitle(title); + dialog->show(); + + return contentLayout; +} + void Widget::copyFriendIdToClipboard(int friendId) { Friend* f = FriendList::findFriend(friendId); @@ -1172,7 +1269,7 @@ void Widget::removeGroup(Group* g, bool fake) GroupList::removeGroup(g->getGroupId(), fake); Nexus::getCore()->removeGroup(g->getGroupId(), fake); delete g; - if (ui->mainHead->layout()->isEmpty()) + if (contentLayout->mainHead->layout()->isEmpty()) onAddClicked(); contactListWidget->reDraw(); diff --git a/src/widget/widget.h b/src/widget/widget.h index 443fd7068..1957fc482 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -51,6 +51,7 @@ class SettingsWidget; class AddFriendForm; class CircleWidget; class QActionGroup; +class ContentLayout; class Widget final : public QMainWindow { @@ -68,6 +69,7 @@ public: bool getIsWindowMinimized(); void updateIcons(); void clearContactsList(); + ContentLayout* createContentDialog(const QString& title) const; static void confirmExecutableOpen(const QFileInfo file); @@ -87,6 +89,8 @@ public: public slots: void onSettingsClicked(); + void onSeparateWindowClicked(bool separate); + void onSeparateWindowChanged(bool separate, bool clicked); void setWindowTitle(const QString& title); void forceShow(); void onConnected(); @@ -185,7 +189,7 @@ private: private: void setActiveToolMenuButton(ActiveToolMenuButton newActiveButton); - void hideMainForms(); + void hideMainForms(GenericChatroomWidget* chatroomWidget); Group *createGroup(int groupId); void removeFriend(Friend* f, bool fake = false); void removeGroup(Group* g, bool fake = false); @@ -225,6 +229,7 @@ private: Ui::MainWindow *ui; QSplitter *centralLayout; QPoint dragPosition; + ContentLayout* contentLayout; AddFriendForm *addFriendForm; ProfileForm *profileForm; SettingsWidget *settingsWidget; From fa3ab02cec6adb5013a357b67fc80ea90787e38c Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Fri, 19 Jun 2015 10:58:48 -0400 Subject: [PATCH 02/24] Multi-window: Tabbed interface --- qtox.pro | 4 +- src/persistence/settings.cpp | 40 +++++ src/persistence/settings.h | 12 ++ src/widget/contentdialog.cpp | 174 ++++++++++++++++++++ src/widget/contentdialog.h | 62 +++++++ src/widget/form/settings/generalform.cpp | 9 + src/widget/form/settings/generalform.h | 1 + src/widget/form/settings/generalsettings.ui | 12 +- src/widget/friendwidget.cpp | 11 +- src/widget/friendwidget.h | 1 + src/widget/genericchatroomwidget.h | 2 + src/widget/widget.cpp | 41 ++++- 12 files changed, 359 insertions(+), 10 deletions(-) create mode 100644 src/widget/contentdialog.cpp create mode 100644 src/widget/contentdialog.h diff --git a/qtox.pro b/qtox.pro index 7b895882a..e53035e15 100644 --- a/qtox.pro +++ b/qtox.pro @@ -496,7 +496,8 @@ SOURCES += \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ src/widget/tool/removefrienddialog.cpp \ - src/widget/contentlayout.cpp + src/widget/contentlayout.cpp \ + src/widget/contentdialog.cpp HEADERS += \ src/audio/audio.h \ @@ -541,4 +542,5 @@ HEADERS += \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ src/widget/contentlayout.h \ + src/widget/contentdialog.h \ src/widget/tool/removefrienddialog.h diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index 81651b17b..2fe9fe5eb 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -175,6 +175,7 @@ void Settings::loadGlobal() QStandardPaths::locate(QStandardPaths::HomeLocation, QString(), QStandardPaths::LocateDirectory) ).toString(); separateWindow = s.value("separateWindow", false).toBool(); + dontGroupWindows = s.value("dontGroupWindows", false).toBool(); groupchatPosition = s.value("groupchatPosition", true).toBool(); s.endGroup(); @@ -217,6 +218,8 @@ void Settings::loadGlobal() windowGeometry = s.value("windowGeometry", QByteArray()).toByteArray(); windowState = s.value("windowState", QByteArray()).toByteArray(); splitterState = s.value("splitterState", QByteArray()).toByteArray(); + dialogGeometry = s.value("dialogGeometry", QByteArray()).toByteArray(); + dialogSplitterState = s.value("dialogSplitterState", QByteArray()).toByteArray(); s.endGroup(); s.beginGroup("Audio"); @@ -377,6 +380,7 @@ void Settings::saveGlobal() s.setValue("groupAlwaysNotify", groupAlwaysNotify); s.setValue("fauxOfflineMessaging", fauxOfflineMessaging); s.setValue("separateWindow", separateWindow); + s.setValue("dontGroupWindows", dontGroupWindows); s.setValue("groupchatPosition", groupchatPosition); s.setValue("autoSaveEnabled", autoSaveEnabled); s.setValue("globalAutoAcceptDir", globalAutoAcceptDir); @@ -1092,6 +1096,30 @@ void Settings::setSplitterState(const QByteArray &value) splitterState = value; } +QByteArray Settings::getDialogGeometry() const +{ + QMutexLocker locker{&bigLock}; + return dialogGeometry; +} + +void Settings::setDialogGeometry(const QByteArray &value) +{ + QMutexLocker locker{&bigLock}; + dialogGeometry = value; +} + +QByteArray Settings::getDialogSplitterState() const +{ + QMutexLocker locker{&bigLock}; + return dialogSplitterState; +} + +void Settings::setDialogSplitterState(const QByteArray &value) +{ + QMutexLocker locker{&bigLock}; + dialogSplitterState = value; +} + bool Settings::isMinimizeOnCloseEnabled() const { QMutexLocker locker{&bigLock}; @@ -1338,6 +1366,18 @@ void Settings::setSeparateWindow(bool value) separateWindow = value; } +bool Settings::getDontGroupWindows() const +{ + QMutexLocker locker{&bigLock}; + return dontGroupWindows; +} + +void Settings::setDontGroupWindows(bool value) +{ + QMutexLocker locker{&bigLock}; + dontGroupWindows = value; +} + bool Settings::getGroupchatPosition() const { QMutexLocker locker{&bigLock}; diff --git a/src/persistence/settings.h b/src/persistence/settings.h index 88e6f18e0..3b6fdf11f 100644 --- a/src/persistence/settings.h +++ b/src/persistence/settings.h @@ -217,6 +217,12 @@ public: QByteArray getSplitterState() const; void setSplitterState(const QByteArray &value); + QByteArray getDialogGeometry() const; + void setDialogGeometry(const QByteArray& value); + + QByteArray getDialogSplitterState() const; + void setDialogSplitterState(const QByteArray &value); + QString getFriendAdress(const QString &publicKey) const; void updateFriendAdress(const QString &newAddr); @@ -240,6 +246,9 @@ public: bool getSeparateWindow() const; void setSeparateWindow(bool value); + bool getDontGroupWindows() const; + void setDontGroupWindows(bool value); + bool getGroupchatPosition() const; void setGroupchatPosition(bool value); @@ -299,6 +308,7 @@ private: bool compactLayout; bool groupchatPosition; bool separateWindow; + bool dontGroupWindows; bool enableIPv6; QString translation; bool makeToxPortable; @@ -338,6 +348,8 @@ private: QByteArray windowGeometry; QByteArray windowState; QByteArray splitterState; + QByteArray dialogGeometry; + QByteArray dialogSplitterState; QString style; bool showSystemTray; diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp new file mode 100644 index 000000000..4ff894870 --- /dev/null +++ b/src/widget/contentdialog.cpp @@ -0,0 +1,174 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + +#include "contentdialog.h" +#include "contentlayout.h" +#include "friendwidget.h" +#include "style.h" +#include "tool/adjustingscrollarea.h" +#include "src/persistence/settings.h" +#include "src/friend.h" +#include "src/friendlist.h" +#include +#include + +ContentDialog* ContentDialog::currentDialog = nullptr; +QHash> ContentDialog::friendList; + +ContentDialog::ContentDialog(QWidget* parent) + : QDialog(parent, Qt::Window) + , activeChatroomWidget(nullptr) +{ + QVBoxLayout* boxLayout = new QVBoxLayout(this); + boxLayout->setMargin(0); + boxLayout->setSpacing(0); + + splitter = new QSplitter(this); + setStyleSheet("QSplitter{color: rgb(255, 255, 255);background-color: rgb(255, 255, 255);alternate-background-color: rgb(255, 255, 255);border-color: rgb(255, 255, 255);gridline-color: rgb(255, 255, 255);selection-color: rgb(255, 255, 255);selection-background-color: rgb(255, 255, 255);}QSplitter:handle{color: rgb(255, 255, 255);background-color: rgb(255, 255, 255);}"); + splitter->setHandleWidth(6); + + QWidget *friendWidget = new QWidget(); + friendWidget->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed); + friendWidget->setAutoFillBackground(true); + + friendLayout = new QVBoxLayout(friendWidget); + friendLayout->setMargin(0); + friendLayout->setSpacing(0); + friendLayout->addStretch(); + + QScrollArea *friendScroll = new QScrollArea(this); + friendScroll->setMinimumWidth(220); + friendScroll->setFrameStyle(QFrame::NoFrame); + friendScroll->setLayoutDirection(Qt::RightToLeft); + friendScroll->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + friendScroll->setStyleSheet(Style::getStylesheet(":/ui/friendList/friendList.css")); + friendScroll->setWidgetResizable(true); + friendScroll->setWidget(friendWidget); + + QWidget* contentWidget = new QWidget(this); + contentWidget->setAutoFillBackground(true); + contentLayout = new ContentLayout(contentWidget); + contentLayout->setMargin(0); + contentLayout->setSpacing(0); + + splitter->addWidget(friendScroll); + splitter->addWidget(contentWidget); + splitter->setStretchFactor(1, 1); + splitter->setCollapsible(1, false); + boxLayout->addWidget(splitter); + + setMinimumSize(775, 420); + setAttribute(Qt::WA_DeleteOnClose); + + //restore window state + restoreGeometry(Settings::getInstance().getDialogGeometry()); + splitter->restoreState(Settings::getInstance().getDialogSplitterState()); + + currentDialog = this; +} + +ContentDialog::~ContentDialog() +{ + if (currentDialog == this) + currentDialog = nullptr; + + auto friendIt = friendList.begin(); + + while (friendIt != friendList.end()) + { + if (std::get<0>(friendIt.value()) == this) + { + friendIt = friendList.erase(friendIt); + continue; + } + ++friendIt; + } +} + +void ContentDialog::addFriend(int friendId, QString id) +{ + FriendWidget* friendWidget = new FriendWidget(friendId, id); + friendLayout->insertWidget(friendLayout->count() - 1, friendWidget); + + onChatroomWidgetClicked(friendWidget); + + connect(friendWidget, &FriendWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); + + friendList.insert(friendId, std::make_tuple(this, friendWidget)); +} + +ContentDialog* ContentDialog::current() +{ + return currentDialog; +} + +bool ContentDialog::showChatroomWidget(int friendId) +{ + auto widgetIt = friendList.find(friendId); + if (widgetIt == friendList.end()) + return false; + + std::get<0>(widgetIt.value())->activateWindow(); + std::get<0>(widgetIt.value())->onChatroomWidgetClicked(std::get<1>(widgetIt.value())); + + return true; +} + +void ContentDialog::resizeEvent(QResizeEvent* event) +{ + Q_UNUSED(event); + saveDialogGeometry(); +} + +void ContentDialog::closeEvent(QCloseEvent* event) +{ + saveDialogGeometry(); + saveSplitterState(); + QWidget::closeEvent(event); +} + +void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget) +{ + contentLayout->clear(); + + if (activeChatroomWidget != nullptr) + activeChatroomWidget->setAsInactiveChatroom(); + + activeChatroomWidget = widget; + + widget->setChatForm(contentLayout); + setWindowTitle(widget->getName()); + widget->setAsActiveChatroom(); + widget->resetEventFlags(); + widget->updateStatusLight(); + QString windowTitle = widget->getName(); + if (!widget->getStatusString().isNull()) + windowTitle += " (" + widget->getStatusString() + ")"; + setWindowTitle(windowTitle); +} + +void ContentDialog::saveDialogGeometry() +{ + Settings::getInstance().setDialogGeometry(saveGeometry()); +} + +void ContentDialog::saveSplitterState() +{ + Settings::getInstance().setDialogSplitterState(splitter->saveState()); +} diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h new file mode 100644 index 000000000..44244d692 --- /dev/null +++ b/src/widget/contentdialog.h @@ -0,0 +1,62 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + +#ifndef CONTENTDIALOG_H +#define CONTENTDIALOG_H + +#include +#include + +template class QHash; + +class QSplitter; +class QVBoxLayout; +class ContentLayout; +class GenericChatroomWidget; + +class ContentDialog : public QDialog +{ + Q_OBJECT +public: + ContentDialog(QWidget* parent = 0); + ~ContentDialog(); + void addFriend(int friendId, QString id); + static ContentDialog* current(); + static bool showChatroomWidget(int friendId); + +protected: + void resizeEvent(QResizeEvent* event) override; + void closeEvent(QCloseEvent* event) override; + +private slots: + void onChatroomWidgetClicked(GenericChatroomWidget* widget); + +private: + void saveDialogGeometry(); + void saveSplitterState(); + + QSplitter* splitter; + QVBoxLayout* friendLayout; + ContentLayout* contentLayout; + GenericChatroomWidget* activeChatroomWidget; + static ContentDialog* currentDialog; + static QHash> friendList; +}; + +#endif // CONTENTDIALOG_H diff --git a/src/widget/form/settings/generalform.cpp b/src/widget/form/settings/generalform.cpp index d919f8555..7e552fd85 100644 --- a/src/widget/form/settings/generalform.cpp +++ b/src/widget/form/settings/generalform.cpp @@ -88,6 +88,8 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : bodyUI->cbFauxOfflineMessaging->setChecked(Settings::getInstance().getFauxOfflineMessaging()); bodyUI->cbCompactLayout->setChecked(Settings::getInstance().getCompactLayout()); bodyUI->cbSeparateWindow->setChecked(Settings::getInstance().getSeparateWindow()); + bodyUI->cbDontGroupWindows->setChecked(Settings::getInstance().getDontGroupWindows()); + bodyUI->cbDontGroupWindows->setEnabled(bodyUI->cbSeparateWindow->isChecked()); bodyUI->cbGroupchatPosition->setChecked(Settings::getInstance().getGroupchatPosition()); for (auto entry : SmileyPack::listSmileyPacks()) @@ -181,6 +183,7 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : connect(bodyUI->cbFauxOfflineMessaging, &QCheckBox::stateChanged, this, &GeneralForm::onFauxOfflineMessaging); connect(bodyUI->cbCompactLayout, &QCheckBox::stateChanged, this, &GeneralForm::onCompactLayout); connect(bodyUI->cbSeparateWindow, &QCheckBox::stateChanged, this, &GeneralForm::onSeparateWindowChanged); + connect(bodyUI->cbDontGroupWindows, &QCheckBox::stateChanged, this, &GeneralForm::onDontGroupWindowsChanged); connect(bodyUI->cbGroupchatPosition, &QCheckBox::stateChanged, this, &GeneralForm::onGroupchatPositionChanged); // prevent stealing mouse whell scroll @@ -434,10 +437,16 @@ void GeneralForm::onCompactLayout() void GeneralForm::onSeparateWindowChanged() { + bodyUI->cbDontGroupWindows->setEnabled(bodyUI->cbSeparateWindow->isChecked()); Settings::getInstance().setSeparateWindow(bodyUI->cbSeparateWindow->isChecked()); emit parent->separateWindowToggled(bodyUI->cbSeparateWindow->isChecked()); } +void GeneralForm::onDontGroupWindowsChanged() +{ + Settings::getInstance().setDontGroupWindows(bodyUI->cbDontGroupWindows->isChecked()); +} + void GeneralForm::onGroupchatPositionChanged() { Settings::getInstance().setGroupchatPosition(bodyUI->cbGroupchatPosition->isChecked()); diff --git a/src/widget/form/settings/generalform.h b/src/widget/form/settings/generalform.h index 31d24b8de..74fc7e19f 100644 --- a/src/widget/form/settings/generalform.h +++ b/src/widget/form/settings/generalform.h @@ -68,6 +68,7 @@ private slots: void onFauxOfflineMessaging(); void onCompactLayout(); void onSeparateWindowChanged(); + void onDontGroupWindowsChanged(); void onGroupchatPositionChanged(); void onThemeColorChanged(int); diff --git a/src/widget/form/settings/generalsettings.ui b/src/widget/form/settings/generalsettings.ui index 54446a19b..d99510599 100644 --- a/src/widget/form/settings/generalsettings.ui +++ b/src/widget/form/settings/generalsettings.ui @@ -377,7 +377,17 @@ instead of system taskbar. - Use separate windows for friend list. + Open chats in separate window. + + + + + + + false + + + Don't group chat wndows. diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index 0f10e96ab..fd298c72e 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -26,6 +26,7 @@ #include "src/core/core.h" #include "form/chatform.h" #include "maskablepixmapwidget.h" +#include "contentdialog.h" #include "src/widget/tool/croppinglabel.h" #include "src/widget/style.h" #include "src/persistence/settings.h" @@ -67,6 +68,9 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) ToxId id = FriendList::findFriend(friendId)->getToxId(); QString dir = Settings::getInstance().getAutoAcceptDir(id); QMenu menu; + menu.addAction(tr("Open Chat")); + menu.addAction(tr("Open Chat in New Window")); + menu.addSeparator(); QMenu* inviteMenu = menu.addMenu(tr("Invite to group","Menu to invite a friend to a groupchat")); QMap groupActions; @@ -296,6 +300,11 @@ QString FriendWidget::getStatusString() return QString::null; } +Friend* FriendWidget::getFriend() const +{ + return FriendList::findFriend(friendId); +} + void FriendWidget::search(const QString &searchString, bool hide) { searchName(searchString, hide); @@ -307,7 +316,7 @@ void FriendWidget::search(const QString &searchString, bool hide) bool FriendWidget::chatFormIsSet() const { Friend* f = FriendList::findFriend(friendId); - return f->getChatForm()->isVisible(); + return f->getChatForm()->isVisible() || ContentDialog::showChatroomWidget(friendId); } void FriendWidget::setChatForm(ContentLayout* contentLayout) diff --git a/src/widget/friendwidget.h b/src/widget/friendwidget.h index e319a9ebf..2ef2b26d2 100644 --- a/src/widget/friendwidget.h +++ b/src/widget/friendwidget.h @@ -36,6 +36,7 @@ public: virtual void setChatForm(ContentLayout* contentLayout) override; virtual void resetEventFlags() override; virtual QString getStatusString() override; + virtual Friend* getFriend() const override; void search(const QString &searchString, bool hide = false); signals: diff --git a/src/widget/genericchatroomwidget.h b/src/widget/genericchatroomwidget.h index b74a0d9ac..73866c2c9 100644 --- a/src/widget/genericchatroomwidget.h +++ b/src/widget/genericchatroomwidget.h @@ -27,6 +27,7 @@ class MaskablePixmapWidget; class QVBoxLayout; class QHBoxLayout; class ContentLayout; +class Friend; class GenericChatroomWidget : public GenericChatItemWidget { @@ -41,6 +42,7 @@ public: virtual void setChatForm(ContentLayout* contentLayout) = 0; virtual void resetEventFlags() = 0; virtual QString getStatusString() = 0; + virtual Friend* getFriend() const{return nullptr;} virtual bool eventFilter(QObject *, QEvent *) final override; diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 4cf5a8de5..7320ae17f 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -22,6 +22,7 @@ #include "ui_mainwindow.h" #include "src/core/core.h" #include "src/persistence/settings.h" +#include "contentdialog.h" #include "src/friend.h" #include "src/friendlist.h" #include "tool/friendrequestdialog.h" @@ -534,6 +535,7 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) if (!separate) { QWindowList windowList = QGuiApplication::topLevelWindows(); + for (QWindow* window : windowList) { qDebug() << window->objectName(); @@ -542,7 +544,6 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) } QWidget* contentWidget = new QWidget(this); - contentWidget->setObjectName("yolo"); contentLayout = new ContentLayout(contentWidget); ui->mainSplitter->addWidget(contentWidget); @@ -553,6 +554,8 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) { if (contentLayout != nullptr) { + contentLayout->clear(); + contentLayout->parentWidget()->setParent(0); // Remove from splitter. contentLayout->parentWidget()->hide(); contentLayout->parentWidget()->deleteLater(); contentLayout->deleteLater(); @@ -563,6 +566,8 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) if (clicked) resize(ui->statusPanel->width(), height()); + + setWindowTitle(QString()); } if (clicked) @@ -571,9 +576,16 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) void Widget::setWindowTitle(const QString& title) { - QString tmp = title; - /// <[^>]*> Regexp to remove HTML tags, in case someone used them in title - QMainWindow::setWindowTitle("qTox - " + tmp.remove(QRegExp("<[^>]*>"))); + if (title.isEmpty()) + { + QMainWindow::setWindowTitle(QApplication::applicationName()); + } + else + { + QString tmp = title; + /// <[^>]*> Regexp to remove HTML tags, in case someone used them in title + QMainWindow::setWindowTitle(QApplication::applicationName() + QStringLiteral(" - ") + tmp.remove(QRegExp("<[^>]*>"))); + } } void Widget::forceShow() @@ -941,12 +953,27 @@ void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget) if (Settings::getInstance().getSeparateWindow()) { if (!widget->chatFormIsSet()) - widget->setChatForm(createContentDialog(widget->getName())); + { + ContentDialog* dialog = nullptr; + + if (!Settings::getInstance().getDontGroupWindows()) + dialog = ContentDialog::current(); + + if (dialog == nullptr) + dialog = new ContentDialog(); + + dialog->show(); + Friend* frnd = widget->getFriend(); + if (frnd != nullptr) + { + dialog->addFriend(frnd->getFriendID(), frnd->getDisplayedName()); + } + } } else { + hideMainForms(widget); widget->setChatForm(contentLayout); - setWindowTitle(widget->getName()); widget->setAsActiveChatroom(); widget->resetEventFlags(); widget->updateStatusLight(); @@ -1799,7 +1826,7 @@ void Widget::retranslateUi() statusOnline->setText(tr("Online", "Button to set your status to 'Online'")); statusAway->setText(tr("Away", "Button to set your status to 'Away'")); statusBusy->setText(tr("Busy", "Button to set your status to 'Busy'")); - setWindowTitle(tr("Settings")); + //setWindowTitle(tr("Settings")); } #ifdef Q_OS_MAC From c684d31d02c7cd292848a96789625becda66d1e4 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 22 Jun 2015 14:01:50 -0400 Subject: [PATCH 03/24] Multi-window: Alert window storing chat, add group widgets, update statuses and avatars --- src/group.cpp | 11 +- src/group.h | 5 + src/widget/contentdialog.cpp | 283 +++++++++++++++++++++++++-- src/widget/contentdialog.h | 38 +++- src/widget/form/chatform.cpp | 4 +- src/widget/friendwidget.cpp | 40 +++- src/widget/friendwidget.h | 4 +- src/widget/genericchatroomwidget.cpp | 10 + src/widget/genericchatroomwidget.h | 9 +- src/widget/groupwidget.cpp | 49 ++++- src/widget/groupwidget.h | 5 +- src/widget/widget.cpp | 158 +++++++++++---- src/widget/widget.h | 9 +- 13 files changed, 547 insertions(+), 78 deletions(-) diff --git a/src/group.cpp b/src/group.cpp index a9a9f68a1..893d08b32 100644 --- a/src/group.cpp +++ b/src/group.cpp @@ -84,6 +84,7 @@ void Group::updatePeer(int peerId, QString name) widget->onUserListChanged(); chatForm->onUserListChanged(); + emit userListChanged(getGroupWidget()); } void Group::setName(const QString& name) @@ -91,7 +92,14 @@ void Group::setName(const QString& name) chatForm->setName(name); if (widget->isActive()) - GUI::setWindowTitle(name); + GUI::setWindowTitle(name); + + emit titleChanged(this->getGroupWidget()); +} + +QString Group::getName() const +{ + return widget->getName(); } void Group::regeneratePeerList() @@ -117,6 +125,7 @@ void Group::regeneratePeerList() widget->onUserListChanged(); chatForm->onUserListChanged(); + emit userListChanged(getGroupWidget()); } bool Group::isAvGroupchat() const diff --git a/src/group.h b/src/group.h index 8ccde2497..8d029bf71 100644 --- a/src/group.h +++ b/src/group.h @@ -61,9 +61,14 @@ public: void updatePeer(int peerId, QString newName); void setName(const QString& name); + QString getName() const; QString resolveToxId(const ToxId &id) const; +signals: + void titleChanged(GroupWidget* widget); + void userListChanged(GroupWidget* widget); + private: GroupWidget* widget; GroupChatForm* chatForm; diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 4ff894870..ac70759f1 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -20,16 +20,23 @@ #include "contentdialog.h" #include "contentlayout.h" #include "friendwidget.h" +#include "groupwidget.h" #include "style.h" +#include "widget.h" #include "tool/adjustingscrollarea.h" #include "src/persistence/settings.h" #include "src/friend.h" #include "src/friendlist.h" +#include "src/group.h" +#include "src/widget/form/chatform.h" +#include "src/core/core.h" #include #include +#include ContentDialog* ContentDialog::currentDialog = nullptr; QHash> ContentDialog::friendList; +QHash> ContentDialog::groupList; ContentDialog::ContentDialog(QWidget* parent) : QDialog(parent, Qt::Window) @@ -101,16 +108,66 @@ ContentDialog::~ContentDialog() } } -void ContentDialog::addFriend(int friendId, QString id) +FriendWidget* ContentDialog::addFriend(int friendId, QString id) { FriendWidget* friendWidget = new FriendWidget(friendId, id); friendLayout->insertWidget(friendLayout->count() - 1, friendWidget); - onChatroomWidgetClicked(friendWidget); + Friend* frnd = friendWidget->getFriend(); + onChatroomWidgetClicked(friendWidget, false); + + connect(frnd, &Friend::displayedNameChanged, this, &ContentDialog::updateFriendWidget); connect(friendWidget, &FriendWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); + connect(friendWidget, SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), frnd->getChatForm(), SLOT(focusInput())); + connect(Core::getInstance(), &Core::friendAvatarChanged, friendWidget, &FriendWidget::onAvatarChange); + connect(Core::getInstance(), &Core::friendAvatarRemoved, friendWidget, &FriendWidget::onAvatarRemoved); friendList.insert(friendId, std::make_tuple(this, friendWidget)); + + return friendWidget; +} + +GroupWidget* ContentDialog::addGroup(int groupId, const QString& name) +{ + GroupWidget* groupWidget = new GroupWidget(groupId, name); + friendLayout->insertWidget(friendLayout->count() - 1, groupWidget); + + Group* group = groupWidget->getGroup(); + connect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); + connect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); + connect(groupWidget, &GroupWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); + + onChatroomWidgetClicked(groupWidget, false); + + groupList.insert(groupId, std::make_tuple(this, groupWidget)); + + return groupWidget; +} + +void ContentDialog::removeFriend(int friendId) +{ + remove(friendId, friendList); +} + +void ContentDialog::removeGroup(int groupId) +{ + remove(groupId, groupList); +} + +bool ContentDialog::hasFriendWidget(int friendId, GenericChatroomWidget* chatroomWidget) +{ + return hasWidget(friendId, chatroomWidget, friendList); +} + +bool ContentDialog::hasGroupWidget(int groupId, GenericChatroomWidget *chatroomWidget) +{ + return hasWidget(groupId, chatroomWidget, groupList); +} + +int ContentDialog::chatroomWidgetCount() const +{ + return friendLayout->count() - 1; // Don't include the stretch. } ContentDialog* ContentDialog::current() @@ -118,16 +175,81 @@ ContentDialog* ContentDialog::current() return currentDialog; } -bool ContentDialog::showChatroomWidget(int friendId) +bool ContentDialog::existsFriendWidget(int friendId, bool focus) { - auto widgetIt = friendList.find(friendId); - if (widgetIt == friendList.end()) - return false; + return existsWidget(friendId, focus, friendList); +} - std::get<0>(widgetIt.value())->activateWindow(); - std::get<0>(widgetIt.value())->onChatroomWidgetClicked(std::get<1>(widgetIt.value())); +bool ContentDialog::existsGroupWidget(int groupId, bool focus) +{ + return existsWidget(groupId, focus, groupList); +} - return true; +void ContentDialog::updateFriendStatus(int friendId) +{ + updateStatus(friendId, friendList); +} + +void ContentDialog::updateFriendStatusMessage(int friendId, const QString &message) +{ + auto iter = friendList.find(friendId); + + if (iter == friendList.end()) + return; + + std::get<1>(iter.value())->setStatusMsg(message); +} + +void ContentDialog::updateGroupStatus(int groupId) +{ + updateStatus(groupId, groupList); +} + +bool ContentDialog::isFriendWidgetActive(int friendId) +{ + return isWidgetActive(friendId, friendList); +} + +bool ContentDialog::isGroupWidgetActive(int groupId) +{ + return isWidgetActive(groupId, groupList); +} + +ContentDialog* ContentDialog::getFriendDialog(int friendId) +{ + return getDialog(friendId, friendList); +} + +ContentDialog* ContentDialog::getGroupDialog(int groupId) +{ + return getDialog(groupId, groupList); +} + +#include +#include +void ContentDialog::dragEnterEvent(QDragEnterEvent *event) +{ + if (event->mimeData()->hasFormat("friend")) + event->acceptProposedAction(); +} + +void ContentDialog::dropEvent(QDropEvent *event) +{ + if (event->mimeData()->hasFormat("friend")) + { + //int friendId = ev->mimeData()->data("friend").toInt(); + //Core::getInstance()->groupInviteFriend(friendId, groupId); + } +} + +void ContentDialog::changeEvent(QEvent *event) +{ + QWidget::changeEvent(event); + if (event->type() == QEvent::ActivationChange) + { + if (isActiveWindow()) + currentDialog = this; + } } void ContentDialog::resizeEvent(QResizeEvent* event) @@ -143,8 +265,27 @@ void ContentDialog::closeEvent(QCloseEvent* event) QWidget::closeEvent(event); } -void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget) +void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) { + if (group) + { + ContentDialog* contentDialog = new ContentDialog(); + contentDialog->show(); + + if (widget->getFriend() != nullptr) + { + removeFriend(widget->getFriend()->getFriendID()); + Widget::getInstance()->addFriendDialog(widget->getFriend(), contentDialog); + } + else + { + removeGroup(widget->getGroup()->getGroupId()); + Widget::getInstance()->addGroupDialog(widget->getGroup(), contentDialog); + } + + return; + } + contentLayout->clear(); if (activeChatroomWidget != nullptr) @@ -153,14 +294,29 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget) activeChatroomWidget = widget; widget->setChatForm(contentLayout); - setWindowTitle(widget->getName()); widget->setAsActiveChatroom(); widget->resetEventFlags(); widget->updateStatusLight(); - QString windowTitle = widget->getName(); - if (!widget->getStatusString().isNull()) - windowTitle += " (" + widget->getStatusString() + ")"; - setWindowTitle(windowTitle); + setWindowTitle(widget->getTitle()); + + if (widget->getFriend() != nullptr) + widget->getFriend()->getFriendWidget()->updateStatusLight(); + else + widget->getGroup()->getGroupWidget()->updateStatusLight(); +} + +void ContentDialog::updateFriendWidget(FriendWidget *w, Status s) +{ + // Note: This argument friend widget is not the one stored here! + Q_UNUSED(s); // Use it after friend list improvements are merged, to sort contacts here. + + std::get<1>(friendList.find(w->friendId).value())->setName(w->getName()); +} + +void ContentDialog::updateGroupWidget(GroupWidget *w) +{ + std::get<1>(groupList.find(w->groupId).value())->setName(w->getName()); + static_cast(std::get<1>(groupList.find(w->groupId).value()))->onUserListChanged(); } void ContentDialog::saveDialogGeometry() @@ -172,3 +328,100 @@ void ContentDialog::saveSplitterState() { Settings::getInstance().setDialogSplitterState(splitter->saveState()); } + +void ContentDialog::remove(int id, const QHash > &list) +{ + auto iter = list.find(id); + + if (iter == list.end()) + return; + + GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value()); + + if (activeChatroomWidget == chatroomWidget) + { + // Need to find replacement to show here instead. + if (chatroomWidgetCount() > 0) + { + int index = friendLayout->indexOf(chatroomWidget) - 1; + + // Don't let it go below 0. If it does, then we're first. Go second. + if (index < 0) + index = 1; + + GenericChatroomWidget* chatroomWidget = static_cast(friendLayout->itemAt(index)->widget()); + onChatroomWidgetClicked(chatroomWidget, false); + } + else + { + contentLayout->clear(); + activeChatroomWidget = nullptr; + deleteLater(); + } + } + + friendLayout->removeWidget(chatroomWidget); + chatroomWidget->deleteLater(); + friendList.remove(id); + update(); +} + +bool ContentDialog::hasWidget(int id, GenericChatroomWidget* chatroomWidget, const QHash>& list) +{ + auto iter = list.find(id); + + if (iter == list.end() || std::get<0>(iter.value()) != this) + return false; + + return chatroomWidget == std::get<1>(iter.value()); +} + +bool ContentDialog::existsWidget(int id, bool focus, const QHash>& list) +{ + auto iter = list.find(id); + if (iter == list.end()) + return false; + + if (focus) + { + std::get<0>(iter.value())->raise(); + std::get<0>(iter.value())->activateWindow(); + std::get<0>(iter.value())->onChatroomWidgetClicked(std::get<1>(iter.value()), false); + } + + return true; +} + +void ContentDialog::updateStatus(int id, const QHash > &list) +{ + auto iter = list.find(id); + + if (iter == list.end()) + return; + + GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value()); + chatroomWidget->updateStatusLight(); + + if(chatroomWidget->isActive()) + std::get<0>(iter.value())->setWindowTitle(chatroomWidget->getTitle()); +} + +bool ContentDialog::isWidgetActive(int id, const QHash > &list) +{ + auto iter = list.find(id); + + if (iter == list.end()) + return false; + + return std::get<0>(iter.value())->activeChatroomWidget == std::get<1>(iter.value()); +} + +ContentDialog* ContentDialog::getDialog(int id, const QHash>& list) +{ + auto iter = list.find(id); + + if (iter == list.end()) + return nullptr; + + return std::get<0>(iter.value()); +} diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index 44244d692..f242830ed 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -22,6 +22,7 @@ #include #include +#include "src/core/corestructs.h" template class QHash; @@ -29,6 +30,8 @@ class QSplitter; class QVBoxLayout; class ContentLayout; class GenericChatroomWidget; +class FriendWidget; +class GroupWidget; class ContentDialog : public QDialog { @@ -36,27 +39,56 @@ class ContentDialog : public QDialog public: ContentDialog(QWidget* parent = 0); ~ContentDialog(); - void addFriend(int friendId, QString id); + + FriendWidget* addFriend(int friendId, QString id); + GroupWidget* addGroup(int groupId, const QString& name); + void removeFriend(int friendId); + void removeGroup(int groupId); + bool hasFriendWidget(int friendId, GenericChatroomWidget* chatroomWidget); + bool hasGroupWidget(int groupId, GenericChatroomWidget* chatroomWidget); + int chatroomWidgetCount() const; + static ContentDialog* current(); - static bool showChatroomWidget(int friendId); + static bool existsFriendWidget(int friendId, bool focus); + static bool existsGroupWidget(int groupId, bool focus); + static void updateFriendStatus(int friendId); + static void updateFriendStatusMessage(int friendId, const QString &message); + static void updateGroupStatus(int groupId); + static bool isFriendWidgetActive(int friendId); + static bool isGroupWidgetActive(int groupId); + static ContentDialog* getFriendDialog(int friendId); + static ContentDialog* getGroupDialog(int groupId); protected: + void dragEnterEvent(QDragEnterEvent* event) final override; + void dropEvent(QDropEvent* event) final override; + void changeEvent(QEvent* event) override; void resizeEvent(QResizeEvent* event) override; void closeEvent(QCloseEvent* event) override; private slots: - void onChatroomWidgetClicked(GenericChatroomWidget* widget); + void onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group); + void updateFriendWidget(FriendWidget* w, Status s); + void updateGroupWidget(GroupWidget* w); private: void saveDialogGeometry(); void saveSplitterState(); + void remove(int id, const QHash>& list); + bool hasWidget(int id, GenericChatroomWidget* chatroomWidget, const QHash>& list); + static bool existsWidget(int id, bool focus, const QHash>& list); + static void updateStatus(int id, const QHash>& list); + static bool isWidgetActive(int id, const QHash>& list); + static ContentDialog* getDialog(int id, const QHash>& list); + QSplitter* splitter; QVBoxLayout* friendLayout; ContentLayout* contentLayout; GenericChatroomWidget* activeChatroomWidget; static ContentDialog* currentDialog; static QHash> friendList; + static QHash> groupList; }; #endif // CONTENTDIALOG_H diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index b20e7480c..bb244af87 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -219,7 +219,7 @@ void ChatForm::onFileRecvRequest(ToxFile file) Widget* w = Widget::getInstance(); if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) { - w->newMessageAlert(f->getFriendWidget()); + w->newFriendMessageAlert(file.friendId); f->setEventFlag(true); f->getFriendWidget()->updateStatusLight(); } @@ -302,7 +302,7 @@ void ChatForm::onAvInvite(uint32_t FriendId, int CallId, bool video) Widget* w = Widget::getInstance(); if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) { - w->newMessageAlert(f->getFriendWidget()); + w->newFriendMessageAlert(FriendId); f->setEventFlag(true); f->getFriendWidget()->updateStatusLight(); } diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index fd298c72e..750c6d2a9 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -68,8 +68,22 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) ToxId id = FriendList::findFriend(friendId)->getToxId(); QString dir = Settings::getInstance().getAutoAcceptDir(id); QMenu menu; - menu.addAction(tr("Open Chat")); - menu.addAction(tr("Open Chat in New Window")); + QAction* openChat = menu.addAction(tr("Open chat")); + QAction* openChatWindow = nullptr; + QAction* removeChatWindow = nullptr; + + if (!Settings::getInstance().getSeparateWindow() || !Settings::getInstance().getDontGroupWindows()) + { + ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); + bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; + + if (contentDialog == nullptr || notAlone) + openChatWindow = menu.addAction(tr("Open chat in new window")); + + if (notAlone && contentDialog->hasFriendWidget(friendId, this)) + removeChatWindow = menu.addAction(tr("Remove chat from this window")); + } + menu.addSeparator(); QMenu* inviteMenu = menu.addMenu(tr("Invite to group","Menu to invite a friend to a groupchat")); QMap groupActions; @@ -160,6 +174,22 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) emit removeFriend(friendId); return; } + else if (selectedItem == openChat) + { + emit chatroomWidgetClicked(this); + return; + } + else if (selectedItem == openChatWindow) + { + emit chatroomWidgetClicked(this, true); + return; + } + else if (selectedItem == removeChatWindow) + { + ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); + contentDialog->removeFriend(friendId); + return; + } else if (selectedItem == autoAccept) { if (!autoAccept->isChecked()) @@ -282,7 +312,7 @@ void FriendWidget::updateStatusLight() statusPic.setMargin(0); } -QString FriendWidget::getStatusString() +QString FriendWidget::getStatusString() const { Friend* f = FriendList::findFriend(friendId); Status status = f->getStatus(); @@ -313,10 +343,10 @@ void FriendWidget::search(const QString &searchString, bool hide) circleWidget->search(searchString); } -bool FriendWidget::chatFormIsSet() const +bool FriendWidget::chatFormIsSet(bool focus) const { Friend* f = FriendList::findFriend(friendId); - return f->getChatForm()->isVisible() || ContentDialog::showChatroomWidget(friendId); + return ContentDialog::existsFriendWidget(friendId, focus) || f->getChatForm()->isVisible(); } void FriendWidget::setChatForm(ContentLayout* contentLayout) diff --git a/src/widget/friendwidget.h b/src/widget/friendwidget.h index 2ef2b26d2..351f10ab0 100644 --- a/src/widget/friendwidget.h +++ b/src/widget/friendwidget.h @@ -32,10 +32,10 @@ public: virtual void setAsActiveChatroom() override; virtual void setAsInactiveChatroom() override; virtual void updateStatusLight() override; - virtual bool chatFormIsSet() const override; + virtual bool chatFormIsSet(bool focus) const override; virtual void setChatForm(ContentLayout* contentLayout) override; virtual void resetEventFlags() override; - virtual QString getStatusString() override; + virtual QString getStatusString() const override; virtual Friend* getFriend() const override; void search(const QString &searchString, bool hide = false); diff --git a/src/widget/genericchatroomwidget.cpp b/src/widget/genericchatroomwidget.cpp index 09983df98..be25f919a 100644 --- a/src/widget/genericchatroomwidget.cpp +++ b/src/widget/genericchatroomwidget.cpp @@ -144,6 +144,16 @@ QString GenericChatroomWidget::getStatusMsg() const return statusMessageLabel->text(); } +QString GenericChatroomWidget::getTitle() const +{ + QString title = getName(); + + if (!getStatusString().isNull()) + title += QStringLiteral(" (") + getStatusString() + QStringLiteral(")"); + + return title; +} + void GenericChatroomWidget::reloadTheme() { QPalette p; diff --git a/src/widget/genericchatroomwidget.h b/src/widget/genericchatroomwidget.h index 73866c2c9..8132e5a9b 100644 --- a/src/widget/genericchatroomwidget.h +++ b/src/widget/genericchatroomwidget.h @@ -28,6 +28,7 @@ class QVBoxLayout; class QHBoxLayout; class ContentLayout; class Friend; +class Group; class GenericChatroomWidget : public GenericChatItemWidget { @@ -38,11 +39,12 @@ public: virtual void setAsActiveChatroom() = 0; virtual void setAsInactiveChatroom() = 0; virtual void updateStatusLight() = 0; - virtual bool chatFormIsSet() const = 0; + virtual bool chatFormIsSet(bool focus) const = 0; virtual void setChatForm(ContentLayout* contentLayout) = 0; virtual void resetEventFlags() = 0; - virtual QString getStatusString() = 0; + virtual QString getStatusString() const = 0; virtual Friend* getFriend() const{return nullptr;} + virtual Group* getGroup() const{return nullptr;} virtual bool eventFilter(QObject *, QEvent *) final override; @@ -52,6 +54,7 @@ public: void setName(const QString& name); void setStatusMsg(const QString& status); QString getStatusMsg() const; + QString getTitle() const; void reloadTheme(); @@ -59,7 +62,7 @@ public slots: void compactChange(bool compact); signals: - void chatroomWidgetClicked(GenericChatroomWidget* widget); + void chatroomWidgetClicked(GenericChatroomWidget* widget, bool group = false); protected: virtual void mouseReleaseEvent(QMouseEvent* event) override; diff --git a/src/widget/groupwidget.cpp b/src/widget/groupwidget.cpp index 36a32d174..adb420b52 100644 --- a/src/widget/groupwidget.cpp +++ b/src/widget/groupwidget.cpp @@ -19,8 +19,10 @@ #include "groupwidget.h" #include "maskablepixmapwidget.h" +#include "contentdialog.h" #include "src/grouplist.h" #include "src/group.h" +#include "src/persistence/settings.h" #include "form/groupchatform.h" #include "src/widget/style.h" #include "src/core/core.h" @@ -62,6 +64,25 @@ void GroupWidget::contextMenuEvent(QContextMenuEvent* event) installEventFilter(this); // Disable leave event. QMenu menu(this); + + QAction* openChat = menu.addAction(tr("Open chat")); + QAction* openChatWindow = nullptr; + QAction* removeChatWindow = nullptr; + + if (!Settings::getInstance().getSeparateWindow() || !Settings::getInstance().getDontGroupWindows()) + { + ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); + bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; + + if (contentDialog == nullptr || notAlone) + openChatWindow = menu.addAction(tr("Open chat in new window")); + + if (notAlone && contentDialog->hasGroupWidget(groupId, this)) + removeChatWindow = menu.addAction(tr("Remove chat from this window")); + } + + menu.addSeparator(); + QAction* setTitle = menu.addAction(tr("Set title...")); QAction* quitGroup = menu.addAction(tr("Quit group","Menu to quit a groupchat")); @@ -78,6 +99,22 @@ void GroupWidget::contextMenuEvent(QContextMenuEvent* event) { emit removeGroup(groupId); } + else if (selectedItem == openChat) + { + emit chatroomWidgetClicked(this); + return; + } + else if (selectedItem == openChatWindow) + { + emit chatroomWidgetClicked(this, true); + return; + } + else if (selectedItem == removeChatWindow) + { + ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); + contentDialog->removeGroup(groupId); + return; + } else if (selectedItem == setTitle) { editName(); @@ -122,7 +159,7 @@ void GroupWidget::updateStatusLight() } } -QString GroupWidget::getStatusString() +QString GroupWidget::getStatusString() const { Group *g = GroupList::findGroup(groupId); @@ -137,10 +174,16 @@ void GroupWidget::editName() nameLabel->editBegin(); } -bool GroupWidget::chatFormIsSet() const +Group* GroupWidget::getGroup() const { + return GroupList::findGroup(groupId); +} + +bool GroupWidget::chatFormIsSet(bool focus) const +{ + (void)focus; Group* g = GroupList::findGroup(groupId); - return g->getChatForm()->isVisible(); + return ContentDialog::existsGroupWidget(groupId, focus) || g->getChatForm()->isVisible(); } void GroupWidget::setChatForm(ContentLayout* contentLayout) diff --git a/src/widget/groupwidget.h b/src/widget/groupwidget.h index 5f7610f32..59a98e058 100644 --- a/src/widget/groupwidget.h +++ b/src/widget/groupwidget.h @@ -30,10 +30,11 @@ public: virtual void setAsInactiveChatroom() final override; virtual void setAsActiveChatroom() final override; virtual void updateStatusLight() final override; - virtual bool chatFormIsSet() const final override; + virtual bool chatFormIsSet(bool focus) const final override; virtual void setChatForm(ContentLayout* contentLayout) override; virtual void resetEventFlags() final override; - virtual QString getStatusString() final override; + virtual QString getStatusString() const final override; + virtual Group* getGroup() const override; void setName(const QString& name); void onUserListChanged(); void editName(); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 7320ae17f..03a50d6c5 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -332,13 +332,12 @@ void Widget::init() contentLayout = nullptr; onSeparateWindowChanged(Settings::getInstance().getSeparateWindow(), false); - if (contentLayout != nullptr) - onAddClicked(); - ui->addButton->setCheckable(true); ui->transferButton->setCheckable(true); ui->settingsButton->setCheckable(true); - setActiveToolMenuButton(Widget::AddButton); + + if (contentLayout != nullptr) + onAddClicked(); //restore window state restoreGeometry(Settings::getInstance().getWindowGeometry()); @@ -568,6 +567,7 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) resize(ui->statusPanel->width(), height()); setWindowTitle(QString()); + setActiveToolMenuButton(None); } if (clicked) @@ -795,7 +795,7 @@ void Widget::addFriend(int friendId, const QString &userId) Core* core = Nexus::getCore(); connect(newfriend, &Friend::displayedNameChanged, this, &Widget::onFriendDisplayChanged); connect(settingsWidget, &SettingsWidget::compactToggled, newfriend->getFriendWidget(), &GenericChatroomWidget::compactChange); - connect(newfriend->getFriendWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), this, SLOT(onChatroomWidgetClicked(GenericChatroomWidget*))); + connect(newfriend->getFriendWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*,bool)), this, SLOT(onChatroomWidgetClicked(GenericChatroomWidget*,bool))); connect(newfriend->getFriendWidget(), SIGNAL(removeFriend(int)), this, SLOT(removeFriend(int))); connect(newfriend->getFriendWidget(), SIGNAL(copyFriendIdToClipboard(int)), this, SLOT(copyFriendIdToClipboard(int))); connect(newfriend->getFriendWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), newfriend->getChatForm(), SLOT(focusInput())); @@ -883,6 +883,8 @@ void Widget::onFriendStatusChanged(int friendId, Status status) setWindowTitle(windowTitle); } + ContentDialog::updateFriendStatus(friendId); + //won't print the message if there were no messages before if (!f->getChatForm()->isEmpty() && Settings::getInstance().getStatusChangeNotificationEnabled()) @@ -921,6 +923,8 @@ void Widget::onFriendStatusMessageChanged(int friendId, const QString& message) QString str = message; str.replace('\n', ' '); str.remove('\r'); str.remove(QChar((char)0)); // null terminator... f->setStatusMessage(str); + + ContentDialog::updateFriendStatusMessage(friendId, message); } void Widget::onFriendUsernameChanged(int friendId, const QString& username) @@ -948,39 +952,46 @@ void Widget::onFriendDisplayChanged(FriendWidget *friendWidget, Status s) } -void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget) +void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) { - if (Settings::getInstance().getSeparateWindow()) + widget->resetEventFlags(); + widget->updateStatusLight(); + + if (widget->chatFormIsSet(true)) + return; + + if (Settings::getInstance().getSeparateWindow() || group) { - if (!widget->chatFormIsSet()) + ContentDialog* dialog = nullptr; + + if (!Settings::getInstance().getDontGroupWindows() && !group) + dialog = ContentDialog::current(); + + if (dialog == nullptr) + dialog = new ContentDialog(); + + dialog->show(); + Friend* frnd = widget->getFriend(); + + if (frnd != nullptr) { - ContentDialog* dialog = nullptr; - - if (!Settings::getInstance().getDontGroupWindows()) - dialog = ContentDialog::current(); - - if (dialog == nullptr) - dialog = new ContentDialog(); - - dialog->show(); - Friend* frnd = widget->getFriend(); - if (frnd != nullptr) - { - dialog->addFriend(frnd->getFriendID(), frnd->getDisplayedName()); - } + addFriendDialog(frnd, dialog); } + else + { + Group* group = widget->getGroup(); + addGroupDialog(group, dialog); + } + + dialog->raise(); + dialog->activateWindow(); } else { hideMainForms(widget); widget->setChatForm(contentLayout); widget->setAsActiveChatroom(); - widget->resetEventFlags(); - widget->updateStatusLight(); - QString windowTitle = widget->getName(); - if (!widget->getStatusString().isNull()) - windowTitle += " (" + widget->getStatusString() + ")"; - setWindowTitle(windowTitle); + setWindowTitle(widget->getTitle()); } } @@ -996,9 +1007,11 @@ void Widget::onFriendMessageReceived(int friendId, const QString& message, bool HistoryKeeper::getInstance()->addChatEntry(f->getToxId().publicKey, isAction ? "/me " + f->getDisplayedName() + " " + message : message, f->getToxId().publicKey, timestamp, true); - f->setEventFlag(f->getFriendWidget() != activeChatroomWidget); - newMessageAlert(f->getFriendWidget()); + f->setEventFlag(f->getFriendWidget() != activeChatroomWidget && !ContentDialog::isFriendWidgetActive(friendId)); + newFriendMessageAlert(friendId); f->getFriendWidget()->updateStatusLight(); + ContentDialog::updateFriendStatus(friendId); + if (f->getFriendWidget()->isActive()) { QString windowTitle = f->getFriendWidget()->getName(); @@ -1017,25 +1030,91 @@ void Widget::onReceiptRecieved(int friendId, int receipt) f->getChatForm()->getOfflineMsgEngine()->dischargeReceipt(receipt); } -void Widget::newMessageAlert(GenericChatroomWidget* chat) +void Widget::addFriendDialog(Friend *frnd, ContentDialog *dialog) +{ + FriendWidget* friendWidget = dialog->addFriend(frnd->getFriendID(), frnd->getDisplayedName()); + + friendWidget->setStatusMsg(frnd->getFriendWidget()->getStatusMsg()); + + connect(friendWidget, SIGNAL(removeFriend(int)), this, SLOT(removeFriend(int))); + connect(friendWidget, SIGNAL(copyFriendIdToClipboard(int)), this, SLOT(copyFriendIdToClipboard(int))); + + connect(Core::getInstance(), &Core::friendAvatarChanged, friendWidget, &FriendWidget::onAvatarChange); + connect(Core::getInstance(), &Core::friendAvatarRemoved, friendWidget, &FriendWidget::onAvatarRemoved); + + QPixmap avatar = Settings::getInstance().getSavedAvatar(frnd->getToxId().toString()); + if (!avatar.isNull()) + friendWidget->onAvatarChange(frnd->getFriendID(), avatar); +} + +void Widget::addGroupDialog(Group *group, ContentDialog *dialog) +{ + GroupWidget* groupWidget = dialog->addGroup(group->getGroupId(), group->getName()); + connect(groupWidget, SIGNAL(removeGroup(int)), this, SLOT(removeGroup(int))); + connect(groupWidget, SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), group->getChatForm(), SLOT(focusInput())); +} + +void Widget::newFriendMessageAlert(int friendId) +{ + bool hasActive; + QWidget* currentWindow; + ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); + + if (contentDialog != nullptr) + { + currentWindow = contentDialog->window(); + hasActive = ContentDialog::isFriendWidgetActive(friendId); + } + else + { + currentWindow = window(); + hasActive = activeChatroomWidget != nullptr && FriendList::findFriend(friendId)->getFriendWidget() == activeChatroomWidget; + } + + newMessageAlert(currentWindow, hasActive); +} + +void Widget::newGroupMessageAlert(int groupId) +{ + bool hasActive; + QWidget* currentWindow; + ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); + + if (contentDialog != nullptr) + { + currentWindow = contentDialog->window(); + hasActive = ContentDialog::isGroupWidgetActive(groupId); + } + else + { + currentWindow = window(); + hasActive = activeChatroomWidget != nullptr && GroupList::findGroup(groupId)->getGroupWidget() == activeChatroomWidget; + } + + newMessageAlert(currentWindow, hasActive); +} + +void Widget::newMessageAlert(QWidget* currentWindow, bool isActive) { bool inactiveWindow = isMinimized() || !isActiveWindow(); - if (!inactiveWindow && activeChatroomWidget != nullptr && chat == activeChatroomWidget) + + if (!inactiveWindow && isActive) return; if (ui->statusButton->property("status").toString() == "busy") return; - QApplication::alert(this); + QApplication::alert(currentWindow); if (inactiveWindow) eventFlag = true; if (Settings::getInstance().getShowWindow()) { - show(); + currentWindow->activateWindow(); + currentWindow->show(); if (inactiveWindow && Settings::getInstance().getShowInFront()) - setWindowState(Qt::WindowActive); + currentWindow->setWindowState(Qt::WindowActive); } if (Settings::getInstance().getNotifySound()) @@ -1052,9 +1131,6 @@ void Widget::newMessageAlert(GenericChatroomWidget* chat) Audio::playMono16Sound(sndData); } - - if (activeChatroomWidget != chat) - ui->friendList->trackWidget(chat); } void Widget::playRingtone() @@ -1212,12 +1288,14 @@ void Widget::onGroupMessageReceived(int groupnumber, int peernumber, const QStri g->setEventFlag(static_cast(g->getGroupWidget()) != activeChatroomWidget); if (targeted || Settings::getInstance().getGroupAlwaysNotify()) - newMessageAlert(g->getGroupWidget()); + newGroupMessageAlert(g->getGroupId()); if (targeted) g->setMentionedFlag(true); // useful for highlighting line or desktop notifications g->getGroupWidget()->updateStatusLight(); + ContentDialog::updateGroupStatus(g->getGroupId()); + if (g->getGroupWidget()->isActive()) { QString windowTitle = g->getGroupWidget()->getName(); @@ -1325,7 +1403,7 @@ Group *Widget::createGroup(int groupId) newgroup->getGroupWidget()->updateStatusLight(); connect(settingsWidget, &SettingsWidget::compactToggled, newgroup->getGroupWidget(), &GenericChatroomWidget::compactChange); - connect(newgroup->getGroupWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), this, SLOT(onChatroomWidgetClicked(GenericChatroomWidget*))); + connect(newgroup->getGroupWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*,bool)), this, SLOT(onChatroomWidgetClicked(GenericChatroomWidget*,bool))); connect(newgroup->getGroupWidget(), SIGNAL(removeGroup(int)), this, SLOT(removeGroup(int))); connect(newgroup->getGroupWidget(), SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), newgroup->getChatForm(), SLOT(focusInput())); connect(newgroup->getChatForm(), &GroupChatForm::sendMessage, core, &Core::sendGroupMessage); diff --git a/src/widget/widget.h b/src/widget/widget.h index 1957fc482..9f797a25e 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -52,6 +52,7 @@ class AddFriendForm; class CircleWidget; class QActionGroup; class ContentLayout; +class ContentDialog; class Widget final : public QMainWindow { @@ -64,7 +65,10 @@ public: QString getUsername(); Camera* getCamera(); static Widget* getInstance(); - void newMessageAlert(GenericChatroomWidget* chat); + void addFriendDialog(Friend* frnd, ContentDialog* dialog); + void addGroupDialog(Group* group, ContentDialog* dialog); + void newFriendMessageAlert(int friendId); + void newGroupMessageAlert(int groupId); bool isFriendWidgetCurActiveWidget(const Friend* f) const; bool getIsWindowMinimized(); void updateIcons(); @@ -147,8 +151,8 @@ private slots: void onTransferClicked(); void showProfile(); void onUsernameChanged(const QString& newUsername, const QString& oldUsername); + void onChatroomWidgetClicked(GenericChatroomWidget *, bool group); void onStatusMessageChanged(const QString& newStatusMessage); - void onChatroomWidgetClicked(GenericChatroomWidget *); void removeFriend(int friendId); void copyFriendIdToClipboard(int friendId); void removeGroup(int groupId); @@ -188,6 +192,7 @@ private: }; private: + void newMessageAlert(QWidget* currentWindow, bool isActive); void setActiveToolMenuButton(ActiveToolMenuButton newActiveButton); void hideMainForms(GenericChatroomWidget* chatroomWidget); Group *createGroup(int groupId); From 7bcf654c72bf3019b96284922a0428c87558efeb Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 23 Jun 2015 12:54:00 -0400 Subject: [PATCH 04/24] Multi-window: Add drag/drop friends to windows, fix call confirm widget --- src/persistence/settings.cpp | 2 +- src/widget/contentdialog.cpp | 69 ++++++++++++++++++++++++--- src/widget/form/chatform.cpp | 21 ++------ src/widget/tool/callconfirmwidget.cpp | 39 +++++++++++---- src/widget/tool/callconfirmwidget.h | 8 ++-- src/widget/widget.cpp | 66 ++++++++++++------------- src/widget/widget.h | 7 ++- 7 files changed, 137 insertions(+), 75 deletions(-) diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index 2fe9fe5eb..9b645bd91 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -752,7 +752,7 @@ void Settings::setStatusChangeNotificationEnabled(bool newValue) bool Settings::getShowInFront() const { QMutexLocker locker{&bigLock}; - return showInFront; + return showInFront; } void Settings::setShowInFront(bool newValue) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index ac70759f1..20bf790a4 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -34,6 +34,39 @@ #include #include +#include +#include + +#include + +/*class ContentFriendList : public QScrollArea +{ +public: + ContentFriendList(QWidget* parent = 0) : QScrollArea(parent) + { + setAcceptDrops(true); + } + +protected: + + void dragEnterEvent(QDragEnterEvent* event) final override + { + qDebug() << "h"; + if (event->mimeData()->hasFormat("friend")) + event->acceptProposedAction(); + QScrollArea::dragEnterEvent(event); + } + + void dropEvent(QDropEvent *event) final override + { + if (event->mimeData()->hasFormat("friend")) + { + int friendId = ev->mimeData()->data("friend").toInt(); + //Core::getInstance()->groupInviteFriend(friendId, groupId); + } + } +};*/ + ContentDialog* ContentDialog::currentDialog = nullptr; QHash> ContentDialog::friendList; QHash> ContentDialog::groupList; @@ -88,6 +121,8 @@ ContentDialog::ContentDialog(QWidget* parent) splitter->restoreState(Settings::getInstance().getDialogSplitterState()); currentDialog = this; + + setAcceptDrops(true); } ContentDialog::~ContentDialog() @@ -225,20 +260,37 @@ ContentDialog* ContentDialog::getGroupDialog(int groupId) return getDialog(groupId, groupList); } -#include -#include void ContentDialog::dragEnterEvent(QDragEnterEvent *event) { if (event->mimeData()->hasFormat("friend")) - event->acceptProposedAction(); + { + int friendId = event->mimeData()->data("friend").toInt(); + + auto iter = friendList.find(friendId); + + // If friend is already in a dialog then you can't drop friend where it already is. + if (iter == friendList.end() || (iter != friendList.end() && std::get<0>(iter.value()) != this)) + event->acceptProposedAction(); + } } void ContentDialog::dropEvent(QDropEvent *event) { if (event->mimeData()->hasFormat("friend")) { - //int friendId = ev->mimeData()->data("friend").toInt(); - //Core::getInstance()->groupInviteFriend(friendId, groupId); + int friendId = event->mimeData()->data("friend").toInt(); + + auto iter = friendList.find(friendId); + + if (iter != friendList.end()) + std::get<0>(iter.value())->removeFriend(friendId); + + Friend* contact = FriendList::findFriend(friendId); + Widget::getInstance()->addFriendDialog(contact, this); + + // Display friend list after dropping, if not already visible. + if (splitter->sizes().at(0) == 0) + splitter->setSizes({1, 1}); } } @@ -283,6 +335,9 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool Widget::getInstance()->addGroupDialog(widget->getGroup(), contentDialog); } + contentDialog->raise(); + contentDialog->activateWindow(); + return; } @@ -341,7 +396,7 @@ void ContentDialog::remove(int id, const QHash 0) + if (chatroomWidgetCount() > 1) { int index = friendLayout->indexOf(chatroomWidget) - 1; @@ -402,7 +457,7 @@ void ContentDialog::updateStatus(int id, const QHash(iter.value()); chatroomWidget->updateStatusLight(); - if(chatroomWidget->isActive()) + if (chatroomWidget->isActive()) std::get<0>(iter.value())->setWindowTitle(chatroomWidget->getTitle()); } diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index bb244af87..2a11fb63c 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -216,13 +216,7 @@ void ChatForm::onFileRecvRequest(ToxFile file) if (file.friendId != f->getFriendID()) return; - Widget* w = Widget::getInstance(); - if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) - { - w->newFriendMessageAlert(file.friendId); - f->setEventFlag(true); - f->getFriendWidget()->updateStatusLight(); - } + Widget::getInstance()->newFriendMessageAlert(file.friendId); QString name; ToxId friendId = f->getToxId(); @@ -265,7 +259,7 @@ void ChatForm::onAvInvite(uint32_t FriendId, int CallId, bool video) if (video) { callConfirm = new CallConfirmWidget(videoButton, *f); - if (Widget::getInstance()->isFriendWidgetCurActiveWidget(f)) + if (f->getFriendWidget()->chatFormIsSet(false)) callConfirm->show(); connect(callConfirm, &CallConfirmWidget::accepted, this, &ChatForm::onAnswerCallTriggered); @@ -280,7 +274,7 @@ void ChatForm::onAvInvite(uint32_t FriendId, int CallId, bool video) else { callConfirm = new CallConfirmWidget(callButton, *f); - if (Widget::getInstance()->isFriendWidgetCurActiveWidget(f)) + if (f->getFriendWidget()->chatFormIsSet(false)) callConfirm->show(); connect(callConfirm, &CallConfirmWidget::accepted, this, &ChatForm::onAnswerCallTriggered); @@ -292,6 +286,7 @@ void ChatForm::onAvInvite(uint32_t FriendId, int CallId, bool video) videoButton->setToolTip(""); connect(callButton, &QPushButton::clicked, this, &ChatForm::onAnswerCallTriggered); } + callButton->style()->polish(callButton); videoButton->style()->polish(videoButton); @@ -299,13 +294,7 @@ void ChatForm::onAvInvite(uint32_t FriendId, int CallId, bool video) ChatMessage::INFO, QDateTime::currentDateTime())); - Widget* w = Widget::getInstance(); - if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) - { - w->newFriendMessageAlert(FriendId); - f->setEventFlag(true); - f->getFriendWidget()->updateStatusLight(); - } + Widget::getInstance()->newFriendMessageAlert(FriendId); } void ChatForm::onAvStart(uint32_t FriendId, int CallId, bool video) diff --git a/src/widget/tool/callconfirmwidget.cpp b/src/widget/tool/callconfirmwidget.cpp index 7a528fbc0..22f21903b 100644 --- a/src/widget/tool/callconfirmwidget.cpp +++ b/src/widget/tool/callconfirmwidget.cpp @@ -19,7 +19,6 @@ #include "callconfirmwidget.h" -#include "src/widget/gui.h" #include "src/widget/widget.h" #include #include @@ -33,7 +32,7 @@ #include CallConfirmWidget::CallConfirmWidget(const QWidget *Anchor, const Friend& f) : - QWidget(GUI::getMainWidget()), anchor(Anchor), f(f), + QWidget(), anchor(Anchor), f(f), rectW{120}, rectH{85}, spikeW{30}, spikeH{15}, roundedFactor{20}, @@ -47,6 +46,7 @@ CallConfirmWidget::CallConfirmWidget(const QWidget *Anchor, const Friend& f) : QVBoxLayout *layout = new QVBoxLayout(this); QLabel *callLabel = new QLabel(QObject::tr("Incoming call..."), this); + callLabel->setStyleSheet("color: white"); callLabel->setWordWrap(true); callLabel->setAlignment(Qt::AlignHCenter); QDialogButtonBox *buttonBox = new QDialogButtonBox(Qt::Horizontal, this); @@ -66,8 +66,6 @@ CallConfirmWidget::CallConfirmWidget(const QWidget *Anchor, const Friend& f) : connect(buttonBox, &QDialogButtonBox::accepted, this, &CallConfirmWidget::accepted); connect(buttonBox, &QDialogButtonBox::rejected, this, &CallConfirmWidget::rejected); - connect(&GUI::getInstance(), &GUI::resized, this, &CallConfirmWidget::reposition); - layout->setMargin(12); layout->addSpacing(spikeH); layout->addWidget(callLabel); @@ -76,10 +74,17 @@ CallConfirmWidget::CallConfirmWidget(const QWidget *Anchor, const Friend& f) : setFixedSize(rectW,rectH+spikeH); reposition(); } - +#include void CallConfirmWidget::reposition() { - QWidget* w = GUI::getMainWidget(); + if (parentWidget()) + parentWidget()->removeEventFilter(this); + + setParent(anchor->window()); + parentWidget()->installEventFilter(this); + + QWidget* w = anchor->window(); + qDebug() << anchor->window(); QPoint pos = anchor->mapToGlobal({(anchor->width()-rectW)/2,anchor->height()})-w->mapToGlobal({0,0}); // We don't want the widget to overflow past the right of the screen @@ -114,11 +119,27 @@ void CallConfirmWidget::showEvent(QShowEvent*) // If someone does show() from Widget or lower, the event will reach us // because it's our parent, and we could show up in the wrong form. // So check here if our friend's form is actually the active one. - if (!Widget::getInstance()->isFriendWidgetCurActiveWidget(&f)) + //if (!Widget::getInstance()->isFriendWidgetCurActiveWidget(&f)) { - QWidget::hide(); - return; + //QWidget::hide(); + //return; } reposition(); update(); } + +void CallConfirmWidget::hideEvent(QHideEvent *) +{ + if (parentWidget()) + parentWidget()->removeEventFilter(this); + + setParent(nullptr); +} + +bool CallConfirmWidget::eventFilter(QObject*, QEvent* event) +{ + if (event->type() == QEvent::Resize) + reposition(); + + return false; +} diff --git a/src/widget/tool/callconfirmwidget.h b/src/widget/tool/callconfirmwidget.h index e04eade0b..3c50fec9a 100644 --- a/src/widget/tool/callconfirmwidget.h +++ b/src/widget/tool/callconfirmwidget.h @@ -43,12 +43,14 @@ signals: void accepted(); void rejected(); +public slots: + void reposition(); ///< Recalculate our positions to track the anchor + protected: virtual void paintEvent(QPaintEvent* event) final override; virtual void showEvent(QShowEvent* event) final override; - -protected slots: - void reposition(); ///< Recalculate our positions to track the anchor + virtual void hideEvent(QHideEvent* event) final override; + virtual bool eventFilter(QObject *, QEvent* event) final override; private: const QWidget* anchor; ///< The widget we're going to be tracking diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 03a50d6c5..a776cfe61 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1007,18 +1007,7 @@ void Widget::onFriendMessageReceived(int friendId, const QString& message, bool HistoryKeeper::getInstance()->addChatEntry(f->getToxId().publicKey, isAction ? "/me " + f->getDisplayedName() + " " + message : message, f->getToxId().publicKey, timestamp, true); - f->setEventFlag(f->getFriendWidget() != activeChatroomWidget && !ContentDialog::isFriendWidgetActive(friendId)); newFriendMessageAlert(friendId); - f->getFriendWidget()->updateStatusLight(); - ContentDialog::updateFriendStatus(friendId); - - if (f->getFriendWidget()->isActive()) - { - QString windowTitle = f->getFriendWidget()->getName(); - if (!f->getFriendWidget()->getStatusString().isNull()) - windowTitle += " (" + f->getFriendWidget()->getStatusString() + ")"; - setWindowTitle(windowTitle); - } } void Widget::onReceiptRecieved(int friendId, int receipt) @@ -1054,11 +1043,12 @@ void Widget::addGroupDialog(Group *group, ContentDialog *dialog) connect(groupWidget, SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), group->getChatForm(), SLOT(focusInput())); } -void Widget::newFriendMessageAlert(int friendId) +bool Widget::newFriendMessageAlert(int friendId) { bool hasActive; QWidget* currentWindow; ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); + Friend* f = FriendList::findFriend(friendId); if (contentDialog != nullptr) { @@ -1068,13 +1058,31 @@ void Widget::newFriendMessageAlert(int friendId) else { currentWindow = window(); - hasActive = activeChatroomWidget != nullptr && FriendList::findFriend(friendId)->getFriendWidget() == activeChatroomWidget; + hasActive = f->getFriendWidget() == activeChatroomWidget; } - newMessageAlert(currentWindow, hasActive); + if (newMessageAlert(currentWindow, hasActive)) + { + f->setEventFlag(true); + f->getFriendWidget()->updateStatusLight(); + + if (contentDialog == nullptr) + { + if (hasActive) + setWindowTitle(f->getFriendWidget()->getTitle()); + } + else + { + ContentDialog::updateFriendStatus(friendId); + } + + return true; + } + + return false; } -void Widget::newGroupMessageAlert(int groupId) +bool Widget::newGroupMessageAlert(int groupId) { bool hasActive; QWidget* currentWindow; @@ -1088,33 +1096,29 @@ void Widget::newGroupMessageAlert(int groupId) else { currentWindow = window(); - hasActive = activeChatroomWidget != nullptr && GroupList::findGroup(groupId)->getGroupWidget() == activeChatroomWidget; + hasActive = GroupList::findGroup(groupId)->getGroupWidget() == activeChatroomWidget; } - newMessageAlert(currentWindow, hasActive); + return newMessageAlert(currentWindow, hasActive); } -void Widget::newMessageAlert(QWidget* currentWindow, bool isActive) +bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive) { bool inactiveWindow = isMinimized() || !isActiveWindow(); if (!inactiveWindow && isActive) - return; + return false; if (ui->statusButton->property("status").toString() == "busy") - return; + return false; QApplication::alert(currentWindow); - if (inactiveWindow) - eventFlag = true; - if (Settings::getInstance().getShowWindow()) { - currentWindow->activateWindow(); currentWindow->show(); if (inactiveWindow && Settings::getInstance().getShowInFront()) - currentWindow->setWindowState(Qt::WindowActive); + currentWindow->activateWindow(); } if (Settings::getInstance().getNotifySound()) @@ -1131,6 +1135,8 @@ void Widget::newMessageAlert(QWidget* currentWindow, bool isActive) Audio::playMono16Sound(sndData); } + + return true; } void Widget::playRingtone() @@ -1138,8 +1144,6 @@ void Widget::playRingtone() if (ui->statusButton->property("status").toString() == "busy") return; - QApplication::alert(this); - // for whatever reason this plays slower/downshifted from what any other program plays the file as... but whatever static QFile sndFile1(":audio/ToxicIncomingCall.pcm"); static QByteArray sndData1; @@ -1425,14 +1429,6 @@ void Widget::onEmptyGroupCreated(int groupId) group->getGroupWidget()->editName(); } -bool Widget::isFriendWidgetCurActiveWidget(const Friend* f) const -{ - if (!f) - return false; - - return (activeChatroomWidget == static_cast(f->getFriendWidget())); -} - bool Widget::event(QEvent * e) { switch (e->type()) diff --git a/src/widget/widget.h b/src/widget/widget.h index 9f797a25e..a8f537cdd 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -67,9 +67,8 @@ public: static Widget* getInstance(); void addFriendDialog(Friend* frnd, ContentDialog* dialog); void addGroupDialog(Group* group, ContentDialog* dialog); - void newFriendMessageAlert(int friendId); - void newGroupMessageAlert(int groupId); - bool isFriendWidgetCurActiveWidget(const Friend* f) const; + bool newFriendMessageAlert(int friendId); + bool newGroupMessageAlert(int groupId); bool getIsWindowMinimized(); void updateIcons(); void clearContactsList(); @@ -192,7 +191,7 @@ private: }; private: - void newMessageAlert(QWidget* currentWindow, bool isActive); + bool newMessageAlert(QWidget* currentWindow, bool isActive); void setActiveToolMenuButton(ActiveToolMenuButton newActiveButton); void hideMainForms(GenericChatroomWidget* chatroomWidget); Group *createGroup(int groupId); From 9867e3b4c57846dfa4d545c90079525da526a105 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Fri, 26 Jun 2015 14:43:08 -0400 Subject: [PATCH 05/24] Multi-window: Allow opening new window when already open, fix drop when already open --- src/widget/contentdialog.cpp | 28 ---------------------------- src/widget/widget.cpp | 6 ++++-- 2 files changed, 4 insertions(+), 30 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 20bf790a4..4072731b9 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -39,34 +39,6 @@ #include -/*class ContentFriendList : public QScrollArea -{ -public: - ContentFriendList(QWidget* parent = 0) : QScrollArea(parent) - { - setAcceptDrops(true); - } - -protected: - - void dragEnterEvent(QDragEnterEvent* event) final override - { - qDebug() << "h"; - if (event->mimeData()->hasFormat("friend")) - event->acceptProposedAction(); - QScrollArea::dragEnterEvent(event); - } - - void dropEvent(QDropEvent *event) final override - { - if (event->mimeData()->hasFormat("friend")) - { - int friendId = ev->mimeData()->data("friend").toInt(); - //Core::getInstance()->groupInviteFriend(friendId, groupId); - } - } -};*/ - ContentDialog* ContentDialog::currentDialog = nullptr; QHash> ContentDialog::friendList; QHash> ContentDialog::groupList; diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index a776cfe61..cf8446320 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -537,7 +537,6 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) for (QWindow* window : windowList) { - qDebug() << window->objectName(); if (window->objectName() == "detachedWindow") window->close(); } @@ -957,7 +956,7 @@ void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) widget->resetEventFlags(); widget->updateStatusLight(); - if (widget->chatFormIsSet(true)) + if (widget->chatFormIsSet(true) && !group) return; if (Settings::getInstance().getSeparateWindow() || group) @@ -1021,6 +1020,9 @@ void Widget::onReceiptRecieved(int friendId, int receipt) void Widget::addFriendDialog(Friend *frnd, ContentDialog *dialog) { + if (!ContentDialog::getFriendDialog(frnd->getFriendID()) && !Settings::getInstance().getSeparateWindow() && activeChatroomWidget == frnd->getFriendWidget()) + onAddClicked(); + FriendWidget* friendWidget = dialog->addFriend(frnd->getFriendID(), frnd->getDisplayedName()); friendWidget->setStatusMsg(frnd->getFriendWidget()->getStatusMsg()); From 730626fc2cce0cd1b2b6845f089d3bca3d986cc9 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Fri, 26 Jun 2015 16:13:20 -0400 Subject: [PATCH 06/24] Multi-window: Sort window chat list, fix group window bugs --- src/widget/contentdialog.cpp | 123 +++++++++++++++++++++++++---- src/widget/contentdialog.h | 5 +- src/widget/friendwidget.cpp | 15 ++-- src/widget/friendwidget.h | 1 - src/widget/genericchatroomwidget.h | 2 + src/widget/groupwidget.cpp | 43 +++++++--- src/widget/groupwidget.h | 2 + 7 files changed, 157 insertions(+), 34 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 4072731b9..9daee6ed4 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -28,17 +28,17 @@ #include "src/friend.h" #include "src/friendlist.h" #include "src/group.h" +#include "src/grouplist.h" #include "src/widget/form/chatform.h" #include "src/core/core.h" +#include "src/widget/genericchatitemlayout.h" +#include "src/widget/friendlistlayout.h" #include #include #include - #include #include -#include - ContentDialog* ContentDialog::currentDialog = nullptr; QHash> ContentDialog::friendList; QHash> ContentDialog::groupList; @@ -59,10 +59,13 @@ ContentDialog::ContentDialog(QWidget* parent) friendWidget->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed); friendWidget->setAutoFillBackground(true); - friendLayout = new QVBoxLayout(friendWidget); + friendLayout = new FriendListLayout(); friendLayout->setMargin(0); friendLayout->setSpacing(0); - friendLayout->addStretch(); + friendWidget->setLayout(friendLayout); + + groupLayout = new GenericChatItemLayout(); + friendLayout->addLayout(groupLayout->getLayout()); QScrollArea *friendScroll = new QScrollArea(this); friendScroll->setMinimumWidth(220); @@ -113,12 +116,24 @@ ContentDialog::~ContentDialog() } ++friendIt; } + + auto groupIt = groupList.begin(); + + while (groupIt != groupList.end()) + { + if (std::get<0>(groupIt.value()) == this) + { + groupIt = groupList.erase(groupIt); + continue; + } + ++groupIt; + } } FriendWidget* ContentDialog::addFriend(int friendId, QString id) { FriendWidget* friendWidget = new FriendWidget(friendId, id); - friendLayout->insertWidget(friendLayout->count() - 1, friendWidget); + friendLayout->addFriendWidget(friendWidget, FriendList::findFriend(friendId)->getStatus()); Friend* frnd = friendWidget->getFriend(); @@ -138,7 +153,7 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) GroupWidget* ContentDialog::addGroup(int groupId, const QString& name) { GroupWidget* groupWidget = new GroupWidget(groupId, name); - friendLayout->insertWidget(friendLayout->count() - 1, groupWidget); + groupLayout->addSortedWidget(groupWidget); Group* group = groupWidget->getGroup(); connect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); @@ -154,7 +169,66 @@ GroupWidget* ContentDialog::addGroup(int groupId, const QString& name) void ContentDialog::removeFriend(int friendId) { - remove(friendId, friendList); + auto iter = friendList.find(friendId); + + if (iter == friendList.end()) + return; + + FriendWidget* chatroomWidget = static_cast(std::get<1>(iter.value())); + + if (activeChatroomWidget == chatroomWidget) + { + // Need to find replacement to show here instead. + if (chatroomWidgetCount() > 1) + { + bool inOnline = true; + int index = friendLayout->getLayoutOnline()->indexOf(chatroomWidget); + + if (index == -1) + { + inOnline = false; + index = friendLayout->getLayoutOffline()->indexOf(chatroomWidget); + } + + // Don't let it go below 0. If it does, then we're first. Go second. + if (index < 0) + index = 1; + + QLayout* fromLayout; + + if (inOnline) + { + if (friendLayout->getLayoutOnline()->count() > 1) + fromLayout = friendLayout->getLayoutOnline(); + else + fromLayout = friendLayout->getLayoutOffline(); + } + else + { + if (friendLayout->getLayoutOffline()->count() > 1) + fromLayout = friendLayout->getLayoutOffline(); + else + fromLayout = friendLayout->getLayoutOnline(); + } + + GenericChatroomWidget* chatroomWidget = static_cast(fromLayout->itemAt(index)->widget()); + + onChatroomWidgetClicked(chatroomWidget, false); + } + else + { + contentLayout->clear(); + activeChatroomWidget = nullptr; + deleteLater(); + } + } + + friendLayout->removeFriendWidget(chatroomWidget, Status::Offline); + friendLayout->removeFriendWidget(chatroomWidget, Status::Online); + + chatroomWidget->deleteLater(); + friendList.remove(friendId); + update(); } void ContentDialog::removeGroup(int groupId) @@ -174,7 +248,7 @@ bool ContentDialog::hasGroupWidget(int groupId, GenericChatroomWidget *chatroomW int ContentDialog::chatroomWidgetCount() const { - return friendLayout->count() - 1; // Don't include the stretch. + return friendLayout->friendTotalCount() + groupLayout->getLayout()->count(); } ContentDialog* ContentDialog::current() @@ -237,13 +311,20 @@ void ContentDialog::dragEnterEvent(QDragEnterEvent *event) if (event->mimeData()->hasFormat("friend")) { int friendId = event->mimeData()->data("friend").toInt(); - auto iter = friendList.find(friendId); // If friend is already in a dialog then you can't drop friend where it already is. if (iter == friendList.end() || (iter != friendList.end() && std::get<0>(iter.value()) != this)) event->acceptProposedAction(); } + else if (event->mimeData()->hasFormat("group")) + { + int groupId = event->mimeData()->data("group").toInt(); + auto iter = groupList.find(groupId); + + if (iter == groupList.end() || (iter != groupList.end() && std::get<0>(iter.value()) != this)) + event->acceptProposedAction(); + } } void ContentDialog::dropEvent(QDropEvent *event) @@ -251,7 +332,6 @@ void ContentDialog::dropEvent(QDropEvent *event) if (event->mimeData()->hasFormat("friend")) { int friendId = event->mimeData()->data("friend").toInt(); - auto iter = friendList.find(friendId); if (iter != friendList.end()) @@ -260,6 +340,21 @@ void ContentDialog::dropEvent(QDropEvent *event) Friend* contact = FriendList::findFriend(friendId); Widget::getInstance()->addFriendDialog(contact, this); + // Display friend list after dropping, if not already visible. + if (splitter->sizes().at(0) == 0) + splitter->setSizes({1, 1}); + } + else if (event->mimeData()->hasFormat("group")) + { + int groupId = event->mimeData()->data("group").toInt(); + auto iter = friendList.find(groupId); + + if (iter != friendList.end()) + std::get<0>(iter.value())->removeGroup(groupId); + + Group* contact = GroupList::findGroup(groupId); + Widget::getInstance()->addGroupDialog(contact, this); + // Display friend list after dropping, if not already visible. if (splitter->sizes().at(0) == 0) splitter->setSizes({1, 1}); @@ -370,13 +465,13 @@ void ContentDialog::remove(int id, const QHash 1) { - int index = friendLayout->indexOf(chatroomWidget) - 1; + int index = groupLayout->indexOfSortedWidget(chatroomWidget) - 1; // Don't let it go below 0. If it does, then we're first. Go second. if (index < 0) index = 1; - GenericChatroomWidget* chatroomWidget = static_cast(friendLayout->itemAt(index)->widget()); + GenericChatroomWidget* chatroomWidget = static_cast(groupLayout->getLayout()->itemAt(index)->widget()); onChatroomWidgetClicked(chatroomWidget, false); } else @@ -387,7 +482,7 @@ void ContentDialog::remove(int id, const QHashremoveWidget(chatroomWidget); + groupLayout->removeSortedWidget(chatroomWidget); chatroomWidget->deleteLater(); friendList.remove(id); update(); diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index f242830ed..3c6cd162f 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -32,6 +32,8 @@ class ContentLayout; class GenericChatroomWidget; class FriendWidget; class GroupWidget; +class FriendListLayout; +class GenericChatItemLayout; class ContentDialog : public QDialog { @@ -83,7 +85,8 @@ private: static ContentDialog* getDialog(int id, const QHash>& list); QSplitter* splitter; - QVBoxLayout* friendLayout; + FriendListLayout* friendLayout; + GenericChatItemLayout* groupLayout; ContentLayout* contentLayout; GenericChatroomWidget* activeChatroomWidget; static ContentDialog* currentDialog; diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index 750c6d2a9..f7ec9515d 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -72,17 +72,14 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) QAction* openChatWindow = nullptr; QAction* removeChatWindow = nullptr; - if (!Settings::getInstance().getSeparateWindow() || !Settings::getInstance().getDontGroupWindows()) - { - ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); - bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; + ContentDialog* contentDialog = ContentDialog::getFriendDialog(friendId); + bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; - if (contentDialog == nullptr || notAlone) - openChatWindow = menu.addAction(tr("Open chat in new window")); + if (contentDialog == nullptr || notAlone) + openChatWindow = menu.addAction(tr("Open chat in new window")); - if (notAlone && contentDialog->hasFriendWidget(friendId, this)) - removeChatWindow = menu.addAction(tr("Remove chat from this window")); - } + if (contentDialog->hasFriendWidget(friendId, this)) + removeChatWindow = menu.addAction(tr("Remove chat from this window")); menu.addSeparator(); QMenu* inviteMenu = menu.addMenu(tr("Invite to group","Menu to invite a friend to a groupchat")); diff --git a/src/widget/friendwidget.h b/src/widget/friendwidget.h index 351f10ab0..9c90bf4cd 100644 --- a/src/widget/friendwidget.h +++ b/src/widget/friendwidget.h @@ -58,7 +58,6 @@ public: int friendId; bool isDefaultAvatar; bool historyLoaded; - QPoint dragStartPos; }; #endif // FRIENDWIDGET_H diff --git a/src/widget/genericchatroomwidget.h b/src/widget/genericchatroomwidget.h index 8132e5a9b..f2670c7e1 100644 --- a/src/widget/genericchatroomwidget.h +++ b/src/widget/genericchatroomwidget.h @@ -69,6 +69,8 @@ protected: virtual void enterEvent(QEvent* e) override; virtual void leaveEvent(QEvent* e) override; + QPoint dragStartPos; + protected: QColor lastColor; QHBoxLayout* mainLayout = nullptr; diff --git a/src/widget/groupwidget.cpp b/src/widget/groupwidget.cpp index adb420b52..7957340a1 100644 --- a/src/widget/groupwidget.cpp +++ b/src/widget/groupwidget.cpp @@ -32,6 +32,8 @@ #include #include #include +#include +#include GroupWidget::GroupWidget(int GroupId, QString Name) : groupId{GroupId} @@ -69,17 +71,14 @@ void GroupWidget::contextMenuEvent(QContextMenuEvent* event) QAction* openChatWindow = nullptr; QAction* removeChatWindow = nullptr; - if (!Settings::getInstance().getSeparateWindow() || !Settings::getInstance().getDontGroupWindows()) - { - ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); - bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; + ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); + bool notAlone = contentDialog != nullptr && contentDialog->chatroomWidgetCount() > 1; - if (contentDialog == nullptr || notAlone) - openChatWindow = menu.addAction(tr("Open chat in new window")); + if (contentDialog == nullptr || notAlone) + openChatWindow = menu.addAction(tr("Open chat in new window")); - if (notAlone && contentDialog->hasGroupWidget(groupId, this)) - removeChatWindow = menu.addAction(tr("Remove chat from this window")); - } + if (contentDialog->hasGroupWidget(groupId, this)) + removeChatWindow = menu.addAction(tr("Remove chat from this window")); menu.addSeparator(); @@ -122,6 +121,32 @@ void GroupWidget::contextMenuEvent(QContextMenuEvent* event) } } +void GroupWidget::mousePressEvent(QMouseEvent *ev) +{ + if (ev->button() == Qt::LeftButton) + dragStartPos = ev->pos(); + + GenericChatroomWidget::mousePressEvent(ev); +} + +void GroupWidget::mouseMoveEvent(QMouseEvent *ev) +{ + if (!(ev->buttons() & Qt::LeftButton)) + return; + + if ((dragStartPos - ev->pos()).manhattanLength() > QApplication::startDragDistance()) + { + QDrag* drag = new QDrag(this); + QMimeData* mdata = new QMimeData; + mdata->setData("group", QString::number(groupId).toLatin1()); + + drag->setMimeData(mdata); + drag->setPixmap(avatar->getPixmap()); + + drag->exec(Qt::CopyAction | Qt::MoveAction); + } +} + void GroupWidget::onUserListChanged() { Group* g = GroupList::findGroup(groupId); diff --git a/src/widget/groupwidget.h b/src/widget/groupwidget.h index 59a98e058..4e70ae6df 100644 --- a/src/widget/groupwidget.h +++ b/src/widget/groupwidget.h @@ -46,6 +46,8 @@ signals: protected: virtual void contextMenuEvent(QContextMenuEvent * event) final override; + virtual void mousePressEvent(QMouseEvent* event) final override; + virtual void mouseMoveEvent(QMouseEvent* event) final override; virtual void dragEnterEvent(QDragEnterEvent* ev) override; virtual void dragLeaveEvent(QDragLeaveEvent* ev); virtual void dropEvent(QDropEvent* ev) override; From c181df63364d8d184517613371d581697120a2af Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 29 Jun 2015 12:26:49 -0400 Subject: [PATCH 07/24] Multi-window: Cycle widgets in opened windows --- src/widget/contentdialog.cpp | 217 ++++++++++++++++++++------ src/widget/contentdialog.h | 16 +- src/widget/friendlistlayout.cpp | 2 +- src/widget/friendlistlayout.h | 2 +- src/widget/friendlistwidget.cpp | 11 -- src/widget/friendlistwidget.h | 1 - src/widget/tool/callconfirmwidget.cpp | 3 +- src/widget/widget.cpp | 2 +- 8 files changed, 187 insertions(+), 67 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 9daee6ed4..f986924b7 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -31,21 +31,23 @@ #include "src/grouplist.h" #include "src/widget/form/chatform.h" #include "src/core/core.h" -#include "src/widget/genericchatitemlayout.h" #include "src/widget/friendlistlayout.h" +#include "src/widget/form/settingswidget.h" #include #include #include #include #include +#include ContentDialog* ContentDialog::currentDialog = nullptr; QHash> ContentDialog::friendList; QHash> ContentDialog::groupList; -ContentDialog::ContentDialog(QWidget* parent) +ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent) : QDialog(parent, Qt::Window) , activeChatroomWidget(nullptr) + , settingsWidget(settingsWidget) { QVBoxLayout* boxLayout = new QVBoxLayout(this); boxLayout->setMargin(0); @@ -64,8 +66,7 @@ ContentDialog::ContentDialog(QWidget* parent) friendLayout->setSpacing(0); friendWidget->setLayout(friendLayout); - groupLayout = new GenericChatItemLayout(); - friendLayout->addLayout(groupLayout->getLayout()); + onGroupchatPositionChanged(Settings::getInstance().getGroupchatPosition()); QScrollArea *friendScroll = new QScrollArea(this); friendScroll->setMinimumWidth(220); @@ -88,6 +89,8 @@ ContentDialog::ContentDialog(QWidget* parent) splitter->setCollapsible(1, false); boxLayout->addWidget(splitter); + connect(settingsWidget, &SettingsWidget::groupchatPositionToggled, this, &ContentDialog::onGroupchatPositionChanged); + setMinimumSize(775, 420); setAttribute(Qt::WA_DeleteOnClose); @@ -98,6 +101,12 @@ ContentDialog::ContentDialog(QWidget* parent) currentDialog = this; setAcceptDrops(true); + + new QShortcut(Qt::CTRL + Qt::Key_Q, this, SLOT(close())); + new QShortcut(Qt::CTRL + Qt::SHIFT + Qt::Key_Tab, this, SLOT(previousContact())); + new QShortcut(Qt::CTRL + Qt::Key_Tab, this, SLOT(nextContact())); + new QShortcut(Qt::CTRL + Qt::Key_PageUp, this, SLOT(previousContact())); + new QShortcut(Qt::CTRL + Qt::Key_PageDown, this, SLOT(nextContact())); } ContentDialog::~ContentDialog() @@ -140,6 +149,7 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) onChatroomWidgetClicked(friendWidget, false); connect(frnd, &Friend::displayedNameChanged, this, &ContentDialog::updateFriendWidget); + connect(settingsWidget, &SettingsWidget::compactToggled, friendWidget, &FriendWidget::compactChange); connect(friendWidget, &FriendWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); connect(friendWidget, SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), frnd->getChatForm(), SLOT(focusInput())); connect(Core::getInstance(), &Core::friendAvatarChanged, friendWidget, &FriendWidget::onAvatarChange); @@ -153,11 +163,12 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) GroupWidget* ContentDialog::addGroup(int groupId, const QString& name) { GroupWidget* groupWidget = new GroupWidget(groupId, name); - groupLayout->addSortedWidget(groupWidget); + groupLayout.addSortedWidget(groupWidget); Group* group = groupWidget->getGroup(); connect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); connect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); + connect(settingsWidget, &SettingsWidget::compactToggled, groupWidget, &GroupWidget::compactChange); connect(groupWidget, &GroupWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); onChatroomWidgetClicked(groupWidget, false); @@ -175,45 +186,14 @@ void ContentDialog::removeFriend(int friendId) return; FriendWidget* chatroomWidget = static_cast(std::get<1>(iter.value())); + disconnect(chatroomWidget->getFriend(), &Friend::displayedNameChanged, this, &ContentDialog::updateFriendWidget); if (activeChatroomWidget == chatroomWidget) { // Need to find replacement to show here instead. if (chatroomWidgetCount() > 1) { - bool inOnline = true; - int index = friendLayout->getLayoutOnline()->indexOf(chatroomWidget); - - if (index == -1) - { - inOnline = false; - index = friendLayout->getLayoutOffline()->indexOf(chatroomWidget); - } - - // Don't let it go below 0. If it does, then we're first. Go second. - if (index < 0) - index = 1; - - QLayout* fromLayout; - - if (inOnline) - { - if (friendLayout->getLayoutOnline()->count() > 1) - fromLayout = friendLayout->getLayoutOnline(); - else - fromLayout = friendLayout->getLayoutOffline(); - } - else - { - if (friendLayout->getLayoutOffline()->count() > 1) - fromLayout = friendLayout->getLayoutOffline(); - else - fromLayout = friendLayout->getLayoutOnline(); - } - - GenericChatroomWidget* chatroomWidget = static_cast(fromLayout->itemAt(index)->widget()); - - onChatroomWidgetClicked(chatroomWidget, false); + cycleContacts(true, false); } else { @@ -233,6 +213,10 @@ void ContentDialog::removeFriend(int friendId) void ContentDialog::removeGroup(int groupId) { + Group* group = GroupList::findGroup(groupId); + disconnect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); + disconnect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); + remove(groupId, groupList); } @@ -248,7 +232,65 @@ bool ContentDialog::hasGroupWidget(int groupId, GenericChatroomWidget *chatroomW int ContentDialog::chatroomWidgetCount() const { - return friendLayout->friendTotalCount() + groupLayout->getLayout()->count(); + return friendLayout->friendTotalCount() + groupLayout.getLayout()->count(); +} + +void ContentDialog::cycleContacts(bool forward, bool loop) +{ + Settings::getInstance().getGroupchatPosition(); + + int index; + QLayout* currentLayout; + if (activeChatroomWidget->getFriend()) + { + currentLayout = friendLayout->getLayoutOnline(); + index = friendLayout->indexOfFriendWidget(activeChatroomWidget, true); + if (index == -1) + { + currentLayout = friendLayout->getLayoutOffline(); + index = friendLayout->indexOfFriendWidget(activeChatroomWidget, false); + } + } + else + { + currentLayout = groupLayout.getLayout(); + index = groupLayout.indexOfSortedWidget(activeChatroomWidget); + } + + index += forward ? 1 : -1; + + for (;;) + { + // Bounds checking. + if (index < 0) + { + currentLayout = nextLayout(currentLayout, forward); + index = currentLayout->count() - 1; + continue; + } + else if (index >= currentLayout->count()) + { + if (!loop && currentLayout == friendLayout->getLayoutOffline()) + { + forward = !forward; // Go backward. + index += forward ? 2 : -2; // Go back to where started and then one. + continue; // Recheck bounds. + } + else + { + currentLayout = nextLayout(currentLayout, forward); + index = 0; + } + continue; + } + + GenericChatroomWidget* chatWidget = dynamic_cast(currentLayout->itemAt(index)->widget()); + + if (chatWidget != nullptr) + emit chatWidget->chatroomWidgetClicked(chatWidget); + + return; + } } ContentDialog* ContentDialog::current() @@ -269,6 +311,12 @@ bool ContentDialog::existsGroupWidget(int groupId, bool focus) void ContentDialog::updateFriendStatus(int friendId) { updateStatus(friendId, friendList); + ContentDialog* contentDialog = getFriendDialog(friendId); + if (contentDialog != nullptr) + { + FriendWidget* friendWidget = static_cast(std::get<1>(friendList.find(friendId).value())); + contentDialog->friendLayout->addFriendWidget(friendWidget, FriendList::findFriend(friendId)->getStatus()); + } } void ContentDialog::updateFriendStatusMessage(int friendId, const QString &message) @@ -306,6 +354,16 @@ ContentDialog* ContentDialog::getGroupDialog(int groupId) return getDialog(groupId, groupList); } +void ContentDialog::previousContact() +{ + cycleContacts(false); +} + +void ContentDialog::nextContact() +{ + cycleContacts(true); +} + void ContentDialog::dragEnterEvent(QDragEnterEvent *event) { if (event->mimeData()->hasFormat("friend")) @@ -388,7 +446,7 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool { if (group) { - ContentDialog* contentDialog = new ContentDialog(); + ContentDialog* contentDialog = new ContentDialog(settingsWidget); contentDialog->show(); if (widget->getFriend() != nullptr) @@ -429,10 +487,9 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool void ContentDialog::updateFriendWidget(FriendWidget *w, Status s) { - // Note: This argument friend widget is not the one stored here! - Q_UNUSED(s); // Use it after friend list improvements are merged, to sort contacts here. - - std::get<1>(friendList.find(w->friendId).value())->setName(w->getName()); + FriendWidget* friendWidget = static_cast(std::get<1>(friendList.find(w->friendId).value())); + friendWidget->setName(w->getName()); + friendLayout->addFriendWidget(friendWidget, s); } void ContentDialog::updateGroupWidget(GroupWidget *w) @@ -441,6 +498,16 @@ void ContentDialog::updateGroupWidget(GroupWidget *w) static_cast(std::get<1>(groupList.find(w->groupId).value()))->onUserListChanged(); } +void ContentDialog::onGroupchatPositionChanged(bool top) +{ + friendLayout->removeItem(groupLayout.getLayout()); + + if (top) + friendLayout->insertLayout(0, groupLayout.getLayout()); + else + friendLayout->insertLayout(1, groupLayout.getLayout()); +} + void ContentDialog::saveDialogGeometry() { Settings::getInstance().setDialogGeometry(saveGeometry()); @@ -465,13 +532,13 @@ void ContentDialog::remove(int id, const QHash 1) { - int index = groupLayout->indexOfSortedWidget(chatroomWidget) - 1; + int index = groupLayout.indexOfSortedWidget(chatroomWidget) - 1; // Don't let it go below 0. If it does, then we're first. Go second. if (index < 0) index = 1; - GenericChatroomWidget* chatroomWidget = static_cast(groupLayout->getLayout()->itemAt(index)->widget()); + GenericChatroomWidget* chatroomWidget = static_cast(groupLayout.getLayout()->itemAt(index)->widget()); onChatroomWidgetClicked(chatroomWidget, false); } else @@ -482,7 +549,7 @@ void ContentDialog::remove(int id, const QHashremoveSortedWidget(chatroomWidget); + groupLayout.removeSortedWidget(chatroomWidget); chatroomWidget->deleteLater(); friendList.remove(id); update(); @@ -547,3 +614,59 @@ ContentDialog* ContentDialog::getDialog(int id, const QHash(iter.value()); } + +QLayout* ContentDialog::nextLayout(QLayout* layout, bool forward) const +{ + if (layout == groupLayout.getLayout()) + { + if (forward) + { + if (Settings::getInstance().getGroupchatPosition()) + return friendLayout->getLayoutOnline(); + + return friendLayout->getLayoutOffline(); + } + else + { + if (Settings::getInstance().getGroupchatPosition()) + return friendLayout->getLayoutOffline(); + + return friendLayout->getLayoutOnline(); + } + } + else if (layout == friendLayout->getLayoutOnline()) + { + if (forward) + { + if (Settings::getInstance().getGroupchatPosition()) + return friendLayout->getLayoutOffline(); + + return groupLayout.getLayout(); + } + else + { + if (Settings::getInstance().getGroupchatPosition()) + return groupLayout.getLayout(); + + return friendLayout->getLayoutOffline(); + } + } + else if (layout == friendLayout->getLayoutOffline()) + { + if (forward) + { + if (Settings::getInstance().getGroupchatPosition()) + return groupLayout.getLayout(); + + return friendLayout->getLayoutOnline(); + } + else + { + if (Settings::getInstance().getGroupchatPosition()) + return friendLayout->getLayoutOnline(); + + return groupLayout.getLayout(); + } + } + return nullptr; +} diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index 3c6cd162f..d05bb4b03 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -23,6 +23,7 @@ #include #include #include "src/core/corestructs.h" +#include "src/widget/genericchatitemlayout.h" template class QHash; @@ -33,13 +34,13 @@ class GenericChatroomWidget; class FriendWidget; class GroupWidget; class FriendListLayout; -class GenericChatItemLayout; +class SettingsWidget; class ContentDialog : public QDialog { Q_OBJECT public: - ContentDialog(QWidget* parent = 0); + ContentDialog(SettingsWidget* settingsWidget, QWidget* parent = 0); ~ContentDialog(); FriendWidget* addFriend(int friendId, QString id); @@ -50,6 +51,8 @@ public: bool hasGroupWidget(int groupId, GenericChatroomWidget* chatroomWidget); int chatroomWidgetCount() const; + void cycleContacts(bool forward, bool loop = true); + static ContentDialog* current(); static bool existsFriendWidget(int friendId, bool focus); static bool existsGroupWidget(int groupId, bool focus); @@ -61,6 +64,10 @@ public: static ContentDialog* getFriendDialog(int friendId); static ContentDialog* getGroupDialog(int groupId); +public slots: + void previousContact(); + void nextContact(); + protected: void dragEnterEvent(QDragEnterEvent* event) final override; void dropEvent(QDropEvent* event) final override; @@ -72,10 +79,12 @@ private slots: void onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group); void updateFriendWidget(FriendWidget* w, Status s); void updateGroupWidget(GroupWidget* w); + void onGroupchatPositionChanged(bool top); private: void saveDialogGeometry(); void saveSplitterState(); + QLayout* nextLayout(QLayout* layout, bool forward) const; void remove(int id, const QHash>& list); bool hasWidget(int id, GenericChatroomWidget* chatroomWidget, const QHash>& list); @@ -86,9 +95,10 @@ private: QSplitter* splitter; FriendListLayout* friendLayout; - GenericChatItemLayout* groupLayout; + GenericChatItemLayout groupLayout; ContentLayout* contentLayout; GenericChatroomWidget* activeChatroomWidget; + SettingsWidget* settingsWidget; static ContentDialog* currentDialog; static QHash> friendList; static QHash> groupList; diff --git a/src/widget/friendlistlayout.cpp b/src/widget/friendlistlayout.cpp index cbe099e25..5b00d71c1 100644 --- a/src/widget/friendlistlayout.cpp +++ b/src/widget/friendlistlayout.cpp @@ -62,7 +62,7 @@ void FriendListLayout::removeFriendWidget(FriendWidget *widget, Status s) friendOnlineLayout.removeSortedWidget(widget); } -int FriendListLayout::indexOfFriendWidget(FriendWidget* widget, bool online) const +int FriendListLayout::indexOfFriendWidget(GenericChatItemWidget* widget, bool online) const { if (online) return friendOnlineLayout.indexOfSortedWidget(widget); diff --git a/src/widget/friendlistlayout.h b/src/widget/friendlistlayout.h index aef147b11..a0e9d6025 100644 --- a/src/widget/friendlistlayout.h +++ b/src/widget/friendlistlayout.h @@ -31,7 +31,7 @@ public: void addFriendWidget(FriendWidget* widget, Status s); void removeFriendWidget(FriendWidget* widget, Status s); - int indexOfFriendWidget(FriendWidget* widget, bool online) const; + int indexOfFriendWidget(GenericChatItemWidget* widget, bool online) const; void moveFriendWidgets(FriendListWidget* listWidget); int friendOnlineCount() const; int friendTotalCount() const; diff --git a/src/widget/friendlistwidget.cpp b/src/widget/friendlistwidget.cpp index 1f0f50591..a9d0a2b02 100644 --- a/src/widget/friendlistwidget.cpp +++ b/src/widget/friendlistwidget.cpp @@ -596,17 +596,6 @@ void FriendListWidget::cycleContacts(GenericChatroomWidget* activeChatroomWidget } } -QVector FriendListWidget::getAllCircles() -{ - QVector vec; - vec.reserve(circleLayout->getLayout()->count()); - for (int i = 0; i < circleLayout->getLayout()->count(); ++i) - { - vec.push_back(dynamic_cast(circleLayout->getLayout()->itemAt(i)->widget())); - } - return vec; -} - void FriendListWidget::dragEnterEvent(QDragEnterEvent* event) { if (event->mimeData()->hasFormat("friend")) diff --git a/src/widget/friendlistwidget.h b/src/widget/friendlistwidget.h index 5c1bc0335..63c4ac20f 100644 --- a/src/widget/friendlistwidget.h +++ b/src/widget/friendlistwidget.h @@ -58,7 +58,6 @@ public: void searchChatrooms(const QString &searchString, bool hideOnline = false, bool hideOffline = false, bool hideGroups = false); void cycleContacts(GenericChatroomWidget* activeChatroomWidget, bool forward); - QVector getAllCircles(); void updateActivityDate(const QDate& date); void reDraw(); diff --git a/src/widget/tool/callconfirmwidget.cpp b/src/widget/tool/callconfirmwidget.cpp index 22f21903b..e61a495c2 100644 --- a/src/widget/tool/callconfirmwidget.cpp +++ b/src/widget/tool/callconfirmwidget.cpp @@ -74,7 +74,7 @@ CallConfirmWidget::CallConfirmWidget(const QWidget *Anchor, const Friend& f) : setFixedSize(rectW,rectH+spikeH); reposition(); } -#include + void CallConfirmWidget::reposition() { if (parentWidget()) @@ -84,7 +84,6 @@ void CallConfirmWidget::reposition() parentWidget()->installEventFilter(this); QWidget* w = anchor->window(); - qDebug() << anchor->window(); QPoint pos = anchor->mapToGlobal({(anchor->width()-rectW)/2,anchor->height()})-w->mapToGlobal({0,0}); // We don't want the widget to overflow past the right of the screen diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index cf8446320..79760b511 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -967,7 +967,7 @@ void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) dialog = ContentDialog::current(); if (dialog == nullptr) - dialog = new ContentDialog(); + dialog = new ContentDialog(settingsWidget); dialog->show(); Friend* frnd = widget->getFriend(); From 014ba45ebbcd19238ffaaf3a8f43777ab977bd0b Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 30 Jun 2015 12:29:09 -0400 Subject: [PATCH 08/24] Multi-window: Keep window size, fix segfault, circle integration, changed defaults --- src/persistence/settings.cpp | 2 +- src/widget/circlewidget.cpp | 37 ++++++++++++++++++++++++++++++-- src/widget/contentdialog.cpp | 38 +++++++++++++++++++++------------ src/widget/form/profileform.cpp | 10 +++++++++ src/widget/form/profileform.h | 1 + src/widget/widget.cpp | 33 ++++++++++++++++++++++------ src/widget/widget.h | 1 + 7 files changed, 99 insertions(+), 23 deletions(-) diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index 9b645bd91..1b4882231 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -175,7 +175,7 @@ void Settings::loadGlobal() QStandardPaths::locate(QStandardPaths::HomeLocation, QString(), QStandardPaths::LocateDirectory) ).toString(); separateWindow = s.value("separateWindow", false).toBool(); - dontGroupWindows = s.value("dontGroupWindows", false).toBool(); + dontGroupWindows = s.value("dontGroupWindows", true).toBool(); groupchatPosition = s.value("groupchatPosition", true).toBool(); s.endGroup(); diff --git a/src/widget/circlewidget.cpp b/src/widget/circlewidget.cpp index 90225a194..079682610 100644 --- a/src/widget/circlewidget.cpp +++ b/src/widget/circlewidget.cpp @@ -24,6 +24,7 @@ #include "src/persistence/settings.h" #include "src/friendlist.h" #include "src/friend.h" +#include "src/widget/contentdialog.h" #include "widget.h" #include #include @@ -84,8 +85,13 @@ void CircleWidget::contextMenuEvent(QContextMenuEvent* event) QMenu menu; QAction* renameAction = menu.addAction(tr("Rename circle", "Menu for renaming a circle")); QAction* removeAction = menu.addAction(tr("Remove circle", "Menu for removing a circle")); + QAction* openAction = nullptr; + + if (friendOfflineLayout()->count() + friendOnlineLayout()->count() > 0) + openAction = menu.addAction(tr("Open all in new window")); QAction* selectedItem = menu.exec(mapToGlobal(event->pos())); + if (selectedItem == renameAction) { editName(); @@ -107,6 +113,33 @@ void CircleWidget::contextMenuEvent(QContextMenuEvent* event) circleList.remove(replacedCircle); } + else if (selectedItem == openAction) + { + ContentDialog* dialog = Widget::getInstance()->createContentDialog(); + + for (int i = 0; i < friendOnlineLayout()->count(); ++i) + { + FriendWidget* friendWidget = dynamic_cast(friendOnlineLayout()->itemAt(i)->widget()); + + if (friendWidget != nullptr) + { + Friend* f = friendWidget->getFriend(); + dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); + } + } + for (int i = 0; i < friendOfflineLayout()->count(); ++i) + { + FriendWidget* friendWidget = dynamic_cast(friendOfflineLayout()->itemAt(i)->widget()); + + if (friendWidget != nullptr) + { + Friend* f = friendWidget->getFriend(); + dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); + } + } + dialog->show(); + } + setContainerAttribute(Qt::WA_UnderMouse, false); } @@ -181,14 +214,14 @@ void CircleWidget::updateID(int index) for (int i = 0; i < friendOnlineLayout()->count(); ++i) { - FriendWidget* friendWidget = dynamic_cast(friendOnlineLayout()->itemAt(i)); + FriendWidget* friendWidget = dynamic_cast(friendOnlineLayout()->itemAt(i)->widget()); if (friendWidget != nullptr) Settings::getInstance().setFriendCircleID(FriendList::findFriend(friendWidget->friendId)->getToxId(), id); } for (int i = 0; i < friendOfflineLayout()->count(); ++i) { - FriendWidget* friendWidget = dynamic_cast(friendOfflineLayout()->itemAt(i)); + FriendWidget* friendWidget = dynamic_cast(friendOfflineLayout()->itemAt(i)->widget()); if (friendWidget != nullptr) Settings::getInstance().setFriendCircleID(FriendList::findFriend(friendWidget->friendId)->getToxId(), id); diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index f986924b7..dce517f72 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -146,7 +146,6 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) Friend* frnd = friendWidget->getFriend(); - onChatroomWidgetClicked(friendWidget, false); connect(frnd, &Friend::displayedNameChanged, this, &ContentDialog::updateFriendWidget); connect(settingsWidget, &SettingsWidget::compactToggled, friendWidget, &FriendWidget::compactChange); @@ -155,7 +154,13 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) connect(Core::getInstance(), &Core::friendAvatarChanged, friendWidget, &FriendWidget::onAvatarChange); connect(Core::getInstance(), &Core::friendAvatarRemoved, friendWidget, &FriendWidget::onAvatarRemoved); + ContentDialog* lastDialog = getFriendDialog(friendId); + + if (lastDialog != nullptr) + lastDialog->removeFriend(friendId); + friendList.insert(friendId, std::make_tuple(this, friendWidget)); + onChatroomWidgetClicked(friendWidget, false); return friendWidget; } @@ -171,9 +176,13 @@ GroupWidget* ContentDialog::addGroup(int groupId, const QString& name) connect(settingsWidget, &SettingsWidget::compactToggled, groupWidget, &GroupWidget::compactChange); connect(groupWidget, &GroupWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); - onChatroomWidgetClicked(groupWidget, false); + ContentDialog* lastDialog = getGroupDialog(groupId); + + if (lastDialog != nullptr) + lastDialog->removeGroup(groupId); groupList.insert(groupId, std::make_tuple(this, groupWidget)); + onChatroomWidgetClicked(groupWidget, false); return groupWidget; } @@ -191,16 +200,7 @@ void ContentDialog::removeFriend(int friendId) if (activeChatroomWidget == chatroomWidget) { // Need to find replacement to show here instead. - if (chatroomWidgetCount() > 1) - { - cycleContacts(true, false); - } - else - { - contentLayout->clear(); - activeChatroomWidget = nullptr; - deleteLater(); - } + cycleContacts(true, false); } friendLayout->removeFriendWidget(chatroomWidget, Status::Offline); @@ -208,7 +208,17 @@ void ContentDialog::removeFriend(int friendId) chatroomWidget->deleteLater(); friendList.remove(friendId); - update(); + + if (chatroomWidgetCount() == 0) + { + contentLayout->clear(); + activeChatroomWidget = nullptr; + deleteLater(); + } + else + { + update(); + } } void ContentDialog::removeGroup(int groupId) @@ -287,7 +297,7 @@ void ContentDialog::cycleContacts(bool forward, bool loop) GenericChatroomWidget* chatWidget = dynamic_cast(currentLayout->itemAt(index)->widget()); if (chatWidget != nullptr) - emit chatWidget->chatroomWidgetClicked(chatWidget); + onChatroomWidgetClicked(chatWidget, false); return; } diff --git a/src/widget/form/profileform.cpp b/src/widget/form/profileform.cpp index ef8a7fa96..fb577f24f 100644 --- a/src/widget/form/profileform.cpp +++ b/src/widget/form/profileform.cpp @@ -134,6 +134,16 @@ ProfileForm::~ProfileForm() head->deleteLater(); } +bool ProfileForm::isShown() const +{ + if (head->isVisible()) + { + return true; + } + + return false; +} + void ProfileForm::show(ContentLayout* contentLayout) { contentLayout->mainHead->layout()->addWidget(head); diff --git a/src/widget/form/profileform.h b/src/widget/form/profileform.h index 2ef32429b..cc995e154 100644 --- a/src/widget/form/profileform.h +++ b/src/widget/form/profileform.h @@ -55,6 +55,7 @@ public: ~ProfileForm(); virtual void show() final{} void show(ContentLayout* contentLayout); + bool isShown() const; signals: void userNameChanged(QString); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 79760b511..74aa3b268 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -550,6 +550,8 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) } else { + int width = ui->friendList->size().width(); + if (contentLayout != nullptr) { contentLayout->clear(); @@ -563,7 +565,7 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) setMinimumWidth(ui->tooliconsZone->sizeHint().width()); if (clicked) - resize(ui->statusPanel->width(), height()); + resize(width, height()); setWindowTitle(QString()); setActiveToolMenuButton(None); @@ -728,15 +730,29 @@ void Widget::onSettingsClicked() void Widget::showProfile() // onAvatarClicked, onUsernameClicked { - hideMainForms(nullptr); - profileForm->show(contentLayout); - setWindowTitle(tr("Profile")); + if (Settings::getInstance().getSeparateWindow()) + { + if (!profileForm->isShown()) + { + profileForm->show(createContentDialog(tr("Profile"))); + setActiveToolMenuButton(Widget::None); + } + } + else + { + hideMainForms(nullptr); + profileForm->show(contentLayout); + setWindowTitle(tr("Profile")); + setActiveToolMenuButton(Widget::None); + } } void Widget::hideMainForms(GenericChatroomWidget* chatroomWidget) { setActiveToolMenuButton(Widget::None); - contentLayout->clear(); + + if (contentLayout != nullptr) + contentLayout->clear(); if (activeChatroomWidget != nullptr) activeChatroomWidget->setAsInactiveChatroom(); @@ -967,7 +983,7 @@ void Widget::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) dialog = ContentDialog::current(); if (dialog == nullptr) - dialog = new ContentDialog(settingsWidget); + dialog = createContentDialog(); dialog->show(); Friend* frnd = widget->getFriend(); @@ -1230,6 +1246,11 @@ void Widget::clearContactsList() removeGroup(g, true); } +ContentDialog* Widget::createContentDialog() const +{ + return new ContentDialog(settingsWidget); +} + ContentLayout* Widget::createContentDialog(const QString &title) const { QDialog* dialog = new QDialog(); diff --git a/src/widget/widget.h b/src/widget/widget.h index a8f537cdd..33deb23b4 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -72,6 +72,7 @@ public: bool getIsWindowMinimized(); void updateIcons(); void clearContactsList(); + ContentDialog* createContentDialog() const; ContentLayout* createContentDialog(const QString& title) const; static void confirmExecutableOpen(const QFileInfo file); From 868099d4848fd7391712197cdd90a7d95bc3e2d2 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 1 Jul 2015 07:49:36 -0400 Subject: [PATCH 09/24] Multi-window: Normalize window size on separate change, notification separate fix, segfault groupchat fix --- src/widget/circlewidget.cpp | 2 ++ src/widget/contentdialog.cpp | 49 ++++++++++++++++++++++++++++++------ src/widget/contentdialog.h | 2 ++ src/widget/widget.cpp | 5 +++- 4 files changed, 49 insertions(+), 9 deletions(-) diff --git a/src/widget/circlewidget.cpp b/src/widget/circlewidget.cpp index 079682610..e59290c5e 100644 --- a/src/widget/circlewidget.cpp +++ b/src/widget/circlewidget.cpp @@ -137,7 +137,9 @@ void CircleWidget::contextMenuEvent(QContextMenuEvent* event) dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); } } + dialog->show(); + dialog->ensureSplitterVisible(); } setContainerAttribute(Qt::WA_UnderMouse, false); diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index dce517f72..220340137 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -245,6 +245,14 @@ int ContentDialog::chatroomWidgetCount() const return friendLayout->friendTotalCount() + groupLayout.getLayout()->count(); } +void ContentDialog::ensureSplitterVisible() +{ + if (splitter->sizes().at(0) == 0) + splitter->setSizes({1, 1}); + + update(); +} + void ContentDialog::cycleContacts(bool forward, bool loop) { Settings::getInstance().getGroupchatPosition(); @@ -374,6 +382,37 @@ void ContentDialog::nextContact() cycleContacts(true); } +bool ContentDialog::event(QEvent* event) +{ + switch (event->type()) + { + case QEvent::WindowActivate: + if (activeChatroomWidget != nullptr) + { + activeChatroomWidget->resetEventFlags(); + activeChatroomWidget->updateStatusLight(); + QString windowTitle = activeChatroomWidget->getName(); + if (!activeChatroomWidget->getStatusString().isNull()) + windowTitle += " (" + activeChatroomWidget->getStatusString() + ")"; + setWindowTitle(windowTitle); + + Friend* frnd = activeChatroomWidget->getFriend(); + + if (frnd) + { + frnd->getFriendWidget()->resetEventFlags(); + frnd->getFriendWidget()->updateStatusLight(); + } + } + + currentDialog = this; + default: + break; + } + + return QWidget::event(event); +} + void ContentDialog::dragEnterEvent(QDragEnterEvent *event) { if (event->mimeData()->hasFormat("friend")) @@ -407,10 +446,7 @@ void ContentDialog::dropEvent(QDropEvent *event) Friend* contact = FriendList::findFriend(friendId); Widget::getInstance()->addFriendDialog(contact, this); - - // Display friend list after dropping, if not already visible. - if (splitter->sizes().at(0) == 0) - splitter->setSizes({1, 1}); + ensureSplitterVisible(); } else if (event->mimeData()->hasFormat("group")) { @@ -422,10 +458,7 @@ void ContentDialog::dropEvent(QDropEvent *event) Group* contact = GroupList::findGroup(groupId); Widget::getInstance()->addGroupDialog(contact, this); - - // Display friend list after dropping, if not already visible. - if (splitter->sizes().at(0) == 0) - splitter->setSizes({1, 1}); + ensureSplitterVisible(); } } diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index d05bb4b03..40e742c08 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -50,6 +50,7 @@ public: bool hasFriendWidget(int friendId, GenericChatroomWidget* chatroomWidget); bool hasGroupWidget(int groupId, GenericChatroomWidget* chatroomWidget); int chatroomWidgetCount() const; + void ensureSplitterVisible(); void cycleContacts(bool forward, bool loop = true); @@ -69,6 +70,7 @@ public slots: void nextContact(); protected: + bool event(QEvent* event) final override; void dragEnterEvent(QDragEnterEvent* event) final override; void dropEvent(QDropEvent* event) final override; void changeEvent(QEvent* event) override; diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 74aa3b268..5653f4101 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -565,7 +565,10 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) setMinimumWidth(ui->tooliconsZone->sizeHint().width()); if (clicked) + { + showNormal(); resize(width, height()); + } setWindowTitle(QString()); setActiveToolMenuButton(None); @@ -1401,7 +1404,7 @@ void Widget::removeGroup(Group* g, bool fake) GroupList::removeGroup(g->getGroupId(), fake); Nexus::getCore()->removeGroup(g->getGroupId(), fake); delete g; - if (contentLayout->mainHead->layout()->isEmpty()) + if (contentLayout != nullptr && contentLayout->mainHead->layout()->isEmpty()) onAddClicked(); contactListWidget->reDraw(); From c5317c6137efdb04a03a41df2b36f6b90400e734 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 1 Jul 2015 09:22:05 -0400 Subject: [PATCH 10/24] Multi-window: Fix dialog null states/geometry --- src/widget/contentdialog.cpp | 30 +++++++++++++++++------------- src/widget/contentdialog.h | 1 - src/widget/widget.cpp | 4 ++-- 3 files changed, 19 insertions(+), 16 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 220340137..c72454767 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -89,14 +89,25 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent) splitter->setCollapsible(1, false); boxLayout->addWidget(splitter); + connect(splitter, &QSplitter::splitterMoved, this, &ContentDialog::saveSplitterState); + connect(settingsWidget, &SettingsWidget::groupchatPositionToggled, this, &ContentDialog::onGroupchatPositionChanged); - setMinimumSize(775, 420); + setMinimumSize(500, 220); setAttribute(Qt::WA_DeleteOnClose); - //restore window state - restoreGeometry(Settings::getInstance().getDialogGeometry()); - splitter->restoreState(Settings::getInstance().getDialogSplitterState()); + QByteArray geometry = Settings::getInstance().getDialogGeometry(); + + if (!geometry.isNull()) + restoreGeometry(geometry); + else + resize(720, 400); + + + QByteArray splitterState = Settings::getInstance().getDialogSplitterState(); + + if (!splitterState.isNull()) + splitter->restoreState(splitterState); currentDialog = this; @@ -146,7 +157,6 @@ FriendWidget* ContentDialog::addFriend(int friendId, QString id) Friend* frnd = friendWidget->getFriend(); - connect(frnd, &Friend::displayedNameChanged, this, &ContentDialog::updateFriendWidget); connect(settingsWidget, &SettingsWidget::compactToggled, friendWidget, &FriendWidget::compactChange); connect(friendWidget, &FriendWidget::chatroomWidgetClicked, this, &ContentDialog::onChatroomWidgetClicked); @@ -478,13 +488,6 @@ void ContentDialog::resizeEvent(QResizeEvent* event) saveDialogGeometry(); } -void ContentDialog::closeEvent(QCloseEvent* event) -{ - saveDialogGeometry(); - saveSplitterState(); - QWidget::closeEvent(event); -} - void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) { if (group) @@ -555,9 +558,10 @@ void ContentDialog::saveDialogGeometry() { Settings::getInstance().setDialogGeometry(saveGeometry()); } - +#include void ContentDialog::saveSplitterState() { + qDebug() << splitter->saveState(); Settings::getInstance().setDialogSplitterState(splitter->saveState()); } diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index 40e742c08..1418d6640 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -75,7 +75,6 @@ protected: void dropEvent(QDropEvent* event) final override; void changeEvent(QEvent* event) override; void resizeEvent(QResizeEvent* event) override; - void closeEvent(QCloseEvent* event) override; private slots: void onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 5653f4101..ab9ca8840 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1125,7 +1125,7 @@ bool Widget::newGroupMessageAlert(int groupId) bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive) { - bool inactiveWindow = isMinimized() || !isActiveWindow(); + bool inactiveWindow = isMinimized() || !currentWindow->isActiveWindow(); if (!inactiveWindow && isActive) return false; @@ -1263,7 +1263,7 @@ ContentLayout* Widget::createContentDialog(const QString &title) const dialog->setLayout(contentLayout); dialog->layout()->setMargin(0); dialog->layout()->setSpacing(0); - dialog->setMinimumSize(775, 420); + dialog->setMinimumSize(720, 400); dialog->setAttribute(Qt::WA_DeleteOnClose); dialog->setWindowTitle(title); dialog->show(); From 29c72549a276351191dd970cbdf988985bed1aff Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 1 Jul 2015 10:19:15 -0400 Subject: [PATCH 11/24] Multi-window: Fix segfault on removing groups and friends who are also in active windows --- src/widget/widget.cpp | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index ab9ca8840..834855a9a 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1221,11 +1221,16 @@ void Widget::removeFriend(Friend* f, bool fake) contactListWidget->removeFriendWidget(f->getFriendWidget()); + ContentDialog* lastDialog = ContentDialog::getFriendDialog(f->getFriendID()); + + if (lastDialog != nullptr) + lastDialog->removeFriend(f->getFriendID()); + FriendList::removeFriend(f->getFriendID(), fake); Nexus::getCore()->removeFriend(f->getFriendID(), fake); delete f; - if (contentLayout->mainHead->layout()->isEmpty()) + if (contentLayout != nullptr && contentLayout->mainHead->layout()->isEmpty()) onAddClicked(); contactListWidget->reDraw(); @@ -1402,6 +1407,12 @@ void Widget::removeGroup(Group* g, bool fake) onAddClicked(); } GroupList::removeGroup(g->getGroupId(), fake); + + ContentDialog* contentDialog = ContentDialog::getGroupDialog(g->getGroupId()); + + if (contentDialog != nullptr) + contentDialog->removeGroup(g->getGroupId()); + Nexus::getCore()->removeGroup(g->getGroupId(), fake); delete g; if (contentLayout != nullptr && contentLayout->mainHead->layout()->isEmpty()) From 6a5fb9f518a77eda2f5306a602a207a88567af65 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Thu, 2 Jul 2015 13:16:11 -0400 Subject: [PATCH 12/24] Multi-window: Dispaly own username on detached windows, fix log out while having detached windows --- src/persistence/settings.cpp | 16 +++++++ src/persistence/settings.h | 4 ++ src/widget/contentdialog.cpp | 36 ++++++++++++---- src/widget/contentdialog.h | 4 ++ src/widget/widget.cpp | 84 +++++++++++++++++++++++++----------- 5 files changed, 110 insertions(+), 34 deletions(-) diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index 1b4882231..be4d7c41f 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -220,6 +220,7 @@ void Settings::loadGlobal() splitterState = s.value("splitterState", QByteArray()).toByteArray(); dialogGeometry = s.value("dialogGeometry", QByteArray()).toByteArray(); dialogSplitterState = s.value("dialogSplitterState", QByteArray()).toByteArray(); + dialogSettingsGeometry = s.value("dialogSettingsGeometry", QByteArray()).toByteArray(); s.endGroup(); s.beginGroup("Audio"); @@ -416,6 +417,9 @@ void Settings::saveGlobal() s.setValue("windowGeometry", windowGeometry); s.setValue("windowState", windowState); s.setValue("splitterState", splitterState); + s.setValue("dialogGeometry", dialogGeometry); + s.setValue("dialogSplitterState", dialogSplitterState); + s.setValue("dialogSettingsGeometry", dialogSettingsGeometry); s.endGroup(); s.beginGroup("Audio"); @@ -1120,6 +1124,18 @@ void Settings::setDialogSplitterState(const QByteArray &value) dialogSplitterState = value; } +QByteArray Settings::getDialogSettingsGeometry() const +{ + QMutexLocker locker{&bigLock}; + return dialogSettingsGeometry; +} + +void Settings::setDialogSettingsGeometry(const QByteArray &value) +{ + QMutexLocker locker{&bigLock}; + dialogSettingsGeometry = value; +} + bool Settings::isMinimizeOnCloseEnabled() const { QMutexLocker locker{&bigLock}; diff --git a/src/persistence/settings.h b/src/persistence/settings.h index 3b6fdf11f..16be7f728 100644 --- a/src/persistence/settings.h +++ b/src/persistence/settings.h @@ -223,6 +223,9 @@ public: QByteArray getDialogSplitterState() const; void setDialogSplitterState(const QByteArray &value); + QByteArray getDialogSettingsGeometry() const; + void setDialogSettingsGeometry(const QByteArray& value); + QString getFriendAdress(const QString &publicKey) const; void updateFriendAdress(const QString &newAddr); @@ -350,6 +353,7 @@ private: QByteArray splitterState; QByteArray dialogGeometry; QByteArray dialogSplitterState; + QByteArray dialogSettingsGeometry; QString style; bool showSystemTray; diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index c72454767..f82172cbc 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -118,6 +118,8 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent) new QShortcut(Qt::CTRL + Qt::Key_Tab, this, SLOT(nextContact())); new QShortcut(Qt::CTRL + Qt::Key_PageUp, this, SLOT(previousContact())); new QShortcut(Qt::CTRL + Qt::Key_PageDown, this, SLOT(nextContact())); + + connect(Core::getInstance(), &Core::usernameSet, this, &ContentDialog::updateTitleUsername); } ContentDialog::~ContentDialog() @@ -382,6 +384,20 @@ ContentDialog* ContentDialog::getGroupDialog(int groupId) return getDialog(groupId, groupList); } +void ContentDialog::updateTitleUsername(const QString& username) +{ + if (displayWidget != nullptr) + setWindowTitle(username + QStringLiteral(" - ") + displayWidget->getTitle()); + else + setWindowTitle(username); +} + +void ContentDialog::updateTitle(GenericChatroomWidget* chatroomWidget) +{ + displayWidget = chatroomWidget; + updateTitleUsername(Core::getInstance()->getUsername()); +} + void ContentDialog::previousContact() { cycleContacts(false); @@ -401,10 +417,7 @@ bool ContentDialog::event(QEvent* event) { activeChatroomWidget->resetEventFlags(); activeChatroomWidget->updateStatusLight(); - QString windowTitle = activeChatroomWidget->getName(); - if (!activeChatroomWidget->getStatusString().isNull()) - windowTitle += " (" + activeChatroomWidget->getStatusString() + ")"; - setWindowTitle(windowTitle); + updateTitle(activeChatroomWidget); Friend* frnd = activeChatroomWidget->getFriend(); @@ -484,8 +497,14 @@ void ContentDialog::changeEvent(QEvent *event) void ContentDialog::resizeEvent(QResizeEvent* event) { - Q_UNUSED(event); saveDialogGeometry(); + QDialog::resizeEvent(event); +} + +void ContentDialog::moveEvent(QMoveEvent* event) +{ + saveDialogGeometry(); + QDialog::moveEvent(event); } void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) @@ -523,7 +542,7 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool widget->setAsActiveChatroom(); widget->resetEventFlags(); widget->updateStatusLight(); - setWindowTitle(widget->getTitle()); + updateTitle(widget); if (widget->getFriend() != nullptr) widget->getFriend()->getFriendWidget()->updateStatusLight(); @@ -558,10 +577,9 @@ void ContentDialog::saveDialogGeometry() { Settings::getInstance().setDialogGeometry(saveGeometry()); } -#include + void ContentDialog::saveSplitterState() { - qDebug() << splitter->saveState(); Settings::getInstance().setDialogSplitterState(splitter->saveState()); } @@ -639,7 +657,7 @@ void ContentDialog::updateStatus(int id, const QHashupdateStatusLight(); if (chatroomWidget->isActive()) - std::get<0>(iter.value())->setWindowTitle(chatroomWidget->getTitle()); + std::get<0>(iter.value())->updateTitle(chatroomWidget); } bool ContentDialog::isWidgetActive(int id, const QHash > &list) diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index 1418d6640..9689bd7bb 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -66,6 +66,8 @@ public: static ContentDialog* getGroupDialog(int groupId); public slots: + void updateTitleUsername(const QString& username); + void updateTitle(GenericChatroomWidget* chatroomWidget); void previousContact(); void nextContact(); @@ -75,6 +77,7 @@ protected: void dropEvent(QDropEvent* event) final override; void changeEvent(QEvent* event) override; void resizeEvent(QResizeEvent* event) override; + void moveEvent(QMoveEvent* event) override; private slots: void onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group); @@ -99,6 +102,7 @@ private: GenericChatItemLayout groupLayout; ContentLayout* contentLayout; GenericChatroomWidget* activeChatroomWidget; + GenericChatroomWidget* displayWidget = nullptr; SettingsWidget* settingsWidget; static ContentDialog* currentDialog; static QHash> friendList; diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 834855a9a..f9f843cb9 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -406,6 +406,14 @@ void Widget::updateIcons() Widget::~Widget() { + QWidgetList windowList = QApplication::topLevelWidgets(); + + for (QWidget* window : windowList) + { + if (window != this) + window->close(); + } + Translator::unregister(this); AutoUpdater::abortUpdates(); if (icon) @@ -545,12 +553,21 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) contentLayout = new ContentLayout(contentWidget); ui->mainSplitter->addWidget(contentWidget); - clicked = true; setMinimumWidth(775); + + onSettingsClicked(); } else { int width = ui->friendList->size().width(); + QSize size; + QPoint pos; + + if (contentLayout) + { + pos = mapToGlobal(ui->mainSplitter->widget(1)->pos()); + size = ui->mainSplitter->widget(1)->size(); + } if (contentLayout != nullptr) { @@ -572,10 +589,16 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) setWindowTitle(QString()); setActiveToolMenuButton(None); - } - if (clicked) - onSettingsClicked(); + if (clicked) + { + ContentLayout* contentLayout = createContentDialog(tr("Settings")); + contentLayout->parentWidget()->resize(size); + contentLayout->parentWidget()->move(pos); + settingsWidget->show(contentLayout); + setActiveToolMenuButton(Widget::None); + } + } } void Widget::setWindowTitle(const QString& title) @@ -894,12 +917,7 @@ void Widget::onFriendStatusChanged(int friendId, Status status) f->setStatus(status); f->getFriendWidget()->updateStatusLight(); if(f->getFriendWidget()->isActive()) - { - QString windowTitle = f->getFriendWidget()->getName(); - if (!f->getFriendWidget()->getStatusString().isNull()) - windowTitle += " (" + f->getFriendWidget()->getStatusString() + ")"; - setWindowTitle(windowTitle); - } + setWindowTitle(f->getFriendWidget()->getTitle()); ContentDialog::updateFriendStatus(friendId); @@ -1213,7 +1231,7 @@ void Widget::removeFriend(Friend* f, bool fake) } f->getFriendWidget()->setAsInactiveChatroom(); - if (static_cast(f->getFriendWidget()) == activeChatroomWidget) + if (f->getFriendWidget() == activeChatroomWidget) { activeChatroomWidget = nullptr; onAddClicked(); @@ -1261,11 +1279,35 @@ ContentDialog* Widget::createContentDialog() const ContentLayout* Widget::createContentDialog(const QString &title) const { - QDialog* dialog = new QDialog(); - ContentLayout* contentLayout = new ContentLayout(dialog); + class Dialog : public QDialog + { + public: + Dialog() + : QDialog() + { + restoreGeometry(Settings::getInstance().getDialogSettingsGeometry()); + } + + protected: + void resizeEvent(QResizeEvent* event) override + { + Settings::getInstance().setDialogSettingsGeometry(saveGeometry()); + QDialog::resizeEvent(event); + } + + void moveEvent(QMoveEvent* event) override + { + Settings::getInstance().setDialogSettingsGeometry(saveGeometry()); + QDialog::moveEvent(event); + } + }; + + QDialog* dialog = new Dialog(); + dialog->setAttribute(Qt::WA_DeleteOnClose); + ContentLayout* contentLayoutDialog = new ContentLayout(dialog); dialog->setObjectName("detached"); - dialog->setLayout(contentLayout); + dialog->setLayout(contentLayoutDialog); dialog->layout()->setMargin(0); dialog->layout()->setSpacing(0); dialog->setMinimumSize(720, 400); @@ -1273,7 +1315,7 @@ ContentLayout* Widget::createContentDialog(const QString &title) const dialog->setWindowTitle(title); dialog->show(); - return contentLayout; + return contentLayoutDialog; } void Widget::copyFriendIdToClipboard(int friendId) @@ -1332,12 +1374,7 @@ void Widget::onGroupMessageReceived(int groupnumber, int peernumber, const QStri ContentDialog::updateGroupStatus(g->getGroupId()); if (g->getGroupWidget()->isActive()) - { - QString windowTitle = g->getGroupWidget()->getName(); - if (!g->getGroupWidget()->getStatusString().isNull()) - windowTitle += " (" + g->getGroupWidget()->getStatusString() + ")"; - setWindowTitle(windowTitle); - } + setWindowTitle(g->getGroupWidget()->getTitle()); } void Widget::onGroupNamelistChanged(int groupnumber, int peernumber, uint8_t Change) @@ -1475,10 +1512,7 @@ bool Widget::event(QEvent * e) { activeChatroomWidget->resetEventFlags(); activeChatroomWidget->updateStatusLight(); - QString windowTitle = activeChatroomWidget->getName(); - if (!activeChatroomWidget->getStatusString().isNull()) - windowTitle += " (" + activeChatroomWidget->getStatusString() + ")"; - setWindowTitle(windowTitle); + setWindowTitle(activeChatroomWidget->getTitle()); } if (eventFlag) { From f825985856f68d01a1ccac1eed80676e1622a5be Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Thu, 2 Jul 2015 14:38:18 -0400 Subject: [PATCH 13/24] Multi-window: Translate all dialogs correctly, don't allow remove friend on dialogs --- src/widget/contentdialog.cpp | 10 ++++ src/widget/contentdialog.h | 2 + src/widget/form/settings/generalsettings.ui | 4 +- src/widget/friendwidget.cpp | 5 +- src/widget/widget.cpp | 66 ++++++++++++++++----- src/widget/widget.h | 12 +++- 6 files changed, 81 insertions(+), 18 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index f82172cbc..a3f46f89c 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -33,6 +33,7 @@ #include "src/core/core.h" #include "src/widget/friendlistlayout.h" #include "src/widget/form/settingswidget.h" +#include "src/widget/translator.h" #include #include #include @@ -120,6 +121,8 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent) new QShortcut(Qt::CTRL + Qt::Key_PageDown, this, SLOT(nextContact())); connect(Core::getInstance(), &Core::usernameSet, this, &ContentDialog::updateTitleUsername); + + Translator::registerHandler(std::bind(&ContentDialog::retranslateUi, this), this); } ContentDialog::~ContentDialog() @@ -150,6 +153,8 @@ ContentDialog::~ContentDialog() } ++groupIt; } + + Translator::unregister(this); } FriendWidget* ContentDialog::addFriend(int friendId, QString id) @@ -573,6 +578,11 @@ void ContentDialog::onGroupchatPositionChanged(bool top) friendLayout->insertLayout(1, groupLayout.getLayout()); } +void ContentDialog::retranslateUi() +{ + updateTitleUsername(Core::getInstance()->getUsername()); +} + void ContentDialog::saveDialogGeometry() { Settings::getInstance().setDialogGeometry(saveGeometry()); diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index 9689bd7bb..ab360ecc8 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -26,6 +26,7 @@ #include "src/widget/genericchatitemlayout.h" template class QHash; +template class QSet; class QSplitter; class QVBoxLayout; @@ -86,6 +87,7 @@ private slots: void onGroupchatPositionChanged(bool top); private: + void retranslateUi(); void saveDialogGeometry(); void saveSplitterState(); QLayout* nextLayout(QLayout* layout, bool forward) const; diff --git a/src/widget/form/settings/generalsettings.ui b/src/widget/form/settings/generalsettings.ui index d99510599..8cc7b04f0 100644 --- a/src/widget/form/settings/generalsettings.ui +++ b/src/widget/form/settings/generalsettings.ui @@ -40,7 +40,7 @@ 0 0 639 - 1221 + 1388 @@ -387,7 +387,7 @@ instead of system taskbar. false - Don't group chat wndows. + Don't group chat windows. diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index f7ec9515d..9cc13ea64 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -147,7 +147,10 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) autoAccept->setChecked(!dir.isEmpty()); menu.addSeparator(); - QAction* removeFriendAction = menu.addAction(tr("Remove friend", "Menu to remove the friend from our friendlist")); + QAction* removeFriendAction = nullptr; + + if (contentDialog == nullptr || !contentDialog->hasFriendWidget(friendId, this)) + removeFriendAction = menu.addAction(tr("Remove friend", "Menu to remove the friend from our friendlist")); QAction* selectedItem = menu.exec(pos); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index f9f843cb9..fe79f3968 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -592,7 +592,7 @@ void Widget::onSeparateWindowChanged(bool separate, bool clicked) if (clicked) { - ContentLayout* contentLayout = createContentDialog(tr("Settings")); + ContentLayout* contentLayout = createContentDialog((SettingDialog)); contentLayout->parentWidget()->resize(size); contentLayout->parentWidget()->move(pos); settingsWidget->show(contentLayout); @@ -628,7 +628,7 @@ void Widget::onAddClicked() { if (!addFriendForm->isShown()) { - addFriendForm->show(createContentDialog(tr("Add friend"))); + addFriendForm->show(createContentDialog(AddDialog)); setActiveToolMenuButton(Widget::None); } } @@ -636,7 +636,7 @@ void Widget::onAddClicked() { hideMainForms(nullptr); addFriendForm->show(contentLayout); - setWindowTitle(tr("Add friend")); + setWindowTitle(fromDialogType(AddDialog)); setActiveToolMenuButton(Widget::AddButton); } } @@ -652,7 +652,7 @@ void Widget::onTransferClicked() { if (!filesForm->isShown()) { - filesForm->show(createContentDialog(tr("File transfers"))); + filesForm->show(createContentDialog(TransferDialog)); setActiveToolMenuButton(Widget::None); } } @@ -660,7 +660,7 @@ void Widget::onTransferClicked() { hideMainForms(nullptr); filesForm->show(contentLayout); - setWindowTitle(tr("File transfers")); + setWindowTitle(fromDialogType(TransferDialog)); setActiveToolMenuButton(Widget::TransferButton); } } @@ -741,7 +741,7 @@ void Widget::onSettingsClicked() { if (!settingsWidget->isShown()) { - settingsWidget->show(createContentDialog(tr("Settings"))); + settingsWidget->show(createContentDialog(SettingDialog)); setActiveToolMenuButton(Widget::None); } } @@ -749,7 +749,7 @@ void Widget::onSettingsClicked() { hideMainForms(nullptr); settingsWidget->show(contentLayout); - setWindowTitle(tr("Settings")); + setWindowTitle(fromDialogType(SettingDialog)); setActiveToolMenuButton(Widget::SettingButton); } } @@ -760,7 +760,7 @@ void Widget::showProfile() // onAvatarClicked, onUsernameClicked { if (!profileForm->isShown()) { - profileForm->show(createContentDialog(tr("Profile"))); + profileForm->show(createContentDialog(ProfileDialog)); setActiveToolMenuButton(Widget::None); } } @@ -768,7 +768,7 @@ void Widget::showProfile() // onAvatarClicked, onUsernameClicked { hideMainForms(nullptr); profileForm->show(contentLayout); - setWindowTitle(tr("Profile")); + setWindowTitle(fromDialogType(ProfileDialog)); setActiveToolMenuButton(Widget::None); } } @@ -1141,6 +1141,23 @@ bool Widget::newGroupMessageAlert(int groupId) return newMessageAlert(currentWindow, hasActive); } +QString Widget::fromDialogType(DialogType type) +{ + switch (type) + { + case AddDialog: + return tr("Add friend"); + case TransferDialog: + return tr("File transfers"); + case SettingDialog: + return tr("Settings"); + case ProfileDialog: + return tr("Profile"); + default: + return QString(); + } +} + bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive) { bool inactiveWindow = isMinimized() || !currentWindow->isActiveWindow(); @@ -1277,15 +1294,32 @@ ContentDialog* Widget::createContentDialog() const return new ContentDialog(settingsWidget); } -ContentLayout* Widget::createContentDialog(const QString &title) const +ContentLayout* Widget::createContentDialog(DialogType type) const { class Dialog : public QDialog { public: - Dialog() + Dialog(DialogType type) : QDialog() + , type(type) { restoreGeometry(Settings::getInstance().getDialogSettingsGeometry()); + Translator::registerHandler(std::bind(&Dialog::retranslateUi, this), this); + retranslateUi(); + + connect(Core::getInstance(), &Core::usernameSet, this, &Dialog::retranslateUi); + } + + ~Dialog() + { + Translator::unregister(this); + } + + public slots: + + void retranslateUi() + { + setWindowTitle(Core::getInstance()->getUsername() + QStringLiteral(" - ") + Widget::fromDialogType(type)); } protected: @@ -1300,9 +1334,12 @@ ContentLayout* Widget::createContentDialog(const QString &title) const Settings::getInstance().setDialogSettingsGeometry(saveGeometry()); QDialog::moveEvent(event); } + + private: + DialogType type; }; - QDialog* dialog = new Dialog(); + QDialog* dialog = new Dialog(type); dialog->setAttribute(Qt::WA_DeleteOnClose); ContentLayout* contentLayoutDialog = new ContentLayout(dialog); @@ -1312,7 +1349,6 @@ ContentLayout* Widget::createContentDialog(const QString &title) const dialog->layout()->setSpacing(0); dialog->setMinimumSize(720, 400); dialog->setAttribute(Qt::WA_DeleteOnClose); - dialog->setWindowTitle(title); dialog->show(); return contentLayoutDialog; @@ -1971,7 +2007,9 @@ void Widget::retranslateUi() statusOnline->setText(tr("Online", "Button to set your status to 'Online'")); statusAway->setText(tr("Away", "Button to set your status to 'Away'")); statusBusy->setText(tr("Busy", "Button to set your status to 'Busy'")); - //setWindowTitle(tr("Settings")); + + if (!Settings::getInstance().getSeparateWindow()) + setWindowTitle(fromDialogType(SettingDialog)); } #ifdef Q_OS_MAC diff --git a/src/widget/widget.h b/src/widget/widget.h index 33deb23b4..71406f70d 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -72,8 +72,18 @@ public: bool getIsWindowMinimized(); void updateIcons(); void clearContactsList(); + + enum DialogType + { + AddDialog, + TransferDialog, + SettingDialog, + ProfileDialog + }; + + static QString fromDialogType(DialogType type); ContentDialog* createContentDialog() const; - ContentLayout* createContentDialog(const QString& title) const; + ContentLayout* createContentDialog(DialogType type) const; static void confirmExecutableOpen(const QFileInfo file); From 1109e0347e05c0823f7e114bc3be8100c06f0db2 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Fri, 3 Jul 2015 09:16:27 -0400 Subject: [PATCH 14/24] Multi-window: Fix segfault for removing last friend on window friend list --- src/widget/contentdialog.cpp | 90 +++++++++++++++++-------------- src/widget/contentdialog.h | 2 +- src/widget/tool/croppinglabel.cpp | 31 +++++++++-- src/widget/tool/croppinglabel.h | 2 +- 4 files changed, 79 insertions(+), 46 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index a3f46f89c..80175d728 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -244,7 +244,33 @@ void ContentDialog::removeGroup(int groupId) disconnect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); disconnect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); - remove(groupId, groupList); + auto iter = groupList.find(groupId); + + if (iter == groupList.end()) + return; + + GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value()); + + if (activeChatroomWidget == chatroomWidget) + { + // Need to find replacement to show here instead. + cycleContacts(true, false); + } + + groupLayout.removeSortedWidget(chatroomWidget); + chatroomWidget->deleteLater(); + groupList.remove(groupId); + + if (chatroomWidgetCount() == 0) + { + contentLayout->clear(); + activeChatroomWidget = nullptr; + deleteLater(); + } + else + { + update(); + } } bool ContentDialog::hasFriendWidget(int friendId, GenericChatroomWidget* chatroomWidget) @@ -292,6 +318,21 @@ void ContentDialog::cycleContacts(bool forward, bool loop) index = groupLayout.indexOfSortedWidget(activeChatroomWidget); } + if (!loop && index == currentLayout->count() - 1) + { + bool groupsOnTop = Settings::getInstance().getGroupchatPosition(); + bool offlineEmpty = friendLayout->getLayoutOffline()->count() == 0; + bool onlineEmpty = offlineEmpty && ((friendLayout->getLayoutOnline()->count() == 0 && groupsOnTop) || !groupsOnTop); + bool groupsEmpty = offlineEmpty && ((groupLayout.getLayout()->count() == 0 && !groupsOnTop) || groupsOnTop); + + if ((currentLayout == friendLayout->getLayoutOffline()) + || (currentLayout == friendLayout->getLayoutOnline() && groupsEmpty) + || (currentLayout == groupLayout.getLayout() && onlineEmpty)) + { + forward = !forward; + } + } + index += forward ? 1 : -1; for (;;) @@ -305,13 +346,13 @@ void ContentDialog::cycleContacts(bool forward, bool loop) } else if (index >= currentLayout->count()) { - if (!loop && currentLayout == friendLayout->getLayoutOffline()) + /*if (!loop && currentLayout == friendLayout->getLayoutOffline()) { forward = !forward; // Go backward. index += forward ? 2 : -2; // Go back to where started and then one. continue; // Recheck bounds. } - else + else*/ { currentLayout = nextLayout(currentLayout, forward); index = 0; @@ -512,6 +553,12 @@ void ContentDialog::moveEvent(QMoveEvent* event) QDialog::moveEvent(event); } +void ContentDialog::keyPressEvent(QKeyEvent* event) +{ + if(event->key() != Qt::Key_Escape) + QDialog::keyPressEvent(event); // Ignore escape keyboard shortcut. +} + void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool group) { if (group) @@ -593,43 +640,6 @@ void ContentDialog::saveSplitterState() Settings::getInstance().setDialogSplitterState(splitter->saveState()); } -void ContentDialog::remove(int id, const QHash > &list) -{ - auto iter = list.find(id); - - if (iter == list.end()) - return; - - GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value()); - - if (activeChatroomWidget == chatroomWidget) - { - // Need to find replacement to show here instead. - if (chatroomWidgetCount() > 1) - { - int index = groupLayout.indexOfSortedWidget(chatroomWidget) - 1; - - // Don't let it go below 0. If it does, then we're first. Go second. - if (index < 0) - index = 1; - - GenericChatroomWidget* chatroomWidget = static_cast(groupLayout.getLayout()->itemAt(index)->widget()); - onChatroomWidgetClicked(chatroomWidget, false); - } - else - { - contentLayout->clear(); - activeChatroomWidget = nullptr; - deleteLater(); - } - } - - groupLayout.removeSortedWidget(chatroomWidget); - chatroomWidget->deleteLater(); - friendList.remove(id); - update(); -} - bool ContentDialog::hasWidget(int id, GenericChatroomWidget* chatroomWidget, const QHash>& list) { auto iter = list.find(id); diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index ab360ecc8..d33512f3a 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -79,6 +79,7 @@ protected: void changeEvent(QEvent* event) override; void resizeEvent(QResizeEvent* event) override; void moveEvent(QMoveEvent* event) override; + void keyPressEvent(QKeyEvent* event) override; private slots: void onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group); @@ -92,7 +93,6 @@ private: void saveSplitterState(); QLayout* nextLayout(QLayout* layout, bool forward) const; - void remove(int id, const QHash>& list); bool hasWidget(int id, GenericChatroomWidget* chatroomWidget, const QHash>& list); static bool existsWidget(int id, bool focus, const QHash>& list); static void updateStatus(int id, const QHash>& list); diff --git a/src/widget/tool/croppinglabel.cpp b/src/widget/tool/croppinglabel.cpp index 7eb6b3a00..ad151462e 100644 --- a/src/widget/tool/croppinglabel.cpp +++ b/src/widget/tool/croppinglabel.cpp @@ -20,6 +20,7 @@ #include "croppinglabel.h" #include #include +#include CroppingLabel::CroppingLabel(QWidget* parent) : QLabel(parent) @@ -29,13 +30,31 @@ CroppingLabel::CroppingLabel(QWidget* parent) { setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred); - textEdit = new QLineEdit(this); + class LineEdit : public QLineEdit + { + public: + LineEdit(QWidget* parent = 0) : + QLineEdit(parent) + {} + + protected: + void keyPressEvent(QKeyEvent* event) override + { + if (event->key() == Qt::Key_Escape) + clearFocus(); + + QLineEdit::keyPressEvent(event); + } + }; + + textEdit = new LineEdit(this); textEdit->hide(); textEdit->setInputMethodHints(Qt::ImhNoAutoUppercase | Qt::ImhNoPredictiveText | Qt::ImhPreferLatin); - connect(textEdit, &QLineEdit::editingFinished, this, &CroppingLabel::editingFinished); + connect(textEdit, &QLineEdit::returnPressed, this, &CroppingLabel::editingFinished); + connect(textEdit, &QLineEdit::editingFinished, this, &CroppingLabel::hideTextEdit); } void CroppingLabel::editBegin() @@ -114,6 +133,12 @@ void CroppingLabel::setElidedText() QLabel::setText(elidedText); } +void CroppingLabel::hideTextEdit() +{ + textEdit->hide(); + blockPaintEvents = false; +} + void CroppingLabel::showTextEdit() { blockPaintEvents = true; @@ -142,7 +167,5 @@ void CroppingLabel::editingFinished() if (origText != newText) emit editFinished(textEdit->text()); - textEdit->hide(); - blockPaintEvents = false; emit editRemoved(); } diff --git a/src/widget/tool/croppinglabel.h b/src/widget/tool/croppinglabel.h index 6a731f662..83d86322b 100644 --- a/src/widget/tool/croppinglabel.h +++ b/src/widget/tool/croppinglabel.h @@ -49,7 +49,7 @@ signals: protected: void paintEvent(QPaintEvent* paintEvent) override; void setElidedText(); - void hideTextEdit(bool acceptText); + void hideTextEdit(); void showTextEdit(); virtual void resizeEvent(QResizeEvent *ev) final override; virtual QSize sizeHint() const final override; From aa9a10fa8bac48fe570d7ea0f1ca66b08d73813d Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Fri, 17 Jul 2015 10:12:00 -0400 Subject: [PATCH 15/24] Multiwindow: Fix segfault on quiting groups, remove dead code --- src/widget/contentdialog.cpp | 15 +++------------ 1 file changed, 3 insertions(+), 12 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 80175d728..39dd275e9 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -346,23 +346,14 @@ void ContentDialog::cycleContacts(bool forward, bool loop) } else if (index >= currentLayout->count()) { - /*if (!loop && currentLayout == friendLayout->getLayoutOffline()) - { - forward = !forward; // Go backward. - index += forward ? 2 : -2; // Go back to where started and then one. - continue; // Recheck bounds. - } - else*/ - { - currentLayout = nextLayout(currentLayout, forward); - index = 0; - } + currentLayout = nextLayout(currentLayout, forward); + index = 0; continue; } GenericChatroomWidget* chatWidget = dynamic_cast(currentLayout->itemAt(index)->widget()); - if (chatWidget != nullptr) + if (chatWidget != nullptr && chatWidget != activeChatroomWidget) onChatroomWidgetClicked(chatWidget, false); return; From 4e5c4cb678750b0b5c2132c5ff1622a908630e6f Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 20 Jul 2015 08:13:51 -0400 Subject: [PATCH 16/24] Multi-window: Fix circle modified from dialog window, fix groupchat leave warnings, flip dialog window title --- src/widget/circlewidget.cpp | 85 +++++++++++++++++---------------- src/widget/contentdialog.cpp | 10 ++-- src/widget/friendlistwidget.cpp | 5 +- src/widget/friendwidget.cpp | 8 ++-- 4 files changed, 59 insertions(+), 49 deletions(-) diff --git a/src/widget/circlewidget.cpp b/src/widget/circlewidget.cpp index e59290c5e..a0c83701b 100644 --- a/src/widget/circlewidget.cpp +++ b/src/widget/circlewidget.cpp @@ -92,54 +92,57 @@ void CircleWidget::contextMenuEvent(QContextMenuEvent* event) QAction* selectedItem = menu.exec(mapToGlobal(event->pos())); - if (selectedItem == renameAction) + if (selectedItem) { - editName(); - } - else if (selectedItem == removeAction) - { - FriendListWidget* friendList = static_cast(parentWidget()); - moveFriendWidgets(friendList); - - friendList->removeCircleWidget(this); - - int replacedCircle = Settings::getInstance().removeCircle(id); - - auto circleReplace = circleList.find(replacedCircle); - if (circleReplace != circleList.end()) - circleReplace.value()->updateID(id); - else - assert(true); // This should never happen. - - circleList.remove(replacedCircle); - } - else if (selectedItem == openAction) - { - ContentDialog* dialog = Widget::getInstance()->createContentDialog(); - - for (int i = 0; i < friendOnlineLayout()->count(); ++i) + if (selectedItem == renameAction) { - FriendWidget* friendWidget = dynamic_cast(friendOnlineLayout()->itemAt(i)->widget()); - - if (friendWidget != nullptr) - { - Friend* f = friendWidget->getFriend(); - dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); - } + editName(); } - for (int i = 0; i < friendOfflineLayout()->count(); ++i) + else if (selectedItem == removeAction) { - FriendWidget* friendWidget = dynamic_cast(friendOfflineLayout()->itemAt(i)->widget()); + FriendListWidget* friendList = static_cast(parentWidget()); + moveFriendWidgets(friendList); - if (friendWidget != nullptr) - { - Friend* f = friendWidget->getFriend(); - dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); - } + friendList->removeCircleWidget(this); + + int replacedCircle = Settings::getInstance().removeCircle(id); + + auto circleReplace = circleList.find(replacedCircle); + if (circleReplace != circleList.end()) + circleReplace.value()->updateID(id); + else + assert(true); // This should never happen. + + circleList.remove(replacedCircle); } + else if (selectedItem == openAction) + { + ContentDialog* dialog = Widget::getInstance()->createContentDialog(); - dialog->show(); - dialog->ensureSplitterVisible(); + for (int i = 0; i < friendOnlineLayout()->count(); ++i) + { + FriendWidget* friendWidget = dynamic_cast(friendOnlineLayout()->itemAt(i)->widget()); + + if (friendWidget != nullptr) + { + Friend* f = friendWidget->getFriend(); + dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); + } + } + for (int i = 0; i < friendOfflineLayout()->count(); ++i) + { + FriendWidget* friendWidget = dynamic_cast(friendOfflineLayout()->itemAt(i)->widget()); + + if (friendWidget != nullptr) + { + Friend* f = friendWidget->getFriend(); + dialog->addFriend(friendWidget->friendId, f->getDisplayedName()); + } + } + + dialog->show(); + dialog->ensureSplitterVisible(); + } } setContainerAttribute(Qt::WA_UnderMouse, false); diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 39dd275e9..7449cfc76 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -241,8 +241,12 @@ void ContentDialog::removeFriend(int friendId) void ContentDialog::removeGroup(int groupId) { Group* group = GroupList::findGroup(groupId); - disconnect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); - disconnect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); + + if (group) + { + disconnect(group, &Group::titleChanged, this, &ContentDialog::updateGroupWidget); + disconnect(group, &Group::userListChanged, this, &ContentDialog::updateGroupWidget); + } auto iter = groupList.find(groupId); @@ -424,7 +428,7 @@ ContentDialog* ContentDialog::getGroupDialog(int groupId) void ContentDialog::updateTitleUsername(const QString& username) { if (displayWidget != nullptr) - setWindowTitle(username + QStringLiteral(" - ") + displayWidget->getTitle()); + setWindowTitle(displayWidget->getTitle() + QStringLiteral(" - ") + username); else setWindowTitle(username); } diff --git a/src/widget/friendlistwidget.cpp b/src/widget/friendlistwidget.cpp index a9d0a2b02..f74fbda18 100644 --- a/src/widget/friendlistwidget.cpp +++ b/src/widget/friendlistwidget.cpp @@ -389,8 +389,11 @@ void FriendListWidget::addCircleWidget(FriendWidget* friendWidget) } Widget::getInstance()->searchCircle(circleWidget); - circleWidget->editName(); + + if (window()->isActiveWindow()) + circleWidget->editName(); } + reDraw(); } void FriendListWidget::removeCircleWidget(CircleWidget* widget) diff --git a/src/widget/friendwidget.cpp b/src/widget/friendwidget.cpp index 9cc13ea64..9459b3ece 100644 --- a/src/widget/friendwidget.cpp +++ b/src/widget/friendwidget.cpp @@ -105,7 +105,7 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) FriendListWidget *friendList; if (circleWidget == nullptr) - friendList = dynamic_cast(parentWidget()); + friendList = dynamic_cast(FriendList::findFriend(friendId)->getFriendWidget()->parentWidget()); else friendList = dynamic_cast(circleWidget->parentWidget()); @@ -214,7 +214,7 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) circleWidget->updateStatus(); if (friendList != nullptr) - friendList->addCircleWidget(this); + friendList->addCircleWidget(FriendList::findFriend(friendId)->getFriendWidget()); else Settings::getInstance().setFriendCircleID(id, Settings::getInstance().addCircle()); } @@ -226,7 +226,7 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) else if (removeCircleAction != nullptr && selectedItem == removeCircleAction) { if (friendList != nullptr) - friendList->moveWidget(this, FriendList::findFriend(friendId)->getStatus(), true); + friendList->moveWidget(FriendList::findFriend(friendId)->getFriendWidget(), FriendList::findFriend(friendId)->getStatus(), true); else Settings::getInstance().setFriendCircleID(id, -1); @@ -242,7 +242,7 @@ void FriendWidget::contextMenuEvent(QContextMenuEvent * event) if (circle != nullptr) { - circle->addFriendWidget(this, FriendList::findFriend(friendId)->getStatus()); + circle->addFriendWidget(FriendList::findFriend(friendId)->getFriendWidget(), FriendList::findFriend(friendId)->getStatus()); circle->setExpanded(true); Widget::getInstance()->searchCircle(circle); Settings::getInstance().savePersonal(); From ee627afc87654593ec46989c49af79e484b1ab9c Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 20 Jul 2015 10:03:01 -0400 Subject: [PATCH 17/24] Multi-window: Fix buttons staying pressed while in separate window mode, make dialogs alert, fix group chat event flags --- src/widget/contentdialog.cpp | 8 +++- src/widget/form/addfriendform.cpp | 2 + src/widget/form/filesform.cpp | 2 + src/widget/form/profileform.cpp | 2 + src/widget/form/settingswidget.cpp | 2 + src/widget/widget.cpp | 68 +++++++++++++----------------- 6 files changed, 44 insertions(+), 40 deletions(-) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 7449cfc76..dfeea2a91 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -467,6 +467,12 @@ bool ContentDialog::event(QEvent* event) frnd->getFriendWidget()->resetEventFlags(); frnd->getFriendWidget()->updateStatusLight(); } + else + { + Group* g = activeChatroomWidget->getGroup(); + g->getGroupWidget()->resetEventFlags(); + g->getGroupWidget()->updateStatusLight(); + } } currentDialog = this; @@ -591,7 +597,7 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool widget->updateStatusLight(); updateTitle(widget); - if (widget->getFriend() != nullptr) + if (widget->getFriend()) widget->getFriend()->getFriendWidget()->updateStatusLight(); else widget->getGroup()->getGroupWidget()->updateStatusLight(); diff --git a/src/widget/form/addfriendform.cpp b/src/widget/form/addfriendform.cpp index d46581202..4c28bb4f1 100644 --- a/src/widget/form/addfriendform.cpp +++ b/src/widget/form/addfriendform.cpp @@ -33,6 +33,7 @@ #include "src/widget/gui.h" #include "src/widget/translator.h" #include "src/widget/contentlayout.h" +#include AddFriendForm::AddFriendForm() { @@ -71,6 +72,7 @@ bool AddFriendForm::isShown() const { if (main->isVisible()) { + head->window()->windowHandle()->alert(0); return true; } diff --git a/src/widget/form/filesform.cpp b/src/widget/form/filesform.cpp index f6a6ea9fb..61c1f949f 100644 --- a/src/widget/form/filesform.cpp +++ b/src/widget/form/filesform.cpp @@ -22,6 +22,7 @@ #include "src/widget/translator.h" #include "src/widget/contentlayout.h" #include +#include FilesForm::FilesForm() : QObject(), doneIcon(":/ui/fileTransferWidget/fileDone.svg") @@ -58,6 +59,7 @@ bool FilesForm::isShown() const { if (main.isVisible()) { + head->window()->windowHandle()->alert(0); return true; } diff --git a/src/widget/form/profileform.cpp b/src/widget/form/profileform.cpp index fb577f24f..4954760cb 100644 --- a/src/widget/form/profileform.cpp +++ b/src/widget/form/profileform.cpp @@ -44,6 +44,7 @@ #include #include #include +#include ProfileForm::ProfileForm(QWidget *parent) : QWidget{parent}, qr{nullptr} @@ -138,6 +139,7 @@ bool ProfileForm::isShown() const { if (head->isVisible()) { + window()->windowHandle()->alert(0); return true; } diff --git a/src/widget/form/settingswidget.cpp b/src/widget/form/settingswidget.cpp index 64ff40b74..c44b94806 100644 --- a/src/widget/form/settingswidget.cpp +++ b/src/widget/form/settingswidget.cpp @@ -29,6 +29,7 @@ #include "src/widget/contentlayout.h" #include #include +#include SettingsWidget::SettingsWidget(QWidget* parent) : QWidget(parent) @@ -91,6 +92,7 @@ bool SettingsWidget::isShown() const { if (body->isVisible()) { + body->window()->windowHandle()->alert(0); return true; } diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index fe79f3968..9c87774a7 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -174,21 +174,7 @@ void Widget::init() ui->searchContactFilterBox->setMenu(filterMenu); - /*ui->mainContent->setLayout(new QVBoxLayout()); - ui->mainHead->setLayout(new QVBoxLayout()); - ui->mainHead->layout()->setMargin(0); - ui->mainHead->layout()->setSpacing(0); - - if (QStyleFactory::keys().contains(Settings::getInstance().getStyle()) - && Settings::getInstance().getStyle() != "None") - { - ui->mainHead->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); - ui->mainContent->setStyle(QStyleFactory::create(Settings::getInstance().getStyle())); - }*/ - #ifndef Q_OS_MAC - //ui->mainHead->setStyleSheet(Style::getStylesheet(":ui/settings/mainHead.css")); - //ui->mainContent->setStyleSheet(Style::getStylesheet(":ui/settings/mainContent.css")); ui->statusHead->setStyleSheet(Style::getStylesheet(":/ui/window/statusPanel.css")); ui->statusPanel->setStyleSheet(Style::getStylesheet(":/ui/window/statusPanel.css")); #endif @@ -627,10 +613,9 @@ void Widget::onAddClicked() if (Settings::getInstance().getSeparateWindow()) { if (!addFriendForm->isShown()) - { addFriendForm->show(createContentDialog(AddDialog)); - setActiveToolMenuButton(Widget::None); - } + + setActiveToolMenuButton(Widget::None); } else { @@ -651,10 +636,9 @@ void Widget::onTransferClicked() if (Settings::getInstance().getSeparateWindow()) { if (!filesForm->isShown()) - { filesForm->show(createContentDialog(TransferDialog)); - setActiveToolMenuButton(Widget::None); - } + + setActiveToolMenuButton(Widget::None); } else { @@ -740,10 +724,9 @@ void Widget::onSettingsClicked() if (Settings::getInstance().getSeparateWindow()) { if (!settingsWidget->isShown()) - { settingsWidget->show(createContentDialog(SettingDialog)); - setActiveToolMenuButton(Widget::None); - } + + setActiveToolMenuButton(Widget::None); } else { @@ -759,10 +742,9 @@ void Widget::showProfile() // onAvatarClicked, onUsernameClicked if (Settings::getInstance().getSeparateWindow()) { if (!profileForm->isShown()) - { profileForm->show(createContentDialog(ProfileDialog)); - setActiveToolMenuButton(Widget::None); - } + + setActiveToolMenuButton(Widget::None); } else { @@ -1126,6 +1108,7 @@ bool Widget::newGroupMessageAlert(int groupId) bool hasActive; QWidget* currentWindow; ContentDialog* contentDialog = ContentDialog::getGroupDialog(groupId); + Group* g = GroupList::findGroup(groupId); if (contentDialog != nullptr) { @@ -1135,10 +1118,28 @@ bool Widget::newGroupMessageAlert(int groupId) else { currentWindow = window(); - hasActive = GroupList::findGroup(groupId)->getGroupWidget() == activeChatroomWidget; + hasActive = g->getGroupWidget() == activeChatroomWidget; } - return newMessageAlert(currentWindow, hasActive); + if (newMessageAlert(currentWindow, hasActive)) + { + g->setEventFlag(true); + g->getGroupWidget()->updateStatusLight(); + + if (contentDialog == nullptr) + { + if (hasActive) + setWindowTitle(g->getGroupWidget()->getTitle()); + } + else + { + ContentDialog::updateGroupStatus(groupId); + } + + return true; + } + + return false; } QString Widget::fromDialogType(DialogType type) @@ -1398,19 +1399,8 @@ void Widget::onGroupMessageReceived(int groupnumber, int peernumber, const QStri else g->getChatForm()->addMessage(author, message, isAction, QDateTime::currentDateTime(), true); - g->setEventFlag(static_cast(g->getGroupWidget()) != activeChatroomWidget); - if (targeted || Settings::getInstance().getGroupAlwaysNotify()) newGroupMessageAlert(g->getGroupId()); - - if (targeted) - g->setMentionedFlag(true); // useful for highlighting line or desktop notifications - - g->getGroupWidget()->updateStatusLight(); - ContentDialog::updateGroupStatus(g->getGroupId()); - - if (g->getGroupWidget()->isActive()) - setWindowTitle(g->getGroupWidget()->getTitle()); } void Widget::onGroupNamelistChanged(int groupnumber, int peernumber, uint8_t Change) From 23e7f9326e2ab75235de1e2f44bcd1dd64191e5e Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 20 Jul 2015 10:33:07 -0400 Subject: [PATCH 18/24] Multi-window: Fix open group in separate window when already open --- src/widget/widget.cpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 9c87774a7..1f2c5813b 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1059,6 +1059,9 @@ void Widget::addFriendDialog(Friend *frnd, ContentDialog *dialog) void Widget::addGroupDialog(Group *group, ContentDialog *dialog) { + if (!ContentDialog::getGroupDialog(group->getGroupId()) && !Settings::getInstance().getSeparateWindow() && activeChatroomWidget == group->getGroupWidget()) + onAddClicked(); + GroupWidget* groupWidget = dialog->addGroup(group->getGroupId(), group->getName()); connect(groupWidget, SIGNAL(removeGroup(int)), this, SLOT(removeGroup(int))); connect(groupWidget, SIGNAL(chatroomWidgetClicked(GenericChatroomWidget*)), group->getChatForm(), SLOT(focusInput())); From 97bdaa5e5f8307d0fd082e307134d633a4410f68 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 29 Jul 2015 19:46:19 -0400 Subject: [PATCH 19/24] Multi-window: OS X window list, fix OS X translations --- qtox.pro | 4 +- src/nexus.cpp | 160 +++++++++++++++++++++++++++-- src/nexus.h | 25 +++++ src/widget/contentdialog.cpp | 8 +- src/widget/contentdialog.h | 9 +- src/widget/loginscreen.cpp | 10 ++ src/widget/loginscreen.h | 7 ++ src/widget/tool/activatedialog.cpp | 36 +++++++ src/widget/tool/activatedialog.h | 36 +++++++ src/widget/widget.cpp | 160 ++++++++++++++++------------- src/widget/widget.h | 20 ++-- 11 files changed, 386 insertions(+), 89 deletions(-) create mode 100644 src/widget/tool/activatedialog.cpp create mode 100644 src/widget/tool/activatedialog.h diff --git a/qtox.pro b/qtox.pro index e53035e15..923317fd0 100644 --- a/qtox.pro +++ b/qtox.pro @@ -497,7 +497,8 @@ SOURCES += \ src/widget/categorywidget.cpp \ src/widget/tool/removefrienddialog.cpp \ src/widget/contentlayout.cpp \ - src/widget/contentdialog.cpp + src/widget/contentdialog.cpp \ + src/widget/tool/activatedialog.cpp HEADERS += \ src/audio/audio.h \ @@ -543,4 +544,5 @@ HEADERS += \ src/widget/categorywidget.h \ src/widget/contentlayout.h \ src/widget/contentdialog.h \ + src/widget/tool/activatedialog.h \ src/widget/tool/removefrienddialog.h diff --git a/src/nexus.cpp b/src/nexus.cpp index 9398f7c45..93f7ebda3 100644 --- a/src/nexus.cpp +++ b/src/nexus.cpp @@ -42,6 +42,8 @@ #ifdef Q_OS_MAC #include #include +#include +#include #endif static Nexus* nexus{nullptr}; @@ -89,26 +91,44 @@ void Nexus::start() qRegisterMetaType("ToxFile::FileDirection"); qRegisterMetaType>("std::shared_ptr"); + loginScreen = new LoginScreen(); + #ifdef Q_OS_MAC globalMenuBar = new QMenuBar(0); + dockMenu = new QMenu(globalMenuBar); - windowMenu = globalMenuBar->addMenu(tr("Window")); + viewMenu = globalMenuBar->addMenu(QString()); + + windowMenu = globalMenuBar->addMenu(QString()); globalMenuBar->addAction(windowMenu->menuAction()); - QAction* minimizeAction = Nexus::getInstance().windowMenu->addAction(tr("Minimize")); + fullscreenAction = viewMenu->addAction(QString()); + fullscreenAction->setShortcut(QKeySequence::FullScreen); + connect(fullscreenAction, &QAction::triggered, this, &Nexus::toggleFullscreen); + + minimizeAction = windowMenu->addAction(QString()); minimizeAction->setShortcut(Qt::CTRL + Qt::Key_M); - connect(minimizeAction, &QAction::triggered, [minimizeAction]() + connect(minimizeAction, &QAction::triggered, [this]() { + minimizeAction->setEnabled(false); QApplication::focusWindow()->showMinimized(); }); - Nexus::getInstance().windowMenu->addSeparator(); + + windowMenu->addSeparator(); + frontAction = windowMenu->addAction(QString()); + connect(frontAction, &QAction::triggered, this, &Nexus::bringAllToFront); QAction* quitAction = new QAction(globalMenuBar); quitAction->setMenuRole(QAction::QuitRole); connect(quitAction, &QAction::triggered, qApp, &QApplication::quit); -#endif - loginScreen = new LoginScreen(); + windowMapper = new QSignalMapper(this); + connect(windowMapper, SIGNAL(mapped(QObject*)), this, SLOT(onOpenWindow(QObject*))); + + connect(loginScreen, &LoginScreen::windowStateChanged, this, &Nexus::onWindowStateChanged); + + retranslateUi(); +#endif if (profile) showMainGUI(); @@ -278,3 +298,131 @@ bool Nexus::tryRemoveFile(const QString& filepath) tmp.remove(); return writable; } + +#ifdef Q_OS_MAC +void Nexus::retranslateUi() +{ + viewMenu->menuAction()->setText(tr("View", "OS X Menu bar")); + windowMenu->menuAction()->setText(tr("Window", "OS X Menu bar")); + minimizeAction->setText(tr("Minimize", "OS X Menu bar")); + frontAction->setText((tr("Bring All to Front", "OS X Menu bar"))); +} + +void Nexus::onWindowStateChanged(Qt::WindowStates state) +{ + minimizeAction->setEnabled(QApplication::activeWindow() != nullptr); + + if (QApplication::activeWindow() != nullptr && sender() == QApplication::activeWindow()) + { + if (state & Qt::WindowFullScreen) + minimizeAction->setEnabled(false); + + if (state & Qt::WindowFullScreen) + fullscreenAction->setText(tr("Exit Fullscreen")); + else + fullscreenAction->setText(tr("Enter Fullscreen")); + + updateWindows(); + } + + updateWindowsStates(); +} + +void Nexus::updateWindows() +{ + updateWindowsArg(nullptr); +} + +void Nexus::updateWindowsArg(QWindow* closedWindow) +{ + QWindowList windowList = QApplication::topLevelWindows(); + delete windowActions; + windowActions = new QActionGroup(this); + + windowMenu->addSeparator(); + + QAction* dockLast; + if (dockMenu->actions().count() != 0) + dockLast = dockMenu->actions().first(); + else + dockLast = nullptr; + + QWindow* activeWindow; + + if (QApplication::activeWindow()) + activeWindow = QApplication::activeWindow()->windowHandle(); + else + activeWindow = nullptr; + + for (int i = 0; i < windowList.size(); ++i) + { + if (closedWindow == windowList[i]) + continue; + + QAction* action = windowActions->addAction(windowList[i]->title()); + action->setCheckable(true); + action->setChecked(windowList[i] == activeWindow); + connect(action, SIGNAL(triggered()), windowMapper, SLOT(map())); + windowMapper->setMapping(action, windowList[i]); + windowMenu->addAction(action); + dockMenu->insertAction(dockLast, action); + } + + if (!dockLast->isSeparator()) + dockMenu->insertSeparator(dockLast); +} + +void Nexus::updateWindowsClosed() +{ + updateWindowsArg(static_cast(sender())->windowHandle()); +} + +void Nexus::updateWindowsStates() +{ + bool exists = false; + QWindowList windowList = QApplication::topLevelWindows(); + + for (QWindow* window : windowList) + { + if (!(window->windowState() & Qt::WindowMinimized)) + { + exists = true; + break; + } + } + + frontAction->setEnabled(exists); +} + +void Nexus::onOpenWindow(QObject* object) +{ + QWindow* window = static_cast(object); + + if (window->windowState() & QWindow::Minimized) + window->showNormal(); + + window->raise(); + window->requestActivate(); +} + +void Nexus::toggleFullscreen() +{ + QWidget* window = QApplication::activeWindow(); + + if (window->isFullScreen()) + window->showNormal(); + else + window->showFullScreen(); +} + +void Nexus::bringAllToFront() +{ + QWindowList windowList = QApplication::topLevelWindows(); + QWindow* focused = QApplication::focusWindow(); + + for (QWindow* window : windowList) + window->raise(); + + focused->raise(); +} +#endif diff --git a/src/nexus.h b/src/nexus.h index 53b2a0df2..95f132223 100644 --- a/src/nexus.h +++ b/src/nexus.h @@ -32,6 +32,10 @@ class Core; #ifdef Q_OS_MAC class QMenuBar; class QMenu; +class QAction; +class QWindow; +class QActionGroup; +class QSignalMapper; #endif /// This class is in charge of connecting various systems together @@ -59,7 +63,28 @@ public: #ifdef Q_OS_MAC QMenuBar* globalMenuBar; + QMenu* viewMenu; QMenu* windowMenu; + QAction* minimizeAction; + QAction* fullscreenAction; + QAction* frontAction; + QMenu* dockMenu; + +public slots: + void retranslateUi(); + void onWindowStateChanged(Qt::WindowStates state); + void updateWindows(); + void updateWindowsClosed(); + void updateWindowsStates(); + void onOpenWindow(QObject* object); + void toggleFullscreen(); + void bringAllToFront(); + +private: + void updateWindowsArg(QWindow *closedWindow); + + QSignalMapper* windowMapper; + QActionGroup* windowActions = nullptr; #endif private: diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index dfeea2a91..22e0301e3 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -46,7 +46,7 @@ QHash> ContentDialog::fr QHash> ContentDialog::groupList; ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent) - : QDialog(parent, Qt::Window) + : ActivateDialog(parent, Qt::Window) , activeChatroomWidget(nullptr) , settingsWidget(settingsWidget) { @@ -476,11 +476,15 @@ bool ContentDialog::event(QEvent* event) } currentDialog = this; + +#ifdef Q_OS_MAC + emit activated(); +#endif default: break; } - return QWidget::event(event); + return ActivateDialog::event(event); } void ContentDialog::dragEnterEvent(QDragEnterEvent *event) diff --git a/src/widget/contentdialog.h b/src/widget/contentdialog.h index d33512f3a..51904f087 100644 --- a/src/widget/contentdialog.h +++ b/src/widget/contentdialog.h @@ -20,7 +20,7 @@ #ifndef CONTENTDIALOG_H #define CONTENTDIALOG_H -#include +#include "src/widget/tool/activatedialog.h" #include #include "src/core/corestructs.h" #include "src/widget/genericchatitemlayout.h" @@ -37,7 +37,7 @@ class GroupWidget; class FriendListLayout; class SettingsWidget; -class ContentDialog : public QDialog +class ContentDialog : public ActivateDialog { Q_OBJECT public: @@ -66,6 +66,11 @@ public: static ContentDialog* getFriendDialog(int friendId); static ContentDialog* getGroupDialog(int groupId); +#ifdef Q_OS_MAC +signals: + void activated(); +#endif + public slots: void updateTitleUsername(const QString& username); void updateTitle(GenericChatroomWidget* chatroomWidget); diff --git a/src/widget/loginscreen.cpp b/src/widget/loginscreen.cpp index bf9ae2cd2..140502f9a 100644 --- a/src/widget/loginscreen.cpp +++ b/src/widget/loginscreen.cpp @@ -97,6 +97,16 @@ void LoginScreen::reset() ui->autoLoginCB->setChecked(Settings::getInstance().getAutoLogin()); } +#ifdef Q_OS_MAC +bool LoginScreen::event(QEvent* event) +{ + if (event->type() == QEvent::WindowActivate || event->type() == QEvent::WindowStateChange) + emit windowStateChanged(windowState()); + + return QWidget::event(event); +} +#endif + void LoginScreen::onNewProfilePageClicked() { ui->stackedWidget->setCurrentIndex(0); diff --git a/src/widget/loginscreen.h b/src/widget/loginscreen.h index 2c3cedcfc..66e027d34 100644 --- a/src/widget/loginscreen.h +++ b/src/widget/loginscreen.h @@ -37,6 +37,13 @@ public: ~LoginScreen(); void reset(); ///< Resets the UI, clears all fields +#ifdef Q_OS_MAC + bool event(QEvent* event) final override; + +signals: + void windowStateChanged(Qt::WindowStates states); +#endif + private slots: void onAutoLoginToggled(int state); void onLoginUsernameSelected(const QString& name); diff --git a/src/widget/tool/activatedialog.cpp b/src/widget/tool/activatedialog.cpp new file mode 100644 index 000000000..1e4426ca8 --- /dev/null +++ b/src/widget/tool/activatedialog.cpp @@ -0,0 +1,36 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + + +#include "activatedialog.h" +#include + +ActivateDialog::ActivateDialog(QWidget *parent, Qt::WindowFlags f) + : QDialog(parent, f) +{ + +} + +bool ActivateDialog::event(QEvent* event) +{ + if (event->type() == QEvent::WindowActivate || event->type() == QEvent::WindowStateChange) + emit windowStateChanged(windowState()); + + return QDialog::event(event); +} diff --git a/src/widget/tool/activatedialog.h b/src/widget/tool/activatedialog.h new file mode 100644 index 000000000..d6b71cfb8 --- /dev/null +++ b/src/widget/tool/activatedialog.h @@ -0,0 +1,36 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre 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 3 of the License, or + (at your option) any later version. + + qTox 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 qTox. If not, see . +*/ + +#ifndef ACTIVATEDIALOG_H +#define ACTIVATEDIALOG_H + +#include + +class ActivateDialog : public QDialog +{ + Q_OBJECT +public: + ActivateDialog(QWidget* parent = 0, Qt::WindowFlags f = 0); + bool event(QEvent* event) override; + +signals: + void windowStateChanged(Qt::WindowStates state); +}; + +#endif // ACTIVATEDIALOG_H diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 1f2c5813b..7ab927bba 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -50,6 +50,7 @@ #include "src/widget/form/profileform.h" #include "src/widget/form/settingswidget.h" #include "tool/removefrienddialog.h" +#include "src/widget/tool/activatedialog.h" #include #include #include @@ -76,6 +77,7 @@ #ifdef Q_OS_MAC #include #include +#include #endif #ifdef Q_OS_ANDROID @@ -243,13 +245,17 @@ void Widget::init() new QShortcut(Qt::CTRL + Qt::Key_PageDown, this, SLOT(nextContact())); #ifdef Q_OS_MAC + QMenuBar* globalMenu = Nexus::getInstance().globalMenuBar; QAction* windowMenu = Nexus::getInstance().windowMenu->menuAction(); - QAction* fileMenu = Nexus::getInstance().globalMenuBar->insertMenu(windowMenu, new QMenu(tr("File"), this)); + QAction* viewMenu = Nexus::getInstance().viewMenu->menuAction(); + QAction* frontAction = Nexus::getInstance().frontAction; - QAction* editProfileAction = fileMenu->menu()->addAction(tr("Edit Profile")); + fileMenu = globalMenu->insertMenu(viewMenu, new QMenu(this)); + + editProfileAction = fileMenu->menu()->addAction(QString()); connect(editProfileAction, &QAction::triggered, this, &Widget::showProfile); - QMenu* changeStatusMenu = fileMenu->menu()->addMenu(tr("Change Status")); + changeStatusMenu = fileMenu->menu()->addMenu(QString()); fileMenu->menu()->addAction(changeStatusMenu->menuAction()); changeStatusMenu->addAction(statusOnline); changeStatusMenu->addSeparator(); @@ -257,50 +263,47 @@ void Widget::init() changeStatusMenu->addAction(statusBusy); fileMenu->menu()->addSeparator(); - QAction* logoutAction = fileMenu->menu()->addAction(tr("Log out")); + logoutAction = fileMenu->menu()->addAction(QString()); connect(logoutAction, &QAction::triggered, [this]() { Nexus::getInstance().showLogin(); }); - QAction* editMenu = Nexus::getInstance().globalMenuBar->insertMenu(windowMenu, new QMenu(tr("Edit"), this)); + editMenu = globalMenu->insertMenu(viewMenu, new QMenu(this)); editMenu->menu()->addSeparator(); - QAction* viewMenu = Nexus::getInstance().globalMenuBar->insertMenu(windowMenu, new QMenu(tr("View"), this)); + viewMenu->menu()->insertMenu(Nexus::getInstance().fullscreenAction, filterMenu); - QMenu* filterMenu = viewMenu->menu()->addMenu(tr("Filter...")); - filterMenu->addAction(filterDisplayName); - filterMenu->addAction(filterDisplayActivity); - filterMenu->addSeparator(); - filterMenu->addAction(filterAllAction); - filterMenu->addAction(filterOnlineAction); - filterMenu->addAction(filterOfflineAction); - filterMenu->addAction(filterFriendsAction); - filterMenu->addAction(filterGroupsAction); + viewMenu->menu()->insertSeparator(Nexus::getInstance().fullscreenAction); - viewMenu->menu()->addSeparator(); - fullscreenAction = viewMenu->menu()->addAction(tr("Enter Fullscreen")); - fullscreenAction->setShortcut(QKeySequence::FullScreen); - connect(fullscreenAction, &QAction::triggered, this, &Widget::toggleFullscreen); + contactMenu = globalMenu->insertMenu(windowMenu, new QMenu(this)); - QAction* contactMenu = Nexus::getInstance().globalMenuBar->insertMenu(windowMenu, new QMenu(tr("Contacts"), this)); - - QAction* addContactAction = contactMenu->menu()->addAction(tr("Add Contact...")); + addContactAction = contactMenu->menu()->addAction(QString()); connect(addContactAction, &QAction::triggered, this, &Widget::onAddClicked); - QAction* nextConversationAction = new QAction(tr("Next Conversation"), this); - Nexus::getInstance().windowMenu->addAction(nextConversationAction); + nextConversationAction = new QAction(this); + Nexus::getInstance().windowMenu->insertAction(frontAction, nextConversationAction); nextConversationAction->setShortcut(QKeySequence::SelectNextPage); - connect(nextConversationAction, &QAction::triggered, this, &Widget::nextContact); + connect(nextConversationAction, &QAction::triggered, [this]() + { + if (ContentDialog::current() == QApplication::activeWindow()) + ContentDialog::current()->cycleContacts(true); + else if (QApplication::activeWindow() == this) + cycleContacts(true); + }); - QAction* previousConversationAction = new QAction(tr("Previous Conversation"), this); - Nexus::getInstance().windowMenu->addAction(previousConversationAction); + previousConversationAction = new QAction(this); + Nexus::getInstance().windowMenu->insertAction(frontAction, previousConversationAction); previousConversationAction->setShortcut(QKeySequence::SelectPreviousPage); - connect(previousConversationAction, &QAction::triggered, this, &Widget::previousContact); + connect(previousConversationAction, &QAction::triggered, [this] + { + if (ContentDialog::current() == QApplication::activeWindow()) + ContentDialog::current()->cycleContacts(false); + else if (QApplication::activeWindow() == this) + cycleContacts(false); + }); - Nexus::getInstance().windowMenu->addSeparator(); - QAction* frontAction = Nexus::getInstance().windowMenu->addAction(tr("Bring All to Front")); - connect(frontAction, &QAction::triggered, this, &Widget::bringAllToFront); + windowMenu->menu()->insertSeparator(frontAction); QAction* preferencesAction = viewMenu->menu()->addAction(QString()); preferencesAction->setMenuRole(QAction::PreferencesRole); @@ -313,6 +316,16 @@ void Widget::init() onSettingsClicked(); settingsWidget->showAbout(); }); + + QMenu* dockChangeStatusMenu = new QMenu(tr("Status"), this); + dockChangeStatusMenu->addAction(statusOnline); + statusOnline->setIconVisibleInMenu(true); + dockChangeStatusMenu->addSeparator(); + dockChangeStatusMenu->addAction(statusAway); + dockChangeStatusMenu->addAction(statusBusy); + Nexus::getInstance().dockMenu->addAction(dockChangeStatusMenu->menuAction()); + + connect(this, &Widget::windowStateChanged, &Nexus::getInstance(), &Nexus::onWindowStateChanged); #endif contentLayout = nullptr; @@ -343,6 +356,8 @@ void Widget::init() if (!Settings::getInstance().getShowSystemTray()) show(); + + Nexus::getInstance().updateWindows(); } bool Widget::eventFilter(QObject *obj, QEvent *event) @@ -357,6 +372,8 @@ bool Widget::eventFilter(QObject *obj, QEvent *event) else wasMaximized = false; } + + emit windowStateChanged(windowState()); } return false; } @@ -1295,16 +1312,24 @@ void Widget::clearContactsList() ContentDialog* Widget::createContentDialog() const { - return new ContentDialog(settingsWidget); + ContentDialog* contentDialog = new ContentDialog(settingsWidget); + contentDialog->show(); +#ifdef Q_OS_MAC + connect(contentDialog, &ContentDialog::destroyed, &Nexus::getInstance(), &Nexus::updateWindowsClosed); + connect(contentDialog, &ContentDialog::windowStateChanged, &Nexus::getInstance(), &Nexus::onWindowStateChanged); + connect(contentDialog->windowHandle(), &QWindow::windowTitleChanged, &Nexus::getInstance(), &Nexus::updateWindows); + Nexus::getInstance().updateWindows(); +#endif + return contentDialog; } -ContentLayout* Widget::createContentDialog(DialogType type) const +ContentLayout* Widget::createContentDialog(DialogType type) { - class Dialog : public QDialog + class Dialog : public ActivateDialog { public: Dialog(DialogType type) - : QDialog() + : ActivateDialog() , type(type) { restoreGeometry(Settings::getInstance().getDialogSettingsGeometry()); @@ -1343,7 +1368,7 @@ ContentLayout* Widget::createContentDialog(DialogType type) const DialogType type; }; - QDialog* dialog = new Dialog(type); + Dialog* dialog = new Dialog(type); dialog->setAttribute(Qt::WA_DeleteOnClose); ContentLayout* contentLayoutDialog = new ContentLayout(dialog); @@ -1355,6 +1380,13 @@ ContentLayout* Widget::createContentDialog(DialogType type) const dialog->setAttribute(Qt::WA_DeleteOnClose); dialog->show(); +#ifdef Q_OS_MAC + connect(dialog, &Dialog::destroyed, &Nexus::getInstance(), &Nexus::updateWindowsClosed); + connect(dialog, &ActivateDialog::windowStateChanged, &Nexus::getInstance(), &Nexus::updateWindowsStates); + connect(dialog->windowHandle(), &QWindow::windowTitleChanged, &Nexus::getInstance(), &Nexus::updateWindows); + Nexus::getInstance().updateWindows(); +#endif + return contentLayoutDialog; } @@ -1549,6 +1581,14 @@ bool Widget::event(QEvent * e) eventIcon = false; updateIcons(); } + +#ifdef Q_OS_MAC + emit windowStateChanged(windowState()); + + case QEvent::WindowStateChange: + Nexus::getInstance().updateWindowsStates(); +#endif + default: break; } @@ -1630,16 +1670,7 @@ void Widget::onTryCreateTrayIcon() } #ifdef Q_OS_MAC - QMenu* changeStatusMenu = new QMenu(tr("Status"), this); - changeStatusMenu->addAction(statusOnline); - statusOnline->setIconVisibleInMenu(true); - changeStatusMenu->addSeparator(); - changeStatusMenu->addAction(statusAway); - changeStatusMenu->addAction(statusBusy); - - QMenu* dockMenu = new QMenu(this); - dockMenu->addAction(changeStatusMenu->menuAction()); - qt_mac_set_dock_menu(dockMenu); + qt_mac_set_dock_menu(Nexus::getInstance().dockMenu); #endif } else if (!isVisible()) @@ -2003,33 +2034,20 @@ void Widget::retranslateUi() if (!Settings::getInstance().getSeparateWindow()) setWindowTitle(fromDialogType(SettingDialog)); -} #ifdef Q_OS_MAC -void Widget::bringAllToFront() -{ - QWindowList windowList = QApplication::topLevelWindows(); - for (QWindow* window : windowList) - { - window->raise(); - window->showNormal(); - window->requestActivate(); - } -} + Nexus::getInstance().retranslateUi(); -void Widget::toggleFullscreen() -{ - QWidget* window = QApplication::activeWindow(); + filterMenu->menuAction()->setText(tr("Filter...")); - if (window->isFullScreen()) - { - window->showNormal(); - fullscreenAction->setText(tr("Enter Fullscreen")); - } - else - { - window->showFullScreen(); - fullscreenAction->setText(tr("Exit Fullscreen")); - } -} + fileMenu->setText(tr("File")); + editMenu->setText(tr("Edit")); + contactMenu->setText(tr("Contacts")); + changeStatusMenu->menuAction()->setText(tr("Change Status")); + editProfileAction->setText(tr("Edit Profile")); + logoutAction->setText(tr("Log out")); + addContactAction->setText(tr("Add Contact...")); + nextConversationAction->setText(tr("Next Conversation")); + previousConversationAction->setText(tr("Previous Conversation")); #endif +} diff --git a/src/widget/widget.h b/src/widget/widget.h index 71406f70d..c29e3bf8c 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -83,7 +83,7 @@ public: static QString fromDialogType(DialogType type); ContentDialog* createContentDialog() const; - ContentLayout* createContentDialog(DialogType type) const; + ContentLayout* createContentDialog(DialogType type); static void confirmExecutableOpen(const QFileInfo file); @@ -147,6 +147,9 @@ signals: void usernameChanged(const QString& username); void statusMessageChanged(const QString& statusMessage); void resized(); +#ifdef Q_OS_MAC + void windowStateChanged(Qt::WindowStates states); +#endif protected: virtual bool eventFilter(QObject *obj, QEvent *event) final override; @@ -178,11 +181,6 @@ private slots: void processOfflineMsgs(); void friendListContextMenu(const QPoint &pos); -#ifdef Q_OS_MAC - void bringAllToFront(); - void toggleFullscreen(); -#endif - private: enum ActiveToolMenuButton { AddButton, @@ -263,7 +261,15 @@ private: bool wasMaximized = false; #ifdef Q_OS_MAC - QAction* fullscreenAction; + QAction* fileMenu; + QAction* editMenu; + QAction* contactMenu; + QMenu* changeStatusMenu; + QAction* editProfileAction; + QAction* logoutAction; + QAction* addContactAction; + QAction* nextConversationAction; + QAction* previousConversationAction; #endif }; From 828dbf436f7dff393aef4ac17095ce4c6fb5473f Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Thu, 30 Jul 2015 07:53:36 -0400 Subject: [PATCH 20/24] Multi-window: Fix compile for non-OS X --- src/widget/widget.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 7ab927bba..8aa12d2fd 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -357,7 +357,9 @@ void Widget::init() if (!Settings::getInstance().getShowSystemTray()) show(); +#ifdef Q_OS_MAC Nexus::getInstance().updateWindows(); +#endif } bool Widget::eventFilter(QObject *obj, QEvent *event) @@ -373,7 +375,9 @@ bool Widget::eventFilter(QObject *obj, QEvent *event) wasMaximized = false; } +#ifdef Q_OS_MAC emit windowStateChanged(windowState()); +#endif } return false; } From 280737fda841b31fda4e6936ab2741a5e1ee7a30 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Thu, 30 Jul 2015 09:37:52 -0400 Subject: [PATCH 21/24] Multi-window: Fix group chat event flag regression --- src/widget/widget.cpp | 46 ++++++++++++++++++++++--------------------- src/widget/widget.h | 4 ++-- 2 files changed, 26 insertions(+), 24 deletions(-) diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 8aa12d2fd..fd1fb2826 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1127,7 +1127,7 @@ bool Widget::newFriendMessageAlert(int friendId) return false; } -bool Widget::newGroupMessageAlert(int groupId) +bool Widget::newGroupMessageAlert(int groupId, bool notify) { bool hasActive; QWidget* currentWindow; @@ -1145,7 +1145,7 @@ bool Widget::newGroupMessageAlert(int groupId) hasActive = g->getGroupWidget() == activeChatroomWidget; } - if (newMessageAlert(currentWindow, hasActive)) + if (newMessageAlert(currentWindow, hasActive, notify)) { g->setEventFlag(true); g->getGroupWidget()->updateStatusLight(); @@ -1183,7 +1183,7 @@ QString Widget::fromDialogType(DialogType type) } } -bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive) +bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive, bool notify) { bool inactiveWindow = isMinimized() || !currentWindow->isActiveWindow(); @@ -1193,28 +1193,31 @@ bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive) if (ui->statusButton->property("status").toString() == "busy") return false; - QApplication::alert(currentWindow); - - if (Settings::getInstance().getShowWindow()) + if (notify) { - currentWindow->show(); - if (inactiveWindow && Settings::getInstance().getShowInFront()) - currentWindow->activateWindow(); - } + QApplication::alert(currentWindow); - if (Settings::getInstance().getNotifySound()) - { - static QFile sndFile(":audio/notification.pcm"); - static QByteArray sndData; - - if (sndData.isEmpty()) + if (Settings::getInstance().getShowWindow()) { - sndFile.open(QIODevice::ReadOnly); - sndData = sndFile.readAll(); - sndFile.close(); + currentWindow->show(); + if (inactiveWindow && Settings::getInstance().getShowInFront()) + currentWindow->activateWindow(); } - Audio::playMono16Sound(sndData); + if (Settings::getInstance().getNotifySound()) + { + static QFile sndFile(":audio/notification.pcm"); + static QByteArray sndData; + + if (sndData.isEmpty()) + { + sndFile.open(QIODevice::ReadOnly); + sndData = sndFile.readAll(); + sndFile.close(); + } + + Audio::playMono16Sound(sndData); + } } return true; @@ -1438,8 +1441,7 @@ void Widget::onGroupMessageReceived(int groupnumber, int peernumber, const QStri else g->getChatForm()->addMessage(author, message, isAction, QDateTime::currentDateTime(), true); - if (targeted || Settings::getInstance().getGroupAlwaysNotify()) - newGroupMessageAlert(g->getGroupId()); + newGroupMessageAlert(g->getGroupId(), targeted || Settings::getInstance().getGroupAlwaysNotify()); } void Widget::onGroupNamelistChanged(int groupnumber, int peernumber, uint8_t Change) diff --git a/src/widget/widget.h b/src/widget/widget.h index c29e3bf8c..5ee59bfa5 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -68,7 +68,7 @@ public: void addFriendDialog(Friend* frnd, ContentDialog* dialog); void addGroupDialog(Group* group, ContentDialog* dialog); bool newFriendMessageAlert(int friendId); - bool newGroupMessageAlert(int groupId); + bool newGroupMessageAlert(int groupId, bool notify); bool getIsWindowMinimized(); void updateIcons(); void clearContactsList(); @@ -200,7 +200,7 @@ private: }; private: - bool newMessageAlert(QWidget* currentWindow, bool isActive); + bool newMessageAlert(QWidget* currentWindow, bool isActive, bool notify = true); void setActiveToolMenuButton(ActiveToolMenuButton newActiveButton); void hideMainForms(GenericChatroomWidget* chatroomWidget); Group *createGroup(int groupId); From 16ebb7f3a9b98f02a57ded09dc931045797f4bd6 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 18 Aug 2015 12:10:08 -0400 Subject: [PATCH 22/24] Multi-window: Unminimize requested window --- src/widget/contentdialog.cpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index 22e0301e3..f63d9a93b 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -663,6 +663,9 @@ bool ContentDialog::existsWidget(int id, bool focus, const QHash(iter.value())->windowState() & Qt::WindowMinimized) + std::get<0>(iter.value())->showNormal(); + std::get<0>(iter.value())->raise(); std::get<0>(iter.value())->activateWindow(); std::get<0>(iter.value())->onChatroomWidgetClicked(std::get<1>(iter.value()), false); From d0e682214b83dac6612fc403287622b8f6a56fbb Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 18 Aug 2015 13:08:32 -0400 Subject: [PATCH 23/24] Multi-window: Fix blinking tray icon regression --- src/widget/widget.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index fd1fb2826..da6e89869 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1196,6 +1196,7 @@ bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive, bool notify) if (notify) { QApplication::alert(currentWindow); + eventFlag = true; if (Settings::getInstance().getShowWindow()) { From 1f81e1f0666b919b412ff5cf4926a8c0fd52bf0c Mon Sep 17 00:00:00 2001 From: tux3 Date: Sat, 5 Sep 2015 23:53:30 +0200 Subject: [PATCH 24/24] Don't reload content dialog if nothing changed Should fix the zetok's Resizing problem in #1938 --- src/widget/contentdialog.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/widget/contentdialog.cpp b/src/widget/contentdialog.cpp index f63d9a93b..86e2e3cdc 100644 --- a/src/widget/contentdialog.cpp +++ b/src/widget/contentdialog.cpp @@ -588,6 +588,10 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget *widget, bool return; } + // If we clicked on the currently active widget, don't reload and relayout everything + if (activeChatroomWidget == widget) + return; + contentLayout->clear(); if (activeChatroomWidget != nullptr)