2018-04-01 08:06:48 +02:00
|
|
|
// Copyright 2018 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-04-17 20:01:14 +02:00
|
|
|
#include <QAction>
|
2018-04-01 08:06:48 +02:00
|
|
|
#include <QIcon>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QStandardItemModel>
|
|
|
|
#include "citra_qt/game_list.h"
|
|
|
|
#include "citra_qt/multiplayer/client_room.h"
|
|
|
|
#include "citra_qt/multiplayer/direct_connect.h"
|
|
|
|
#include "citra_qt/multiplayer/host_room.h"
|
|
|
|
#include "citra_qt/multiplayer/lobby.h"
|
|
|
|
#include "citra_qt/multiplayer/message.h"
|
|
|
|
#include "citra_qt/multiplayer/state.h"
|
|
|
|
#include "citra_qt/util/clickable_label.h"
|
|
|
|
#include "common/announce_multiplayer_room.h"
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
|
2018-04-17 20:01:14 +02:00
|
|
|
MultiplayerState::MultiplayerState(QWidget* parent, QStandardItemModel* game_list_model,
|
|
|
|
QAction* leave_room, QAction* show_room)
|
|
|
|
: QWidget(parent), game_list_model(game_list_model), leave_room(leave_room),
|
|
|
|
show_room(show_room) {
|
2018-04-01 08:06:48 +02:00
|
|
|
if (auto member = Network::GetRoomMember().lock()) {
|
|
|
|
// register the network structs to use in slots and signals
|
|
|
|
state_callback_handle = member->BindOnStateChanged(
|
|
|
|
[this](const Network::RoomMember::State& state) { emit NetworkStateChanged(state); });
|
|
|
|
connect(this, &MultiplayerState::NetworkStateChanged, this,
|
|
|
|
&MultiplayerState::OnNetworkStateChanged);
|
2018-11-24 09:13:46 +01:00
|
|
|
error_callback_handle = member->BindOnError(
|
|
|
|
[this](const Network::RoomMember::Error& error) { emit NetworkError(error); });
|
|
|
|
connect(this, &MultiplayerState::NetworkError, this, &MultiplayerState::OnNetworkError);
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
|
|
|
|
2018-04-05 20:07:11 +02:00
|
|
|
qRegisterMetaType<Network::RoomMember::State>();
|
2018-11-24 09:13:46 +01:00
|
|
|
qRegisterMetaType<Network::RoomMember::Error>();
|
2018-04-01 08:06:48 +02:00
|
|
|
qRegisterMetaType<Common::WebResult>();
|
|
|
|
announce_multiplayer_session = std::make_shared<Core::AnnounceMultiplayerSession>();
|
|
|
|
announce_multiplayer_session->BindErrorCallback(
|
|
|
|
[this](const Common::WebResult& result) { emit AnnounceFailed(result); });
|
|
|
|
connect(this, &MultiplayerState::AnnounceFailed, this, &MultiplayerState::OnAnnounceFailed);
|
|
|
|
|
|
|
|
status_text = new ClickableLabel(this);
|
|
|
|
status_icon = new ClickableLabel(this);
|
|
|
|
status_text->setToolTip(tr("Current connection status"));
|
|
|
|
status_text->setText(tr("Not Connected. Click here to find a room!"));
|
|
|
|
status_icon->setPixmap(QIcon::fromTheme("disconnected").pixmap(16));
|
|
|
|
|
|
|
|
connect(status_text, &ClickableLabel::clicked, this, &MultiplayerState::OnOpenNetworkRoom);
|
|
|
|
connect(status_icon, &ClickableLabel::clicked, this, &MultiplayerState::OnOpenNetworkRoom);
|
|
|
|
}
|
|
|
|
|
|
|
|
MultiplayerState::~MultiplayerState() {
|
|
|
|
if (state_callback_handle) {
|
|
|
|
if (auto member = Network::GetRoomMember().lock()) {
|
|
|
|
member->Unbind(state_callback_handle);
|
|
|
|
}
|
|
|
|
}
|
2018-11-24 09:13:46 +01:00
|
|
|
|
|
|
|
if (error_callback_handle) {
|
|
|
|
if (auto member = Network::GetRoomMember().lock()) {
|
|
|
|
member->Unbind(error_callback_handle);
|
|
|
|
}
|
|
|
|
}
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::Close() {
|
|
|
|
if (host_room)
|
|
|
|
host_room->close();
|
|
|
|
if (direct_connect)
|
|
|
|
direct_connect->close();
|
|
|
|
if (client_room)
|
|
|
|
client_room->close();
|
|
|
|
if (lobby)
|
|
|
|
lobby->close();
|
|
|
|
}
|
|
|
|
|
2018-07-24 12:57:45 +02:00
|
|
|
void MultiplayerState::retranslateUi() {
|
|
|
|
status_text->setToolTip(tr("Current connection status"));
|
|
|
|
|
|
|
|
if (current_state == Network::RoomMember::State::Uninitialized) {
|
|
|
|
status_text->setText(tr("Not Connected. Click here to find a room!"));
|
|
|
|
} else if (current_state == Network::RoomMember::State::Joined) {
|
|
|
|
status_text->setText(tr("Connected"));
|
|
|
|
} else {
|
|
|
|
status_text->setText(tr("Not Connected"));
|
|
|
|
}
|
2018-10-09 17:08:33 +02:00
|
|
|
|
|
|
|
if (lobby)
|
|
|
|
lobby->RetranslateUi();
|
|
|
|
if (host_room)
|
|
|
|
host_room->RetranslateUi();
|
|
|
|
if (client_room)
|
|
|
|
client_room->RetranslateUi();
|
|
|
|
if (direct_connect)
|
|
|
|
direct_connect->RetranslateUi();
|
2018-07-24 12:57:45 +02:00
|
|
|
}
|
|
|
|
|
2018-04-01 08:06:48 +02:00
|
|
|
void MultiplayerState::OnNetworkStateChanged(const Network::RoomMember::State& state) {
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_DEBUG(Frontend, "Network State: {}", Network::GetStateStr(state));
|
2018-11-24 09:13:46 +01:00
|
|
|
if (state == Network::RoomMember::State::Joined) {
|
|
|
|
OnOpenNetworkRoom();
|
|
|
|
status_icon->setPixmap(QIcon::fromTheme("connected").pixmap(16));
|
|
|
|
status_text->setText(tr("Connected"));
|
|
|
|
leave_room->setEnabled(true);
|
|
|
|
show_room->setEnabled(true);
|
|
|
|
} else {
|
|
|
|
status_icon->setPixmap(QIcon::fromTheme("disconnected").pixmap(16));
|
|
|
|
status_text->setText(tr("Not Connected"));
|
|
|
|
leave_room->setEnabled(false);
|
|
|
|
show_room->setEnabled(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
current_state = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnNetworkError(const Network::RoomMember::Error& error) {
|
|
|
|
LOG_DEBUG(Frontend, "Network Error: {}", Network::GetErrorStr(error));
|
|
|
|
switch (error) {
|
|
|
|
case Network::RoomMember::Error::LostConnection:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::LOST_CONNECTION);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::HostKicked:
|
|
|
|
NetworkMessage::ShowError(NetworkMessage::HOST_KICKED);
|
|
|
|
break;
|
|
|
|
case Network::RoomMember::Error::CouldNotConnect:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::UNABLE_TO_CONNECT);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::NameCollision:
|
2018-11-02 13:17:25 +01:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::USERNAME_NOT_VALID_SERVER);
|
2018-04-19 08:47:11 +02:00
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::MacCollision:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::MAC_COLLISION);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::ConsoleIdCollision:
|
2018-10-31 16:07:03 +01:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::CONSOLE_ID_COLLISION);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::RoomIsFull:
|
2018-04-20 09:34:37 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::ROOM_IS_FULL);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::WrongPassword:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::WRONG_PASSWORD);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::WrongVersion:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::WRONG_VERSION);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::HostBanned:
|
|
|
|
NetworkMessage::ShowError(NetworkMessage::HOST_BANNED);
|
|
|
|
break;
|
|
|
|
case Network::RoomMember::Error::UnknownError:
|
2018-04-19 08:47:11 +02:00
|
|
|
NetworkMessage::ShowError(NetworkMessage::UNABLE_TO_CONNECT);
|
|
|
|
break;
|
2018-11-24 09:13:46 +01:00
|
|
|
case Network::RoomMember::Error::PermissionDenied:
|
|
|
|
NetworkMessage::ShowError(NetworkMessage::PERMISSION_DENIED);
|
|
|
|
break;
|
|
|
|
case Network::RoomMember::Error::NoSuchUser:
|
|
|
|
NetworkMessage::ShowError(NetworkMessage::NO_SUCH_USER);
|
2018-04-19 08:47:11 +02:00
|
|
|
break;
|
|
|
|
}
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnAnnounceFailed(const Common::WebResult& result) {
|
|
|
|
announce_multiplayer_session->Stop();
|
2018-04-18 07:06:02 +02:00
|
|
|
QMessageBox::warning(
|
|
|
|
this, tr("Error"),
|
|
|
|
tr("Failed to announce the room to the public lobby. In order to host a room publicly, you "
|
|
|
|
"must have a valid Citra account configured in Emulation -> Configure -> Web. If you do "
|
|
|
|
"not want to publish a room in the public lobby, then select Unlisted instead.\n"
|
|
|
|
"Debug Message: ") +
|
|
|
|
QString::fromStdString(result.result_string),
|
|
|
|
QMessageBox::Ok);
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
|
|
|
|
2018-07-27 04:51:33 +02:00
|
|
|
void MultiplayerState::UpdateThemedIcons() {
|
|
|
|
if (current_state == Network::RoomMember::State::Joined) {
|
|
|
|
status_icon->setPixmap(QIcon::fromTheme("connected").pixmap(16));
|
|
|
|
} else {
|
|
|
|
status_icon->setPixmap(QIcon::fromTheme("disconnected").pixmap(16));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-01 08:06:48 +02:00
|
|
|
static void BringWidgetToFront(QWidget* widget) {
|
|
|
|
widget->show();
|
|
|
|
widget->activateWindow();
|
|
|
|
widget->raise();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnViewLobby() {
|
|
|
|
if (lobby == nullptr) {
|
|
|
|
lobby = new Lobby(this, game_list_model, announce_multiplayer_session);
|
|
|
|
}
|
|
|
|
BringWidgetToFront(lobby);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnCreateRoom() {
|
|
|
|
if (host_room == nullptr) {
|
|
|
|
host_room = new HostRoomWindow(this, game_list_model, announce_multiplayer_session);
|
|
|
|
}
|
|
|
|
BringWidgetToFront(host_room);
|
|
|
|
}
|
|
|
|
|
2018-04-18 19:04:32 +02:00
|
|
|
bool MultiplayerState::OnCloseRoom() {
|
2018-04-17 20:01:14 +02:00
|
|
|
if (!NetworkMessage::WarnCloseRoom())
|
2018-04-18 19:04:32 +02:00
|
|
|
return false;
|
2018-04-01 08:06:48 +02:00
|
|
|
if (auto room = Network::GetRoom().lock()) {
|
2018-04-17 20:01:14 +02:00
|
|
|
// if you are in a room, leave it
|
|
|
|
if (auto member = Network::GetRoomMember().lock()) {
|
|
|
|
member->Leave();
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_DEBUG(Frontend, "Left the room (as a client)");
|
2018-04-17 20:01:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// if you are hosting a room, also stop hosting
|
2018-04-09 19:00:56 +02:00
|
|
|
if (room->GetState() != Network::Room::State::Open) {
|
2018-04-18 19:04:32 +02:00
|
|
|
return true;
|
2018-04-09 19:00:56 +02:00
|
|
|
}
|
2018-04-17 20:01:14 +02:00
|
|
|
room->Destroy();
|
|
|
|
announce_multiplayer_session->Stop();
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_DEBUG(Frontend, "Closed the room (as a server)");
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
2018-04-18 19:04:32 +02:00
|
|
|
return true;
|
2018-04-01 08:06:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnOpenNetworkRoom() {
|
|
|
|
if (auto member = Network::GetRoomMember().lock()) {
|
|
|
|
if (member->IsConnected()) {
|
|
|
|
if (client_room == nullptr) {
|
|
|
|
client_room = new ClientRoomWindow(this);
|
|
|
|
}
|
2018-11-24 09:22:14 +01:00
|
|
|
const std::string host_username = member->GetRoomInformation().host_username;
|
|
|
|
if (host_username.empty()) {
|
|
|
|
client_room->SetModPerms(false);
|
|
|
|
} else {
|
|
|
|
client_room->SetModPerms(member->GetUsername() == host_username);
|
|
|
|
}
|
2018-04-01 08:06:48 +02:00
|
|
|
BringWidgetToFront(client_room);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If the user is not a member of a room, show the lobby instead.
|
|
|
|
// This is currently only used on the clickable label in the status bar
|
|
|
|
OnViewLobby();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MultiplayerState::OnDirectConnectToRoom() {
|
|
|
|
if (direct_connect == nullptr) {
|
|
|
|
direct_connect = new DirectConnectWindow(this);
|
|
|
|
}
|
|
|
|
BringWidgetToFront(direct_connect);
|
|
|
|
}
|