diff --git a/amxmodx/meta_api.cpp b/amxmodx/meta_api.cpp index 6a4e10ad..50614ca7 100755 --- a/amxmodx/meta_api.cpp +++ b/amxmodx/meta_api.cpp @@ -34,6 +34,7 @@ #include "CoreConfig.h" #include #include +#include plugin_info_t Plugin_info = { @@ -157,6 +158,9 @@ int FF_ClientConnectEx = -1; IFileSystem* g_FileSystem; HLTypeConversion TypeConversion; +bool HasReHLDS; +server_t *Server; + bool ColoredMenus(const char *ModName) { const char * pModNames[] = { "cstrike", "czero", "dmc", "dod", "tfc", "valve" }; @@ -679,6 +683,121 @@ void C_ServerActivate_Post(edict_t *pEdictList, int edictCount, int clientMax) g_memreport_enabled = true; #endif + if (!HasReHLDS && Server) + { + const auto precacheTextures = [](const char *model, studiohdr_t *pStudioHeader) + { + if (pStudioHeader->textureindex) + { + return; // Skip if textures are internal + } + + const auto modelLength = strlen(model); + + if (modelLength > MAX_MODEL_NAME - 2) + { + return; // Skip if model length goes above the max minus 'T' + EOS + } + + char textureModel[MAX_MODEL_NAME]; + ke::SafeStrcpy(textureModel, sizeof textureModel, model); + + const auto modelExtensionLength = sizeof ".mdl" - 1; + const auto modelExtension = &textureModel[modelLength - modelExtensionLength]; + ke::SafeStrcpy(modelExtension, sizeof textureModel - modelLength - modelExtensionLength, "T.mdl"); + + #if defined PLATFORM_WINDOWS + if (!g_FileSystem->FileExists(textureModel)) + { + modelExtension[0] = 't'; + } + #endif + + PRECACHE_GENERIC(textureModel); + }; + + const auto precacheGroups = [](studiohdr_t *pStudioHeader) + { + if (pStudioHeader->numseqgroups <= 1) + { + return; // Skip if no available group + } + + const auto pSeqGroup = reinterpret_cast(reinterpret_cast(pStudioHeader) + pStudioHeader->seqgroupindex); + + for (auto group = 1; group < pStudioHeader->numseqgroups; group++) + { + if (pSeqGroup[group].name[0] == '\0') + { + continue; + } + + char seqGroupName[MAX_QPATH]; + const auto length = ke::SafeStrcpy(seqGroupName, sizeof seqGroupName, pSeqGroup[group].name); + + for (auto index = 0u; index < length; index++) + { + if (seqGroupName[index] == '\\') + { + seqGroupName[index] = '/'; + } + } + + PRECACHE_GENERIC(seqGroupName); + } + }; + + const auto precacheSounds = [](studiohdr_t *pStudioHeader) + { + const auto pSeqDesc = reinterpret_cast(reinterpret_cast(pStudioHeader) + pStudioHeader->seqindex); + + for (auto sequence = 0; sequence < pStudioHeader->numseq; sequence++) + { + const auto pEvent = reinterpret_cast(reinterpret_cast(pStudioHeader) + pSeqDesc[sequence].eventindex); + + for (auto index = 0; index < pSeqDesc[sequence].numevents; index++) + { + if (pEvent[index].event != 5004 || pEvent[index].options[0] == '\0') + { + continue; + } + + char sound[MAX_QPATH]; + ke::SafeSprintf(sound, sizeof sound, "sound/%s", pEvent[index].options); + + PRECACHE_GENERIC(sound); + } + } + }; + + const auto pEntity = CREATE_ENTITY(); + + for (auto index = 1u; index < ARRAYSIZE(Server->model_precache); index++) + { + const auto model = Server->model_precache[index]; + + if (!model) + { + break; + } + + if (Server->models[index]->type != mod_studio) + { + continue; + } + + pEntity->v.modelindex = index; + + const auto pStudioHeader = static_cast(GET_MODEL_PTR(pEntity)); + + precacheTextures(model, pStudioHeader); + precacheGroups(pStudioHeader); + precacheSounds(pStudioHeader); + } + + REMOVE_ENTITY(pEntity); + } + g_activated = true; RETURN_META(MRES_IGNORED); @@ -1670,14 +1789,30 @@ C_DLLEXPORT int Meta_Attach(PLUG_LOADTIME now, META_FUNCTIONS *pFunctionTable, m if (RehldsApi_Init()) { + HasReHLDS = true; + RehldsHookchains->SV_DropClient()->registerHook(SV_DropClient_RH); g_isDropClientHookAvailable = true; g_isDropClientHookEnabled = true; + + } else { + HasReHLDS = false; + void *address = nullptr; + if (CommonConfig->GetAddress("sv", &address) && address) + { + Server = *reinterpret_cast(address); + } + else + { + AMXXLOG_Log("Automatic precaching of model textures and groups has been disabled - check your gamedata files."); + } + + if (CommonConfig && CommonConfig->GetMemSig("SV_DropClient", &address) && address) { DropClientDetour = DETOUR_CREATE_STATIC_FIXED(SV_DropClient, address);