Netplay: Wired up netplay chat.
This commit is contained in:
parent
f8a06969a8
commit
ddaa7172c8
|
@ -48,6 +48,11 @@ enum class ControlMessage : u32
|
||||||
SynchronizeComplete,
|
SynchronizeComplete,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum class SessionMessage : u32
|
||||||
|
{
|
||||||
|
ChatMessage,
|
||||||
|
};
|
||||||
|
|
||||||
#pragma pack(push, 1)
|
#pragma pack(push, 1)
|
||||||
struct ControlMessageHeader
|
struct ControlMessageHeader
|
||||||
{
|
{
|
||||||
|
@ -55,6 +60,13 @@ struct ControlMessageHeader
|
||||||
u32 size;
|
u32 size;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#pragma pack(push, 1)
|
||||||
|
struct SessionMessageHeader
|
||||||
|
{
|
||||||
|
SessionMessage type;
|
||||||
|
u32 size;
|
||||||
|
};
|
||||||
|
|
||||||
struct ControlConnectResponseMessage
|
struct ControlConnectResponseMessage
|
||||||
{
|
{
|
||||||
enum class Result : u32
|
enum class Result : u32
|
||||||
|
@ -94,6 +106,15 @@ struct ControlSynchronizeCompleteMessage
|
||||||
|
|
||||||
static ControlMessage MessageType() { return ControlMessage::SynchronizeComplete; }
|
static ControlMessage MessageType() { return ControlMessage::SynchronizeComplete; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct SessionChatMessage
|
||||||
|
{
|
||||||
|
SessionMessageHeader header;
|
||||||
|
|
||||||
|
u32 chat_message_size;
|
||||||
|
|
||||||
|
static SessionMessage MessageType() { return SessionMessage::ChatMessage; }
|
||||||
|
};
|
||||||
#pragma pack(pop)
|
#pragma pack(pop)
|
||||||
|
|
||||||
using SaveStateBuffer = std::unique_ptr<System::MemorySaveState>;
|
using SaveStateBuffer = std::unique_ptr<System::MemorySaveState>;
|
||||||
|
@ -142,6 +163,10 @@ static void HandleConnectResponseMessage(s32 player_id, const ENetPacket* pkt);
|
||||||
static void HandleSynchronizeSessionMessage(s32 player_id, const ENetPacket* pkt);
|
static void HandleSynchronizeSessionMessage(s32 player_id, const ENetPacket* pkt);
|
||||||
static void HandleSynchronizeCompleteMessage(s32 player_id, const ENetPacket* pkt);
|
static void HandleSynchronizeCompleteMessage(s32 player_id, const ENetPacket* pkt);
|
||||||
|
|
||||||
|
// Sessionpackets
|
||||||
|
static void HandleSessionMessage(s32 player_id, const ENetPacket* pkt);
|
||||||
|
static void HandleSessionChatMessage(s32 player_id, const ENetPacket* pkt);
|
||||||
|
|
||||||
// l = local, r = remote
|
// l = local, r = remote
|
||||||
static bool CreateGGPOSession();
|
static bool CreateGGPOSession();
|
||||||
static void DestroyGGPOSession();
|
static void DestroyGGPOSession();
|
||||||
|
@ -170,7 +195,6 @@ static void Throttle();
|
||||||
|
|
||||||
// Desync Detection
|
// Desync Detection
|
||||||
static void GenerateChecksumForFrame(int* checksum, int frame, unsigned char* buffer, int buffer_size);
|
static void GenerateChecksumForFrame(int* checksum, int frame, unsigned char* buffer, int buffer_size);
|
||||||
static void GenerateDesyncReport(s32 desync_frame);
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
// Variables
|
// Variables
|
||||||
|
@ -252,7 +276,34 @@ static bool SendControlPacket(s32 player_id, const PacketWrapper<T>& pkt)
|
||||||
DebugAssert(player_id >= 0 && player_id < MAX_PLAYERS && s_peers[player_id].peer);
|
DebugAssert(player_id >= 0 && player_id < MAX_PLAYERS && s_peers[player_id].peer);
|
||||||
return SendControlPacket<T>(s_peers[player_id].peer, pkt);
|
return SendControlPacket<T>(s_peers[player_id].peer, pkt);
|
||||||
}
|
}
|
||||||
|
template<typename T>
|
||||||
|
static PacketWrapper<T> NewSessionPacket(u32 size = sizeof(T), u32 flags = ENET_PACKET_FLAG_RELIABLE)
|
||||||
|
{
|
||||||
|
PacketWrapper<T> ret = NewWrappedPacket<T>(size, flags);
|
||||||
|
SessionMessageHeader* hdr = reinterpret_cast<SessionMessageHeader*>(ret.pkt->data);
|
||||||
|
hdr->type = T::MessageType();
|
||||||
|
hdr->size = size;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
template<typename T>
|
||||||
|
static bool SendSessionPacket(ENetPeer* peer, const PacketWrapper<T>& pkt)
|
||||||
|
{
|
||||||
|
const int rc = enet_peer_send(peer, ENET_CHANNEL_SESSION, pkt.pkt);
|
||||||
|
if (rc != 0)
|
||||||
|
{
|
||||||
|
Log_ErrorPrintf("enet_peer_send() failed: %d", rc);
|
||||||
|
enet_packet_destroy(pkt.pkt);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
template<typename T>
|
||||||
|
static bool SendSessionPacket(s32 player_id, const PacketWrapper<T>& pkt)
|
||||||
|
{
|
||||||
|
DebugAssert(player_id >= 0 && player_id < MAX_PLAYERS && s_peers[player_id].peer);
|
||||||
|
return SendSessionPacket<T>(s_peers[player_id].peer, pkt);
|
||||||
|
}
|
||||||
} // namespace Netplay
|
} // namespace Netplay
|
||||||
|
|
||||||
// Netplay Impl
|
// Netplay Impl
|
||||||
|
@ -472,6 +523,10 @@ void Netplay::HandleEnetEvent(const ENetEvent* event)
|
||||||
{
|
{
|
||||||
HandleControlMessage(player_id, event->packet);
|
HandleControlMessage(player_id, event->packet);
|
||||||
}
|
}
|
||||||
|
else if (event->channelID == ENET_CHANNEL_SESSION)
|
||||||
|
{
|
||||||
|
HandleSessionMessage(player_id, event->packet);
|
||||||
|
}
|
||||||
else if (event->channelID == ENET_CHANNEL_GGPO)
|
else if (event->channelID == ENET_CHANNEL_GGPO)
|
||||||
{
|
{
|
||||||
Log_TracePrintf("Received %zu ggpo bytes from player %d", event->packet->dataLength, player_id);
|
Log_TracePrintf("Received %zu ggpo bytes from player %d", event->packet->dataLength, player_id);
|
||||||
|
@ -505,6 +560,10 @@ void Netplay::PollEnet(Common::Timer::Value until_time)
|
||||||
const u32 enet_timeout = (current_time >= until_time) ?
|
const u32 enet_timeout = (current_time >= until_time) ?
|
||||||
0 :
|
0 :
|
||||||
static_cast<u32>(Common::Timer::ConvertValueToMilliseconds(until_time - current_time));
|
static_cast<u32>(Common::Timer::ConvertValueToMilliseconds(until_time - current_time));
|
||||||
|
|
||||||
|
// make sure s_enet_host exists
|
||||||
|
Assert(s_enet_host);
|
||||||
|
|
||||||
const int res = enet_host_service(s_enet_host, &event, enet_timeout);
|
const int res = enet_host_service(s_enet_host, &event, enet_timeout);
|
||||||
if (res > 0)
|
if (res > 0)
|
||||||
{
|
{
|
||||||
|
@ -514,7 +573,6 @@ void Netplay::PollEnet(Common::Timer::Value until_time)
|
||||||
current_time = Common::Timer::GetCurrentValue();
|
current_time = Common::Timer::GetCurrentValue();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// exit once we're nonblocking
|
// exit once we're nonblocking
|
||||||
current_time = Common::Timer::GetCurrentValue();
|
current_time = Common::Timer::GetCurrentValue();
|
||||||
if (enet_timeout == 0 || current_time >= until_time)
|
if (enet_timeout == 0 || current_time >= until_time)
|
||||||
|
@ -1003,6 +1061,43 @@ void Netplay::HandleSynchronizeCompleteMessage(s32 player_id, const ENetPacket*
|
||||||
CheckForCompleteResynchronize();
|
CheckForCompleteResynchronize();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Netplay::HandleSessionMessage(s32 player_id, const ENetPacket* pkt)
|
||||||
|
{
|
||||||
|
if (pkt->dataLength < sizeof(ControlMessageHeader))
|
||||||
|
{
|
||||||
|
Log_ErrorPrintf("Invalid control packet from player %d of size %zu", player_id, pkt->dataLength);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const SessionMessageHeader* hdr = reinterpret_cast<const SessionMessageHeader*>(pkt->data);
|
||||||
|
switch (hdr->type)
|
||||||
|
{
|
||||||
|
case SessionMessage::ChatMessage:
|
||||||
|
HandleSessionChatMessage(player_id, pkt);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
Log_ErrorPrintf("Unhandled session packet %u from player %d of size %zu", hdr->type, player_id, pkt->dataLength);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Netplay::HandleSessionChatMessage(s32 player_id, const ENetPacket* pkt)
|
||||||
|
{
|
||||||
|
const SessionChatMessage* msg = reinterpret_cast<const SessionChatMessage*>(pkt->data);
|
||||||
|
if (pkt->dataLength < sizeof(SessionChatMessage) ||
|
||||||
|
pkt->dataLength < (sizeof(SessionChatMessage) + msg->chat_message_size))
|
||||||
|
{
|
||||||
|
// invalid chat message. ignore.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string message(pkt->data + sizeof(SessionChatMessage),
|
||||||
|
pkt->data + sizeof(SessionChatMessage) + msg->chat_message_size);
|
||||||
|
|
||||||
|
Host::OnNetplayMessage(fmt::format("Player {}: {}", player_id + 1, message));
|
||||||
|
}
|
||||||
|
|
||||||
void Netplay::CheckForCompleteResynchronize()
|
void Netplay::CheckForCompleteResynchronize()
|
||||||
{
|
{
|
||||||
if (s_synchronized_players == s_num_players)
|
if (s_synchronized_players == s_num_players)
|
||||||
|
@ -1037,8 +1132,6 @@ void Netplay::SetSettings()
|
||||||
|
|
||||||
// no block linking, it degrades savestate loading performance
|
// no block linking, it degrades savestate loading performance
|
||||||
si.SetBoolValue("CPU", "RecompilerBlockLinking", false);
|
si.SetBoolValue("CPU", "RecompilerBlockLinking", false);
|
||||||
// not sure its needed but enabled for now... TODO
|
|
||||||
si.SetBoolValue("GPU", "UseSoftwareRendererForReadbacks", true);
|
|
||||||
|
|
||||||
Host::Internal::SetNetplaySettingsLayer(&si);
|
Host::Internal::SetNetplaySettingsLayer(&si);
|
||||||
System::ApplySettings(false);
|
System::ApplySettings(false);
|
||||||
|
@ -1065,6 +1158,9 @@ void Netplay::UpdateThrottlePeriod()
|
||||||
|
|
||||||
void Netplay::HandleTimeSyncEvent(float frame_delta, int update_interval)
|
void Netplay::HandleTimeSyncEvent(float frame_delta, int update_interval)
|
||||||
{
|
{
|
||||||
|
// only activate timesync if its worth correcting.
|
||||||
|
if (std::abs(frame_delta) < 1.0f)
|
||||||
|
return;
|
||||||
// Distribute the frame difference over the next N * 0.75 frames.
|
// Distribute the frame difference over the next N * 0.75 frames.
|
||||||
// only part of the interval time is used since we want to come back to normal speed.
|
// only part of the interval time is used since we want to come back to normal speed.
|
||||||
// otherwise we will keep spiraling into unplayable gameplay.
|
// otherwise we will keep spiraling into unplayable gameplay.
|
||||||
|
@ -1140,37 +1236,6 @@ void Netplay::GenerateChecksumForFrame(int* checksum, int frame, unsigned char*
|
||||||
// Log_VerbosePrintf("Netplay Checksum: f:%d wf:%d c:%u", frame, frame % num_group_of_pages, *checksum);
|
// Log_VerbosePrintf("Netplay Checksum: f:%d wf:%d c:%u", frame, frame % num_group_of_pages, *checksum);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Netplay::GenerateDesyncReport(s32 desync_frame)
|
|
||||||
{
|
|
||||||
std::string path = "\\netplaylogs\\desync_frame_" + std::to_string(desync_frame) + "_p" +
|
|
||||||
std::to_string(s_local_handle) + "_" + System::GetRunningSerial() + "_.txt";
|
|
||||||
std::string filename = EmuFolders::Dumps + path;
|
|
||||||
|
|
||||||
std::unique_ptr<ByteStream> stream =
|
|
||||||
ByteStream::OpenFile(filename.c_str(), BYTESTREAM_OPEN_CREATE | BYTESTREAM_OPEN_WRITE | BYTESTREAM_OPEN_TRUNCATE |
|
|
||||||
BYTESTREAM_OPEN_ATOMIC_UPDATE | BYTESTREAM_OPEN_STREAMED);
|
|
||||||
if (!stream)
|
|
||||||
{
|
|
||||||
Log_VerbosePrint("desync log creation failed to create stream");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!ByteStream::WriteBinaryToStream(stream.get(),
|
|
||||||
s_save_buffer_pool.back().get()->state_stream.get()->GetMemoryPointer(),
|
|
||||||
s_save_buffer_pool.back().get()->state_stream.get()->GetMemorySize()))
|
|
||||||
{
|
|
||||||
Log_VerbosePrint("desync log creation failed to write the stream");
|
|
||||||
stream->Discard();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
/* stream->Write(s_save_buffer_pool.back().get()->state_stream.get()->GetMemoryPointer(),
|
|
||||||
s_save_buffer_pool.back().get()->state_stream.get()->GetMemorySize());*/
|
|
||||||
|
|
||||||
stream->Commit();
|
|
||||||
|
|
||||||
Log_VerbosePrintf("desync log created for frame %d", desync_frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Netplay::AdvanceFrame()
|
void Netplay::AdvanceFrame()
|
||||||
{
|
{
|
||||||
ggpo_advance_frame(s_ggpo, 0);
|
ggpo_advance_frame(s_ggpo, 0);
|
||||||
|
@ -1178,19 +1243,13 @@ void Netplay::AdvanceFrame()
|
||||||
|
|
||||||
void Netplay::RunFrame()
|
void Netplay::RunFrame()
|
||||||
{
|
{
|
||||||
|
PollEnet(0);
|
||||||
|
|
||||||
|
if (!s_ggpo)
|
||||||
|
return;
|
||||||
// housekeeping
|
// housekeeping
|
||||||
// TODO: get rid of double polling
|
|
||||||
PollEnet(0);
|
|
||||||
if (!s_ggpo)
|
|
||||||
return;
|
|
||||||
|
|
||||||
ggpo_network_idle(s_ggpo);
|
ggpo_network_idle(s_ggpo);
|
||||||
PollEnet(0);
|
|
||||||
if (!s_ggpo)
|
|
||||||
return;
|
|
||||||
|
|
||||||
ggpo_idle(s_ggpo);
|
ggpo_idle(s_ggpo);
|
||||||
|
|
||||||
// run game
|
// run game
|
||||||
auto result = GGPO_OK;
|
auto result = GGPO_OK;
|
||||||
int disconnect_flags = 0;
|
int disconnect_flags = 0;
|
||||||
|
@ -1238,7 +1297,36 @@ Netplay::Input Netplay::ReadLocalInput()
|
||||||
return inp;
|
return inp;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Netplay::SendMsg(const char* msg) {}
|
void Netplay::SendMsg(std::string msg)
|
||||||
|
{
|
||||||
|
SessionChatMessage header{};
|
||||||
|
const size_t msg_size = msg.size();
|
||||||
|
|
||||||
|
header.header.type = SessionMessage::ChatMessage;
|
||||||
|
header.header.size = sizeof(SessionChatMessage) + msg_size;
|
||||||
|
header.chat_message_size = msg_size;
|
||||||
|
|
||||||
|
ENetPacket* pkt = enet_packet_create(nullptr, sizeof(header) + msg_size, ENET_PACKET_FLAG_RELIABLE);
|
||||||
|
std::memcpy(pkt->data, &header, sizeof(header));
|
||||||
|
std::memcpy(pkt->data + sizeof(header), msg.c_str(), msg_size);
|
||||||
|
|
||||||
|
for (s32 i = 0; i < MAX_PLAYERS; i++)
|
||||||
|
{
|
||||||
|
if (!s_peers[i].peer)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const int err = enet_peer_send(s_peers[i].peer, ENET_CHANNEL_SESSION, pkt);
|
||||||
|
if (err != 0)
|
||||||
|
{
|
||||||
|
// failed to send netplay message? just clean it up.
|
||||||
|
Log_ErrorPrint("Failed to send netplay message");
|
||||||
|
enet_packet_destroy(pkt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// add own netplay message locally to netplay messages
|
||||||
|
Host::OnNetplayMessage(fmt::format("Player {}: {}", s_local_handle, msg));
|
||||||
|
}
|
||||||
|
|
||||||
GGPOErrorCode Netplay::SyncInput(Netplay::Input inputs[2], int* disconnect_flags)
|
GGPOErrorCode Netplay::SyncInput(Netplay::Input inputs[2], int* disconnect_flags)
|
||||||
{
|
{
|
||||||
|
@ -1377,8 +1465,7 @@ bool Netplay::NpAdvFrameCb(void* ctx, int flags)
|
||||||
bool Netplay::NpSaveFrameCb(void* ctx, unsigned char** buffer, int* len, int* checksum, int frame)
|
bool Netplay::NpSaveFrameCb(void* ctx, unsigned char** buffer, int* len, int* checksum, int frame)
|
||||||
{
|
{
|
||||||
SaveStateBuffer our_buffer;
|
SaveStateBuffer our_buffer;
|
||||||
// min size is 2 because otherwise the desync logger doesnt have enough time to dump the state.
|
if (s_save_buffer_pool.empty())
|
||||||
if (s_save_buffer_pool.size() < 2)
|
|
||||||
{
|
{
|
||||||
our_buffer = std::make_unique<System::MemorySaveState>();
|
our_buffer = std::make_unique<System::MemorySaveState>();
|
||||||
}
|
}
|
||||||
|
@ -1465,7 +1552,6 @@ bool Netplay::NpOnEventCb(void* ctx, GGPOEvent* ev)
|
||||||
CurrentFrame(), ev->u.desync.nFrameOfDesync,
|
CurrentFrame(), ev->u.desync.nFrameOfDesync,
|
||||||
CurrentFrame() - ev->u.desync.nFrameOfDesync, ev->u.desync.ourCheckSum,
|
CurrentFrame() - ev->u.desync.nFrameOfDesync, ev->u.desync.ourCheckSum,
|
||||||
ev->u.desync.remoteChecksum));
|
ev->u.desync.remoteChecksum));
|
||||||
GenerateDesyncReport(ev->u.desync.nFrameOfDesync);
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
Host::OnNetplayMessage(fmt::format("Netplay Event Code: {}", static_cast<int>(ev->code)));
|
Host::OnNetplayMessage(fmt::format("Netplay Event Code: {}", static_cast<int>(ev->code)));
|
||||||
|
|
|
@ -8,7 +8,7 @@ namespace Netplay {
|
||||||
enum : s32
|
enum : s32
|
||||||
{
|
{
|
||||||
// Maximum number of emulated controllers.
|
// Maximum number of emulated controllers.
|
||||||
MAX_PLAYERS = 2,
|
MAX_PLAYERS = 2,
|
||||||
// Maximum netplay prediction frames
|
// Maximum netplay prediction frames
|
||||||
MAX_ROLLBACK_FRAMES = 8,
|
MAX_ROLLBACK_FRAMES = 8,
|
||||||
};
|
};
|
||||||
|
@ -17,6 +17,7 @@ enum : u8
|
||||||
{
|
{
|
||||||
ENET_CHANNEL_CONTROL = 0,
|
ENET_CHANNEL_CONTROL = 0,
|
||||||
ENET_CHANNEL_GGPO = 1,
|
ENET_CHANNEL_GGPO = 1,
|
||||||
|
ENET_CHANNEL_SESSION = 2,
|
||||||
|
|
||||||
NUM_ENET_CHANNELS,
|
NUM_ENET_CHANNELS,
|
||||||
};
|
};
|
||||||
|
@ -36,7 +37,7 @@ void ExecuteNetplay();
|
||||||
|
|
||||||
void CollectInput(u32 slot, u32 bind, float value);
|
void CollectInput(u32 slot, u32 bind, float value);
|
||||||
|
|
||||||
void SendMsg(const char* msg);
|
void SendMsg(std::string msg);
|
||||||
|
|
||||||
s32 GetPing();
|
s32 GetPing();
|
||||||
u32 GetMaxPrediction();
|
u32 GetMaxPrediction();
|
||||||
|
|
|
@ -1079,12 +1079,6 @@ void EmuThread::startNetplaySession(int local_handle, quint16 local_port, const
|
||||||
Q_ARG(quint16, remote_port), Q_ARG(int, input_delay), Q_ARG(const QString&, game_path));
|
Q_ARG(quint16, remote_port), Q_ARG(int, input_delay), Q_ARG(const QString&, game_path));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// disable block linking and disable rewind and runahead during a netplay session
|
|
||||||
g_settings.cpu_recompiler_block_linking = false;
|
|
||||||
g_settings.rewind_enable = false;
|
|
||||||
g_settings.runahead_frames = 0;
|
|
||||||
|
|
||||||
Log_WarningPrintf("Disabling block linking, runahead and rewind due to rollback.");
|
|
||||||
|
|
||||||
auto remAddr = remote_addr.trimmed().toStdString();
|
auto remAddr = remote_addr.trimmed().toStdString();
|
||||||
auto gamePath = game_path.trimmed().toStdString();
|
auto gamePath = game_path.trimmed().toStdString();
|
||||||
|
@ -1101,7 +1095,8 @@ void EmuThread::sendNetplayMessage(const QString& message)
|
||||||
QMetaObject::invokeMethod(this, "sendNetplayMessage", Qt::QueuedConnection, Q_ARG(const QString&, message));
|
QMetaObject::invokeMethod(this, "sendNetplayMessage", Qt::QueuedConnection, Q_ARG(const QString&, message));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Netplay::SendMsg(message.toStdString().c_str());
|
// TODO REDO NETPLAY UI
|
||||||
|
// Netplay::SendMsg(message.toStdString().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void EmuThread::stopNetplaySession()
|
void EmuThread::stopNetplaySession()
|
||||||
|
|
|
@ -175,8 +175,10 @@ void ImGuiManager::DrawNetplayChatDialog()
|
||||||
const bool close_chat =
|
const bool close_chat =
|
||||||
send_message || (s_netplay_chat_message.empty() && (ImGui::IsKeyPressed(ImGuiKey_Backspace)) ||
|
send_message || (s_netplay_chat_message.empty() && (ImGui::IsKeyPressed(ImGuiKey_Backspace)) ||
|
||||||
ImGui::IsKeyPressed(ImGuiKey_Escape));
|
ImGui::IsKeyPressed(ImGuiKey_Escape));
|
||||||
|
|
||||||
|
// sending netplay message
|
||||||
if (send_message && !s_netplay_chat_message.empty())
|
if (send_message && !s_netplay_chat_message.empty())
|
||||||
Netplay::SendMsg(s_netplay_chat_message.c_str());
|
Netplay::SendMsg(s_netplay_chat_message);
|
||||||
|
|
||||||
const ImGuiIO& io = ImGui::GetIO();
|
const ImGuiIO& io = ImGui::GetIO();
|
||||||
const ImGuiStyle& style = ImGui::GetStyle();
|
const ImGuiStyle& style = ImGui::GetStyle();
|
||||||
|
|
Loading…
Reference in New Issue