| 
									
										
										
										
											2022-07-25 17:18:30 +02:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2017 Citra Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <QInputDialog>
 | 
					
						
							|  |  |  | #include <QList>
 | 
					
						
							|  |  |  | #include <QtConcurrent/QtConcurrentRun>
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2022-08-27 03:41:19 +02:00
										 |  |  | #include "core/internal_network/network_interface.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  | #include "network/network.h"
 | 
					
						
							|  |  |  | #include "ui_lobby.h"
 | 
					
						
							|  |  |  | #include "yuzu/game_list_p.h"
 | 
					
						
							|  |  |  | #include "yuzu/main.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/client_room.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/lobby.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/lobby_p.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/message.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/state.h"
 | 
					
						
							|  |  |  | #include "yuzu/multiplayer/validation.h"
 | 
					
						
							|  |  |  | #include "yuzu/uisettings.h"
 | 
					
						
							|  |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							|  |  |  | #include "web_service/web_backend.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Lobby::Lobby(QWidget* parent, QStandardItemModel* list, | 
					
						
							| 
									
										
										
										
											2022-08-27 03:41:19 +02:00
										 |  |  |              std::shared_ptr<Core::AnnounceMultiplayerSession> session, Core::System& system_) | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |     : QDialog(parent, Qt::WindowTitleHint | Qt::WindowCloseButtonHint | Qt::WindowSystemMenuHint), | 
					
						
							| 
									
										
										
										
											2022-07-22 16:31:13 +02:00
										 |  |  |       ui(std::make_unique<Ui::Lobby>()), | 
					
						
							| 
									
										
										
										
											2022-08-27 03:41:19 +02:00
										 |  |  |       announce_multiplayer_session(session), system{system_}, room_network{ | 
					
						
							|  |  |  |                                                                   system.GetRoomNetwork()} { | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |     ui->setupUi(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // setup the watcher for background connections
 | 
					
						
							|  |  |  |     watcher = new QFutureWatcher<void>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     model = new QStandardItemModel(ui->room_list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Create a proxy to the game list to get the list of games owned
 | 
					
						
							|  |  |  |     game_list = new QStandardItemModel; | 
					
						
							|  |  |  |     UpdateGameList(list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     proxy = new LobbyFilterProxyModel(this, game_list); | 
					
						
							|  |  |  |     proxy->setSourceModel(model); | 
					
						
							|  |  |  |     proxy->setDynamicSortFilter(true); | 
					
						
							|  |  |  |     proxy->setFilterCaseSensitivity(Qt::CaseInsensitive); | 
					
						
							|  |  |  |     proxy->setSortLocaleAware(true); | 
					
						
							|  |  |  |     ui->room_list->setModel(proxy); | 
					
						
							|  |  |  |     ui->room_list->header()->setSectionResizeMode(QHeaderView::Interactive); | 
					
						
							|  |  |  |     ui->room_list->header()->stretchLastSection(); | 
					
						
							|  |  |  |     ui->room_list->setAlternatingRowColors(true); | 
					
						
							|  |  |  |     ui->room_list->setSelectionMode(QHeaderView::SingleSelection); | 
					
						
							|  |  |  |     ui->room_list->setSelectionBehavior(QHeaderView::SelectRows); | 
					
						
							|  |  |  |     ui->room_list->setVerticalScrollMode(QHeaderView::ScrollPerPixel); | 
					
						
							|  |  |  |     ui->room_list->setHorizontalScrollMode(QHeaderView::ScrollPerPixel); | 
					
						
							|  |  |  |     ui->room_list->setSortingEnabled(true); | 
					
						
							|  |  |  |     ui->room_list->setEditTriggers(QHeaderView::NoEditTriggers); | 
					
						
							|  |  |  |     ui->room_list->setExpandsOnDoubleClick(false); | 
					
						
							|  |  |  |     ui->room_list->setContextMenuPolicy(Qt::CustomContextMenu); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ui->nickname->setValidator(validation.GetNickname()); | 
					
						
							| 
									
										
										
										
											2022-07-15 19:45:35 +02:00
										 |  |  |     ui->nickname->setText(UISettings::values.multiplayer_nickname.GetValue()); | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |     if (ui->nickname->text().isEmpty() && !Settings::values.yuzu_username.GetValue().empty()) { | 
					
						
							|  |  |  |         // Use yuzu Web Service user name as nickname by default
 | 
					
						
							|  |  |  |         ui->nickname->setText(QString::fromStdString(Settings::values.yuzu_username.GetValue())); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // UI Buttons
 | 
					
						
							|  |  |  |     connect(ui->refresh_list, &QPushButton::clicked, this, &Lobby::RefreshLobby); | 
					
						
							|  |  |  |     connect(ui->games_owned, &QCheckBox::toggled, proxy, &LobbyFilterProxyModel::SetFilterOwned); | 
					
						
							|  |  |  |     connect(ui->hide_full, &QCheckBox::toggled, proxy, &LobbyFilterProxyModel::SetFilterFull); | 
					
						
							|  |  |  |     connect(ui->search, &QLineEdit::textChanged, proxy, &LobbyFilterProxyModel::SetFilterSearch); | 
					
						
							|  |  |  |     connect(ui->room_list, &QTreeView::doubleClicked, this, &Lobby::OnJoinRoom); | 
					
						
							|  |  |  |     connect(ui->room_list, &QTreeView::clicked, this, &Lobby::OnExpandRoom); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Actions
 | 
					
						
							|  |  |  |     connect(&room_list_watcher, &QFutureWatcher<AnnounceMultiplayerRoom::RoomList>::finished, this, | 
					
						
							|  |  |  |             &Lobby::OnRefreshLobby); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // manually start a refresh when the window is opening
 | 
					
						
							|  |  |  |     // TODO(jroweboy): if this refresh is slow for people with bad internet, then don't do it as
 | 
					
						
							|  |  |  |     // part of the constructor, but offload the refresh until after the window shown. perhaps emit a
 | 
					
						
							|  |  |  |     // refreshroomlist signal from places that open the lobby
 | 
					
						
							|  |  |  |     RefreshLobby(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Lobby::~Lobby() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::UpdateGameList(QStandardItemModel* list) { | 
					
						
							|  |  |  |     game_list->clear(); | 
					
						
							|  |  |  |     for (int i = 0; i < list->rowCount(); i++) { | 
					
						
							|  |  |  |         auto parent = list->item(i, 0); | 
					
						
							|  |  |  |         for (int j = 0; j < parent->rowCount(); j++) { | 
					
						
							|  |  |  |             game_list->appendRow(parent->child(j)->clone()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (proxy) | 
					
						
							|  |  |  |         proxy->UpdateGameList(game_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::RetranslateUi() { | 
					
						
							|  |  |  |     ui->retranslateUi(this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString Lobby::PasswordPrompt() { | 
					
						
							|  |  |  |     bool ok; | 
					
						
							|  |  |  |     const QString text = | 
					
						
							|  |  |  |         QInputDialog::getText(this, tr("Password Required to Join"), tr("Password:"), | 
					
						
							|  |  |  |                               QLineEdit::Password, QString(), &ok); | 
					
						
							|  |  |  |     return ok ? text : QString(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::OnExpandRoom(const QModelIndex& index) { | 
					
						
							|  |  |  |     QModelIndex member_index = proxy->index(index.row(), Column::MEMBER); | 
					
						
							|  |  |  |     auto member_list = proxy->data(member_index, LobbyItemMemberList::MemberListRole).toList(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::OnJoinRoom(const QModelIndex& source) { | 
					
						
							| 
									
										
										
										
											2022-08-27 03:41:19 +02:00
										 |  |  |     if (!Network::GetSelectedNetworkInterface()) { | 
					
						
							|  |  |  |         NetworkMessage::ErrorManager::ShowError( | 
					
						
							|  |  |  |             NetworkMessage::ErrorManager::NO_INTERFACE_SELECTED); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (system.IsPoweredOn()) { | 
					
						
							|  |  |  |         if (!NetworkMessage::WarnGameRunning()) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-22 16:31:13 +02:00
										 |  |  |     if (const auto member = room_network.GetRoomMember().lock()) { | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |         // Prevent the user from trying to join a room while they are already joining.
 | 
					
						
							|  |  |  |         if (member->GetState() == Network::RoomMember::State::Joining) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } else if (member->IsConnected()) { | 
					
						
							|  |  |  |             // And ask if they want to leave the room if they are already in one.
 | 
					
						
							|  |  |  |             if (!NetworkMessage::WarnDisconnect()) { | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     QModelIndex index = source; | 
					
						
							|  |  |  |     // If the user double clicks on a child row (aka the player list) then use the parent instead
 | 
					
						
							|  |  |  |     if (source.parent() != QModelIndex()) { | 
					
						
							|  |  |  |         index = source.parent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!ui->nickname->hasAcceptableInput()) { | 
					
						
							|  |  |  |         NetworkMessage::ErrorManager::ShowError(NetworkMessage::ErrorManager::USERNAME_NOT_VALID); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get a password to pass if the room is password protected
 | 
					
						
							|  |  |  |     QModelIndex password_index = proxy->index(index.row(), Column::ROOM_NAME); | 
					
						
							|  |  |  |     bool has_password = proxy->data(password_index, LobbyItemName::PasswordRole).toBool(); | 
					
						
							|  |  |  |     const std::string password = has_password ? PasswordPrompt().toStdString() : ""; | 
					
						
							|  |  |  |     if (has_password && password.empty()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QModelIndex connection_index = proxy->index(index.row(), Column::HOST); | 
					
						
							|  |  |  |     const std::string nickname = ui->nickname->text().toStdString(); | 
					
						
							|  |  |  |     const std::string ip = | 
					
						
							|  |  |  |         proxy->data(connection_index, LobbyItemHost::HostIPRole).toString().toStdString(); | 
					
						
							|  |  |  |     int port = proxy->data(connection_index, LobbyItemHost::HostPortRole).toInt(); | 
					
						
							| 
									
										
										
										
											2022-07-25 17:08:20 +02:00
										 |  |  |     const std::string verify_uid = | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |         proxy->data(connection_index, LobbyItemHost::HostVerifyUIDRole).toString().toStdString(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // attempt to connect in a different thread
 | 
					
						
							| 
									
										
										
										
											2022-07-25 17:08:20 +02:00
										 |  |  |     QFuture<void> f = QtConcurrent::run([nickname, ip, port, password, verify_uid, this] { | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |         std::string token; | 
					
						
							|  |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							| 
									
										
										
										
											2022-07-15 19:45:35 +02:00
										 |  |  |         if (!Settings::values.yuzu_username.GetValue().empty() && | 
					
						
							|  |  |  |             !Settings::values.yuzu_token.GetValue().empty()) { | 
					
						
							|  |  |  |             WebService::Client client(Settings::values.web_api_url.GetValue(), | 
					
						
							|  |  |  |                                       Settings::values.yuzu_username.GetValue(), | 
					
						
							|  |  |  |                                       Settings::values.yuzu_token.GetValue()); | 
					
						
							| 
									
										
										
										
											2022-07-25 17:08:20 +02:00
										 |  |  |             token = client.GetExternalJWT(verify_uid).returned_data; | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |             if (token.empty()) { | 
					
						
							|  |  |  |                 LOG_ERROR(WebService, "Could not get external JWT, verification may fail"); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 LOG_INFO(WebService, "Successfully requested external JWT: size={}", token.size()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-07-22 16:31:13 +02:00
										 |  |  |         if (auto room_member = room_network.GetRoomMember().lock()) { | 
					
						
							| 
									
										
										
										
											2022-07-30 05:58:23 +02:00
										 |  |  |             room_member->Join(nickname, ip.c_str(), port, 0, Network::NoPreferredIP, password, | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |                               token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     watcher->setFuture(f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO(jroweboy): disable widgets and display a connecting while we wait
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Save settings
 | 
					
						
							| 
									
										
										
										
											2022-07-15 19:45:35 +02:00
										 |  |  |     UISettings::values.multiplayer_nickname = ui->nickname->text(); | 
					
						
							|  |  |  |     UISettings::values.multiplayer_ip = | 
					
						
							|  |  |  |         proxy->data(connection_index, LobbyItemHost::HostIPRole).toString(); | 
					
						
							|  |  |  |     UISettings::values.multiplayer_port = | 
					
						
							|  |  |  |         proxy->data(connection_index, LobbyItemHost::HostPortRole).toInt(); | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::ResetModel() { | 
					
						
							|  |  |  |     model->clear(); | 
					
						
							|  |  |  |     model->insertColumns(0, Column::TOTAL); | 
					
						
							|  |  |  |     model->setHeaderData(Column::EXPAND, Qt::Horizontal, QString(), Qt::DisplayRole); | 
					
						
							|  |  |  |     model->setHeaderData(Column::ROOM_NAME, Qt::Horizontal, tr("Room Name"), Qt::DisplayRole); | 
					
						
							|  |  |  |     model->setHeaderData(Column::GAME_NAME, Qt::Horizontal, tr("Preferred Game"), Qt::DisplayRole); | 
					
						
							|  |  |  |     model->setHeaderData(Column::HOST, Qt::Horizontal, tr("Host"), Qt::DisplayRole); | 
					
						
							|  |  |  |     model->setHeaderData(Column::MEMBER, Qt::Horizontal, tr("Players"), Qt::DisplayRole); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::RefreshLobby() { | 
					
						
							|  |  |  |     if (auto session = announce_multiplayer_session.lock()) { | 
					
						
							|  |  |  |         ResetModel(); | 
					
						
							|  |  |  |         ui->refresh_list->setEnabled(false); | 
					
						
							|  |  |  |         ui->refresh_list->setText(tr("Refreshing")); | 
					
						
							|  |  |  |         room_list_watcher.setFuture( | 
					
						
							|  |  |  |             QtConcurrent::run([session]() { return session->GetRoomList(); })); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // TODO(jroweboy): Display an error box about announce couldn't be started
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Lobby::OnRefreshLobby() { | 
					
						
							|  |  |  |     AnnounceMultiplayerRoom::RoomList new_room_list = room_list_watcher.result(); | 
					
						
							|  |  |  |     for (auto room : new_room_list) { | 
					
						
							|  |  |  |         // find the icon for the game if this person owns that game.
 | 
					
						
							|  |  |  |         QPixmap smdh_icon; | 
					
						
							|  |  |  |         for (int r = 0; r < game_list->rowCount(); ++r) { | 
					
						
							|  |  |  |             auto index = game_list->index(r, 0); | 
					
						
							|  |  |  |             auto game_id = game_list->data(index, GameListItemPath::ProgramIdRole).toULongLong(); | 
					
						
							| 
									
										
										
										
											2022-07-17 22:53:44 -05:00
										 |  |  |             if (game_id != 0 && room.information.preferred_game.id == game_id) { | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |                 smdh_icon = game_list->data(index, Qt::DecorationRole).value<QPixmap>(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QList<QVariant> members; | 
					
						
							|  |  |  |         for (auto member : room.members) { | 
					
						
							|  |  |  |             QVariant var; | 
					
						
							|  |  |  |             var.setValue(LobbyMember{QString::fromStdString(member.username), | 
					
						
							| 
									
										
										
										
											2022-07-17 22:53:44 -05:00
										 |  |  |                                      QString::fromStdString(member.nickname), member.game.id, | 
					
						
							|  |  |  |                                      QString::fromStdString(member.game.name)}); | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |             members.append(var); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto first_item = new LobbyItem(); | 
					
						
							|  |  |  |         auto row = QList<QStandardItem*>({ | 
					
						
							|  |  |  |             first_item, | 
					
						
							| 
									
										
										
										
											2022-07-15 21:11:09 +02:00
										 |  |  |             new LobbyItemName(room.has_password, QString::fromStdString(room.information.name)), | 
					
						
							| 
									
										
										
										
											2022-07-17 22:53:44 -05:00
										 |  |  |             new LobbyItemGame(room.information.preferred_game.id, | 
					
						
							|  |  |  |                               QString::fromStdString(room.information.preferred_game.name), | 
					
						
							|  |  |  |                               smdh_icon), | 
					
						
							| 
									
										
										
										
											2022-07-15 21:11:09 +02:00
										 |  |  |             new LobbyItemHost(QString::fromStdString(room.information.host_username), | 
					
						
							|  |  |  |                               QString::fromStdString(room.ip), room.information.port, | 
					
						
							| 
									
										
										
										
											2022-07-25 17:08:20 +02:00
										 |  |  |                               QString::fromStdString(room.verify_uid)), | 
					
						
							| 
									
										
										
										
											2022-07-15 21:11:09 +02:00
										 |  |  |             new LobbyItemMemberList(members, room.information.member_slots), | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |         }); | 
					
						
							|  |  |  |         model->appendRow(row); | 
					
						
							|  |  |  |         // To make the rows expandable, add the member data as a child of the first column of the
 | 
					
						
							|  |  |  |         // rows with people in them and have qt set them to colspan after the model is finished
 | 
					
						
							|  |  |  |         // resetting
 | 
					
						
							| 
									
										
										
										
											2022-07-15 21:11:09 +02:00
										 |  |  |         if (!room.information.description.empty()) { | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |             first_item->appendRow( | 
					
						
							| 
									
										
										
										
											2022-07-15 21:11:09 +02:00
										 |  |  |                 new LobbyItemDescription(QString::fromStdString(room.information.description))); | 
					
						
							| 
									
										
										
										
											2021-12-25 20:27:52 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (!room.members.empty()) { | 
					
						
							|  |  |  |             first_item->appendRow(new LobbyItemExpandedMemberList(members)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reenable the refresh button and resize the columns
 | 
					
						
							|  |  |  |     ui->refresh_list->setEnabled(true); | 
					
						
							|  |  |  |     ui->refresh_list->setText(tr("Refresh List")); | 
					
						
							|  |  |  |     ui->room_list->header()->stretchLastSection(); | 
					
						
							|  |  |  |     for (int i = 0; i < Column::TOTAL - 1; ++i) { | 
					
						
							|  |  |  |         ui->room_list->resizeColumnToContents(i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set the member list child items to span all columns
 | 
					
						
							|  |  |  |     for (int i = 0; i < proxy->rowCount(); i++) { | 
					
						
							|  |  |  |         auto parent = model->item(i, 0); | 
					
						
							|  |  |  |         for (int j = 0; j < parent->rowCount(); j++) { | 
					
						
							|  |  |  |             ui->room_list->setFirstColumnSpanned(j, proxy->index(i, 0), true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LobbyFilterProxyModel::LobbyFilterProxyModel(QWidget* parent, QStandardItemModel* list) | 
					
						
							|  |  |  |     : QSortFilterProxyModel(parent), game_list(list) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LobbyFilterProxyModel::UpdateGameList(QStandardItemModel* list) { | 
					
						
							|  |  |  |     game_list = list; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool LobbyFilterProxyModel::filterAcceptsRow(int sourceRow, const QModelIndex& sourceParent) const { | 
					
						
							|  |  |  |     // Prioritize filters by fastest to compute
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // pass over any child rows (aka row that shows the players in the room)
 | 
					
						
							|  |  |  |     if (sourceParent != QModelIndex()) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // filter by filled rooms
 | 
					
						
							|  |  |  |     if (filter_full) { | 
					
						
							|  |  |  |         QModelIndex member_list = sourceModel()->index(sourceRow, Column::MEMBER, sourceParent); | 
					
						
							|  |  |  |         int player_count = | 
					
						
							|  |  |  |             sourceModel()->data(member_list, LobbyItemMemberList::MemberListRole).toList().size(); | 
					
						
							|  |  |  |         int max_players = | 
					
						
							|  |  |  |             sourceModel()->data(member_list, LobbyItemMemberList::MaxPlayerRole).toInt(); | 
					
						
							|  |  |  |         if (player_count >= max_players) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // filter by search parameters
 | 
					
						
							|  |  |  |     if (!filter_search.isEmpty()) { | 
					
						
							|  |  |  |         QModelIndex game_name = sourceModel()->index(sourceRow, Column::GAME_NAME, sourceParent); | 
					
						
							|  |  |  |         QModelIndex room_name = sourceModel()->index(sourceRow, Column::ROOM_NAME, sourceParent); | 
					
						
							|  |  |  |         QModelIndex host_name = sourceModel()->index(sourceRow, Column::HOST, sourceParent); | 
					
						
							|  |  |  |         bool preferred_game_match = sourceModel() | 
					
						
							|  |  |  |                                         ->data(game_name, LobbyItemGame::GameNameRole) | 
					
						
							|  |  |  |                                         .toString() | 
					
						
							|  |  |  |                                         .contains(filter_search, filterCaseSensitivity()); | 
					
						
							|  |  |  |         bool room_name_match = sourceModel() | 
					
						
							|  |  |  |                                    ->data(room_name, LobbyItemName::NameRole) | 
					
						
							|  |  |  |                                    .toString() | 
					
						
							|  |  |  |                                    .contains(filter_search, filterCaseSensitivity()); | 
					
						
							|  |  |  |         bool username_match = sourceModel() | 
					
						
							|  |  |  |                                   ->data(host_name, LobbyItemHost::HostUsernameRole) | 
					
						
							|  |  |  |                                   .toString() | 
					
						
							|  |  |  |                                   .contains(filter_search, filterCaseSensitivity()); | 
					
						
							|  |  |  |         if (!preferred_game_match && !room_name_match && !username_match) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // filter by game owned
 | 
					
						
							|  |  |  |     if (filter_owned) { | 
					
						
							|  |  |  |         QModelIndex game_name = sourceModel()->index(sourceRow, Column::GAME_NAME, sourceParent); | 
					
						
							|  |  |  |         QList<QModelIndex> owned_games; | 
					
						
							|  |  |  |         for (int r = 0; r < game_list->rowCount(); ++r) { | 
					
						
							|  |  |  |             owned_games.append(QModelIndex(game_list->index(r, 0))); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         auto current_id = sourceModel()->data(game_name, LobbyItemGame::TitleIDRole).toLongLong(); | 
					
						
							|  |  |  |         if (current_id == 0) { | 
					
						
							|  |  |  |             // TODO(jroweboy): homebrew often doesn't have a game id and this hides them
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         bool owned = false; | 
					
						
							|  |  |  |         for (const auto& game : owned_games) { | 
					
						
							|  |  |  |             auto game_id = game_list->data(game, GameListItemPath::ProgramIdRole).toLongLong(); | 
					
						
							|  |  |  |             if (current_id == game_id) { | 
					
						
							|  |  |  |                 owned = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!owned) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LobbyFilterProxyModel::sort(int column, Qt::SortOrder order) { | 
					
						
							|  |  |  |     sourceModel()->sort(column, order); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LobbyFilterProxyModel::SetFilterOwned(bool filter) { | 
					
						
							|  |  |  |     filter_owned = filter; | 
					
						
							|  |  |  |     invalidate(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LobbyFilterProxyModel::SetFilterFull(bool filter) { | 
					
						
							|  |  |  |     filter_full = filter; | 
					
						
							|  |  |  |     invalidate(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LobbyFilterProxyModel::SetFilterSearch(const QString& filter) { | 
					
						
							|  |  |  |     filter_search = filter; | 
					
						
							|  |  |  |     invalidate(); | 
					
						
							|  |  |  | } |