2022-02-27 07:53:44 -05:00
# include <STDInclude.hpp>
2023-01-03 07:16:44 -05:00
2023-03-22 14:47:13 -04:00
# include "ArenaLength.hpp"
2023-01-03 07:16:44 -05:00
# include "FastFiles.hpp"
2022-12-26 07:07:24 -05:00
# include "RawFiles.hpp"
# include "StartupMessages.hpp"
# include "Theatre.hpp"
2017-01-16 11:42:50 -05:00
namespace Components
{
2017-04-06 16:22:30 -04:00
Maps : : UserMapContainer Maps : : UserMap ;
2017-01-16 11:42:50 -05:00
std : : string Maps : : CurrentMainZone ;
std : : vector < std : : pair < std : : string , std : : string > > Maps : : DependencyList ;
std : : vector < std : : string > Maps : : CurrentDependencies ;
2023-03-17 08:46:29 -04:00
std : : vector < std : : string > Maps : : FoundCustomMaps ;
2017-01-16 11:42:50 -05:00
2022-06-16 10:15:26 -04:00
Dvar : : Var Maps : : RListSModels ;
2017-04-06 16:22:30 -04:00
bool Maps : : SPMap ;
2017-01-16 11:42:50 -05:00
std : : vector < Maps : : DLC > Maps : : DlcPacks ;
2017-04-08 09:32:51 -04:00
const char * Maps : : UserMapFiles [ 4 ] =
{
" .ff " ,
" _load.ff " ,
" .iwd " ,
" .arena " ,
} ;
2017-04-06 16:22:30 -04:00
Maps : : UserMapContainer * Maps : : GetUserMap ( )
{
return & Maps : : UserMap ;
}
void Maps : : UserMapContainer : : loadIwd ( )
{
if ( this - > isValid ( ) & & ! this - > searchPath . iwd )
{
2022-12-11 12:54:24 -05:00
auto iwdName = std : : format ( " {}.iwd " , this - > mapname ) ;
2023-01-05 04:59:09 -05:00
auto path = std : : format ( " {} \\ usermaps \\ {} \\ {} " , ( * Game : : fs_basepath ) - > current . string , this - > mapname , iwdName ) ;
2017-04-06 16:22:30 -04:00
2017-04-09 12:27:58 -04:00
if ( Utils : : IO : : FileExists ( path ) )
2017-04-06 16:22:30 -04:00
{
2017-04-09 12:27:58 -04:00
this - > searchPath . iwd = Game : : FS_IsShippedIWD ( path . data ( ) , iwdName . data ( ) ) ;
if ( this - > searchPath . iwd )
{
this - > searchPath . bLocalized = false ;
this - > searchPath . ignore = 0 ;
this - > searchPath . ignorePureCheck = 0 ;
this - > searchPath . language = 0 ;
this - > searchPath . dir = nullptr ;
this - > searchPath . next = * Game : : fs_searchpaths ;
* Game : : fs_searchpaths = & this - > searchPath ;
}
2017-04-06 16:22:30 -04:00
}
}
}
void Maps : : UserMapContainer : : reloadIwd ( )
{
if ( this - > isValid ( ) & & this - > wasFreed )
{
this - > wasFreed = false ;
this - > searchPath . iwd = nullptr ;
this - > loadIwd ( ) ;
}
}
void Maps : : UserMapContainer : : handlePackfile ( void * packfile )
{
2017-04-07 15:37:01 -04:00
if ( this - > isValid ( ) & & this - > searchPath . iwd = = packfile )
2017-04-06 16:22:30 -04:00
{
this - > wasFreed = true ;
}
}
void Maps : : UserMapContainer : : freeIwd ( )
{
2017-04-07 15:37:01 -04:00
if ( this - > isValid ( ) & & this - > searchPath . iwd & & ! this - > wasFreed )
2017-04-06 16:22:30 -04:00
{
this - > wasFreed = true ;
// Unchain our searchpath
2023-01-02 12:04:34 -05:00
for ( auto * * pathPtr = Game : : fs_searchpaths ; * pathPtr ; pathPtr = & ( * pathPtr ) - > next )
2017-04-06 16:22:30 -04:00
{
2017-04-07 15:37:01 -04:00
if ( * pathPtr = = & this - > searchPath )
2017-04-06 16:22:30 -04:00
{
* pathPtr = ( * pathPtr ) - > next ;
break ;
}
}
Game : : unzClose ( this - > searchPath . iwd - > handle ) ;
auto _free = Utils : : Hook : : Call < void ( void * ) > ( 0x6B5CF2 ) ;
_free ( this - > searchPath . iwd - > buildBuffer ) ;
_free ( this - > searchPath . iwd ) ;
ZeroMemory ( & this - > searchPath , sizeof this - > searchPath ) ;
}
}
2017-04-08 09:32:51 -04:00
const char * Maps : : LoadArenaFileStub ( const char * name , char * buffer , int size )
{
2022-08-24 17:46:07 -04:00
std : : string data = RawFiles : : ReadRawFile ( name , buffer , size ) ;
2017-04-08 09:32:51 -04:00
2022-08-24 17:46:07 -04:00
if ( Maps : : UserMap . isValid ( ) )
2017-04-08 09:32:51 -04:00
{
2023-03-17 08:46:29 -04:00
const auto mapname = Maps : : UserMap . getName ( ) ;
const auto arena = GetArenaPath ( mapname ) ;
2017-04-08 09:32:51 -04:00
if ( Utils : : IO : : FileExists ( arena ) )
{
data . append ( Utils : : IO : : ReadFile ( arena ) ) ;
}
}
2022-08-24 17:46:07 -04:00
strncpy_s ( buffer , size , data . data ( ) , _TRUNCATE ) ;
2017-04-08 09:32:51 -04:00
return buffer ;
}
2017-04-07 15:37:01 -04:00
void Maps : : UnloadMapZones ( Game : : XZoneInfo * zoneInfo , unsigned int zoneCount , int sync )
2017-04-06 16:22:30 -04:00
{
Game : : DB_LoadXAssets ( zoneInfo , zoneCount , sync ) ;
if ( Maps : : UserMap . isValid ( ) )
{
Maps : : UserMap . freeIwd ( ) ;
Maps : : UserMap . clear ( ) ;
}
}
2017-04-07 15:37:01 -04:00
void Maps : : LoadMapZones ( Game : : XZoneInfo * zoneInfo , unsigned int zoneCount , int sync )
2017-01-16 11:42:50 -05:00
{
if ( ! zoneInfo ) return ;
2017-04-06 16:22:30 -04:00
Maps : : SPMap = false ;
2017-01-16 11:42:50 -05:00
Maps : : CurrentMainZone = zoneInfo - > name ;
Maps : : CurrentDependencies . clear ( ) ;
2022-07-20 15:47:54 -04:00
auto dependencies = GetDependenciesForMap ( zoneInfo - > name ) ;
2017-01-16 11:42:50 -05:00
std : : vector < Game : : XZoneInfo > data ;
Utils : : Merge ( & data , zoneInfo , zoneCount ) ;
2022-07-20 15:47:54 -04:00
Utils : : Memory : : Allocator allocator ;
2019-10-02 02:08:11 -04:00
2022-07-20 15:47:54 -04:00
if ( dependencies . requiresTeamZones )
{
auto teams = dependencies . requiredTeams ;
Game : : XZoneInfo team ;
team . allocFlags = zoneInfo - > allocFlags ;
team . freeFlags = zoneInfo - > freeFlags ;
2017-01-16 11:42:50 -05:00
2022-12-11 12:54:24 -05:00
team . name = allocator . duplicateString ( std : : format ( " iw4x_team_{} " , teams . first ) ) ;
2022-07-20 15:47:54 -04:00
data . push_back ( team ) ;
2017-01-16 11:42:50 -05:00
2022-12-11 12:54:24 -05:00
team . name = allocator . duplicateString ( std : : format ( " iw4x_team_{} " , teams . second ) ) ;
2022-07-20 15:47:54 -04:00
data . push_back ( team ) ;
}
2017-01-16 11:42:50 -05:00
2022-07-20 15:47:54 -04:00
Utils : : Merge ( & Maps : : CurrentDependencies , dependencies . requiredMaps . data ( ) , dependencies . requiredMaps . size ( ) ) ;
2017-01-16 11:42:50 -05:00
for ( unsigned int i = 0 ; i < Maps : : CurrentDependencies . size ( ) ; + + i )
{
Game : : XZoneInfo info ;
info . name = ( & Maps : : CurrentDependencies [ i ] ) - > data ( ) ;
info . allocFlags = zoneInfo - > allocFlags ;
info . freeFlags = zoneInfo - > freeFlags ;
data . push_back ( info ) ;
}
// Load patch files
2022-12-11 12:54:24 -05:00
auto patchZone = std : : format ( " patch_{} " , zoneInfo - > name ) ;
2017-01-16 11:42:50 -05:00
if ( FastFiles : : Exists ( patchZone ) )
{
2017-04-07 15:37:01 -04:00
data . push_back ( { patchZone . data ( ) , zoneInfo - > allocFlags , zoneInfo - > freeFlags } ) ;
2017-01-16 11:42:50 -05:00
}
return FastFiles : : LoadLocalizeZones ( data . data ( ) , data . size ( ) , sync ) ;
}
void Maps : : OverrideMapEnts ( Game : : MapEnts * ents )
{
auto callback = [ ] ( Game : : XAssetHeader header , void * ents )
2023-03-17 08:46:29 -04:00
{
Game : : MapEnts * mapEnts = reinterpret_cast < Game : : MapEnts * > ( ents ) ;
Game : : clipMap_t * clipMap = header . clipMap ;
2017-04-07 15:37:01 -04:00
2023-03-17 08:46:29 -04:00
if ( clipMap & & mapEnts & & ! _stricmp ( mapEnts - > name , clipMap - > name ) )
{
clipMap - > mapEnts = mapEnts ;
//*Game::marMapEntsPtr = mapEnts;
//Game::G_SpawnEntitiesFromString();
}
} ;
2017-01-16 11:42:50 -05:00
// Internal doesn't lock the thread, as locking is impossible, due to executing this in the thread that holds the current lock
2018-05-09 06:04:20 -04:00
Game : : DB_EnumXAssets_Internal ( Game : : XAssetType : : ASSET_TYPE_CLIPMAP_MP , callback , ents , true ) ;
Game : : DB_EnumXAssets_Internal ( Game : : XAssetType : : ASSET_TYPE_CLIPMAP_SP , callback , ents , true ) ;
2017-01-16 11:42:50 -05:00
}
2018-12-17 08:29:18 -05:00
void Maps : : LoadAssetRestrict ( Game : : XAssetType type , Game : : XAssetHeader asset , const std : : string & name , bool * restrict )
2017-01-16 11:42:50 -05:00
{
if ( std : : find ( Maps : : CurrentDependencies . begin ( ) , Maps : : CurrentDependencies . end ( ) , FastFiles : : Current ( ) ) ! = Maps : : CurrentDependencies . end ( )
& & ( FastFiles : : Current ( ) ! = " mp_shipment_long " | | Maps : : CurrentMainZone ! = " mp_shipment " ) ) // Shipment is a special case
{
2022-01-30 09:03:13 -05:00
switch ( type )
2017-01-16 11:42:50 -05:00
{
2022-01-30 09:03:13 -05:00
case Game : : XAssetType : : ASSET_TYPE_CLIPMAP_MP :
case Game : : XAssetType : : ASSET_TYPE_CLIPMAP_SP :
case Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_SP :
case Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_MP :
case Game : : XAssetType : : ASSET_TYPE_GFXWORLD :
case Game : : XAssetType : : ASSET_TYPE_MAP_ENTS :
case Game : : XAssetType : : ASSET_TYPE_COMWORLD :
case Game : : XAssetType : : ASSET_TYPE_FXWORLD :
2017-01-16 11:42:50 -05:00
* restrict = true ;
return ;
}
}
if ( type = = Game : : XAssetType : : ASSET_TYPE_ADDON_MAP_ENTS )
{
* restrict = true ;
return ;
}
if ( type = = Game : : XAssetType : : ASSET_TYPE_WEAPON )
{
if ( ( ! strstr ( name . data ( ) , " _mp " ) & & name ! = " none " & & name ! = " destructible_car " ) | | Zones : : Version ( ) > = VERSION_ALPHA2 )
{
* restrict = true ;
return ;
}
}
if ( type = = Game : : XAssetType : : ASSET_TYPE_STRINGTABLE )
{
if ( FastFiles : : Current ( ) = = " mp_cross_fire " )
{
* restrict = true ;
return ;
}
}
if ( type = = Game : : XAssetType : : ASSET_TYPE_MAP_ENTS )
{
if ( Flags : : HasFlag ( " dump " ) )
{
2022-01-27 09:32:47 -05:00
Utils : : IO : : WriteFile ( Utils : : String : : VA ( " raw/%s.ents " , name . data ( ) ) , asset . mapEnts - > entityString , true ) ;
2017-01-16 11:42:50 -05:00
}
2022-02-08 11:41:49 -05:00
static std : : string mapEntities ;
2022-07-20 18:22:49 -04:00
FileSystem : : File ents ( name + " .ents " , Game : : FS_THREAD_DATABASE ) ;
2017-01-16 11:42:50 -05:00
if ( ents . exists ( ) )
{
2022-02-08 11:41:49 -05:00
mapEntities = ents . getBuffer ( ) ;
2022-01-31 07:53:57 -05:00
asset . mapEnts - > entityString = mapEntities . data ( ) ;
2017-01-16 11:42:50 -05:00
asset . mapEnts - > numEntityChars = mapEntities . size ( ) + 1 ;
}
}
2022-01-31 07:53:57 -05:00
2017-01-16 11:42:50 -05:00
// This is broken
if ( ( type = = Game : : XAssetType : : ASSET_TYPE_MENU | | type = = Game : : XAssetType : : ASSET_TYPE_MENULIST ) & & Zones : : Version ( ) > = 359 )
{
* restrict = true ;
return ;
}
}
Game : : G_GlassData * Maps : : GetWorldData ( )
{
2017-05-28 16:21:26 -04:00
Logger : : Print ( " Waiting for database... \n " ) ;
while ( ! Game : : Sys_IsDatabaseReady ( ) ) std : : this_thread : : sleep_for ( 10 ms ) ;
2019-10-02 02:08:11 -04:00
if ( ! Game : : DB_XAssetPool [ Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_MP ] . gameWorldMp | | ! Game : : DB_XAssetPool [ Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_MP ] . gameWorldMp - > name | |
! Game : : DB_XAssetPool [ Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_MP ] . gameWorldMp - > g_glassData | | Maps : : SPMap )
2017-01-16 11:42:50 -05:00
{
2019-10-02 02:08:11 -04:00
return Game : : DB_XAssetPool [ Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_SP ] . gameWorldSp - > g_glassData ;
2017-01-16 11:42:50 -05:00
}
2019-10-02 02:08:11 -04:00
return Game : : DB_XAssetPool [ Game : : XAssetType : : ASSET_TYPE_GAMEWORLD_MP ] . gameWorldMp - > g_glassData ;
2017-01-16 11:42:50 -05:00
}
__declspec ( naked ) void Maps : : GetWorldDataStub ( )
{
__asm
{
push eax
pushad
call Maps : : GetWorldData
mov [ esp + 20 h ] , eax
popad
pop eax
retn
}
}
void Maps : : LoadRawSun ( )
{
Game : : R_FlushSun ( ) ;
Game : : GfxWorld * world = * reinterpret_cast < Game : : GfxWorld * * > ( 0x66DEE94 ) ;
2022-12-11 12:54:24 -05:00
if ( FileSystem : : File ( std : : format ( " sun/{}.sun " , Maps : : CurrentMainZone ) ) )
2017-01-16 11:42:50 -05:00
{
Game : : R_LoadSunThroughDvars ( Maps : : CurrentMainZone . data ( ) , & world - > sun ) ;
}
}
void Maps : : GetBSPName ( char * buffer , size_t size , const char * format , const char * mapname )
{
if ( ! Utils : : String : : StartsWith ( mapname , " mp_ " ) & & ! Utils : : String : : StartsWith ( mapname , " zm_ " ) )
{
format = " maps/%s.d3dbsp " ;
}
// Redirect shipment to shipment long
if ( mapname = = " mp_shipment " s )
{
mapname = " mp_shipment_long " ;
}
2022-01-30 09:03:13 -05:00
_snprintf_s ( buffer , size , _TRUNCATE , format , mapname ) ;
2017-01-16 11:42:50 -05:00
}
void Maps : : HandleAsSPMap ( )
{
2017-04-06 16:22:30 -04:00
Maps : : SPMap = true ;
2017-01-16 11:42:50 -05:00
}
int Maps : : IgnoreEntityStub ( const char * entity )
{
return ( Utils : : String : : StartsWith ( entity , " dyn_ " ) | | Utils : : String : : StartsWith ( entity , " node_ " ) | | Utils : : String : : StartsWith ( entity , " actor_ " ) ) ;
}
2023-03-17 08:46:29 -04:00
std : : unordered_map < std : : string , std : : string > Maps : : ParseCustomMapArena ( const std : : string & singleMapArena )
{
static const std : : regex regex ( " ( \\ w*) * \" ?((?: \\ w| )*) \" ? " ) ;
std : : unordered_map < std : : string , std : : string > arena ;
std : : smatch m ;
std : : string : : const_iterator search_start ( singleMapArena . cbegin ( ) ) ;
while ( std : : regex_search ( search_start , singleMapArena . cend ( ) , m , regex ) )
{
if ( m . size ( ) > 2 )
{
arena . emplace ( m [ 1 ] . str ( ) , m [ 2 ] . str ( ) ) ;
search_start = m . suffix ( ) . first ;
}
}
return arena ;
}
2022-07-20 15:47:54 -04:00
Maps : : MapDependencies Maps : : GetDependenciesForMap ( const std : : string & map )
2017-01-16 11:42:50 -05:00
{
2022-07-20 15:47:54 -04:00
std : : string teamAxis = " opforce_composite " ;
std : : string teamAllies = " us_army " ;
2017-01-16 11:42:50 -05:00
2023-03-17 08:46:29 -04:00
Maps : : MapDependencies dependencies ;
2017-01-16 11:42:50 -05:00
2022-07-27 04:54:36 -04:00
// True by default - cause some maps won't have an arenafile entry
dependencies . requiresTeamZones = true ;
2017-01-16 11:42:50 -05:00
for ( int i = 0 ; i < * Game : : arenaCount ; + + i )
{
Game : : newMapArena_t * arena = & ArenaLength : : NewArenas [ i ] ;
if ( arena - > mapName = = map )
{
2022-07-27 04:54:36 -04:00
// If it's in the arena file, surely it's a vanilla map that doesn't need teams...
dependencies . requiresTeamZones = false ;
2022-07-20 15:47:54 -04:00
for ( std : : size_t j = 0 ; j < std : : extent_v < decltype ( Game : : newMapArena_t : : keys ) > ; + + j )
2017-01-16 11:42:50 -05:00
{
2022-07-20 15:47:54 -04:00
const auto * key = arena - > keys [ j ] ;
const auto * value = arena - > values [ j ] ;
if ( key = = " dependency " s )
2017-01-16 11:42:50 -05:00
{
2022-07-20 15:47:54 -04:00
dependencies . requiredMaps = Utils : : String : : Split ( arena - > values [ j ] , ' ' ) ;
2017-01-16 11:42:50 -05:00
}
2022-07-20 15:47:54 -04:00
else if ( key = = " allieschar " s )
2017-01-16 11:42:50 -05:00
{
2022-07-20 15:47:54 -04:00
teamAllies = value ;
}
else if ( key = = " axischar " s )
{
teamAxis = value ;
}
else if ( key = = " useteamzones " s )
{
2022-07-27 04:54:36 -04:00
// ... unless it specifies so! This allows loading of CODO/COD4 zones that might not have the correct teams
2022-07-20 15:47:54 -04:00
dependencies . requiresTeamZones = Utils : : String : : ToLower ( value ) = = " true " s ;
2017-01-16 11:42:50 -05:00
}
}
break ;
}
}
2022-07-20 15:47:54 -04:00
dependencies . requiredTeams = std : : make_pair ( teamAllies , teamAxis ) ;
return dependencies ;
2017-01-16 11:42:50 -05:00
}
2017-04-06 16:22:30 -04:00
void Maps : : PrepareUsermap ( const char * mapname )
{
2017-04-07 15:37:01 -04:00
if ( Maps : : UserMap . isValid ( ) )
2017-04-06 16:22:30 -04:00
{
Maps : : UserMap . freeIwd ( ) ;
Maps : : UserMap . clear ( ) ;
}
2017-07-12 13:36:18 -04:00
if ( Maps : : IsUserMap ( mapname ) )
2017-04-06 16:22:30 -04:00
{
Maps : : UserMap = Maps : : UserMapContainer ( mapname ) ;
Maps : : UserMap . loadIwd ( ) ;
}
else
{
Maps : : UserMap . clear ( ) ;
}
}
2018-12-17 08:29:18 -05:00
unsigned int Maps : : GetUsermapHash ( const std : : string & map )
2017-04-06 16:22:30 -04:00
{
2022-12-11 12:54:24 -05:00
if ( Utils : : IO : : DirectoryExists ( std : : format ( " usermaps/{} " , map ) ) )
2017-04-06 16:22:30 -04:00
{
2017-04-08 09:32:51 -04:00
std : : string hash ;
2017-04-06 16:22:30 -04:00
2022-11-21 19:34:17 -05:00
for ( std : : size_t i = 0 ; i < ARRAYSIZE ( Maps : : UserMapFiles ) ; + + i )
2017-04-06 16:22:30 -04:00
{
2022-12-11 12:54:24 -05:00
auto filePath = std : : format ( " usermaps/{}/{}{} " , map , map , Maps : : UserMapFiles [ i ] ) ;
2017-04-08 09:32:51 -04:00
if ( Utils : : IO : : FileExists ( filePath ) )
{
hash . append ( Utils : : Cryptography : : SHA256 : : Compute ( Utils : : IO : : ReadFile ( filePath ) ) ) ;
}
2017-04-06 16:22:30 -04:00
}
2017-04-08 09:32:51 -04:00
return Utils : : Cryptography : : JenkinsOneAtATime : : Compute ( hash ) ;
2017-04-06 16:22:30 -04:00
}
return 0 ;
}
2017-05-30 15:49:13 -04:00
void Maps : : LoadNewMapCommand ( char * buffer , size_t size , const char * /*format*/ , const char * mapname , const char * gametype )
{
unsigned int hash = Maps : : GetUsermapHash ( mapname ) ;
2022-01-30 09:03:13 -05:00
_snprintf_s ( buffer , size , _TRUNCATE , " loadingnewmap \n %s \n %s \n %d " , mapname , gametype , hash ) ;
2017-05-30 15:49:13 -04:00
}
int Maps : : TriggerReconnectForMap ( Game : : msg_t * msg , const char * mapname )
2017-04-07 15:37:01 -04:00
{
Theatre : : StopRecording ( ) ;
2017-05-30 15:49:13 -04:00
char hashBuf [ 100 ] = { 0 } ;
unsigned int hash = atoi ( Game : : MSG_ReadStringLine ( msg , hashBuf , sizeof hashBuf ) ) ;
if ( ! Maps : : CheckMapInstalled ( mapname , false , true ) | | hash & & hash ! = Maps : : GetUsermapHash ( mapname ) )
2017-04-07 15:37:01 -04:00
{
// Reconnecting forces the client to download the new map
Command : : Execute ( " disconnect " , false ) ;
Command : : Execute ( " awaitDatabase " , false ) ; // Wait for the database to load
Command : : Execute ( " wait 100 " , false ) ;
Command : : Execute ( " openmenu popup_reconnectingtoparty " , false ) ;
2017-05-25 14:18:20 -04:00
Command : : Execute ( " delayReconnect " , false ) ;
2017-04-07 15:37:01 -04:00
return true ;
}
return false ;
}
__declspec ( naked ) void Maps : : RotateCheckStub ( )
{
__asm
{
push eax
pushad
push [ esp + 28 h ]
2017-05-30 15:49:13 -04:00
push ebp
2017-04-07 15:37:01 -04:00
call Maps : : TriggerReconnectForMap
2017-05-30 15:49:13 -04:00
add esp , 8 h
2017-04-07 15:37:01 -04:00
2017-06-14 06:06:04 -04:00
mov [ esp + 20 h ] , eax
2017-04-07 15:37:01 -04:00
popad
pop eax
test eax , eax
jnz skipRotation
push 487 C50h // Rotate map
skipRotation :
retn
}
}
2017-04-06 16:22:30 -04:00
__declspec ( naked ) void Maps : : SpawnServerStub ( )
{
__asm
{
pushad
push [ esp + 24 h ]
call Maps : : PrepareUsermap
pop eax
popad
push 4 A7120h // SV_SpawnServer
retn
}
}
__declspec ( naked ) void Maps : : LoadMapLoadscreenStub ( )
{
__asm
{
pushad
2017-04-07 15:37:01 -04:00
push [ esp + 24 h ]
2017-04-06 16:22:30 -04:00
call Maps : : PrepareUsermap
pop eax
popad
push 4 D8030h // LoadMapLoadscreen
retn
}
}
2017-01-16 11:42:50 -05:00
void Maps : : AddDlc ( Maps : : DLC dlc )
{
for ( auto & pack : Maps : : DlcPacks )
{
if ( pack . index = = dlc . index )
{
pack . maps = dlc . maps ;
Maps : : UpdateDlcStatus ( ) ;
return ;
}
}
2022-06-18 13:42:56 -04:00
Dvar : : Register < bool > ( Utils : : String : : VA ( " isDlcInstalled_%d " , dlc . index ) , false , Game : : DVAR_EXTERNAL | Game : : DVAR_INIT , " " ) ;
2017-01-16 11:42:50 -05:00
Maps : : DlcPacks . p ush_back ( dlc ) ;
Maps : : UpdateDlcStatus ( ) ;
}
void Maps : : UpdateDlcStatus ( )
{
bool hasAllDlcs = true ;
2017-02-10 08:06:42 -05:00
std : : vector < bool > hasDlc ;
2017-01-16 11:42:50 -05:00
for ( auto & pack : Maps : : DlcPacks )
{
bool hasAllMaps = true ;
for ( auto map : pack . maps )
{
if ( ! FastFiles : : Exists ( map ) )
{
hasAllMaps = false ;
hasAllDlcs = false ;
break ;
}
}
2017-02-10 08:06:42 -05:00
hasDlc . push_back ( hasAllMaps ) ;
2021-09-10 06:40:53 -04:00
Dvar : : Var ( Utils : : String : : VA ( " isDlcInstalled_%d " , pack . index ) ) . set ( hasAllMaps ? true : false ) ;
2017-01-16 11:42:50 -05:00
}
2017-01-30 21:57:31 -05:00
// Must have all of dlc 3 to 5 or it causes issues
static bool sentMessage = false ;
2017-02-10 08:06:42 -05:00
if ( hasDlc . size ( ) > = 5 & & ( hasDlc [ 2 ] | | hasDlc [ 3 ] | | hasDlc [ 4 ] ) & & ( ! hasDlc [ 2 ] | | ! hasDlc [ 3 ] | | ! hasDlc [ 4 ] ) & & ! sentMessage )
2017-01-30 21:57:31 -05:00
{
StartupMessages : : AddMessage ( " Warning: \n You only have some of DLCs 3-5 which are all required to be installed to work. There may be issues with those maps. " ) ;
sentMessage = true ;
}
2021-09-10 06:40:53 -04:00
Dvar : : Var ( " isDlcInstalled_All " ) . set ( hasAllDlcs ? true : false ) ;
2017-01-16 11:42:50 -05:00
}
2017-05-01 08:57:17 -04:00
bool Maps : : IsCustomMap ( )
{
Game : : GfxWorld * & gameWorld = * reinterpret_cast < Game : : GfxWorld * * > ( 0x66DEE94 ) ;
2022-12-14 03:40:15 -05:00
if ( gameWorld ) return ( gameWorld - > checksum & 0xFFFF0000 ) = = 0xC0D40000 ;
2017-05-01 08:57:17 -04:00
return false ;
}
2018-12-17 08:29:18 -05:00
bool Maps : : IsUserMap ( const std : : string & mapname )
2017-07-12 13:36:18 -04:00
{
2022-12-11 12:54:24 -05:00
return Utils : : IO : : DirectoryExists ( std : : format ( " usermaps/{} " , mapname ) ) & & Utils : : IO : : FileExists ( std : : format ( " usermaps/{}/{}.ff " , mapname , mapname ) ) ;
2017-07-12 13:36:18 -04:00
}
2023-03-17 08:46:29 -04:00
void Maps : : ScanCustomMaps ( )
{
FoundCustomMaps . clear ( ) ;
2023-03-17 09:47:33 -04:00
Logger : : Print ( " Looking for custom maps... \n " ) ;
2023-03-17 08:46:29 -04:00
2023-03-17 09:24:00 -04:00
std : : filesystem : : path basePath = ( * Game : : fs_basepath ) - > current . string ;
basePath / = " usermaps " ;
2023-03-17 08:46:29 -04:00
2023-03-17 09:24:00 -04:00
if ( ! std : : filesystem : : exists ( basePath ) )
{
return ;
}
const auto entries = Utils : : IO : : ListFiles ( basePath ) ;
2023-03-17 08:46:29 -04:00
for ( const auto & entry : entries )
{
if ( entry . is_directory ( ) )
{
2023-03-17 18:40:52 -04:00
const auto zoneName = entry . path ( ) . filename ( ) . string ( ) ;
const auto mapPath = std : : format ( " {} \\ {}.ff " , entry . path ( ) . string ( ) , zoneName ) ;
2023-03-17 08:46:29 -04:00
if ( Utils : : IO : : FileExists ( mapPath ) )
{
FoundCustomMaps . push_back ( zoneName ) ;
Logger : : Print ( " Discovered custom map {} \n " , zoneName ) ;
}
}
}
}
std : : string Maps : : GetArenaPath ( const std : : string & mapName )
{
return std : : format ( " usermaps/{}/{}.arena " , mapName , mapName ) ;
}
const std : : vector < std : : string > & Maps : : GetCustomMaps ( )
{
return FoundCustomMaps ;
}
2017-05-01 07:08:34 -04:00
Game : : XAssetEntry * Maps : : GetAssetEntryPool ( )
{
2017-06-21 13:56:20 -04:00
return * reinterpret_cast < Game : : XAssetEntry * * > ( 0x48E6F4 ) ;
2017-01-16 11:42:50 -05:00
}
2017-04-07 15:37:01 -04:00
// dlcIsTrue serves as a check if the map is a custom map and if it's missing
2022-12-31 09:03:33 -05:00
bool Maps : : CheckMapInstalled ( const std : : string & mapname , bool error , bool dlcIsTrue )
2017-01-29 12:27:11 -05:00
{
if ( FastFiles : : Exists ( mapname ) ) return true ;
for ( auto & pack : Maps : : DlcPacks )
{
for ( auto map : pack . maps )
{
2022-12-31 09:03:33 -05:00
if ( map = = mapname )
2017-01-29 12:27:11 -05:00
{
2017-04-07 15:37:01 -04:00
if ( error )
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_DISCONNECT , " Missing DLC pack {} ({}) containing map {} ({}). \n Please download it to play this map. " ,
2023-03-22 14:47:13 -04:00
pack . name , pack . index , Localization : : LocalizeMapName ( mapname . data ( ) ) , mapname ) ;
2017-04-07 15:37:01 -04:00
}
return dlcIsTrue ;
2017-01-29 12:27:11 -05:00
}
}
}
2022-06-12 17:07:53 -04:00
if ( error )
{
Logger : : Error ( Game : : ERR_DISCONNECT ,
" Missing map file {}. \n You may have a damaged installation or are attempting to load a non-existent map. " , mapname ) ;
}
2017-01-29 12:27:11 -05:00
return false ;
}
2017-04-24 15:13:54 -04:00
void Maps : : HideModel ( )
{
Game : : GfxWorld * & gameWorld = * reinterpret_cast < Game : : GfxWorld * * > ( 0x66DEE94 ) ;
if ( ! gameWorld ) return ;
std : : string model = Dvar : : Var ( " r_hideModel " ) . get < std : : string > ( ) ;
if ( model . empty ( ) ) return ;
for ( unsigned int i = 0 ; i < gameWorld - > dpvs . smodelCount ; + + i )
{
2022-12-14 03:40:15 -05:00
if ( model = = " all " s | | gameWorld - > dpvs . smodelDrawInsts [ i ] . model - > name = = model )
2017-04-24 15:13:54 -04:00
{
gameWorld - > dpvs . smodelVisData [ 0 ] [ i ] = 0 ;
gameWorld - > dpvs . smodelVisData [ 1 ] [ i ] = 0 ;
gameWorld - > dpvs . smodelVisData [ 2 ] [ i ] = 0 ;
}
}
}
__declspec ( naked ) void Maps : : HideModelStub ( )
{
__asm
{
pushad
call Maps : : HideModel
popad
push 541E40 h
retn
}
}
2019-10-02 02:08:11 -04:00
void Maps : : G_SpawnTurretHook ( Game : : gentity_s * ent , int unk , int unk2 )
{
if ( Maps : : CurrentMainZone = = " mp_backlot_sh " s | | Maps : : CurrentMainZone = = " mp_con_spring " s | |
Maps : : CurrentMainZone = = " mp_mogadishu_sh " s | | Maps : : CurrentMainZone = = " mp_nightshift_sh " s )
{
return ;
}
Utils : : Hook : : Call < void ( Game : : gentity_s * , int , int ) > ( 0x408910 ) ( ent , unk , unk2 ) ;
}
2021-04-02 10:13:51 -04:00
bool Maps : : SV_SetTriggerModelHook ( Game : : gentity_s * ent ) {
2021-04-05 12:13:00 -04:00
// Use me for debugging
//std::string classname = Game::SL_ConvertToString(ent->script_classname);
//std::string targetname = Game::SL_ConvertToString(ent->targetname);
2021-04-02 10:13:51 -04:00
return Utils : : Hook : : Call < bool ( Game : : gentity_s * ) > ( 0x5050C0 ) ( ent ) ;
}
2023-02-17 06:26:40 -05:00
unsigned short Maps : : CM_TriggerModelBounds_Hk ( unsigned int triggerIndex , Game : : Bounds * bounds )
{
auto * ents = * reinterpret_cast < Game : : MapEnts * * > ( 0x1AA651C ) ; // Use me for debugging
if ( ents )
{
if ( triggerIndex > = ents - > trigger . count )
{
Logger : : Error ( Game : : errorParm_t : : ERR_DROP , " Invalid trigger index ({}) in entities exceeds the maximum trigger count ({}) defined in the clipmap. Check your map ents, or your clipmap! " , triggerIndex , ents - > trigger . count ) ;
return 0 ;
}
else
{
return Utils : : Hook : : Call < unsigned short ( int , Game : : Bounds * ) > ( 0x4416C0 ) ( triggerIndex , bounds ) ;
}
}
return 0 ;
2021-04-02 10:13:51 -04:00
}
2019-10-02 02:08:11 -04:00
2017-01-16 11:42:50 -05:00
Maps : : Maps ( )
{
2022-05-27 06:19:28 -04:00
Scheduler : : Once ( [ ]
2017-04-07 16:07:44 -04:00
{
2022-06-18 13:42:56 -04:00
Dvar : : Register < bool > ( " isDlcInstalled_All " , false , Game : : DVAR_EXTERNAL | Game : : DVAR_INIT , " " ) ;
2022-06-16 10:15:26 -04:00
Maps : : RListSModels = Dvar : : Register < bool > ( " r_listSModels " , false , Game : : DVAR_NONE , " Display a list of visible SModels " ) ;
2017-04-07 16:07:44 -04:00
Maps : : AddDlc ( { 1 , " Stimulus Pack " , { " mp_complex " , " mp_compact " , " mp_storm " , " mp_overgrown " , " mp_crash " } } ) ;
2017-04-24 15:13:54 -04:00
Maps : : AddDlc ( { 2 , " Resurgence Pack " , { " mp_abandon " , " mp_vacant " , " mp_trailerpark " , " mp_strike " , " mp_fuel2 " } } ) ;
2023-03-17 20:29:46 -04:00
Maps : : AddDlc ( { 3 , " IW4x Classics " , { " mp_nuked " , " mp_cross_fire " , " mp_cargoship " , " mp_bloc " , " mp_killhouse " , " mp_bog_sh " , " mp_cargoship_sh " , " mp_shipment " , " mp_shipment_long " , " mp_rust_long " , " mp_firingrange " , " mp_bloc_sh " , " mp_crash_tropical " , " mp_estate_tropical " , " mp_fav_tropical " , " mp_storm_spring " } } ) ;
2023-03-17 08:46:29 -04:00
Maps : : AddDlc ( { 4 , " Call Of Duty 4 Pack " , { " mp_farm " , " mp_backlot " , " mp_pipeline " , " mp_countdown " , " mp_crash_snow " , " mp_carentan " , " mp_broadcast " , " mp_showdown " , " mp_convoy " } } ) ;
Maps : : AddDlc ( { 5 , " Modern Warfare 3 Pack " , { " mp_dome " , " mp_hardhat " , " mp_paris " , " mp_seatown " , " mp_bravo " , " mp_underground " , " mp_plaza2 " , " mp_village " , " mp_alpha " } } ) ;
2017-04-07 16:07:44 -04:00
Maps : : UpdateDlcStatus ( ) ;
2022-08-24 10:38:14 -04:00
UIScript : : Add ( " downloadDLC " , [ ] ( [[maybe_unused]] const UIScript::Token& token, [[maybe_unused]] const Game : : uiInfo_s * info )
2017-04-07 16:07:44 -04:00
{
int dlc = token . get < int > ( ) ;
2017-01-16 11:42:50 -05:00
2022-02-01 07:15:59 -05:00
for ( const auto & pack : Maps : : DlcPacks )
2017-04-07 16:07:44 -04:00
{
if ( pack . index = = dlc )
{
2022-02-01 07:15:59 -05:00
ShellExecuteW ( 0 , 0 , L " https://xlabs.dev/support_iw4x_client.html " , 0 , 0 , SW_SHOW ) ;
2017-04-07 16:07:44 -04:00
return ;
}
}
Game : : ShowMessageBox ( Utils : : String : : VA ( " DLC %d does not exist! " , dlc ) , " ERROR " ) ;
2017-01-16 11:42:50 -05:00
} ) ;
2022-05-27 06:19:28 -04:00
} , Scheduler : : Pipeline : : MAIN ) ;
2017-01-16 11:42:50 -05:00
2019-10-02 02:08:11 -04:00
// disable turrets on CoD:OL 448+ maps for now
Utils : : Hook ( 0x5EE577 , Maps : : G_SpawnTurretHook , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x44A4D5 , Maps : : G_SpawnTurretHook , HOOK_CALL ) . install ( ) - > quick ( ) ;
2021-04-02 10:13:51 -04:00
2023-02-17 06:26:40 -05:00
// Catch trigger errors before they're critical
Utils : : Hook ( 0x5050D4 , Maps : : CM_TriggerModelBounds_Hk , HOOK_CALL ) . install ( ) - > quick ( ) ;
2021-04-05 12:13:00 -04:00
# ifdef DEBUG
2021-04-02 10:13:51 -04:00
// Check trigger models
Utils : : Hook ( 0x5FC0F1 , Maps : : SV_SetTriggerModelHook , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x5FC2671 , Maps : : SV_SetTriggerModelHook , HOOK_CALL ) . install ( ) - > quick ( ) ;
2021-04-05 12:13:00 -04:00
# endif
2021-04-02 10:13:51 -04:00
//
2019-10-02 02:08:11 -04:00
2017-04-24 15:13:54 -04:00
//#define SORT_SMODELS
# if !defined(DEBUG) || !defined(SORT_SMODELS)
2017-04-16 08:29:35 -04:00
// Don't sort static models
Utils : : Hook : : Nop ( 0x53D815 , 2 ) ;
# endif
2017-04-20 15:27:38 -04:00
// Don't draw smodels
//Utils::Hook::Set<BYTE>(0x541E40, 0xC3);
2017-01-16 11:42:50 -05:00
// Restrict asset loading
AssetHandler : : OnLoad ( Maps : : LoadAssetRestrict ) ;
// hunk size (was 300 MiB)
Utils : : Hook : : Set < DWORD > ( 0x64A029 , 0x1C200000 ) ; // 450 MiB
Utils : : Hook : : Set < DWORD > ( 0x64A057 , 0x1C200000 ) ;
# if DEBUG
// Hunk debugging
Utils : : Hook : : Set < BYTE > ( 0x4FF57B , 0xCC ) ;
Utils : : Hook : : Nop ( 0x4FF57C , 4 ) ;
2017-05-27 11:26:12 -04:00
# else
// Temporarily disable distortion warnings
Utils : : Hook : : Nop ( 0x50DBFF , 5 ) ;
Utils : : Hook : : Nop ( 0x50DC4F , 5 ) ;
Utils : : Hook : : Nop ( 0x50DCA3 , 5 ) ;
Utils : : Hook : : Nop ( 0x50DCFE , 5 ) ;
2017-01-16 11:42:50 -05:00
# endif
// Intercept BSP name resolving
Utils : : Hook ( 0x4C5979 , Maps : : GetBSPName , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-04-06 16:22:30 -04:00
// Intercept map zone loading/unloading
2017-01-16 11:42:50 -05:00
Utils : : Hook ( 0x42C2AF , Maps : : LoadMapZones , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-04-06 16:22:30 -04:00
Utils : : Hook ( 0x60B477 , Maps : : UnloadMapZones , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-01-16 11:42:50 -05:00
// Ignore SP entities
Utils : : Hook ( 0x444810 , Maps : : IgnoreEntityStub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
// WorldData pointer replacement
Utils : : Hook ( 0x4D90B6 , Maps : : GetWorldDataStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
// Allow loading raw suns
Utils : : Hook ( 0x51B46A , Maps : : LoadRawSun , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-04-06 16:22:30 -04:00
// Intercept map loading for usermap initialization
Utils : : Hook ( 0x6245E3 , Maps : : SpawnServerStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x62493E , Maps : : SpawnServerStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x42CF58 , Maps : : LoadMapLoadscreenStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x487CDD , Maps : : LoadMapLoadscreenStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x4CA3E9 , Maps : : LoadMapLoadscreenStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x5A9D51 , Maps : : LoadMapLoadscreenStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x5B34DD , Maps : : LoadMapLoadscreenStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
2022-06-16 10:15:26 -04:00
Command : : Add ( " delayReconnect " , [ ] ( [[maybe_unused]] Command : : Params * params )
2017-05-25 14:18:20 -04:00
{
2022-05-05 10:03:14 -04:00
Scheduler : : Once ( [ ]
2017-05-25 14:18:20 -04:00
{
Command : : Execute ( " closemenu popup_reconnectingtoparty " , false ) ;
Command : : Execute ( " reconnect " , false ) ;
2022-05-05 10:03:14 -04:00
} , Scheduler : : Pipeline : : CLIENT , 10 s ) ;
2017-05-25 14:18:20 -04:00
} ) ;
2023-02-18 07:18:41 -05:00
if ( Dedicated : : IsEnabled ( ) )
2017-05-30 15:49:13 -04:00
{
Utils : : Hook ( 0x4A7251 , Maps : : LoadNewMapCommand , HOOK_CALL ) . install ( ) - > quick ( ) ;
}
2017-04-07 15:37:01 -04:00
// Download the map before a maprotation if necessary
// Conflicts with Theater's SV map rotation check, but this one is safer!
Utils : : Hook ( 0x5AA91C , Maps : : RotateCheckStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-04-08 09:32:51 -04:00
// Load usermap arena file
Utils : : Hook ( 0x630A88 , Maps : : LoadArenaFileStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-04-24 15:13:54 -04:00
// Allow hiding specific smodels
Utils : : Hook ( 0x50E67C , Maps : : HideModelStub , HOOK_CALL ) . install ( ) - > quick ( ) ;
2022-06-16 10:15:26 -04:00
if ( Dedicated : : IsEnabled ( ) | | ZoneBuilder : : IsEnabled ( ) )
{
return ;
}
// Client only
2022-05-05 10:03:14 -04:00
Scheduler : : Loop ( [ ]
2017-04-24 15:13:54 -04:00
{
2022-05-05 10:03:14 -04:00
auto * & gameWorld = * reinterpret_cast < Game : : GfxWorld * * > ( 0x66DEE94 ) ;
2022-06-16 10:15:26 -04:00
if ( ! Game : : CL_IsCgameInitialized ( ) | | ! gameWorld | | ! Maps : : RListSModels . get < bool > ( ) ) return ;
2017-04-24 15:13:54 -04:00
std : : map < std : : string , int > models ;
for ( unsigned int i = 0 ; i < gameWorld - > dpvs . smodelCount ; + + i )
{
if ( gameWorld - > dpvs . smodelVisData [ 0 ] [ i ] )
{
std : : string name = gameWorld - > dpvs . smodelDrawInsts [ i ] . model - > name ;
2022-05-31 12:38:09 -04:00
if ( ! models . contains ( name ) ) models [ name ] = 1 ;
2017-04-24 15:13:54 -04:00
else models [ name ] + + ;
}
}
2018-05-09 08:33:52 -04:00
Game : : Font_s * font = Game : : R_RegisterFont ( " fonts/smallFont " , 0 ) ;
2022-05-05 10:03:14 -04:00
auto height = Game : : R_TextHeight ( font ) ;
auto scale = 0.75f ;
float color [ 4 ] = { 0.0f , 1.0f , 0.0f , 1.0f } ;
2017-04-24 15:13:54 -04:00
unsigned int i = 0 ;
for ( auto & model : models )
{
2022-12-11 12:54:24 -05:00
Game : : R_AddCmdDrawText ( Utils : : String : : VA ( " %d %s " , model . second , model . first . data ( ) ) , std : : numeric_limits < int > : : max ( ) , font , 15.0f , ( height * scale + 1 ) * ( i + + + 1 ) + 15.0f , scale , scale , 0.0f , color , Game : : ITEM_TEXTSTYLE_NORMAL ) ;
2017-04-24 15:13:54 -04:00
}
2022-05-05 10:03:14 -04:00
} , Scheduler : : Pipeline : : RENDERER ) ;
2017-01-16 11:42:50 -05:00
}
Maps : : ~ Maps ( )
{
Maps : : DlcPacks . clear ( ) ;
Maps : : DependencyList . clear ( ) ;
Maps : : CurrentMainZone . clear ( ) ;
Maps : : CurrentDependencies . clear ( ) ;
}
}