mirror of
https://github.com/CookiePLMonster/SilentPatch.git
synced 2025-01-04 02:03:02 +05:00
879 lines
24 KiB
C++
879 lines
24 KiB
C++
#include "StdAfxSA.h"
|
|
|
|
#include <functional>
|
|
#include <algorithm>
|
|
#include <map>
|
|
#include "VehicleSA.h"
|
|
#include "TimerSA.h"
|
|
#include "PedSA.h"
|
|
#include "Utils/DelimStringReader.h"
|
|
#include "PlayerInfoSA.h"
|
|
|
|
static constexpr float PHOENIX_FLUTTER_PERIOD = 70.0f;
|
|
static constexpr float PHOENIX_FLUTTER_AMP = 0.13f;
|
|
static constexpr float SWEEPER_BRUSH_SPEED = 0.3f;
|
|
|
|
static constexpr float PI = 3.14159265358979323846f;
|
|
|
|
float CAutomobile::ms_engineCompSpeed;
|
|
|
|
|
|
namespace SVF {
|
|
enum class Feature
|
|
{
|
|
NO_FEATURE,
|
|
|
|
// Those are fully controlled by SilentPatch
|
|
PHOENIX_FLUTTER,
|
|
SWEEPER_BRUSHES,
|
|
NEWSVAN_DISH,
|
|
EXTRA_AILERONS1, // Like on Beagle
|
|
EXTRA_AILERONS2, // Like on Stuntplane
|
|
DOUBLE_TRAILER, // Like on artict3
|
|
|
|
// Those are partially controlled by SilentPatch (only affected by minor fixes)
|
|
VORTEX_EXHAUST,
|
|
TOWTRUCK_HOOK,
|
|
TRACTOR_HOOK,
|
|
RHINO_WHEELS,
|
|
FIRELA_LADDER,
|
|
|
|
// Internal SP use only, formerly "rotor exceptions"
|
|
// Unreachable from RegisterSpecialVehicleFeature
|
|
NO_ROTOR_FADE,
|
|
NO_LIGHTBEAM_BFC_FIX,
|
|
FORCE_DOUBLE_RWHEELS_OFF,
|
|
FORCE_DOUBLE_RWHEELS_ON,
|
|
};
|
|
|
|
Feature GetFeatureFromName( const char* featureName )
|
|
{
|
|
constexpr std::pair< const char*, Feature > features[] = {
|
|
{ "PHOENIX_FLUTTER", Feature::PHOENIX_FLUTTER },
|
|
{ "SWEEPER_BRUSHES", Feature::SWEEPER_BRUSHES },
|
|
{ "NEWSVAN_DISH", Feature::NEWSVAN_DISH },
|
|
{ "EXTRA_AILERONS1", Feature::EXTRA_AILERONS1 },
|
|
{ "EXTRA_AILERONS2", Feature::EXTRA_AILERONS2 },
|
|
{ "DOUBLE_TRAILER", Feature::DOUBLE_TRAILER },
|
|
{ "VORTEX_EXHAUST", Feature::VORTEX_EXHAUST },
|
|
{ "TOWTRUCK_HOOK", Feature::TOWTRUCK_HOOK },
|
|
{ "TRACTOR_HOOK", Feature::TRACTOR_HOOK },
|
|
{ "RHINO_WHEELS", Feature::RHINO_WHEELS },
|
|
{ "FIRELA_LADDER", Feature::FIRELA_LADDER },
|
|
};
|
|
|
|
auto it = std::find_if( std::begin(features), std::end(features), [featureName]( const auto& e ) {
|
|
return _stricmp( e.first, featureName ) == 0;
|
|
});
|
|
|
|
if ( it == std::end(features) ) return Feature::NO_FEATURE;
|
|
return it->second;
|
|
}
|
|
|
|
static int32_t nextFeatureCookie = 0;
|
|
int32_t _getCookie()
|
|
{
|
|
return nextFeatureCookie++;
|
|
}
|
|
|
|
static int32_t highestStockCookie = 0;
|
|
int32_t _getCookieStockID()
|
|
{
|
|
return highestStockCookie = _getCookie();
|
|
}
|
|
|
|
auto _registerFeatureInternal( int32_t modelID, Feature feature )
|
|
{
|
|
return std::make_pair( modelID, std::make_tuple( feature, _getCookieStockID() ) );
|
|
}
|
|
|
|
static std::multimap<int32_t, std::tuple<Feature, int32_t> > specialVehFeatures = {
|
|
_registerFeatureInternal( 432, Feature::RHINO_WHEELS ),
|
|
_registerFeatureInternal( 511, Feature::EXTRA_AILERONS1 ),
|
|
_registerFeatureInternal( 513, Feature::EXTRA_AILERONS2 ),
|
|
_registerFeatureInternal( 525, Feature::TOWTRUCK_HOOK ),
|
|
_registerFeatureInternal( 531, Feature::TRACTOR_HOOK ),
|
|
_registerFeatureInternal( 539, Feature::VORTEX_EXHAUST ),
|
|
_registerFeatureInternal( 544, Feature::FIRELA_LADDER ),
|
|
_registerFeatureInternal( 574, Feature::SWEEPER_BRUSHES ),
|
|
_registerFeatureInternal( 582, Feature::NEWSVAN_DISH ),
|
|
_registerFeatureInternal( 591, Feature::DOUBLE_TRAILER ),
|
|
_registerFeatureInternal( 603, Feature::PHOENIX_FLUTTER ),
|
|
};
|
|
|
|
int32_t RegisterFeature( int32_t modelID, Feature feature )
|
|
{
|
|
if ( feature == Feature::NO_FEATURE ) return -1;
|
|
|
|
const int32_t cookie = _getCookie();
|
|
specialVehFeatures.emplace( modelID, std::make_tuple(feature, cookie) );
|
|
return cookie;
|
|
}
|
|
|
|
void DeleteFeature( int32_t cookie )
|
|
{
|
|
for ( auto it = specialVehFeatures.begin(); it != specialVehFeatures.end(); ++it )
|
|
{
|
|
if ( std::get<int32_t>(it->second) == cookie )
|
|
{
|
|
specialVehFeatures.erase( it );
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void DisableStockVehiclesForFeature( Feature feature )
|
|
{
|
|
if ( feature == Feature::NO_FEATURE ) return;
|
|
for ( auto it = specialVehFeatures.begin(); it != specialVehFeatures.end(); )
|
|
{
|
|
if ( std::get<Feature>(it->second) == feature && std::get<int32_t>(it->second) <= highestStockCookie )
|
|
{
|
|
it = specialVehFeatures.erase( it );
|
|
}
|
|
else
|
|
{
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool ModelHasFeature( int32_t modelID, Feature feature )
|
|
{
|
|
auto results = specialVehFeatures.equal_range( modelID );
|
|
return std::find_if( results.first, results.second, [feature] ( const auto& e ) {
|
|
return std::get<Feature>(e.second) == feature;
|
|
} ) != results.second;
|
|
}
|
|
|
|
template<typename Pred>
|
|
Pred ForAllModelFeatures( int32_t modelID, Pred&& pred )
|
|
{
|
|
auto results = specialVehFeatures.equal_range( modelID );
|
|
for ( auto it = results.first; it != results.second; ++it )
|
|
{
|
|
std::forward<Pred>(pred)(std::get<Feature>(it->second));
|
|
}
|
|
return std::forward<Pred>(pred);
|
|
}
|
|
}
|
|
|
|
bool ReadDoubleRearWheels(const wchar_t* pPath)
|
|
{
|
|
bool listedAny = false;
|
|
|
|
constexpr size_t SCRATCH_PAD_SIZE = 32767;
|
|
WideDelimStringReader reader( SCRATCH_PAD_SIZE );
|
|
|
|
GetPrivateProfileSectionW( L"DoubleRearWheels", reader.GetBuffer(), reader.GetSize(), pPath );
|
|
while ( const wchar_t* str = reader.GetString() )
|
|
{
|
|
wchar_t textLine[64];
|
|
wchar_t* context = nullptr;
|
|
wchar_t* token;
|
|
|
|
wcscpy_s( textLine, str );
|
|
token = wcstok_s( textLine, L"=", &context );
|
|
|
|
int32_t toList = wcstol( token, nullptr, 0 );
|
|
if ( toList <= 0 ) continue;
|
|
|
|
wchar_t* begin = wcstok_s( nullptr, L"=", &context );
|
|
if ( begin == nullptr ) continue;
|
|
|
|
wchar_t* end = nullptr;
|
|
bool value = wcstoul( begin, &end, 0 ) != 0;
|
|
if ( begin != end )
|
|
{
|
|
SVF::RegisterFeature( toList, value ? SVF::Feature::FORCE_DOUBLE_RWHEELS_ON : SVF::Feature::FORCE_DOUBLE_RWHEELS_OFF );
|
|
listedAny = true;
|
|
}
|
|
}
|
|
return listedAny;
|
|
}
|
|
|
|
// 1.0 ONLY!
|
|
bool __stdcall CheckDoubleRWheelsList( void* modelInfo, uint8_t* handlingData )
|
|
{
|
|
static void* lastModelInfo = nullptr;
|
|
static bool lastResult = false;
|
|
|
|
if ( modelInfo == lastModelInfo ) return lastResult;
|
|
lastModelInfo = modelInfo;
|
|
|
|
const uint32_t numModelInfoPtrs = *(uint32_t*)0x4C5956+2;
|
|
int32_t modelID = std::distance( ms_modelInfoPtrs, std::find( ms_modelInfoPtrs, ms_modelInfoPtrs+numModelInfoPtrs, modelInfo ) );
|
|
|
|
bool foundFeature = false;
|
|
bool featureStatus = false;
|
|
SVF::ForAllModelFeatures( modelID, [&]( SVF::Feature f ) {
|
|
if ( foundFeature ) return;
|
|
|
|
if ( f == SVF::Feature::FORCE_DOUBLE_RWHEELS_OFF )
|
|
{
|
|
foundFeature = true;
|
|
featureStatus = false;
|
|
}
|
|
else if ( f == SVF::Feature::FORCE_DOUBLE_RWHEELS_ON )
|
|
{
|
|
foundFeature = true;
|
|
featureStatus = true;
|
|
}
|
|
} );
|
|
if ( !foundFeature )
|
|
{
|
|
uint32_t flags = *(uint32_t*)(handlingData+0xCC);
|
|
lastResult = (flags & 0x20000000) != 0;
|
|
return lastResult;
|
|
}
|
|
|
|
lastResult = featureStatus;
|
|
return lastResult;
|
|
}
|
|
|
|
|
|
// Now left only for "backwards compatibility"
|
|
bool CVehicle::IgnoresRotorFix() const
|
|
{
|
|
if ( ms_rotorFixOverride != 0 )
|
|
{
|
|
return ms_rotorFixOverride < 0;
|
|
}
|
|
return SVF::ModelHasFeature( m_nModelIndex.Get(), SVF::Feature::NO_ROTOR_FADE );
|
|
}
|
|
|
|
static void* varVehicleRender = AddressByVersion<void*>(0x6D0E60, 0x6D1680, 0x70C0B0);
|
|
WRAPPER void CVehicle::Render() { VARJMP(varVehicleRender); }
|
|
static void* varIsLawEnforcementVehicle = AddressByVersion<void*>(0x6D2370, 0x6D2BA0, 0x70D8C0);
|
|
WRAPPER bool CVehicle::IsLawEnforcementVehicle() { VARJMP(varIsLawEnforcementVehicle); }
|
|
|
|
auto GetFrameHierarchyId = AddressByVersion<int32_t(*)(RwFrame*)>(0x732A20, 0x733250, 0x76CC30);
|
|
|
|
void (CAutomobile::*CAutomobile::orgAutomobilePreRender)();
|
|
void (CPlane::*CPlane::orgPlanePreRender)();
|
|
CVehicle* (CStoredCar::*CStoredCar::orgRestoreCar)();
|
|
|
|
static int32_t random(int32_t from, int32_t to)
|
|
{
|
|
return from + ( Int32Rand() % (to-from) );
|
|
}
|
|
|
|
static RwObject* GetCurrentAtomicObject( RwFrame* frame )
|
|
{
|
|
RwObject* obj = nullptr;
|
|
RwFrameForAllObjects( frame, [&obj]( RwObject* object ) -> RwObject* {
|
|
if ( RpAtomicGetFlags(object) & rpATOMICRENDER )
|
|
{
|
|
obj = object;
|
|
return nullptr;
|
|
}
|
|
return object;
|
|
} );
|
|
return obj;
|
|
}
|
|
|
|
RwFrame* GetFrameFromName( RwFrame* topFrame, const char* name )
|
|
{
|
|
class GetFramePredicate
|
|
{
|
|
public:
|
|
RwFrame* foundFrame = nullptr;
|
|
|
|
GetFramePredicate( const char* name )
|
|
: m_name( name )
|
|
{
|
|
}
|
|
|
|
RwFrame* operator() ( RwFrame* frame )
|
|
{
|
|
if ( _stricmp( m_name, GetFrameNodeName(frame) ) == 0 )
|
|
{
|
|
foundFrame = frame;
|
|
return nullptr;
|
|
}
|
|
RwFrameForAllChildren( frame, std::forward<GetFramePredicate>(*this) );
|
|
return foundFrame != nullptr ? nullptr : frame;
|
|
}
|
|
|
|
private:
|
|
const char* const m_name;
|
|
};
|
|
;
|
|
return RwFrameForAllChildren( topFrame, GetFramePredicate(name) ).foundFrame;
|
|
}
|
|
|
|
RwFrame* GetFrameFromID( RwFrame* topFrame, int32_t ID )
|
|
{
|
|
class GetFramePredicate
|
|
{
|
|
public:
|
|
RwFrame* foundFrame = nullptr;
|
|
|
|
GetFramePredicate( int32_t ID )
|
|
: ID( ID )
|
|
{
|
|
}
|
|
|
|
RwFrame* operator() ( RwFrame* frame )
|
|
{
|
|
if ( ID == GetFrameHierarchyId(frame) )
|
|
{
|
|
foundFrame = frame;
|
|
return nullptr;
|
|
}
|
|
RwFrameForAllChildren( frame, std::forward<GetFramePredicate>(*this) );
|
|
return foundFrame != nullptr ? nullptr : frame;
|
|
}
|
|
|
|
private:
|
|
const int32_t ID;
|
|
};
|
|
return RwFrameForAllChildren( topFrame, GetFramePredicate(ID) ).foundFrame;
|
|
}
|
|
|
|
void ReadRotorFixExceptions(const wchar_t* pPath)
|
|
{
|
|
constexpr size_t SCRATCH_PAD_SIZE = 32767;
|
|
WideDelimStringReader reader( SCRATCH_PAD_SIZE );
|
|
|
|
GetPrivateProfileSectionW( L"RotorFixExceptions", reader.GetBuffer(), reader.GetSize(), pPath );
|
|
while ( const wchar_t* str = reader.GetString() )
|
|
{
|
|
int32_t toList = wcstol( str, nullptr, 0 );
|
|
if ( toList > 0 )
|
|
SVF::RegisterFeature( toList, SVF::Feature::NO_ROTOR_FADE );
|
|
}
|
|
}
|
|
|
|
void ReadLightbeamFixExceptions(const wchar_t* pPath)
|
|
{
|
|
constexpr size_t SCRATCH_PAD_SIZE = 32767;
|
|
WideDelimStringReader reader( SCRATCH_PAD_SIZE );
|
|
|
|
GetPrivateProfileSectionW( L"LightbeamFixExceptions", reader.GetBuffer(), reader.GetSize(), pPath );
|
|
while ( const wchar_t* str = reader.GetString() )
|
|
{
|
|
int32_t toList = wcstol( str, nullptr, 0 );
|
|
if ( toList > 0 )
|
|
SVF::RegisterFeature( toList, SVF::Feature::NO_LIGHTBEAM_BFC_FIX );
|
|
}
|
|
}
|
|
|
|
bool CVehicle::HasFirelaLadder() const
|
|
{
|
|
return SVF::ModelHasFeature( m_nModelIndex.Get(), SVF::Feature::FIRELA_LADDER );
|
|
}
|
|
|
|
void* CVehicle::PlayPedHitSample_GetColModel()
|
|
{
|
|
if ( this == FindPlayerVehicle() )
|
|
{
|
|
CPed *pPassenger = PickRandomPassenger();
|
|
if ( pPassenger != nullptr )
|
|
{
|
|
pPassenger->Say( CONTEXT_GLOBAL_CAR_HIT_PED );
|
|
}
|
|
}
|
|
|
|
return GetColModel();
|
|
}
|
|
|
|
void CVehicle::SetComponentAtomicAlpha(RpAtomic* pAtomic, int nAlpha)
|
|
{
|
|
RpGeometry* pGeometry = RpAtomicGetGeometry(pAtomic);
|
|
pGeometry->flags |= rpGEOMETRYMODULATEMATERIALCOLOR;
|
|
|
|
RpGeometryForAllMaterials( pGeometry, [nAlpha] (RpMaterial* material) {
|
|
material->color.alpha = RwUInt8(nAlpha);
|
|
return material;
|
|
} );
|
|
}
|
|
|
|
bool CVehicle::IgnoresLightbeamFix() const
|
|
{
|
|
if ( ms_lightbeamFixOverride != 0 )
|
|
{
|
|
return ms_lightbeamFixOverride < 0;
|
|
}
|
|
return SVF::ModelHasFeature( m_nModelIndex.Get(), SVF::Feature::NO_LIGHTBEAM_BFC_FIX );
|
|
}
|
|
|
|
|
|
void CVehicle::DoHeadLightBeam_LightBeamFixSaveObj(int type, CMatrix& m, bool right)
|
|
{
|
|
LightbeamFix::SetCurrentVehicle( this );
|
|
DoHeadLightBeam( type, m, right );
|
|
LightbeamFix::SetCurrentVehicle( nullptr );
|
|
}
|
|
|
|
bool CVehicle::CustomCarPlate_TextureCreate(CVehicleModelInfo* pModelInfo)
|
|
{
|
|
char PlateText[CVehicleModelInfo::PLATE_TEXT_LEN+1];
|
|
const char* pOverrideText = pModelInfo->GetCustomCarPlateText();
|
|
|
|
if ( pOverrideText )
|
|
strncpy_s(PlateText, pOverrideText, CVehicleModelInfo::PLATE_TEXT_LEN);
|
|
else
|
|
CCustomCarPlateMgr::GeneratePlateText(PlateText, CVehicleModelInfo::PLATE_TEXT_LEN);
|
|
|
|
PlateText[CVehicleModelInfo::PLATE_TEXT_LEN] = '\0';
|
|
PlateTexture = CCustomCarPlateMgr::CreatePlateTexture(PlateText, pModelInfo->m_nPlateType);
|
|
if ( pModelInfo->m_nPlateType != -1 )
|
|
PlateDesign = pModelInfo->m_nPlateType;
|
|
else if ( IsLawEnforcementVehicle() )
|
|
PlateDesign = CCustomCarPlateMgr::GetMapRegionPlateDesign();
|
|
else
|
|
PlateDesign = random(0, 20) == 0 ? int8_t(random(0, 3)) : CCustomCarPlateMgr::GetMapRegionPlateDesign();
|
|
|
|
assert(PlateDesign >= 0 && PlateDesign < 3);
|
|
|
|
pModelInfo->m_plateText[0] = '\0';
|
|
pModelInfo->m_nPlateType = -1;
|
|
|
|
return true;
|
|
}
|
|
|
|
void CVehicle::CustomCarPlate_BeforeRenderingStart(CVehicleModelInfo* pModelInfo)
|
|
{
|
|
for ( size_t i = 0; i < pModelInfo->m_apPlateMaterials->m_numPlates; i++ )
|
|
{
|
|
RpMaterialSetTexture(pModelInfo->m_apPlateMaterials->m_plates[i], PlateTexture);
|
|
}
|
|
|
|
for ( size_t i = 0; i < pModelInfo->m_apPlateMaterials->m_numPlatebacks; i++ )
|
|
{
|
|
CCustomCarPlateMgr::SetupMaterialPlatebackTexture(pModelInfo->m_apPlateMaterials->m_platebacks[i], PlateDesign);
|
|
}
|
|
}
|
|
|
|
void CVehicle::SetComponentRotation( RwFrame* component, eRotAxis axis, float angle, bool absolute )
|
|
{
|
|
if ( component == nullptr ) return;
|
|
|
|
CMatrix matrix( RwFrameGetMatrix(component) );
|
|
if ( absolute )
|
|
{
|
|
if ( axis == ROT_AXIS_X ) matrix.SetRotateXOnly(angle);
|
|
else if ( axis == ROT_AXIS_Y ) matrix.SetRotateYOnly(angle);
|
|
else if ( axis == ROT_AXIS_Z ) matrix.SetRotateZOnly(angle);
|
|
}
|
|
else
|
|
{
|
|
const CVector pos = matrix.GetPos();
|
|
matrix.SetTranslateOnly(0.0f, 0.0f, 0.0f);
|
|
|
|
if ( axis == ROT_AXIS_X ) matrix.RotateX(angle);
|
|
else if ( axis == ROT_AXIS_Y ) matrix.RotateY(angle);
|
|
else if ( axis == ROT_AXIS_Z ) matrix.RotateZ(angle);
|
|
|
|
matrix.GetPos() += pos;
|
|
}
|
|
matrix.UpdateRW();
|
|
}
|
|
|
|
CPed* CVehicle::PickRandomPassenger()
|
|
{
|
|
const unsigned int randomNum = static_cast<unsigned int>((static_cast<double>(rand()) / RAND_MAX) * 8.0);
|
|
for ( size_t i = 0; i < 8; i++ )
|
|
{
|
|
const size_t index = (i + randomNum) % 8;
|
|
if ( m_apPassengers[index] != nullptr ) return m_apPassengers[index];
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
int32_t CVehicle::GetRemapIndex()
|
|
{
|
|
int32_t remapTxd = m_remapTxdSlot.Get();
|
|
if ( remapTxd == -1 )
|
|
{
|
|
// Original code never checked that variable, hence the bug
|
|
remapTxd = m_remapTxdSlotToLoad.Get();
|
|
}
|
|
if ( remapTxd == -1 )
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
const CVehicleModelInfo* modelInfo = static_cast<CVehicleModelInfo*>(ms_modelInfoPtrs[ m_nModelIndex.Get() ]);
|
|
for ( int32_t i = 0, j = modelInfo->GetNumRemaps(); i < j; i++ )
|
|
{
|
|
if ( modelInfo->m_awRemapTxds[i].Get() == remapTxd )
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void CHeli::Render()
|
|
{
|
|
double dRotorsSpeed, dMovingRotorSpeed;
|
|
const bool bDisplayRotors = !IgnoresRotorFix();
|
|
const bool bHasMovingRotor = m_pCarNode[13] != nullptr && bDisplayRotors;
|
|
const bool bHasMovingRotor2 = m_pCarNode[15] != nullptr && bDisplayRotors;
|
|
|
|
m_nTimeTillWeNeedThisCar = CTimer::m_snTimeInMilliseconds + 3000;
|
|
|
|
if ( m_fRotorSpeed > 0.0 )
|
|
dRotorsSpeed = std::min(1.7 * (1.0/0.22) * m_fRotorSpeed, 1.5);
|
|
else
|
|
dRotorsSpeed = 0.0;
|
|
|
|
dMovingRotorSpeed = dRotorsSpeed - 0.4;
|
|
if ( dMovingRotorSpeed < 0.0 )
|
|
dMovingRotorSpeed = 0.0;
|
|
|
|
int nStaticRotorAlpha = static_cast<int>(std::min((1.5-dRotorsSpeed) * 255.0, 255.0));
|
|
int nMovingRotorAlpha = static_cast<int>(std::min(dMovingRotorSpeed * 175.0, 175.0));
|
|
|
|
if ( m_pCarNode[12] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[12] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingRotor ? nStaticRotorAlpha : 255);
|
|
}
|
|
|
|
if ( m_pCarNode[14] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[14] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingRotor2 ? nStaticRotorAlpha : 255);
|
|
}
|
|
|
|
if ( m_pCarNode[13] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[13] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingRotor ? nMovingRotorAlpha : 0);
|
|
}
|
|
|
|
if ( m_pCarNode[15] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[15] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingRotor2 ? nMovingRotorAlpha : 0);
|
|
}
|
|
|
|
CEntity::Render();
|
|
}
|
|
|
|
void CPlane::Render()
|
|
{
|
|
double dRotorsSpeed, dMovingRotorSpeed;
|
|
const bool bDisplayRotors = !IgnoresRotorFix();
|
|
const bool bHasMovingProp = m_pCarNode[13] != nullptr && bDisplayRotors;
|
|
const bool bHasMovingProp2 = m_pCarNode[15] != nullptr && bDisplayRotors;
|
|
|
|
m_nTimeTillWeNeedThisCar = CTimer::m_snTimeInMilliseconds + 3000;
|
|
|
|
if ( m_fPropellerSpeed > 0.0 )
|
|
dRotorsSpeed = std::min(1.7 * (1.0/0.31) * m_fPropellerSpeed, 1.5);
|
|
else
|
|
dRotorsSpeed = 0.0;
|
|
|
|
dMovingRotorSpeed = dRotorsSpeed - 0.4;
|
|
if ( dMovingRotorSpeed < 0.0 )
|
|
dMovingRotorSpeed = 0.0;
|
|
|
|
int nStaticRotorAlpha = static_cast<int>(std::min((1.5-dRotorsSpeed) * 255.0, 255.0));
|
|
int nMovingRotorAlpha = static_cast<int>(std::min(dMovingRotorSpeed * 175.0, 175.0));
|
|
|
|
if ( m_pCarNode[12] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[12] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingProp ? nStaticRotorAlpha : 255);
|
|
}
|
|
|
|
if ( m_pCarNode[14] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[14] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingProp2 ? nStaticRotorAlpha : 255);
|
|
}
|
|
|
|
if ( m_pCarNode[13] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[13] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingProp ? nMovingRotorAlpha : 0);
|
|
}
|
|
|
|
if ( m_pCarNode[15] != nullptr )
|
|
{
|
|
RpAtomic* pOutAtomic = (RpAtomic*)GetCurrentAtomicObject( m_pCarNode[15] );
|
|
if ( pOutAtomic != nullptr )
|
|
SetComponentAtomicAlpha(pOutAtomic, bHasMovingProp2 ? nMovingRotorAlpha : 0);
|
|
}
|
|
|
|
CVehicle::Render();
|
|
}
|
|
|
|
void CPlane::Fix_SilentPatch()
|
|
{
|
|
// Reset bouncing panels
|
|
// No reset on Vortex
|
|
for ( ptrdiff_t i = SVF::ModelHasFeature( m_nModelIndex.Get(), SVF::Feature::VORTEX_EXHAUST ) ? 1 : 0; i < 3; i++ )
|
|
{
|
|
m_aBouncingPanel[i].m_nNodeIndex = -1;
|
|
}
|
|
}
|
|
|
|
void CPlane::PreRender()
|
|
{
|
|
(this->*(orgPlanePreRender))();
|
|
|
|
const int32_t extID = m_nModelIndex.Get();
|
|
|
|
auto copyRotation = [&]( size_t src, size_t dest ) {
|
|
if ( m_pCarNode[src] != nullptr && m_pCarNode[dest] != nullptr )
|
|
{
|
|
RwMatrix* lhs = RwFrameGetMatrix( m_pCarNode[dest] );
|
|
const RwMatrix* rhs = RwFrameGetMatrix( m_pCarNode[src] );
|
|
|
|
lhs->at = rhs->at;
|
|
lhs->up = rhs->up;
|
|
lhs->right = rhs->right;
|
|
RwMatrixUpdate( lhs );
|
|
}
|
|
};
|
|
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::EXTRA_AILERONS1 ) )
|
|
{
|
|
copyRotation( 18, 21 );
|
|
}
|
|
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::EXTRA_AILERONS2 ) )
|
|
{
|
|
copyRotation( 19, 23 );
|
|
copyRotation( 20, 24 );
|
|
}
|
|
}
|
|
|
|
RwFrame* CAutomobile::GetTowBarFrame() const
|
|
{
|
|
RwFrame* towBar = m_pCarNode[20];
|
|
if ( towBar == nullptr )
|
|
{
|
|
towBar = m_pCarNode[21];
|
|
}
|
|
return towBar;
|
|
}
|
|
|
|
void CAutomobile::PreRender()
|
|
{
|
|
// For rotating engine components
|
|
ms_engineCompSpeed = m_nVehicleFlags.bEngineOn ? CTimer::m_fTimeStep : 0.0f;
|
|
|
|
(this->*(orgAutomobilePreRender))();
|
|
|
|
const int32_t extID = m_nModelIndex.Get();
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::PHOENIX_FLUTTER ) )
|
|
{
|
|
ProcessPhoenixBlower( extID );
|
|
}
|
|
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::SWEEPER_BRUSHES ) )
|
|
{
|
|
ProcessSweeper();
|
|
}
|
|
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::NEWSVAN_DISH ) )
|
|
{
|
|
ProcessNewsvan();
|
|
}
|
|
}
|
|
|
|
void CAutomobile::Fix_SilentPatch()
|
|
{
|
|
ResetFrames();
|
|
|
|
// Reset bouncing panels
|
|
const int32_t extID = m_nModelIndex.Get();
|
|
for ( ptrdiff_t i = (m_pCarNode[21] != nullptr && SVF::ModelHasFeature( extID, SVF::Feature::TOWTRUCK_HOOK )) || (m_pCarNode[17] != nullptr && SVF::ModelHasFeature( extID, SVF::Feature::TRACTOR_HOOK )) ? 1 : 0; i < 3; i++ )
|
|
{
|
|
// Towtruck/Tractor fix
|
|
m_aBouncingPanel[i].m_nNodeIndex = -1;
|
|
}
|
|
|
|
// Reset Rhino middle wheels state
|
|
if ( SVF::ModelHasFeature( extID, SVF::Feature::RHINO_WHEELS ) )
|
|
{
|
|
Door[REAR_LEFT_DOOR].SetExtraWheelPositions( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
Door[REAR_RIGHT_DOOR].SetExtraWheelPositions( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
|
|
if ( m_pCarNode[3] != nullptr )
|
|
{
|
|
RwObject* object = GetFirstObject( m_pCarNode[3] );
|
|
RpAtomicSetFlags( object, 0 );
|
|
}
|
|
|
|
if ( m_pCarNode[6] != nullptr )
|
|
{
|
|
RwObject* object = GetFirstObject( m_pCarNode[6] );
|
|
RpAtomicSetFlags( object, 0 );
|
|
}
|
|
}
|
|
}
|
|
|
|
void CAutomobile::ResetFrames()
|
|
{
|
|
RpClump* pOrigClump = reinterpret_cast<RpClump*>(ms_modelInfoPtrs[ m_nModelIndex.Get() ]->pRwObject);
|
|
if ( pOrigClump != nullptr )
|
|
{
|
|
// Instead of setting frame rotation to (0,0,0) like R* did, obtain the original frame matrix from CBaseNodelInfo clump
|
|
for ( ptrdiff_t i = 8; i < 25; i++ )
|
|
{
|
|
if ( m_pCarNode[i] != nullptr )
|
|
{
|
|
// Find a frame in CBaseModelInfo object
|
|
RwFrame* origFrame = GetFrameFromID( RpClumpGetFrame(pOrigClump), static_cast<int32_t>(i) );
|
|
if ( origFrame != nullptr )
|
|
{
|
|
// Found a frame, reset it
|
|
*RwFrameGetMatrix(m_pCarNode[i]) = *RwFrameGetMatrix(origFrame);
|
|
RwMatrixUpdate(RwFrameGetMatrix(m_pCarNode[i]));
|
|
}
|
|
else
|
|
{
|
|
// Same as original code
|
|
CMatrix matrix( RwFrameGetMatrix(m_pCarNode[i]) );
|
|
const CVector pos( matrix.GetPos() );
|
|
matrix.SetTranslate( pos.x, pos.y, pos.z );
|
|
matrix.UpdateRW();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CAutomobile::ProcessPhoenixBlower( int32_t modelID )
|
|
{
|
|
if ( m_pCarNode[20] == nullptr ) return;
|
|
if ( !m_nVehicleFlags.bEngineOn ) return;
|
|
|
|
RpClump* pOrigClump = reinterpret_cast<RpClump*>(ms_modelInfoPtrs[ modelID ]->pRwObject);
|
|
if ( pOrigClump != nullptr )
|
|
{
|
|
RwFrame* origFrame = GetFrameFromID( RpClumpGetFrame(pOrigClump), 20 );
|
|
if ( origFrame != nullptr )
|
|
{
|
|
*RwFrameGetMatrix(m_pCarNode[20]) = *RwFrameGetMatrix(origFrame);
|
|
}
|
|
}
|
|
|
|
float finalAngle = 0.0f;
|
|
if ( std::abs(m_fGasPedal) > 0.0f )
|
|
{
|
|
if ( m_fSpecialComponentAngle < 1.3f )
|
|
{
|
|
finalAngle = m_fSpecialComponentAngle = std::min( m_fSpecialComponentAngle + 0.1f * CTimer::m_fTimeStep, 1.3f );
|
|
}
|
|
else
|
|
{
|
|
finalAngle = m_fSpecialComponentAngle + (std::sin( (CTimer::m_snTimeInMilliseconds % 10000) / PHOENIX_FLUTTER_PERIOD ) * PHOENIX_FLUTTER_AMP);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( m_fSpecialComponentAngle > 0.0f )
|
|
{
|
|
finalAngle = m_fSpecialComponentAngle = std::max( m_fSpecialComponentAngle - 0.05f * CTimer::m_fTimeStep, 0.0f );
|
|
}
|
|
}
|
|
|
|
SetComponentRotation( m_pCarNode[20], ROT_AXIS_X, finalAngle, false );
|
|
}
|
|
|
|
void CAutomobile::ProcessSweeper()
|
|
{
|
|
if ( !m_nVehicleFlags.bEngineOn ) return;
|
|
|
|
if ( GetStatus() == STATUS_PLAYER || GetStatus() == STATUS_PHYSICS || GetStatus() == STATUS_SIMPLE )
|
|
{
|
|
const float angle = CTimer::m_fTimeStep * SWEEPER_BRUSH_SPEED;
|
|
|
|
SetComponentRotation( m_pCarNode[20], ROT_AXIS_Z, angle, false );
|
|
SetComponentRotation( m_pCarNode[21], ROT_AXIS_Z, -angle, false );
|
|
}
|
|
}
|
|
|
|
void CAutomobile::ProcessNewsvan()
|
|
{
|
|
if ( GetStatus() == STATUS_PLAYER || GetStatus() == STATUS_PHYSICS || GetStatus() == STATUS_SIMPLE )
|
|
{
|
|
// TODO: Point at something? Like nearest collectable or safehouse
|
|
m_fGunOrientation += CTimer::m_fTimeStep * 0.05f;
|
|
if ( m_fGunOrientation > 2.0f * PI ) m_fGunOrientation -= 2.0f * PI;
|
|
SetComponentRotation( m_pCarNode[20], ROT_AXIS_Z, m_fGunOrientation );
|
|
}
|
|
}
|
|
|
|
bool CTrailer::GetTowBarPos(CVector& posnOut, bool defaultPos, CVehicle* trailer)
|
|
{
|
|
const int32_t modelID = m_nModelIndex.Get();
|
|
if ( SVF::ModelHasFeature( modelID, SVF::Feature::DOUBLE_TRAILER ) )
|
|
{
|
|
if ( m_pCarNode[21] != nullptr )
|
|
{
|
|
const RwMatrix* ltm = RwFrameGetLTM( m_pCarNode[21] );
|
|
posnOut.x = ltm->pos.x;
|
|
posnOut.y = ltm->pos.y;
|
|
posnOut.z = ltm->pos.z;
|
|
return true;
|
|
}
|
|
|
|
// Fallback, same as in original CTrailer::GetTowBarPos
|
|
if ( defaultPos )
|
|
{
|
|
posnOut = *GetMatrix() * CVector(0.0f, ms_modelInfoPtrs[ modelID ]->pColModel->boundingBox.vecMin.y - 0.05f, 0.5f - m_fHeightAboveRoad);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return GetTowBarPos_GTA(posnOut, defaultPos, trailer);
|
|
}
|
|
|
|
CVehicle* CStoredCar::RestoreCar_SilentPatch()
|
|
{
|
|
CVehicle* vehicle = (this->*(orgRestoreCar))();
|
|
if ( vehicle == nullptr ) return nullptr;
|
|
|
|
if ( m_bombType != 0 )
|
|
{
|
|
// Fixup bomb stuff
|
|
if ( vehicle->GetClass() == VEHICLE_AUTOMOBILE || vehicle->GetClass() == VEHICLE_BIKE )
|
|
{
|
|
vehicle->SetBombOnBoard( m_bombType );
|
|
vehicle->SetBombOwner( FindPlayerPed() );
|
|
}
|
|
}
|
|
|
|
return vehicle;
|
|
}
|
|
|
|
|
|
// Returns "feature cookie" on success, -1 on failure
|
|
extern "C" {
|
|
|
|
__declspec(dllexport) int32_t RegisterSpecialVehicleFeature( int32_t modelID, const char* featureName )
|
|
{
|
|
if ( featureName == nullptr ) return -1;
|
|
return SVF::RegisterFeature( modelID, SVF::GetFeatureFromName(featureName) );
|
|
}
|
|
|
|
__declspec(dllexport) void DeleteSpecialVehicleFeature( int32_t cookie )
|
|
{
|
|
if ( cookie == -1 ) return;
|
|
SVF::DeleteFeature( cookie );
|
|
}
|
|
|
|
__declspec(dllexport) void DisableStockVehiclesForSpecialVehicleFeature( const char* featureName )
|
|
{
|
|
if ( featureName == nullptr ) return;
|
|
SVF::DisableStockVehiclesForFeature( SVF::GetFeatureFromName(featureName) );
|
|
}
|
|
|
|
}
|