1
0
mirror of https://github.com/qTox/qTox.git synced 2024-03-22 14:00:36 +08:00

refactor: ContentDialog refactoring

This commit is contained in:
Diadlo 2017-02-26 17:22:28 +03:00
parent c7202c8b24
commit add8d51a29
No known key found for this signature in database
GPG Key ID: 5AF9F2E29107C727
6 changed files with 319 additions and 268 deletions

View File

@ -7,7 +7,7 @@
class QFile;
class QTimer;
enum class Status : int
enum class Status
{
Online = 0,
Away,

View File

@ -44,38 +44,50 @@
#include "src/widget/translator.h"
#include "tool/adjustingscrollarea.h"
QString ContentDialog::username = "";
ContentDialog* ContentDialog::currentDialog = nullptr;
QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>> ContentDialog::friendList;
QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>> ContentDialog::groupList;
static const int minWidget = 220;
static const int minHeight = 220;
static const QSize minSize(minHeight, minWidget);
static const QSize defaultSize(720, 400);
ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent)
: ActivateDialog(parent, Qt::Window)
, splitter{new QSplitter(this)}
, friendLayout{new FriendListLayout(this)}
, activeChatroomWidget(nullptr)
, settingsWidget(settingsWidget)
, videoSurfaceSize(QSize())
, videoCount(0)
{
QVBoxLayout* boxLayout = new QVBoxLayout(this);
boxLayout->setMargin(0);
boxLayout->setSpacing(0);
splitter = new QSplitter(this);
const Settings& s = Settings::getInstance();
setStyleSheet(Style::getStylesheet(":/ui/contentDialog/contentDialog.css"));
splitter->setHandleWidth(6);
friendLayout->setMargin(0);
friendLayout->setSpacing(0);
layouts = {
friendLayout->getLayoutOnline(),
groupLayout.getLayout(),
friendLayout->getLayoutOffline()
};
if (s.getGroupchatPosition()) {
layouts.swap(0, 1);
}
QWidget* friendWidget = new QWidget();
friendWidget->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
friendWidget->setAutoFillBackground(true);
friendLayout = new FriendListLayout();
friendLayout->setMargin(0);
friendLayout->setSpacing(0);
friendWidget->setLayout(friendLayout);
onGroupchatPositionChanged(Settings::getInstance().getGroupchatPosition());
onGroupchatPositionChanged(s.getGroupchatPosition());
QScrollArea* friendScroll = new QScrollArea(this);
friendScroll->setMinimumWidth(220);
friendScroll->setMinimumWidth(minWidget);
friendScroll->setFrameStyle(QFrame::NoFrame);
friendScroll->setLayoutDirection(Qt::RightToLeft);
friendScroll->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
@ -85,6 +97,7 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent)
QWidget* contentWidget = new QWidget(this);
contentWidget->setAutoFillBackground(true);
contentLayout = new ContentLayout(contentWidget);
contentLayout->setMargin(0);
contentLayout->setSpacing(0);
@ -93,13 +106,13 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent)
splitter->addWidget(contentWidget);
splitter->setStretchFactor(1, 1);
splitter->setCollapsible(1, false);
QVBoxLayout* boxLayout = new QVBoxLayout(this);
boxLayout->setMargin(0);
boxLayout->setSpacing(0);
boxLayout->addWidget(splitter);
const Settings& s = Settings::getInstance();
connect(&s, &Settings::groupchatPositionChanged, this, &ContentDialog::onGroupchatPositionChanged);
connect(splitter, &QSplitter::splitterMoved, this, &ContentDialog::saveSplitterState);
setMinimumSize(500, 220);
setMinimumSize(minSize);
setAttribute(Qt::WA_DeleteOnClose);
QByteArray geometry = s.getDialogGeometry();
@ -107,12 +120,11 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent)
if (!geometry.isNull()) {
restoreGeometry(geometry);
} else {
resize(720, 400);
resize(defaultSize);
}
QByteArray splitterState = s.getDialogSplitterState();
SplitterRestorer restorer(splitter);
restorer.restore(splitterState, size());
restorer.restore(s.getDialogSettingsGeometry(), size());
currentDialog = this;
setAcceptDrops(true);
@ -123,35 +135,30 @@ ContentDialog::ContentDialog(SettingsWidget* settingsWidget, QWidget* parent)
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::updateTitleAndStatusIcon);
connect(&s, &Settings::groupchatPositionChanged, this, &ContentDialog::onGroupchatPositionChanged);
connect(splitter, &QSplitter::splitterMoved, this, &ContentDialog::saveSplitterState);
Translator::registerHandler(std::bind(&ContentDialog::retranslateUi, this), this);
}
void ContentDialog::removeCurrent(QHash<int, ContactInfo>& infos) {
for (auto it = infos.begin(); it != infos.end(); ) {
if (std::get<0>(*it) == this) {
it = infos.erase(it);
} else {
++it;
}
}
}
ContentDialog::~ContentDialog()
{
if (currentDialog == this)
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;
}
auto groupIt = groupList.begin();
while (groupIt != groupList.end()) {
if (std::get<0>(groupIt.value()) == this) {
groupIt = groupList.erase(groupIt);
continue;
}
++groupIt;
}
removeCurrent(friendList);
removeCurrent(groupList);
Translator::unregister(this);
}
@ -159,24 +166,19 @@ ContentDialog::~ContentDialog()
FriendWidget* ContentDialog::addFriend(int friendId, QString id)
{
FriendWidget* friendWidget = new FriendWidget(friendId, id);
friendLayout->addFriendWidget(friendWidget, FriendList::findFriend(friendId)->getStatus());
Friend* frnd = friendWidget->getFriend();
friendLayout->addFriendWidget(friendWidget, frnd->getStatus());
connect(frnd, &Friend::aliasChanged, 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);
ContentDialog* lastDialog = getFriendDialog(friendId);
if (lastDialog != nullptr)
if (lastDialog) {
lastDialog->removeFriend(friendId);
}
friendList.insert(friendId, std::make_tuple(this, friendWidget));
// FIXME: emit should be removed
@ -198,8 +200,9 @@ GroupWidget* ContentDialog::addGroup(int groupId, const QString& name)
ContentDialog* lastDialog = getGroupDialog(groupId);
if (lastDialog != nullptr)
if (lastDialog) {
lastDialog->removeGroup(groupId);
}
groupList.insert(groupId, std::make_tuple(this, groupWidget));
// FIXME: emit should be removed
@ -211,17 +214,18 @@ GroupWidget* ContentDialog::addGroup(int groupId, const QString& name)
void ContentDialog::removeFriend(int friendId)
{
auto iter = friendList.find(friendId);
if (iter == friendList.end())
if (iter == friendList.end()) {
return;
}
FriendWidget* chatroomWidget = static_cast<FriendWidget*>(std::get<1>(iter.value()));
disconnect(chatroomWidget->getFriend(), &Friend::aliasChanged, this,
&ContentDialog::updateFriendWidget);
// Need to find replacement to show here instead.
if (activeChatroomWidget == chatroomWidget)
if (activeChatroomWidget == chatroomWidget) {
cycleContacts(true, false);
}
friendLayout->removeFriendWidget(chatroomWidget, Status::Offline);
friendLayout->removeFriendWidget(chatroomWidget, Status::Online);
@ -248,15 +252,16 @@ void ContentDialog::removeGroup(int groupId)
}
auto iter = groupList.find(groupId);
if (iter == groupList.end())
if (iter == groupList.end()) {
return;
}
GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value());
// Need to find replacement to show here instead.
if (activeChatroomWidget == chatroomWidget)
if (activeChatroomWidget == chatroomWidget) {
cycleContacts(true, false);
}
groupLayout.removeSortedWidget(chatroomWidget);
chatroomWidget->deleteLater();
@ -288,75 +293,90 @@ int ContentDialog::chatroomWidgetCount() const
void ContentDialog::ensureSplitterVisible()
{
if (splitter->sizes().at(0) == 0)
if (splitter->sizes().at(0) == 0) {
splitter->setSizes({1, 1});
}
update();
}
void ContentDialog::cycleContacts(bool forward, bool loop)
int ContentDialog::getCurrentLayout(QLayout*& layout)
{
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);
layout = friendLayout->getLayoutOnline();
int index = friendLayout->indexOfFriendWidget(activeChatroomWidget, true);
if (index != -1) {
return index;
}
if (!loop && index == currentLayout->count() - 1) {
layout = friendLayout->getLayoutOffline();
index = friendLayout->indexOfFriendWidget(activeChatroomWidget, false);
if (index != -1) {
return index;
}
layout = groupLayout.getLayout();
index = groupLayout.indexOfSortedWidget(activeChatroomWidget);
if (index != -1) {
return index;
}
layout = nullptr;
return -1;
}
void ContentDialog::cycleContacts(bool forward, bool inverse)
{
QLayout* currentLayout;
int index = getCurrentLayout(currentLayout);
if (!currentLayout || index == -1) {
return;
}
if (!inverse && index == currentLayout->count() - 1) {
bool groupsOnTop = Settings::getInstance().getGroupchatPosition();
bool offlineEmpty = friendLayout->getLayoutOffline()->isEmpty();
bool onlineEmpty =
offlineEmpty && (friendLayout->getLayoutOnline()->isEmpty() || !groupsOnTop);
bool groupsEmpty = offlineEmpty && (groupLayout.getLayout()->isEmpty() || groupsOnTop);
bool onlineEmpty = friendLayout->getLayoutOnline()->isEmpty();
bool groupsEmpty = groupLayout.getLayout()->isEmpty();
bool isCurOffline = currentLayout == friendLayout->getLayoutOffline();
bool isCurOnline = currentLayout == friendLayout->getLayoutOnline();
bool isCurGroup = currentLayout == groupLayout.getLayout();
bool nextIsEmpty = isCurOnline && offlineEmpty && (groupsEmpty || groupsOnTop) ||
isCurGroup && offlineEmpty && (onlineEmpty || !groupsOnTop) ||
isCurOffline;
if ((currentLayout == friendLayout->getLayoutOffline())
|| (currentLayout == friendLayout->getLayoutOnline() && groupsEmpty)
|| (currentLayout == groupLayout.getLayout() && onlineEmpty)) {
if (nextIsEmpty) {
forward = !forward;
}
}
index += forward ? 1 : -1;
for (;;) {
// Bounds checking.
// If goes out of the layout, then go to the next and skip empty. This loop goes more
// then 1 time, because for empty layout index will be out of interval (0 < 0 || 0 >= 0)
while (index < 0 || index >= currentLayout->count()) {
int oldCount = currentLayout->count();
currentLayout = nextLayout(currentLayout, forward);
int newCount = currentLayout->count();
if (index < 0) {
currentLayout = nextLayout(currentLayout, forward);
index = currentLayout->count() - 1;
continue;
} else if (index >= currentLayout->count()) {
currentLayout = nextLayout(currentLayout, forward);
index = newCount - 1;
} else if (index >= oldCount) {
index = 0;
continue;
}
}
GenericChatroomWidget* chatWidget =
qobject_cast<GenericChatroomWidget*>(currentLayout->itemAt(index)->widget());
if (chatWidget != nullptr && chatWidget != activeChatroomWidget) {
// FIXME: emit should be removed
emit chatWidget->chatroomWidgetClicked(chatWidget, false);
}
return;
QWidget* widget = currentLayout->itemAt(index)->widget();
GenericChatroomWidget* chatWidget = qobject_cast<GenericChatroomWidget*>(widget);
if (chatWidget && chatWidget != activeChatroomWidget) {
// FIXME: emit should be removed
emit chatWidget->chatroomWidgetClicked(chatWidget, false);
}
}
void ContentDialog::onVideoShow(QSize size)
{
++videoCount;
if (videoCount > 1)
if (videoCount > 1) {
return;
}
videoSurfaceSize = size;
QSize minSize = minimumSize();
@ -366,8 +386,9 @@ void ContentDialog::onVideoShow(QSize size)
void ContentDialog::onVideoHide()
{
videoCount--;
if (videoCount > 0)
if (videoCount > 0) {
return;
}
QSize minSize = minimumSize();
setMinimumSize(minSize - videoSurfaceSize);
@ -379,25 +400,37 @@ ContentDialog* ContentDialog::current()
return currentDialog;
}
bool ContentDialog::existsFriendWidget(int friendId, bool focus)
bool ContentDialog::existsFriendWidget(int friendId)
{
return existsWidget(friendId, focus, friendList);
return existsWidget(friendId, friendList);
}
bool ContentDialog::existsGroupWidget(int groupId, bool focus)
bool ContentDialog::existsGroupWidget(int groupId)
{
return existsWidget(groupId, focus, groupList);
return existsWidget(groupId, groupList);
}
void ContentDialog::focusFriend(int friendId)
{
focusDialog(friendId, friendList);
}
void ContentDialog::focusGroup(int groupId)
{
focusDialog(groupId, groupList);
}
void ContentDialog::updateFriendStatus(int friendId)
{
updateStatus(friendId, friendList);
ContentDialog* contentDialog = getFriendDialog(friendId);
if (contentDialog != nullptr) {
FriendWidget* friendWidget =
static_cast<FriendWidget*>(std::get<1>(friendList.find(friendId).value()));
contentDialog->friendLayout->addFriendWidget(friendWidget,
FriendList::findFriend(friendId)->getStatus());
if (contentDialog) {
auto iter = friendList.find(friendId).value();
GenericChatroomWidget* widget = std::get<1>(iter);
FriendWidget* friendWidget = static_cast<FriendWidget*>(widget);
Friend* f = FriendList::findFriend(friendId);
contentDialog->friendLayout->addFriendWidget(friendWidget, f->getStatus());
}
}
@ -405,8 +438,9 @@ void ContentDialog::updateFriendStatusMessage(int friendId, const QString& messa
{
auto iter = friendList.find(friendId);
if (iter == friendList.end())
if (iter == friendList.end()) {
return;
}
std::get<1>(iter.value())->setStatusMsg(message);
}
@ -436,42 +470,43 @@ ContentDialog* ContentDialog::getGroupDialog(int groupId)
return getDialog(groupId, groupList);
}
void ContentDialog::updateTitleAndStatusIcon(const QString& username)
void ContentDialog::updateTitleAndStatusIcon()
{
if (displayWidget != nullptr) {
setWindowTitle(displayWidget->getTitle() + QStringLiteral(" - ") + username);
// it's null when it's a groupchat
if (displayWidget->getFriend() == nullptr) {
setWindowIcon(QIcon(":/img/group.svg"));
return;
}
Status currentStatus = displayWidget->getFriend()->getStatus();
switch (currentStatus) {
case Status::Online:
setWindowIcon(QIcon(":/img/status/dot_online.svg"));
break;
case Status::Away:
setWindowIcon(QIcon(":/img/status/dot_away.svg"));
break;
case Status::Busy:
setWindowIcon(QIcon(":/img/status/dot_busy.svg"));
break;
case Status::Offline:
setWindowIcon(QIcon(":/img/status/dot_offline.svg"));
break;
}
} else
if (!activeChatroomWidget) {
setWindowTitle(username);
return;
}
setWindowTitle(activeChatroomWidget->getTitle() + QStringLiteral(" - ") + username);
bool isGroupchat = activeChatroomWidget->getGroup() != nullptr;
if (isGroupchat) {
setWindowIcon(QIcon(":/img/group.svg"));
return;
}
Status currentStatus = activeChatroomWidget->getFriend()->getStatus();
QMap<Status, QIcon> icons {
{Status::Online, QIcon(":/img/status/dot_online.svg")},
{Status::Away, QIcon(":/img/status/dot_away.svg")},
{Status::Busy, QIcon(":/img/status/dot_busy.svg")},
{Status::Offline, QIcon(":/img/status/dot_offline.svg")}
};
setWindowIcon(icons[currentStatus]);
}
void ContentDialog::updateTitle(GenericChatroomWidget* chatroomWidget)
/**
* @brief Update layouts order according to settings.
* @param groupOnTop If true move groupchat layout on the top. Move under online otherwise.
*/
void ContentDialog::reorderLayouts(bool newGroupOnTop)
{
displayWidget = chatroomWidget;
updateTitleAndStatusIcon(Core::getInstance()->getUsername());
bool oldGroupOnTop = layouts.first() == groupLayout.getLayout();
if (newGroupOnTop != oldGroupOnTop) {
layouts.swap(0, 1);
}
}
void ContentDialog::previousContact()
@ -484,14 +519,21 @@ void ContentDialog::nextContact()
cycleContacts(true);
}
void ContentDialog::setUsername(const QString& newName)
{
username = newName;
updateTitleAndStatusIcon();
}
bool ContentDialog::event(QEvent* event)
{
switch (event->type()) {
case QEvent::WindowActivate:
if (activeChatroomWidget != nullptr) {
if (activeChatroomWidget) {
activeChatroomWidget->resetEventFlags();
activeChatroomWidget->updateStatusLight();
updateTitle(activeChatroomWidget);
updateTitleAndStatusIcon();
Friend* frnd = activeChatroomWidget->getFriend();
Group* group = activeChatroomWidget->getGroup();
@ -523,27 +565,32 @@ void ContentDialog::dragEnterEvent(QDragEnterEvent* event)
if (frnd) {
ToxId toxId(event->mimeData()->text());
Friend* contact = FriendList::findFriend(toxId.getPublicKey());
if (!contact)
if (!contact) {
return;
}
int friendId = contact->getFriendId();
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() || std::get<0>(iter.value()) != this)
if (iter == friendList.end() || std::get<0>(iter.value()) != this) {
event->acceptProposedAction();
}
} else if (group) {
if (!event->mimeData()->hasFormat("groupId"))
if (!event->mimeData()->hasFormat("groupId")) {
return;
}
int groupId = event->mimeData()->data("groupId").toInt();
Group* contact = GroupList::findGroup(groupId);
if (!contact)
if (!contact) {
return;
}
auto iter = groupList.find(groupId);
if (iter == groupList.end() || std::get<0>(iter.value()) != this)
if (iter == groupList.end() || std::get<0>(iter.value()) != this) {
event->acceptProposedAction();
}
}
}
@ -555,28 +602,33 @@ void ContentDialog::dropEvent(QDropEvent* event)
if (frnd) {
ToxId toxId(event->mimeData()->text());
Friend* contact = FriendList::findFriend(toxId.getPublicKey());
if (!contact)
if (!contact) {
return;
}
int friendId = contact->getFriendId();
auto iter = friendList.find(friendId);
if (iter != friendList.end())
if (iter != friendList.end()) {
std::get<0>(iter.value())->removeFriend(friendId);
}
Widget::getInstance()->addFriendDialog(contact, this);
ensureSplitterVisible();
} else if (group) {
if (!event->mimeData()->hasFormat("groupId"))
if (!event->mimeData()->hasFormat("groupId")) {
return;
}
int groupId = event->mimeData()->data("groupId").toInt();
Group* contact = GroupList::findGroup(groupId);
if (!contact)
if (!contact) {
return;
}
auto iter = friendList.find(groupId);
if (iter != friendList.end())
if (iter != friendList.end()) {
std::get<0>(iter.value())->removeGroup(groupId);
}
Widget::getInstance()->addGroupDialog(contact, this);
ensureSplitterVisible();
@ -587,8 +639,9 @@ void ContentDialog::changeEvent(QEvent* event)
{
QWidget::changeEvent(event);
if (event->type() == QEvent::ActivationChange) {
if (isActiveWindow())
if (isActiveWindow()) {
currentDialog = this;
}
}
}
@ -606,8 +659,10 @@ void ContentDialog::moveEvent(QMoveEvent* event)
void ContentDialog::keyPressEvent(QKeyEvent* event)
{
if (event->key() != Qt::Key_Escape)
QDialog::keyPressEvent(event); // Ignore escape keyboard shortcut.
// Ignore escape keyboard shortcut.
if (event->key() != Qt::Key_Escape) {
QDialog::keyPressEvent(event);
}
}
void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget* widget, bool group)
@ -616,7 +671,7 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget* widget, bool
ContentDialog* contentDialog = new ContentDialog(settingsWidget);
contentDialog->show();
if (widget->getFriend() != nullptr) {
if (widget->getFriend()) {
removeFriend(widget->getFriend()->getFriendId());
Widget::getInstance()->addFriendDialog(widget->getFriend(), contentDialog);
} else {
@ -631,13 +686,15 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget* widget, bool
}
// If we clicked on the currently active widget, don't reload and relayout everything
if (activeChatroomWidget == widget)
if (activeChatroomWidget == widget) {
return;
}
contentLayout->clear();
if (activeChatroomWidget != nullptr)
if (activeChatroomWidget) {
activeChatroomWidget->setAsInactiveChatroom();
}
activeChatroomWidget = widget;
@ -645,7 +702,8 @@ void ContentDialog::onChatroomWidgetClicked(GenericChatroomWidget* widget, bool
widget->setAsActiveChatroom();
widget->resetEventFlags();
widget->updateStatusLight();
updateTitle(widget);
updateTitleAndStatusIcon();
}
void ContentDialog::updateFriendWidget(uint32_t friendId, QString alias)
@ -661,23 +719,23 @@ void ContentDialog::updateFriendWidget(uint32_t friendId, QString alias)
void ContentDialog::updateGroupWidget(GroupWidget* w)
{
std::get<1>(groupList.find(w->groupId).value())->setName(w->getName());
static_cast<GroupWidget*>(std::get<1>(groupList.find(w->groupId).value()))->onUserListChanged();
ContactInfo info = groupList.find(w->groupId).value();
GroupWidget* widget = static_cast<GroupWidget*>(std::get<1>(info));
QString name = w->getName();
widget->setName(name);
widget->onUserListChanged();
}
void ContentDialog::onGroupchatPositionChanged(bool top)
{
friendLayout->removeItem(groupLayout.getLayout());
if (top)
friendLayout->insertLayout(0, groupLayout.getLayout());
else
friendLayout->insertLayout(1, groupLayout.getLayout());
friendLayout->insertLayout(top ? 0 : 1, groupLayout.getLayout());
}
void ContentDialog::retranslateUi()
{
updateTitleAndStatusIcon(Core::getInstance()->getUsername());
updateTitleAndStatusIcon();
}
void ContentDialog::saveDialogGeometry()
@ -691,111 +749,86 @@ void ContentDialog::saveSplitterState()
}
bool ContentDialog::hasWidget(int id, GenericChatroomWidget* chatroomWidget,
const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list)
const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
if (iter == list.end() || std::get<0>(iter.value()) != this)
if (iter == list.end()) {
return false;
return chatroomWidget == std::get<1>(iter.value());
}
bool ContentDialog::existsWidget(int id, bool focus,
const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list)
{
auto iter = list.find(id);
if (iter == list.end())
return false;
if (focus) {
if (std::get<0>(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);
}
return true;
return std::get<0>(*iter) == this &&
std::get<1>(*iter) == chatroomWidget;
}
void ContentDialog::updateStatus(int id,
const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list)
void ContentDialog::focusDialog(int id, const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
if (iter == list.end())
if (iter == list.end()) {
return;
}
GenericChatroomWidget* chatroomWidget = std::get<1>(iter.value());
ContentDialog* dialog = std::get<0>(*iter);
if (dialog->windowState() & Qt::WindowMinimized) {
dialog->showNormal();
}
dialog->raise();
dialog->activateWindow();
dialog->onChatroomWidgetClicked(std::get<1>(iter.value()), false);
}
bool ContentDialog::existsWidget(int id, const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
return iter != list.end();
}
void ContentDialog::updateStatus(int id, const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
if (iter == list.end()) {
return;
}
GenericChatroomWidget* chatroomWidget = std::get<1>(*iter);
chatroomWidget->updateStatusLight();
if (chatroomWidget->isActive())
std::get<0>(iter.value())->updateTitle(chatroomWidget);
if (chatroomWidget->isActive()) {
ContentDialog* dialog = std::get<0>(*iter);
dialog->updateTitleAndStatusIcon();
}
}
bool ContentDialog::isWidgetActive(
int id, const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list)
bool ContentDialog::isWidgetActive(int id, const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
if (iter == list.end())
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<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list)
ContentDialog* ContentDialog::getDialog(int id, const QHash<int, ContactInfo>& list)
{
auto iter = list.find(id);
if (iter == list.end())
if (iter == list.end()) {
return nullptr;
}
return std::get<0>(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();
}
int index = layouts.indexOf(layout);
if (index == -1) {
return nullptr;
}
return nullptr;
int next = forward ? index + 1 : index - 1;
size_t size = layouts.size();
next = (next + size) % size;
return layouts[next];
}

View File

@ -33,6 +33,7 @@ class QSet;
class QSplitter;
class QVBoxLayout;
class ContentDialog;
class ContentLayout;
class GenericChatroomWidget;
class FriendWidget;
@ -42,6 +43,8 @@ class SettingsWidget;
class Friend;
class Group;
using ContactInfo = std::tuple<ContentDialog*, GenericChatroomWidget*>;
class ContentDialog : public ActivateDialog
{
Q_OBJECT
@ -57,14 +60,17 @@ public:
bool hasGroupWidget(int groupId, GenericChatroomWidget* chatroomWidget);
int chatroomWidgetCount() const;
void ensureSplitterVisible();
void updateTitleAndStatusIcon();
void cycleContacts(bool forward, bool loop = true);
void onVideoShow(QSize size);
void onVideoHide();
static ContentDialog* current();
static bool existsFriendWidget(int friendId, bool focus);
static bool existsGroupWidget(int groupId, bool focus);
static bool existsFriendWidget(int friendId);
static bool existsGroupWidget(int groupId);
static void focusFriend(int friendId);
static void focusGroup(int groupId);
static void updateFriendStatus(int friendId);
static void updateFriendStatusMessage(int friendId, const QString& message);
static void updateGroupStatus(int groupId);
@ -79,10 +85,10 @@ signals:
void activated();
public slots:
void updateTitleAndStatusIcon(const QString& username);
void updateTitle(GenericChatroomWidget* chatroomWidget);
void reorderLayouts(bool newGroupOnTop);
void previousContact();
void nextContact();
void setUsername(const QString& newName);
protected:
bool event(QEvent* event) final override;
@ -104,31 +110,31 @@ private:
void saveDialogGeometry();
void saveSplitterState();
QLayout* nextLayout(QLayout* layout, bool forward) const;
int getCurrentLayout(QLayout*& layout);
bool hasWidget(int id, GenericChatroomWidget* chatroomWidget,
const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list);
static bool existsWidget(int id, bool focus,
const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list);
static void
updateStatus(int id, const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list);
static bool
isWidgetActive(int id, const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list);
static ContentDialog*
getDialog(int id, const QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>>& list);
const QHash<int, ContactInfo>& list);
void removeCurrent(QHash<int, ContactInfo>& infos);
static bool existsWidget(int id, const QHash<int, ContactInfo>& list);
static void focusDialog(int id, const QHash<int, ContactInfo>& list);
static void updateStatus(int id, const QHash<int, ContactInfo>& list);
static bool isWidgetActive(int id, const QHash<int, ContactInfo>& list);
static ContentDialog* getDialog(int id, const QHash<int, ContactInfo>& list);
QList<QLayout*> layouts;
QSplitter* splitter;
FriendListLayout* friendLayout;
GenericChatItemLayout groupLayout;
ContentLayout* contentLayout;
GenericChatroomWidget* activeChatroomWidget;
GenericChatroomWidget* displayWidget = nullptr;
SettingsWidget* settingsWidget;
QSize videoSurfaceSize;
int videoCount;
static QString username;
static ContentDialog* currentDialog;
static QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>> friendList;
static QHash<int, std::tuple<ContentDialog*, GenericChatroomWidget*>> groupList;
static QHash<int, ContactInfo> friendList;
static QHash<int, ContactInfo> groupList;
};
#endif // CONTENTDIALOG_H

View File

@ -363,7 +363,12 @@ void FriendWidget::search(const QString& searchString, bool hide)
bool FriendWidget::chatFormIsSet(bool focus) const
{
Friend* f = FriendList::findFriend(friendId);
return ContentDialog::existsFriendWidget(friendId, focus) || f->getChatForm()->isVisible();
if (focus) {
ContentDialog::focusFriend(friendId);
}
bool exist = ContentDialog::existsFriendWidget(friendId);
return exist || f->getChatForm()->isVisible();
}
void FriendWidget::setChatForm(ContentLayout* contentLayout)

View File

@ -200,9 +200,13 @@ Group* GroupWidget::getGroup() const
bool GroupWidget::chatFormIsSet(bool focus) const
{
(void)focus;
Group* g = GroupList::findGroup(groupId);
return ContentDialog::existsGroupWidget(groupId, focus) || g->getChatForm()->isVisible();
if (focus) {
ContentDialog::focusGroup(groupId);
}
bool exist = ContentDialog::existsGroupWidget(groupId);
return exist || g->getChatForm()->isVisible();
}
void GroupWidget::setChatForm(ContentLayout* contentLayout)

View File

@ -1457,16 +1457,19 @@ ContentDialog* Widget::createContentDialog() const
ContentDialog* contentDialog = new ContentDialog(settingsWidget);
connect(contentDialog, &ContentDialog::friendDialogShown, this, &Widget::onFriendDialogShown);
connect(contentDialog, &ContentDialog::groupDialogShown, this, &Widget::onGroupDialogShown);
connect(Core::getInstance(), &Core::usernameSet, contentDialog, &ContentDialog::setUsername);
Settings& s = Settings::getInstance();
connect(&s, &Settings::groupchatPositionChanged, contentDialog, &ContentDialog::reorderLayouts);
#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();
Nexus &n = Nexus::getInstance();
connect(contentDialog, &ContentDialog::destroyed, &n, &Nexus::updateWindowsClosed);
connect(contentDialog, &ContentDialog::windowStateChanged, &n, &Nexus::onWindowStateChanged);
connect(contentDialog->windowHandle(), &QWindow::windowTitleChanged, &n, &Nexus::updateWindows);
n.updateWindows();
#endif
return contentDialog;
}