iw4x-client/src/Components/Modules/ServerInfo.cpp

338 lines
10 KiB
C++
Raw Normal View History

2022-02-27 07:53:44 -05:00
#include <STDInclude.hpp>
#include <Utils/InfoString.hpp>
#include "Gamepad.hpp"
#include "Party.hpp"
#include "ServerInfo.hpp"
#include "ServerList.hpp"
#include "UIFeeder.hpp"
#include "Voice.hpp"
2017-01-19 16:23:59 -05:00
#include <version.hpp>
2017-01-19 16:23:59 -05:00
namespace Components
{
ServerInfo::Container ServerInfo::PlayerContainer;
unsigned int ServerInfo::GetPlayerCount()
{
return PlayerContainer.playerList.size();
2017-01-19 16:23:59 -05:00
}
const char* ServerInfo::GetPlayerText(unsigned int index, int column)
{
if (index < PlayerContainer.playerList.size())
2017-01-19 16:23:59 -05:00
{
switch (column)
{
case 0:
return Utils::String::VA("%d", index);
case 1:
return PlayerContainer.playerList[index].name.data();
2017-01-19 16:23:59 -05:00
case 2:
return Utils::String::VA("%d", PlayerContainer.playerList[index].score);
2017-01-19 16:23:59 -05:00
case 3:
return Utils::String::VA("%d", PlayerContainer.playerList[index].ping);
default:
break;
2017-01-19 16:23:59 -05:00
}
}
return "";
}
void ServerInfo::SelectPlayer(unsigned int index)
{
PlayerContainer.currentPlayer = index;
2017-01-19 16:23:59 -05:00
}
2022-08-24 10:38:14 -04:00
void ServerInfo::ServerStatus([[maybe_unused]] const UIScript::Token& token, [[maybe_unused]] const Game::uiInfo_s* info)
2017-01-19 16:23:59 -05:00
{
PlayerContainer.currentPlayer = 0;
PlayerContainer.playerList.clear();
2017-01-19 16:23:59 -05:00
2022-08-24 10:38:14 -04:00
auto* serverInfo = ServerList::GetCurrentServer();
2017-01-19 16:23:59 -05:00
if (info)
2017-01-19 16:23:59 -05:00
{
2022-08-24 10:38:14 -04:00
Dvar::Var("uiSi_ServerName").set(serverInfo->hostname);
Dvar::Var("uiSi_MaxClients").set(serverInfo->clients);
Dvar::Var("uiSi_Version").set(serverInfo->shortversion);
Dvar::Var("uiSi_SecurityLevel").set(serverInfo->securityLevel);
Dvar::Var("uiSi_isPrivate").set(serverInfo->password ? "@MENU_YES" : "@MENU_NO");
Dvar::Var("uiSi_Hardcore").set(serverInfo->hardcore ? "@MENU_ENABLED" : "@MENU_DISABLED");
2017-01-19 16:23:59 -05:00
Dvar::Var("uiSi_KillCam").set("@MENU_NO");
Dvar::Var("uiSi_ffType").set("@MENU_DISABLED");
2022-08-24 10:38:14 -04:00
Dvar::Var("uiSi_MapName").set(serverInfo->mapname);
Dvar::Var("uiSi_MapNameLoc").set(Game::UI_LocalizeMapName(serverInfo->mapname.data()));
Dvar::Var("uiSi_GameType").set(Game::UI_LocalizeGameType(serverInfo->gametype.data()));
2017-01-19 16:23:59 -05:00
Dvar::Var("uiSi_ModName").set("");
2022-08-24 10:38:14 -04:00
Dvar::Var("uiSi_aimAssist").set(serverInfo->aimassist ? "@MENU_YES" : "@MENU_NO");
Dvar::Var("uiSi_voiceChat").set(serverInfo->voice ? "@MENU_YES" : "@MENU_NO");
2017-01-19 16:23:59 -05:00
2022-08-24 10:38:14 -04:00
if (serverInfo->mod.size() > 5)
2017-01-19 16:23:59 -05:00
{
2022-08-24 10:38:14 -04:00
Dvar::Var("uiSi_ModName").set(serverInfo->mod.data() + 5);
2017-01-19 16:23:59 -05:00
}
PlayerContainer.target = serverInfo->addr;
Network::SendCommand(PlayerContainer.target, "getstatus");
2017-01-19 16:23:59 -05:00
}
}
2021-09-07 19:53:25 -04:00
void ServerInfo::DrawScoreboardInfo(int localClientNum)
2017-01-19 16:23:59 -05:00
{
2018-05-09 08:33:52 -04:00
Game::Font_s* font = Game::R_RegisterFont("fonts/bigfont", 0);
2022-04-12 17:15:50 -04:00
const auto* cxt = Game::ScrPlace_GetActivePlacement(localClientNum);
auto addressText = Network::Address(*Game::connectedHost).getString();
2017-01-19 16:23:59 -05:00
if (addressText == "0.0.0.0:0"s || addressText == "loopback"s)
{
addressText = "Listen Server"s;
}
2017-01-19 16:23:59 -05:00
2022-04-12 17:15:50 -04:00
// Get x positions
auto y = (480.0f - (*Game::cg_scoreboardHeight)->current.value) * 0.5f;
y += (*Game::cg_scoreboardHeight)->current.value + 6.0f;
2017-01-19 16:23:59 -05:00
const auto x = 320.0f - (*Game::cg_scoreboardWidth)->current.value * 0.5f;
const auto x2 = 320.0f + (*Game::cg_scoreboardWidth)->current.value * 0.5f;
2017-01-19 16:23:59 -05:00
2022-04-12 17:15:50 -04:00
// Draw only when stream friendly ui is not enabled
if (!Friends::UIStreamFriendly.get<bool>())
2017-01-19 16:23:59 -05:00
{
2022-04-12 17:15:50 -04:00
constexpr auto fontSize = 0.35f;
Game::UI_DrawText(cxt, reinterpret_cast<const char*>(0x7ED3F8), std::numeric_limits<int>::max(), font, x, y, 0, 0, fontSize, reinterpret_cast<float*>(0x747F34), 3);
Game::UI_DrawText(cxt, addressText.data(), std::numeric_limits<int>::max(), font, x2 - Game::UI_TextWidth(addressText.data(), 0, font, fontSize), y, 0, 0, fontSize, reinterpret_cast<float*>(0x747F34), 3);
2017-01-19 16:23:59 -05:00
}
}
__declspec(naked) void ServerInfo::DrawScoreboardStub()
{
__asm
{
2017-02-01 07:44:25 -05:00
pushad
2017-01-19 16:23:59 -05:00
push eax
call DrawScoreboardInfo
2017-01-19 16:23:59 -05:00
pop eax
2017-02-01 07:44:25 -05:00
popad
push 591B70h
retn
2017-01-19 16:23:59 -05:00
}
}
2019-01-09 15:40:56 -05:00
Utils::InfoString ServerInfo::GetHostInfo()
{
Utils::InfoString info;
info.set("admin", Dvar::Var("_Admin").get<const char*>());
info.set("website", Dvar::Var("_Website").get<const char*>());
info.set("email", Dvar::Var("_Email").get<const char*>());
info.set("location", Dvar::Var("_Location").get<const char*>());
return info;
}
2017-01-19 16:23:59 -05:00
Utils::InfoString ServerInfo::GetInfo()
{
auto maxClientCount = *Game::svs_clientCount;
const auto* password = *Game::g_password ? (*Game::g_password)->current.string : "";
2017-01-19 16:23:59 -05:00
if (!maxClientCount)
2017-01-19 16:23:59 -05:00
{
maxClientCount = *Game::party_maxplayers ? (*Game::party_maxplayers)->current.integer : 18;
2017-01-19 16:23:59 -05:00
}
Utils::InfoString info(Game::Dvar_InfoString_Big(Game::DVAR_SERVERINFO));
2017-01-19 16:23:59 -05:00
info.set("gamename", "IW4");
info.set("sv_maxclients", std::to_string(maxClientCount));
info.set("protocol", std::to_string(PROTOCOL));
2017-01-19 16:23:59 -05:00
info.set("shortversion", SHORTVERSION);
info.set("version", (*Game::version)->current.string);
info.set("mapname", (*Game::sv_mapname)->current.string);
2022-12-25 12:23:53 -05:00
info.set("isPrivate", *password ? "1" : "0");
info.set("checksum", Utils::String::VA("%X", Utils::Cryptography::JenkinsOneAtATime::Compute(std::to_string(Game::Sys_Milliseconds()))));
info.set("aimAssist", (Gamepad::sv_allowAimAssist.get<bool>() ? "1" : "0"));
info.set("voiceChat", (Voice::SV_VoiceEnabled() ? "1" : "0"));
2017-01-19 16:23:59 -05:00
// Ensure mapname is set
if (info.get("mapname").empty())
{
2023-01-27 18:05:26 -05:00
info.set("mapname", (*Game::ui_mapname)->current.string);
2017-01-19 16:23:59 -05:00
}
// Set matchtype
// 0 - No match, connecting not possible
// 1 - Party, use Steam_JoinLobby to connect
// 2 - Match, use CL_ConnectFromParty to connect
if (Party::IsEnabled() && Dvar::Var("party_host").get<bool>()) // Party hosting
2017-01-19 16:23:59 -05:00
{
info.set("matchtype", "1");
}
else if (Dedicated::IsRunning()) // Match hosting
2017-01-19 16:23:59 -05:00
{
info.set("matchtype", "2");
}
else
{
info.set("matchtype", "0");
}
return info;
}
ServerInfo::ServerInfo()
{
PlayerContainer.currentPlayer = 0;
2017-01-19 16:23:59 -05:00
// Draw IP and hostname on the scoreboard
Utils::Hook(0x4FC6EA, DrawScoreboardStub, HOOK_CALL).install()->quick();
2017-01-19 16:23:59 -05:00
// Ignore native getStatus implementation
Utils::Hook::Nop(0x62654E, 6);
// Add uiscript
UIScript::Add("ServerStatus", ServerStatus);
2017-01-19 16:23:59 -05:00
// Add uifeeder
UIFeeder::Add(13.0f, GetPlayerCount, GetPlayerText, SelectPlayer);
2017-01-19 16:23:59 -05:00
2022-09-24 13:04:37 -04:00
Network::OnClientPacket("getStatus", [](const Network::Address& address, [[maybe_unused]] const std::string& data)
2017-01-19 16:23:59 -05:00
{
std::string playerList;
Utils::InfoString info = GetInfo();
2017-01-19 16:23:59 -05:00
info.set("challenge", Utils::ParseChallenge(data));
for (std::size_t i = 0; i < Game::MAX_CLIENTS; ++i)
2017-01-19 16:23:59 -05:00
{
2022-05-20 18:12:46 -04:00
auto score = 0;
auto ping = 0;
2017-01-19 16:23:59 -05:00
std::string name;
if (Dedicated::IsRunning())
2017-01-19 16:23:59 -05:00
{
if (Game::svs_clients[i].header.state < Game::CS_ACTIVE) continue;
if (!Game::svs_clients[i].gentity || !Game::svs_clients[i].gentity->client) continue;
const auto* client = Game::svs_clients[i].gentity->client;
const auto team = client->sess.cs.team;
if (team == Game::TEAM_SPECTATOR)
{
continue;
}
2017-01-19 16:23:59 -05:00
score = Game::SV_GameClientNum_Score(static_cast<int>(i));
2017-01-19 16:23:59 -05:00
ping = Game::svs_clients[i].ping;
name = Game::svs_clients[i].name;
}
else
{
// Score and ping are irrelevant
const auto* namePtr = Game::PartyHost_GetMemberName(reinterpret_cast<Game::PartyData*>(0x1081C00), i);
2017-01-19 16:23:59 -05:00
if (!namePtr || !namePtr[0]) continue;
name = namePtr;
}
playerList.append(Utils::String::VA("%i %i \"%s\"\n", score, ping, name.data()));
}
Network::SendCommand(address, "statusResponse", "\\" + info.build() + "\n" + playerList + "\n");
2017-01-19 16:23:59 -05:00
});
2022-08-19 19:10:35 -04:00
Network::OnClientPacket("statusResponse", [](const Network::Address& address, [[maybe_unused]] const std::string& data)
2017-01-19 16:23:59 -05:00
{
if (PlayerContainer.target != address)
2017-01-19 16:23:59 -05:00
{
return;
}
2017-01-19 16:23:59 -05:00
const Utils::InfoString info(data.substr(0, data.find_first_of('\n')));
Dvar::Var("uiSi_ServerName").set(info.get("sv_hostname"));
Dvar::Var("uiSi_MaxClients").set(info.get("sv_maxclients"));
Dvar::Var("uiSi_Version").set(info.get("shortversion"));
Dvar::Var("uiSi_SecurityLevel").set(info.get("sv_securityLevel"));
Dvar::Var("uiSi_isPrivate").set(info.get("isPrivate") == "0" ? "@MENU_NO" : "@MENU_YES");
Dvar::Var("uiSi_Hardcore").set(info.get("g_hardcore") == "0" ? "@MENU_DISABLED" : "@MENU_ENABLED");
Dvar::Var("uiSi_KillCam").set(info.get("scr_game_allowkillcam") == "0" ? "@MENU_NO" : "@MENU_YES");
Dvar::Var("uiSi_MapName").set(info.get("mapname"));
Dvar::Var("uiSi_MapNameLoc").set(Game::UI_LocalizeMapName(info.get("mapname").data()));
Dvar::Var("uiSi_GameType").set(Game::UI_LocalizeGameType(info.get("g_gametype").data()));
Dvar::Var("uiSi_ModName").set("");
Dvar::Var("uiSi_aimAssist").set(info.get("aimAssist") == "0" ? "@MENU_DISABLED" : "@MENU_ENABLED");
Dvar::Var("uiSi_voiceChat").set(info.get("voiceChat") == "0" ? "@MENU_DISABLED" : "@MENU_ENABLED");
2017-01-19 16:23:59 -05:00
switch (std::strtol(info.get("scr_team_fftype").data(), nullptr, 10))
{
default:
Dvar::Var("uiSi_ffType").set("@MENU_DISABLED");
break;
case 1:
Dvar::Var("uiSi_ffType").set("@MENU_ENABLED");
break;
case 2:
Dvar::Var("uiSi_ffType").set("@MPUI_RULES_REFLECT");
break;
case 3:
Dvar::Var("uiSi_ffType").set("@MPUI_RULES_SHARED");
break;
}
2017-01-19 16:23:59 -05:00
if (Utils::String::StartsWith(info.get("fs_game"), "mods/"))
{
const auto mod = info.get("fs_game");
Dvar::Var("uiSi_ModName").set(mod.substr(5));
}
2017-01-19 16:23:59 -05:00
const auto lines = Utils::String::Split(data, '\n');
2017-01-19 16:23:59 -05:00
if (lines.size() <= 1) return;
2017-01-19 16:23:59 -05:00
for (std::size_t i = 1; i < lines.size(); ++i)
{
Container::Player player;
2017-01-19 16:23:59 -05:00
std::string currentData = lines[i];
2017-01-19 16:23:59 -05:00
if (currentData.size() < 3) continue;
2017-01-19 16:23:59 -05:00
// Insert score
player.score = atoi(currentData.substr(0, currentData.find_first_of(' ')).data());
2017-01-19 16:23:59 -05:00
// Remove score
currentData = currentData.substr(currentData.find_first_of(' ') + 1);
2017-01-19 16:23:59 -05:00
// Insert ping
player.ping = atoi(currentData.substr(0, currentData.find_first_of(' ')).data());
2017-01-19 16:23:59 -05:00
// Remove ping
currentData = currentData.substr(currentData.find_first_of(' ') + 1);
2017-01-19 16:23:59 -05:00
if (currentData[0] == '\"')
{
currentData = currentData.substr(1);
}
2017-01-19 16:23:59 -05:00
if (currentData.back() == '\"')
{
currentData.pop_back();
}
2017-01-19 16:23:59 -05:00
player.name = currentData;
2017-01-19 16:23:59 -05:00
PlayerContainer.playerList.push_back(player);
2017-01-19 16:23:59 -05:00
}
});
}
ServerInfo::~ServerInfo()
{
PlayerContainer.playerList.clear();
2017-01-19 16:23:59 -05:00
}
}