#include "StdAfxSA.h" #include "GeneralSA.h" #include "PedSA.h" #include "ModelInfoSA.h" #include "PoolsSA.h" // Wrappers static void* EntityRender = AddressByVersion(0x534310, 0x5347B0, 0x545B30); WRAPPER void CEntity::Render() { VARJMP(EntityRender); } static void* varEntityIsVisible = AddressByVersion( 0x536BC0, 0, 0 ); // TODO WRAPPER bool CEntity::IsVisible() { VARJMP(varEntityIsVisible); } static void* varInvertRaster = AddressByVersion(0x705660, 0x705E90, 0x7497A0); WRAPPER void CShadowCamera::InvertRaster() { VARJMP(varInvertRaster); } CWeaponInfo* (*CWeaponInfo::GetWeaponInfo)(eWeaponType, signed char) = AddressByVersion(0x743C60, 0x744490, 0x77D940); static RwTexture*& ms_pRemapTexture = **AddressByVersion(0x59F1BD, 0x6D6E53, 0x5B811D); auto SetEditableMaterialsCB = AddressByVersion(0x4C83E0, 0x4C8460, 0x4D2CE0); static void ResetEditableMaterials(std::pair* pData) { for ( auto* i = pData; i->first != nullptr; i++ ) *static_cast(i->first) = i->second; } RpAtomic* ShadowCameraRenderCB(RpAtomic* pAtomic) { if ( RpAtomicGetFlags(pAtomic) & rpATOMICRENDER ) { RpGeometry* pGeometry = RpAtomicGetGeometry(pAtomic); if ( pGeometry->repEntry != nullptr ) // Only switch to optimized flags if already instanced so as not to break the instanced model { RwUInt32 geometryFlags = RpGeometryGetFlags(pGeometry); pGeometry->flags &= ~(rpGEOMETRYTEXTURED|rpGEOMETRYPRELIT|rpGEOMETRYNORMALS|rpGEOMETRYLIGHT|rpGEOMETRYMODULATEMATERIALCOLOR|rpGEOMETRYTEXTURED2); pAtomic = AtomicDefaultRenderCallBack(pAtomic); RpGeometrySetFlags(pGeometry, geometryFlags); } else { pAtomic = AtomicDefaultRenderCallBack(pAtomic); } } return pAtomic; } void CObject::Render() { if ( m_bDoNotRender || !m_pRwObject ) return; bool bCallRestore; std::pair materialRestoreData[16]; const int32_t carPartModelIndex = m_wCarPartModelIndex.Get(); if ( carPartModelIndex != -1 && m_objectCreatedBy == TEMP_OBJECT && bObjectFlag7 && RwObjectGetType(m_pRwObject) == rpATOMIC ) { auto* pData = materialRestoreData; ms_pRemapTexture = m_pPaintjobTex; static_cast(ms_modelInfoPtrs[ carPartModelIndex ])->SetVehicleColour( m_nCarColor[0].Get(), m_nCarColor[1].Get(), m_nCarColor[2].Get(), m_nCarColor[3].Get() ); SetEditableMaterialsCB(reinterpret_cast(m_pRwObject), &pData); pData->first = nullptr; // Disable backface culling for the part #ifdef _DEBUG RwCullMode oldCullMode; RwRenderStateGet(rwRENDERSTATECULLMODE, &oldCullMode); assert(oldCullMode == rwCULLMODECULLBACK); #endif RwRenderStateSet(rwRENDERSTATECULLMODE, reinterpret_cast(rwCULLMODECULLNONE)); bCallRestore = true; } else bCallRestore = false; CEntity::Render(); if ( bCallRestore ) { ResetEditableMaterials(materialRestoreData); RwRenderStateSet(rwRENDERSTATECULLMODE, reinterpret_cast(rwCULLMODECULLBACK)); } } extern void (*WorldRemove)(CEntity*); void CObject::TryToFreeUpTempObjects_SilentPatch( int numObjects ) { int numProcessed, numFreed; std::tie( numProcessed, numFreed ) = TryOrFreeUpTempObjects( numObjects, false ); if ( numProcessed >= numObjects && numObjects > numFreed ) { TryOrFreeUpTempObjects( numObjects - numFreed, true ); } } std::tuple CObject::TryOrFreeUpTempObjects( int numObjects, bool force ) { int numProcessed = 0, numFreed = 0; auto& pool = CPools::GetObjectPool(); for ( auto& obj : pool ) { if ( numFreed >= numObjects ) break; CObject* const objPtr = &obj; if ( pool.IsValidPtr( objPtr ) ) { if ( objPtr->m_objectCreatedBy == TEMP_OBJECT ) { numProcessed++; if ( force || !objPtr->IsVisible() ) { numFreed++; WorldRemove( objPtr ); delete objPtr; } } } } return std::make_tuple( numProcessed, numFreed ); } RwCamera* CShadowCamera::Update(CEntity* pEntity) { RwRGBA ClearColour = { 255, 255, 255, 0 }; RwCameraClear(m_pCamera, &ClearColour, rwCAMERACLEARIMAGE|rwCAMERACLEARZ); if ( RwCameraBeginUpdate(m_pCamera ) ) { if ( pEntity ) { if ( pEntity->nType == 3 ) static_cast(pEntity)->RenderForShadow(); else RpClumpForAllAtomics(reinterpret_cast(pEntity->m_pRwObject), ShadowCameraRenderCB); } InvertRaster(); RwCameraEndUpdate(m_pCamera); } return m_pCamera; } std::vector CEscalator::ms_entitiesToRemove; void CEscalator::SwitchOffNoRemove() { if ( !m_bExists ) return; for ( ptrdiff_t i = 0, j = field_7C + field_80 + field_84; i < j; ++i ) { if ( m_pSteps[i] != nullptr ) { ms_entitiesToRemove.push_back( m_pSteps[i] ); m_pSteps[i] = nullptr; } } m_bExists = false; }