Files
itgmania212121/src/Actor.cpp
T

2232 lines
70 KiB
C++
Raw Normal View History

#include "global.h"
#include "Actor.h"
#include "ActorFrame.h"
#include "RageDisplay.h"
#include "RageUtil.h"
#include "RageMath.h"
#include "RageLog.h"
#include "XmlFile.h"
#include "LuaBinding.h"
#include "ThemeManager.h"
#include "LuaReference.h"
#include "MessageManager.h"
#include "LightsManager.h" // for NUM_CabinetLight
#include "ActorUtil.h"
#include "Preference.h"
#include "GameLoop.h"
2023-04-19 14:22:59 +02:00
#include <cmath>
2023-04-19 23:04:25 +02:00
#include <cstddef>
#include <typeinfo>
2023-04-20 19:02:13 +02:00
#include <vector>
static Preference<bool> g_bShowMasks("ShowMasks", false);
static const float default_effect_period= 1.0f;
/**
* @brief Set up a hidden Actor that won't be drawn.
*
* It's useful to be able to construct a basic Actor in XML, in
* order to simply delay a Transition, or receive and send broadcasts.
* Since these actors will never draw, set them hidden by default. */
class HiddenActor: public Actor
{
public:
HiddenActor() { SetVisible(false); }
virtual HiddenActor *Copy() const;
};
REGISTER_ACTOR_CLASS_WITH_NAME( HiddenActor, Actor );
float Actor::g_fCurrentBGMTime = 0, Actor::g_fCurrentBGMBeat;
float Actor::g_fCurrentBGMTimeNoOffset = 0, Actor::g_fCurrentBGMBeatNoOffset = 0;
std::vector<float> Actor::g_vfCurrentBGMBeatPlayer(NUM_PlayerNumber, 0);
std::vector<float> Actor::g_vfCurrentBGMBeatPlayerNoOffset(NUM_PlayerNumber, 0);
Actor *Actor::Copy() const { return new Actor(*this); }
static float g_fCabinetLights[NUM_CabinetLight];
static const char *HorizAlignNames[] = {
"Left",
"Center",
"Right"
};
XToString( HorizAlign );
LuaXType( HorizAlign );
static const char *VertAlignNames[] = {
"Top",
"Middle",
"Bottom"
};
XToString( VertAlign );
LuaXType( VertAlign );
void Actor::SetBGMTime( float fTime, float fBeat, float fTimeNoOffset, float fBeatNoOffset )
{
g_fCurrentBGMTime = fTime;
g_fCurrentBGMBeat = fBeat;
/* This timer is generally only used for effects tied to the background music
* when GameSoundManager is aligning music beats. Alignment doesn't handle
* g_fVisualDelaySeconds. */
g_fCurrentBGMTimeNoOffset = fTimeNoOffset;
g_fCurrentBGMBeatNoOffset = fBeatNoOffset;
}
void Actor::SetPlayerBGMBeat( PlayerNumber pn, float fBeat, float fBeatNoOffset )
{
g_vfCurrentBGMBeatPlayer[pn] = fBeat;
g_vfCurrentBGMBeatPlayerNoOffset[pn] = fBeatNoOffset;
}
void Actor::SetBGMLight( int iLightNumber, float fCabinetLights )
{
ASSERT( iLightNumber < NUM_CabinetLight );
g_fCabinetLights[iLightNumber] = fCabinetLights;
}
void Actor::InitState()
{
2014-04-14 21:29:19 -05:00
this->StopTweening();
2019-06-22 12:35:38 -07:00
m_pTempState = nullptr;
m_baseRotation = RageVector3( 0, 0, 0 );
m_baseScale = RageVector3( 1, 1, 1 );
m_fBaseAlpha = 1;
2013-05-28 21:46:15 -04:00
m_internalDiffuse = RageColor( 1, 1, 1, 1 );
2013-05-28 20:52:01 -04:00
m_internalGlow = RageColor( 0, 0, 0, 0 );
m_start.Init();
m_current.Init();
m_fHorizAlign = 0.5f;
m_fVertAlign = 0.5f;
#if defined(SSC_FUTURES)
for( unsigned i = 0; i < m_Effects.size(); ++i )
m_Effects[i] = no_effect;
#else
m_Effect = no_effect;
#endif
m_fSecsIntoEffect = 0;
m_fEffectDelta = 0;
SetEffectPeriod(default_effect_period);
m_fEffectOffset = 0;
m_EffectClock = CLOCK_TIMER;
m_vEffectMagnitude = RageVector3(0,0,10);
m_effectColor1 = RageColor(1,1,1,1);
m_effectColor2 = RageColor(1,1,1,1);
m_bVisible = true;
m_fShadowLengthX = 0;
m_fShadowLengthY = 0;
m_ShadowColor = RageColor(0,0,0,0.5f);
m_bIsAnimating = true;
m_fHibernateSecondsLeft = 0;
m_iDrawOrder = 0;
m_bTextureWrapping = false;
m_bTextureFiltering = true;
2014-02-20 21:17:10 -06:00
m_texTranslate = RageVector2( 0, 0 );
m_BlendMode = BLEND_NORMAL;
m_fZBias = 0;
m_bClearZBuffer = false;
m_ZTestMode = ZTEST_OFF;
m_bZWrite = false;
m_CullMode = CULL_NONE;
}
static bool GetMessageNameFromCommandName( const RString &sCommandName, RString &sMessageNameOut )
{
if( sCommandName.Right(7) == "Message" )
{
sMessageNameOut = sCommandName.Left(sCommandName.size()-7);
return true;
}
else
{
return false;
}
}
Actor::Actor()
{
m_pLuaInstance = new LuaClass;
Lua *L = LUA->Get();
m_pLuaInstance->PushSelf( L );
lua_newtable( L );
lua_pushvalue( L, -1 );
lua_setmetatable( L, -2 );
lua_setfield( L, -2, "ctx" );
lua_pop( L, 1 );
LUA->Release( L );
2023-04-19 14:22:59 +02:00
m_size = RageVector2( 1, 1 );
InitState();
2019-06-22 12:35:38 -07:00
m_pParent = nullptr;
m_FakeParent = nullptr;
m_bFirstUpdate = true;
m_tween_uses_effect_delta = false;
rate_scaling_enabled_ = true;
}
Actor::~Actor()
{
StopTweening();
UnsubscribeAll();
2024-10-01 00:43:07 -07:00
for(size_t i= 0; i < m_WrapperStates.size(); ++i)
{
RageUtil::SafeDelete(m_WrapperStates[i]);
}
m_WrapperStates.clear();
}
Actor::Actor( const Actor &cpy ):
MessageSubscriber( cpy )
{
/* Don't copy an Actor in the middle of rendering. */
2019-06-22 12:35:38 -07:00
ASSERT( cpy.m_pTempState == nullptr );
m_pTempState = nullptr;
#define CPY(x) x = cpy.x
CPY( m_sName );
2025-04-07 17:14:10 -04:00
CPY(alias_);
CPY( m_pParent );
CPY( m_FakeParent );
CPY( m_pLuaInstance );
m_WrapperStates.resize(cpy.m_WrapperStates.size());
2024-10-01 00:43:07 -07:00
for(size_t i= 0; i < m_WrapperStates.size(); ++i)
{
m_WrapperStates[i]= new ActorFrame(*dynamic_cast<ActorFrame*>(cpy.m_WrapperStates[i]));
}
CPY( m_baseRotation );
CPY( m_baseScale );
CPY( m_fBaseAlpha );
2013-05-28 21:46:15 -04:00
CPY( m_internalDiffuse );
2013-05-28 20:52:01 -04:00
CPY( m_internalGlow );
CPY( m_size );
CPY( m_current );
CPY( m_start );
for( unsigned i = 0; i < cpy.m_Tweens.size(); ++i )
m_Tweens.push_back( new TweenStateAndInfo(*cpy.m_Tweens[i]) );
CPY( m_bFirstUpdate );
CPY( m_fHorizAlign );
CPY( m_fVertAlign );
#if defined(SSC_FUTURES)
// I'm a bit worried about this -aj
for( unsigned i = 0; i < cpy.m_Effects.size(); ++i )
m_Effects.push_back( (*cpy.m_Effects[i]) );
#else
CPY( m_Effect );
#endif
CPY( m_fSecsIntoEffect );
CPY( m_fEffectDelta );
CPY(m_effect_ramp_to_half);
CPY(m_effect_hold_at_half);
CPY(m_effect_ramp_to_full);
CPY(m_effect_hold_at_full);
CPY(m_effect_hold_at_zero);
CPY(m_effect_period);
CPY( m_fEffectOffset );
CPY( m_EffectClock );
CPY( m_effectColor1 );
CPY( m_effectColor2 );
CPY( m_vEffectMagnitude );
CPY( m_bVisible );
CPY( m_fHibernateSecondsLeft );
CPY( m_fShadowLengthX );
CPY( m_fShadowLengthY );
CPY( m_ShadowColor );
CPY( m_bIsAnimating );
CPY( m_iDrawOrder );
CPY( m_bTextureWrapping );
CPY( m_bTextureFiltering );
CPY( m_BlendMode );
CPY( m_bClearZBuffer );
CPY( m_ZTestMode );
CPY( m_bZWrite );
CPY( m_fZBias );
CPY( m_CullMode );
CPY( m_mapNameToCommands );
#undef CPY
}
2017-06-18 11:55:16 -04:00
Actor &Actor::operator=(Actor other)
{
/* Don't copy an Actor in the middle of rendering. */
ASSERT( other.m_pTempState == nullptr );
m_pTempState = nullptr;
using std::swap;
#define SWAP(x) swap(x, other.x)
SWAP( m_sName );
2025-04-07 17:14:10 -04:00
SWAP(alias_);
2017-06-18 11:55:16 -04:00
SWAP( m_pParent );
SWAP( m_FakeParent );
SWAP( m_pLuaInstance );
SWAP( m_WrapperStates );
SWAP( m_baseRotation );
SWAP( m_baseScale );
SWAP( m_fBaseAlpha );
SWAP( m_internalDiffuse );
SWAP( m_internalGlow );
SWAP( m_size );
SWAP( m_current );
SWAP( m_start );
SWAP( m_Tweens );
SWAP( m_bFirstUpdate );
SWAP( m_fHorizAlign );
SWAP( m_fVertAlign );
#if defined(SSC_FUTURES)
SWAP( M_Effects );
#else
SWAP( m_Effect );
#endif
SWAP( m_fSecsIntoEffect );
SWAP( m_fEffectDelta );
SWAP(m_effect_ramp_to_half);
SWAP(m_effect_hold_at_half);
SWAP(m_effect_ramp_to_full);
SWAP(m_effect_hold_at_full);
SWAP(m_effect_hold_at_zero);
SWAP(m_effect_period);
SWAP( m_fEffectOffset );
SWAP( m_EffectClock );
SWAP( m_effectColor1 );
SWAP( m_effectColor2 );
SWAP( m_vEffectMagnitude );
SWAP( m_bVisible );
SWAP( m_fHibernateSecondsLeft );
SWAP( m_fShadowLengthX );
SWAP( m_fShadowLengthY );
SWAP( m_ShadowColor );
SWAP( m_bIsAnimating );
SWAP( m_iDrawOrder );
SWAP( m_bTextureWrapping );
SWAP( m_bTextureFiltering );
SWAP( m_BlendMode );
SWAP( m_bClearZBuffer );
SWAP( m_ZTestMode );
SWAP( m_bZWrite );
SWAP( m_fZBias );
SWAP( m_CullMode );
SWAP( m_mapNameToCommands );
#undef SWAP
return *this;
}
2025-04-07 17:14:10 -04:00
bool Actor::IsAlias(const std::string& name) {
if (alias_.empty()) {
return false;
}
return name == alias_;
}
/* XXX: This calls InitCommand, which must happen after all other
* initialization (eg. ActorFrame loading children). However, it
* also loads input variables, which should happen first. The
* former is more important. */
void Actor::LoadFromNode( const XNode* pNode )
{
Lua *L = LUA->Get();
FOREACH_CONST_Attr( pNode, pAttr )
{
// Load Name, if any.
const RString &sKeyName = pAttr->first;
const XNodeValue *pValue = pAttr->second;
2015-03-30 17:45:52 -06:00
if( EndsWith(sKeyName,"Command") )
{
LuaReference *pRef = new LuaReference;
pValue->PushValue( L );
pRef->SetFromStack( L );
RString sCmdName = sKeyName.Left( sKeyName.size()-7 );
AddCommand( sCmdName, apActorCommands( pRef ) );
}
2015-03-30 17:45:52 -06:00
else if( sKeyName == "Name" ) SetName( pValue->GetValue<RString>() );
else if( sKeyName == "BaseRotationX" ) SetBaseRotationX( pValue->GetValue<float>() );
else if( sKeyName == "BaseRotationY" ) SetBaseRotationY( pValue->GetValue<float>() );
else if( sKeyName == "BaseRotationZ" ) SetBaseRotationZ( pValue->GetValue<float>() );
else if( sKeyName == "BaseZoomX" ) SetBaseZoomX( pValue->GetValue<float>() );
else if( sKeyName == "BaseZoomY" ) SetBaseZoomY( pValue->GetValue<float>() );
else if( sKeyName == "BaseZoomZ" ) SetBaseZoomZ( pValue->GetValue<float>() );
}
LUA->Release( L );
2023-04-19 14:22:59 +02:00
// Don't recurse Init. It gets called once for every Actor when the
// Actor is loaded, and we don't want to call it again.
PlayCommandNoRecurse( Message("Init") );
}
bool Actor::PartiallyOpaque()
{
return m_pTempState->diffuse[0].a > 0 || m_pTempState->diffuse[1].a > 0 ||
m_pTempState->diffuse[2].a > 0 || m_pTempState->diffuse[3].a > 0 ||
m_pTempState->glow.a > 0;
}
void Actor::Draw()
{
if( !m_bVisible ||
2023-04-19 14:22:59 +02:00
m_fHibernateSecondsLeft > 0 ||
this->EarlyAbortDraw() )
{
return; // early abort
}
if(m_FakeParent)
{
if(!m_FakeParent->m_bVisible || m_FakeParent->m_fHibernateSecondsLeft > 0
|| m_FakeParent->EarlyAbortDraw())
{
return;
}
m_FakeParent->PreDraw();
if(!m_FakeParent->PartiallyOpaque())
{
m_FakeParent->PostDraw();
return;
}
}
if(m_FakeParent)
{
m_FakeParent->BeginDraw();
}
2024-10-01 00:43:07 -07:00
size_t wrapper_states_used= 0;
RageColor last_diffuse;
RageColor last_glow;
bool use_last_diffuse= false;
// dont_abort_draw exists because if one of the layers is invisible,
// there's no point in continuing. -Kyz
bool dont_abort_draw= true;
// abort_with_end_draw exists because PreDraw happens before the
// opaqueness test, so if we abort at the opaqueness test, there isn't
// a BeginDraw to match the EndDraw. -Kyz
bool abort_with_end_draw= true;
// It's more intuitive to apply the highest index wrappers first.
// On the lua side, it looks like this:
// wrapper[3] is the outermost frame. wrapper[2] is inside wrapper[3].
// wrapper[1] is inside wrapper[2]. The actor is inside wrapper[1].
// -Kyz
2024-10-01 00:43:07 -07:00
for(size_t i= m_WrapperStates.size(); i > 0 && dont_abort_draw; --i)
{
Actor* state= m_WrapperStates[i-1];
if(!state->m_bVisible || state->m_fHibernateSecondsLeft > 0 ||
state->EarlyAbortDraw())
{
dont_abort_draw= false;
}
else
{
state->PreDraw();
if(state->PartiallyOpaque())
{
state->BeginDraw();
last_diffuse= state->m_pTempState->diffuse[0];
last_glow= state->m_pTempState->glow;
use_last_diffuse= true;
if(i > 1)
{
m_WrapperStates[i-2]->SetInternalDiffuse(last_diffuse);
m_WrapperStates[i-2]->SetInternalGlow(last_glow);
}
}
else
{
dont_abort_draw= false;
abort_with_end_draw= false;
}
++wrapper_states_used;
}
}
// call the most-derived versions
if(dont_abort_draw)
{
if(use_last_diffuse)
{
this->SetInternalDiffuse(last_diffuse);
this->SetInternalGlow(last_glow);
}
this->PreDraw();
2019-06-22 12:35:38 -07:00
ASSERT( m_pTempState != nullptr );
if(PartiallyOpaque())
{
this->BeginDraw();
this->DrawPrimitives();
this->EndDraw();
}
this->PostDraw();
}
2024-10-01 00:43:07 -07:00
for(size_t i= 0; i < wrapper_states_used; ++i)
{
Actor* state= m_WrapperStates[i];
if(abort_with_end_draw)
{
state->EndDraw();
}
abort_with_end_draw= true;
state->PostDraw();
2019-06-22 12:35:38 -07:00
state->m_pTempState= nullptr;
}
if(m_FakeParent)
{
m_FakeParent->EndDraw();
m_FakeParent->PostDraw();
2019-06-22 12:35:38 -07:00
m_FakeParent->m_pTempState= nullptr;
}
2019-06-22 12:35:38 -07:00
m_pTempState = nullptr;
}
void Actor::PostDraw() // reset internal diffuse and glow
{
m_internalDiffuse = RageColor(1, 1, 1, 1);
m_internalGlow.a = 0;
}
void Actor::PreDraw() // calculate actor properties
{
// Somthing below may set m_pTempState to tempState
m_pTempState = &m_current;
// todo: account for SSC_FUTURES -aj
if( m_Effect == no_effect )
{
}
else
{
m_pTempState= & m_current_with_effects;
m_current_with_effects= m_current;
const float fTotalPeriod = GetEffectPeriod();
ASSERT( fTotalPeriod > 0 );
const float fTimeIntoEffect = fmodfp( m_fSecsIntoEffect+m_fEffectOffset, fTotalPeriod );
float fPercentThroughEffect;
const float rup_plus_ath= m_effect_ramp_to_half + m_effect_hold_at_half;
const float rupath_plus_rdown= rup_plus_ath + m_effect_ramp_to_full;
const float rupathrdown_plus_atf= rupath_plus_rdown + m_effect_hold_at_full;
if(fTimeIntoEffect < m_effect_ramp_to_half)
{
fPercentThroughEffect = SCALE(fTimeIntoEffect,
0, m_effect_ramp_to_half, 0.0f, 0.5f);
}
else if(fTimeIntoEffect < rup_plus_ath)
{
fPercentThroughEffect = 0.5f;
}
else if(fTimeIntoEffect < rupath_plus_rdown)
{
fPercentThroughEffect = SCALE(fTimeIntoEffect,
rup_plus_ath, rupath_plus_rdown, 0.5f, 1.0f);
}
else if(fTimeIntoEffect < rupathrdown_plus_atf)
{
fPercentThroughEffect= 1.0f;
}
else
{
fPercentThroughEffect = 0;
}
2023-04-19 14:22:59 +02:00
ASSERT_M( fPercentThroughEffect >= 0 && fPercentThroughEffect <= 1,
ssprintf("PercentThroughEffect: %f", fPercentThroughEffect) );
bool bBlinkOn = fPercentThroughEffect > 0.5f;
2024-07-17 21:37:27 -07:00
float fPercentBetweenColors = std::sin( (fPercentThroughEffect + 0.25f) * 2 * PI ) / 2 + 0.5f;
ASSERT_M( fPercentBetweenColors >= 0 && fPercentBetweenColors <= 1,
ssprintf("PercentBetweenColors: %f, PercentThroughEffect: %f", fPercentBetweenColors, fPercentThroughEffect) );
float fOriginalAlpha = m_current_with_effects.diffuse[0].a;
// todo: account for SSC_FUTURES -aj
switch( m_Effect )
{
case diffuse_blink:
2023-04-19 14:22:59 +02:00
/* XXX: Should diffuse_blink and diffuse_shift multiply the m_current_with_effects color?
* (That would have the same effect with 1,1,1,1, and allow tweening the diffuse
* while blinking and shifting.) */
for(int i=0; i<NUM_DIFFUSE_COLORS; i++)
{
m_current_with_effects.diffuse[i] = bBlinkOn ? m_effectColor1 : m_effectColor2;
m_current_with_effects.diffuse[i].a *= fOriginalAlpha; // multiply the alphas so we can fade even while an effect is playing
}
break;
case diffuse_shift:
for(int i=0; i<NUM_DIFFUSE_COLORS; i++)
{
m_current_with_effects.diffuse[i] = m_effectColor1*fPercentBetweenColors + m_effectColor2*(1.0f-fPercentBetweenColors);
m_current_with_effects.diffuse[i].a *= fOriginalAlpha; // multiply the alphas so we can fade even while an effect is playing
}
break;
case diffuse_ramp:
for(int i=0; i<NUM_DIFFUSE_COLORS; i++)
{
m_current_with_effects.diffuse[i] = m_effectColor1*fPercentThroughEffect + m_effectColor2*(1.0f-fPercentThroughEffect);
m_current_with_effects.diffuse[i].a *= fOriginalAlpha; // multiply the alphas so we can fade even while an effect is playing
}
break;
case glow_blink:
m_current_with_effects.glow = bBlinkOn ? m_effectColor1 : m_effectColor2;
m_current_with_effects.glow.a *= fOriginalAlpha; // don't glow if the Actor is transparent!
break;
case glow_shift:
m_current_with_effects.glow = m_effectColor1*fPercentBetweenColors + m_effectColor2*(1.0f-fPercentBetweenColors);
m_current_with_effects.glow.a *= fOriginalAlpha; // don't glow if the Actor is transparent!
break;
case glow_ramp:
m_current_with_effects.glow = m_effectColor1*fPercentThroughEffect + m_effectColor2*(1.0f-fPercentThroughEffect);
m_current_with_effects.glow.a *= fOriginalAlpha; // don't glow if the Actor is transparent!
break;
case rainbow:
m_current_with_effects.diffuse[0] = RageColor(
2024-07-17 21:37:27 -07:00
std::cos( fPercentBetweenColors*2*PI ) * 0.5f + 0.5f,
std::cos( fPercentBetweenColors*2*PI + PI * 2.0f / 3.0f ) * 0.5f + 0.5f,
std::cos( fPercentBetweenColors*2*PI + PI * 4.0f / 3.0f) * 0.5f + 0.5f,
fOriginalAlpha );
for( int i=1; i<NUM_DIFFUSE_COLORS; i++ )
m_current_with_effects.diffuse[i] = m_current_with_effects.diffuse[0];
break;
case wag:
2024-07-17 21:37:27 -07:00
m_current_with_effects.rotation += m_vEffectMagnitude * std::sin( fPercentThroughEffect * 2.0f * PI );
break;
case spin:
// nothing needs to be here
break;
case vibrate:
m_current_with_effects.pos.x += m_vEffectMagnitude.x * randomf(-1.0f, 1.0f) * GetZoom();
m_current_with_effects.pos.y += m_vEffectMagnitude.y * randomf(-1.0f, 1.0f) * GetZoom();
m_current_with_effects.pos.z += m_vEffectMagnitude.z * randomf(-1.0f, 1.0f) * GetZoom();
break;
case bounce:
{
2024-07-17 21:37:27 -07:00
float fPercentOffset = std::sin( fPercentThroughEffect*PI );
m_current_with_effects.pos += m_vEffectMagnitude * fPercentOffset;
}
break;
case bob:
{
2024-07-17 21:37:27 -07:00
float fPercentOffset = std::sin( fPercentThroughEffect*PI*2 );
m_current_with_effects.pos += m_vEffectMagnitude * fPercentOffset;
}
break;
case pulse:
{
float fMinZoom = m_vEffectMagnitude[0];
float fMaxZoom = m_vEffectMagnitude[1];
2024-07-17 21:37:27 -07:00
float fPercentOffset = std::sin( fPercentThroughEffect*PI );
float fZoom = SCALE( fPercentOffset, 0.f, 1.f, fMinZoom, fMaxZoom );
m_current_with_effects.scale *= fZoom;
// Use the color as a Vector3 to scale the effect for added control
RageColor c = SCALE( fPercentOffset, 0.f, 1.f, m_effectColor1, m_effectColor2 );
m_current_with_effects.scale.x *= c.r;
m_current_with_effects.scale.y *= c.g;
m_current_with_effects.scale.z *= c.b;
}
break;
default:
2012-12-27 16:59:35 -05:00
FAIL_M(ssprintf("Invalid effect: %i", m_Effect));
}
}
if( m_fBaseAlpha != 1 )
2013-05-28 21:46:15 -04:00
m_internalDiffuse.a *= m_fBaseAlpha;
if( m_internalDiffuse != RageColor(1, 1, 1, 1) )
{
if( m_pTempState != &m_current_with_effects )
{
m_pTempState = &m_current_with_effects;
m_current_with_effects = m_current;
}
for( int i=0; i<NUM_DIFFUSE_COLORS; i++ )
2013-05-28 21:46:15 -04:00
{
m_current_with_effects.diffuse[i] *= m_internalDiffuse;
2013-05-28 21:46:15 -04:00
}
}
2013-05-28 20:52:01 -04:00
if( m_internalGlow.a > 0 )
{
if( m_pTempState != &m_current_with_effects )
2013-05-28 20:52:01 -04:00
{
m_pTempState = &m_current_with_effects;
m_current_with_effects = m_current;
2013-05-28 20:52:01 -04:00
}
// Blend using Screen mode
m_current_with_effects.glow = m_current_with_effects.glow + m_internalGlow - m_internalGlow * m_current_with_effects.glow;
2013-05-28 20:52:01 -04:00
}
}
2024-05-25 03:39:46 -07:00
void Actor::BeginDraw()
{
2024-05-25 03:39:46 -07:00
DISPLAY->PushMatrix(); // Save the current transformation matrix
2013-05-28 20:52:01 -04:00
2024-05-25 03:39:46 -07:00
// Get the position of the actor
const float posX = m_pTempState->pos.x;
const float posY = m_pTempState->pos.y;
const float posZ = m_pTempState->pos.z;
if (posX != 0 || posY != 0 || posZ != 0)
{
RageMatrix m;
2024-05-25 03:39:46 -07:00
RageMatrixTranslate(&m, posX, posY, posZ);
DISPLAY->PreMultMatrix(m);
}
2024-05-25 03:39:46 -07:00
// Get the rotation of the actor
{
2024-05-25 03:39:46 -07:00
const float rotationX = m_pTempState->rotation.x + m_baseRotation.x;
const float rotationY = m_pTempState->rotation.y + m_baseRotation.y;
const float rotationZ = m_pTempState->rotation.z + m_baseRotation.z;
if (rotationX != 0 || rotationY != 0 || rotationZ != 0)
{
RageMatrix m;
2024-05-25 03:39:46 -07:00
RageMatrixRotationXYZ(&m, rotationX, rotationY, rotationZ);
DISPLAY->PreMultMatrix(m);
}
}
2024-05-25 03:39:46 -07:00
// Get the scale of the actor
{
2024-05-25 03:39:46 -07:00
const float scaleX = m_pTempState->scale.x * m_baseScale.x;
const float scaleY = m_pTempState->scale.y * m_baseScale.y;
const float scaleZ = m_pTempState->scale.z * m_baseScale.z;
2024-05-25 03:39:46 -07:00
if (scaleX != 1 || scaleY != 1 || scaleZ != 1)
{
RageMatrix m;
2024-05-25 03:39:46 -07:00
RageMatrixScale(&m, scaleX, scaleY, scaleZ);
DISPLAY->PreMultMatrix(m);
}
}
2024-05-25 03:39:46 -07:00
// Adjust the alignment of the actor
if (unlikely(m_fHorizAlign != 0.5f || m_fVertAlign != 0.5f))
{
2024-05-25 03:39:46 -07:00
float fX = SCALE(m_fHorizAlign, 0.0f, 1.0f, +m_size.x / 2.0f, -m_size.x / 2.0f);
float fY = SCALE(m_fVertAlign, 0.0f, 1.0f, +m_size.y / 2.0f, -m_size.y / 2.0f);
RageMatrix m;
2024-05-25 03:39:46 -07:00
RageMatrixTranslate(&m, fX, fY, 0);
DISPLAY->PreMultMatrix(m);
}
2024-05-25 03:39:46 -07:00
// Get the quaternion of the actor
const float quatX = m_pTempState->quat.x;
const float quatY = m_pTempState->quat.y;
const float quatZ = m_pTempState->quat.z;
const float quatW = m_pTempState->quat.w;
if (quatX != 0 || quatY != 0 || quatZ != 0 || quatW != 1)
{
RageMatrix mat;
2024-05-25 03:39:46 -07:00
RageMatrixFromQuat(&mat, m_pTempState->quat);
DISPLAY->MultMatrix(mat);
}
2024-05-25 03:39:46 -07:00
// Get the skew of the actor along the X-axis
const float skewX = m_pTempState->fSkewX;
if (skewX != 0)
{
2024-05-25 03:39:46 -07:00
DISPLAY->SkewX(skewX);
}
2024-05-25 03:39:46 -07:00
// Get the skew of the actor along the Y-axis
const float skewY = m_pTempState->fSkewY;
if (skewY != 0)
{
2024-05-25 03:39:46 -07:00
DISPLAY->SkewY(skewY);
}
2024-05-25 03:39:46 -07:00
// If the texture is not at the origin, translate the texture
if (m_texTranslate.x != 0 || m_texTranslate.y != 0)
2014-02-20 21:17:10 -06:00
{
DISPLAY->TexturePushMatrix();
2024-05-25 03:39:46 -07:00
DISPLAY->TextureTranslate(m_texTranslate.x, m_texTranslate.y);
2014-02-20 21:17:10 -06:00
}
}
void Actor::SetGlobalRenderStates()
{
// set Actor-defined render states
if( !g_bShowMasks.Get() || m_BlendMode != BLEND_NO_EFFECT )
DISPLAY->SetBlendMode( m_BlendMode );
DISPLAY->SetZWrite( m_bZWrite );
DISPLAY->SetZTestMode( m_ZTestMode );
// BLEND_NO_EFFECT is used to draw masks to the Z-buffer, which always wants
// Z-bias enabled.
if( m_fZBias == 0 && m_BlendMode == BLEND_NO_EFFECT )
DISPLAY->SetZBias( 1.0f );
else
DISPLAY->SetZBias( m_fZBias );
if( m_bClearZBuffer )
DISPLAY->ClearZBuffer();
DISPLAY->SetCullMode( m_CullMode );
}
void Actor::SetTextureRenderStates()
{
DISPLAY->SetTextureWrapping( TextureUnit_1, m_bTextureWrapping );
DISPLAY->SetTextureFiltering( TextureUnit_1, m_bTextureFiltering );
}
void Actor::EndDraw()
{
DISPLAY->PopMatrix();
2014-02-20 21:17:10 -06:00
if( m_texTranslate.x != 0 || m_texTranslate.y != 0 )
DISPLAY->TexturePopMatrix();
}
void Actor::CalcPercentThroughTween()
{
2024-05-25 03:39:46 -07:00
TweenState& TS = m_Tweens[0]->state;
TweenInfo& TI = m_Tweens[0]->info;
const float percent_through = 1 - (TI.m_fTimeLeftInTween / TI.m_fTweenTime);
// distort the percentage if appropriate
float percent_along = TI.m_pTween->Tween(percent_through);
TweenState::MakeWeightedAverage(m_current, m_start, TS, percent_along);
UpdatePercentThroughTween(percent_along);
}
2024-05-25 03:39:46 -07:00
void Actor::UpdateTweening(float fDeltaTime)
{
2024-05-25 03:39:46 -07:00
if (fDeltaTime < 0.0 && !m_Tweens.empty())
{
2024-05-25 03:39:46 -07:00
m_Tweens[0]->info.m_fTimeLeftInTween -= fDeltaTime;
CalcPercentThroughTween();
return;
}
2024-05-25 03:39:46 -07:00
while (!m_Tweens.empty() // something to do
&& fDeltaTime > 0) // something will change
{
// update current tween state
// earliest tween
2024-05-25 03:39:46 -07:00
auto& firstTween = m_Tweens[0];
TweenState& TS = firstTween->state;
TweenInfo& TI = firstTween->info;
bool bBeginning = TI.m_fTimeLeftInTween == TI.m_fTweenTime;
2024-05-25 03:39:46 -07:00
float fSecsToSubtract = std::min(TI.m_fTimeLeftInTween, fDeltaTime);
TI.m_fTimeLeftInTween -= fSecsToSubtract;
fDeltaTime -= fSecsToSubtract;
RString sCommand = TI.m_sCommandName;
2024-05-25 03:39:46 -07:00
if (bBeginning) // we are just beginning this tween
2014-04-14 21:29:19 -05:00
{
2024-05-25 03:39:46 -07:00
m_start = m_current; // set the start position
2014-04-14 21:29:19 -05:00
SetCurrentTweenStart();
}
2023-04-19 14:22:59 +02:00
2024-05-25 03:39:46 -07:00
if (TI.m_fTimeLeftInTween == 0) // Current tween is over. Stop.
{
m_current = TS;
// delete the head tween
2024-05-25 03:39:46 -07:00
delete firstTween;
m_Tweens.erase(m_Tweens.begin());
2014-04-14 21:29:19 -05:00
EraseHeadTween();
}
2024-05-25 03:39:46 -07:00
else // in the middle of tweening. Recalcute the current position.
{
CalcPercentThroughTween();
}
2024-05-25 03:39:46 -07:00
if (bBeginning)
{
// Execute the command in this tween (if any). Do this last, and don't
// access TI or TS after, since this may modify the tweening queue.
2024-05-25 03:39:46 -07:00
if (!sCommand.empty())
{
if (sCommand.Left(1) == "!")
2024-05-25 03:39:46 -07:00
MESSAGEMAN->Broadcast(sCommand.substr(1));
else
2024-05-25 03:39:46 -07:00
this->PlayCommand(sCommand);
}
}
}
}
bool Actor::IsFirstUpdate() const
{
return m_bFirstUpdate;
}
void Actor::Update( float fDeltaTime )
{
// LOG->Trace( "Actor::Update( %f )", fDeltaTime );
float rate = GameLoop::GetUpdateRate();
if (rate != 1 && !rate_scaling_enabled_) {
// Prevent divide by 0 when tab + tilde are both pressed.
if (rate != 0) {
fDeltaTime *= (1 / rate);
}
}
ASSERT_M( fDeltaTime >= 0, ssprintf("DeltaTime: %f",fDeltaTime) );
if( m_fHibernateSecondsLeft > 0 )
{
m_fHibernateSecondsLeft -= fDeltaTime;
if( m_fHibernateSecondsLeft > 0 )
{
return;
}
// Grab the leftover time.
fDeltaTime = -m_fHibernateSecondsLeft;
m_fHibernateSecondsLeft = 0;
}
2024-10-01 00:43:07 -07:00
for(size_t i= 0; i < m_WrapperStates.size(); ++i)
{
m_WrapperStates[i]->Update(fDeltaTime);
}
this->UpdateInternal( fDeltaTime );
}
static void generic_global_timer_update(float new_time, float& effect_delta_time, float& time_into_effect)
{
effect_delta_time= new_time - time_into_effect;
time_into_effect= new_time;
}
void Actor::UpdateInternal(float delta_time)
{
2024-05-25 03:39:46 -07:00
if (m_bFirstUpdate)
m_bFirstUpdate = false;
2024-05-25 03:39:46 -07:00
const float effectPeriod = GetEffectPeriod();
switch (m_EffectClock)
{
2024-05-25 03:39:46 -07:00
case CLOCK_TIMER:
m_fSecsIntoEffect += delta_time;
m_fEffectDelta = delta_time;
2024-05-28 23:18:30 -07:00
// Wrap the counter, so it doesn't increase indefinitely (causing loss
// of precision if a screen is left to sit for a day).
2024-05-25 03:39:46 -07:00
if (m_fSecsIntoEffect > effectPeriod)
{
m_fSecsIntoEffect -= effectPeriod;
}
break;
case CLOCK_TIMER_GLOBAL:
2024-09-08 23:13:50 -07:00
generic_global_timer_update(RageTimer::GetTimeSinceStartMicroseconds(), m_fEffectDelta, m_fSecsIntoEffect);
2024-05-25 03:39:46 -07:00
break;
case CLOCK_BGM_BEAT:
generic_global_timer_update(g_fCurrentBGMBeat, m_fEffectDelta, m_fSecsIntoEffect);
break;
case CLOCK_BGM_BEAT_PLAYER1:
generic_global_timer_update(g_vfCurrentBGMBeatPlayer[PLAYER_1], m_fEffectDelta, m_fSecsIntoEffect);
break;
case CLOCK_BGM_BEAT_PLAYER2:
generic_global_timer_update(g_vfCurrentBGMBeatPlayer[PLAYER_2], m_fEffectDelta, m_fSecsIntoEffect);
break;
case CLOCK_BGM_TIME:
generic_global_timer_update(g_fCurrentBGMTime, m_fEffectDelta, m_fSecsIntoEffect);
break;
case CLOCK_BGM_BEAT_NO_OFFSET:
generic_global_timer_update(g_fCurrentBGMBeatNoOffset, m_fEffectDelta, m_fSecsIntoEffect);
break;
case CLOCK_BGM_TIME_NO_OFFSET:
generic_global_timer_update(g_fCurrentBGMTimeNoOffset, m_fEffectDelta, m_fSecsIntoEffect);
break;
default:
if (m_EffectClock >= CLOCK_LIGHT_1 && m_EffectClock <= CLOCK_LIGHT_LAST)
{
int lightIndex = m_EffectClock - CLOCK_LIGHT_1;
generic_global_timer_update(g_fCabinetLights[lightIndex], m_fEffectDelta, m_fSecsIntoEffect);
}
break;
}
// update effect
2024-05-25 03:39:46 -07:00
switch (m_Effect)
{
2024-05-25 03:39:46 -07:00
case spin:
m_current.rotation += m_fEffectDelta * m_vEffectMagnitude;
wrap(m_current.rotation.x, 360);
wrap(m_current.rotation.y, 360);
wrap(m_current.rotation.z, 360);
break;
default: break;
}
2024-05-25 03:39:46 -07:00
if (m_tween_uses_effect_delta)
{
2024-05-25 03:39:46 -07:00
delta_time = m_fEffectDelta;
}
this->UpdateTweening(delta_time);
}
RString Actor::GetLineage() const
{
RString sPath;
2023-04-19 14:22:59 +02:00
if( m_pParent )
sPath = m_pParent->GetLineage() + '/';
sPath += ssprintf( "<type %s> %s", typeid(*this).name(), m_sName.c_str() );
return sPath;
}
void Actor::AddWrapperState()
{
ActorFrame* wrapper= new ActorFrame;
wrapper->InitState();
m_WrapperStates.push_back(wrapper);
}
2024-10-01 00:43:07 -07:00
void Actor::RemoveWrapperState(size_t i)
{
ASSERT(i < m_WrapperStates.size());
RageUtil::SafeDelete(m_WrapperStates[i]);
m_WrapperStates.erase(m_WrapperStates.begin()+i);
}
2024-10-01 00:43:07 -07:00
Actor* Actor::GetWrapperState(size_t i)
{
ASSERT(i < m_WrapperStates.size());
return m_WrapperStates[i];
}
void Actor::BeginTweening( float time, ITween *pTween )
{
ASSERT( time >= 0 );
2023-04-19 14:22:59 +02:00
// If the number of tweens to ever gets this large, there's probably an infinitely
// recursing ActorCommand.
if( m_Tweens.size() > 50 )
{
LuaHelpers::ReportScriptErrorFmt("Tween overflow: \"%s\"; infinitely recursing ActorCommand?", GetLineage().c_str());
2014-04-14 21:29:19 -05:00
this->FinishTweening();
}
// add a new TweenState to the tail, and initialize it
m_Tweens.push_back( new TweenStateAndInfo );
// latest
TweenState &TS = m_Tweens.back()->state;
TweenInfo &TI = m_Tweens.back()->info;
if( m_Tweens.size() >= 2 ) // if there was already a TS on the stack
{
// initialize the new TS from the last TS in the list
TS = m_Tweens[m_Tweens.size()-2]->state;
}
else
{
// This new TS is the only TS.
// Set our tween starting and ending values to the current position.
TS = m_current;
}
TI.m_pTween = pTween;
TI.m_fTweenTime = time;
TI.m_fTimeLeftInTween = time;
}
void Actor::BeginTweening( float time, TweenType tt )
{
2013-10-20 18:02:13 -04:00
ASSERT( time >= 0 );
ITween *pTween = ITween::CreateFromType( tt );
2014-04-14 21:29:19 -05:00
this->BeginTweening( time, pTween );
}
void Actor::StopTweening()
{
for( unsigned i = 0; i < m_Tweens.size(); ++i )
delete m_Tweens[i];
m_Tweens.clear();
}
void Actor::FinishTweening()
{
if( !m_Tweens.empty() )
m_current = DestTweenState();
2014-04-14 21:29:19 -05:00
this->StopTweening();
}
void Actor::HurryTweening( float factor )
{
for( unsigned i = 0; i < m_Tweens.size(); ++i )
{
m_Tweens[i]->info.m_fTimeLeftInTween *= factor;
m_Tweens[i]->info.m_fTweenTime *= factor;
}
}
void Actor::ScaleTo( const RectF &rect, StretchType st )
{
// width and height of rectangle
float rect_width = rect.GetWidth();
float rect_height = rect.GetHeight();
if( rect_width < 0 ) SetRotationY( 180 );
if( rect_height < 0 ) SetRotationX( 180 );
// zoom fActor needed to scale the Actor to fill the rectangle
2023-04-19 14:22:59 +02:00
float fNewZoomX = std::abs(rect_width / m_size.x);
float fNewZoomY = std::abs(rect_height / m_size.y);
float fNewZoom = 0.f;
2024-09-04 01:07:29 -07:00
switch (st)
{
2024-09-22 23:40:04 -07:00
case StretchType::kCover:
2024-09-04 01:07:29 -07:00
fNewZoom = fNewZoomX > fNewZoomY ? fNewZoomX : fNewZoomY; // use larger zoom
break;
2024-09-22 23:40:04 -07:00
case StretchType::kFitInside:
2024-09-04 01:07:29 -07:00
fNewZoom = fNewZoomX > fNewZoomY ? fNewZoomY : fNewZoomX; // use smaller zoom
break;
}
SetX( rect.left + rect_width * m_fHorizAlign );
SetY( rect.top + rect_height * m_fVertAlign );
SetZoom( fNewZoom );
}
void Actor::SetEffectClockString( const RString &s )
{
if (s.EqualsNoCase("timer")) this->SetEffectClock( CLOCK_TIMER );
else if(s.EqualsNoCase("timerglobal")) this->SetEffectClock( CLOCK_TIMER_GLOBAL );
else if(s.EqualsNoCase("beat")) this->SetEffectClock( CLOCK_BGM_BEAT );
else if(s.EqualsNoCase("music")) this->SetEffectClock( CLOCK_BGM_TIME );
else if(s.EqualsNoCase("bgm")) this->SetEffectClock( CLOCK_BGM_BEAT ); // compat, deprecated
else if(s.EqualsNoCase("musicnooffset"))this->SetEffectClock( CLOCK_BGM_TIME_NO_OFFSET );
else if(s.EqualsNoCase("beatnooffset")) this->SetEffectClock( CLOCK_BGM_BEAT_NO_OFFSET );
else
{
CabinetLight cl = StringToCabinetLight( s );
2012-12-27 16:59:35 -05:00
if( cl == CabinetLight_Invalid )
{
LuaHelpers::ReportScriptErrorFmt("String '%s' is not an effect clock string or the name of a cabinet light.", s.c_str());
}
else
{
this->SetEffectClock(static_cast<EffectClock>(Enum::to_integral(cl) + CLOCK_LIGHT_1));
}
}
}
void Actor::StretchTo( const RectF &r )
{
// width and height of rectangle
float width = r.GetWidth();
float height = r.GetHeight();
// center of the rectangle
float cx = r.left + width/2.0f;
float cy = r.top + height/2.0f;
// zoom fActor needed to scale the Actor to fill the rectangle
float fNewZoomX = width / m_size.x;
float fNewZoomY = height / m_size.y;
SetXY( cx, cy );
SetZoomX( fNewZoomX );
SetZoomY( fNewZoomY );
}
void Actor::RecalcEffectPeriod()
{
m_effect_period= m_effect_ramp_to_half + m_effect_hold_at_half +
m_effect_ramp_to_full + m_effect_hold_at_full + m_effect_hold_at_zero;
}
void Actor::SetEffectPeriod(float time)
{
ASSERT(time > 0);
m_effect_ramp_to_half= time/2;
m_effect_hold_at_half= 0;
m_effect_ramp_to_full= time/2;
m_effect_hold_at_full= 0;
m_effect_hold_at_zero= 0;
RecalcEffectPeriod();
}
bool Actor::SetEffectTiming(float ramp_toh, float at_half, float ramp_tof, float at_full, float at_zero, RString& err)
{
// No negative timings
2024-05-25 03:39:46 -07:00
if (ramp_toh < 0 || at_half < 0 || ramp_tof < 0 || at_full < 0 || at_zero < 0)
{
2024-05-25 03:39:46 -07:00
err = ssprintf("Effect timings (%f,%f,%f,%f,%f) must not be negative;", ramp_toh, at_half, ramp_tof, at_zero, at_full);
return false;
}
// and at least one positive timing.
2024-05-25 03:39:46 -07:00
if (ramp_toh <= 0 && at_half <= 0 && ramp_tof <= 0 && at_full <= 0 && at_zero <= 0)
{
2024-05-25 03:39:46 -07:00
err = "Effect timings (0,0,0,0,0) must not all be zero;";
return false;
}
2024-05-25 03:39:46 -07:00
m_effect_ramp_to_half = ramp_toh;
m_effect_hold_at_half = at_half;
m_effect_ramp_to_full = ramp_tof;
m_effect_hold_at_full = at_full;
m_effect_hold_at_zero = at_zero;
RecalcEffectPeriod();
return true;
}
bool Actor::SetEffectHoldAtFull(float haf, RString& err)
{
return SetEffectTiming(m_effect_ramp_to_half, m_effect_hold_at_half,
m_effect_ramp_to_full, haf, m_effect_hold_at_zero, err);
}
// effect "macros"
void Actor::ResetEffectTimeIfDifferent(Effect new_effect)
{
if(m_Effect != new_effect)
{
m_Effect= new_effect;
m_fSecsIntoEffect = 0;
}
}
void Actor::SetEffectDiffuseBlink( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(diffuse_blink);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectDiffuseShift( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(diffuse_shift);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectDiffuseRamp( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(diffuse_ramp);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectGlowBlink( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(glow_blink);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectGlowShift( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(glow_shift);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectGlowRamp( float fEffectPeriodSeconds, RageColor c1, RageColor c2 )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(glow_ramp);
SetEffectPeriod( fEffectPeriodSeconds );
m_effectColor1 = c1;
m_effectColor2 = c2;
}
void Actor::SetEffectRainbow( float fEffectPeriodSeconds )
{
2013-10-20 16:59:24 -04:00
ASSERT( fEffectPeriodSeconds > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(rainbow);
SetEffectPeriod( fEffectPeriodSeconds );
}
void Actor::SetEffectWag( float fPeriod, RageVector3 vect )
{
2013-10-20 16:59:24 -04:00
ASSERT( fPeriod > 0 );
// todo: account for SSC_FUTURES -aj
ResetEffectTimeIfDifferent(wag);
SetEffectPeriod( fPeriod );
m_vEffectMagnitude = vect;
}
void Actor::SetEffectBounce( float fPeriod, RageVector3 vect )
{
2013-10-20 16:59:24 -04:00
ASSERT( fPeriod > 0 );
// todo: account for SSC_FUTURES -aj
m_Effect = bounce;
SetEffectPeriod( fPeriod );
m_vEffectMagnitude = vect;
m_fSecsIntoEffect = 0;
}
void Actor::SetEffectBob( float fPeriod, RageVector3 vect )
{
2013-10-20 16:59:24 -04:00
ASSERT( fPeriod > 0 );
// todo: account for SSC_FUTURES -aj
if( m_Effect!=bob || GetEffectPeriod() != fPeriod )
{
m_Effect = bob;
SetEffectPeriod( fPeriod );
m_fSecsIntoEffect = 0;
}
m_vEffectMagnitude = vect;
}
void Actor::SetEffectSpin( RageVector3 vect )
{
// todo: account for SSC_FUTURES -aj
m_Effect = spin;
m_vEffectMagnitude = vect;
}
void Actor::SetEffectVibrate( RageVector3 vect )
{
// todo: account for SSC_FUTURES -aj
m_Effect = vibrate;
m_vEffectMagnitude = vect;
}
void Actor::SetEffectPulse( float fPeriod, float fMinZoom, float fMaxZoom )
{
2013-10-20 16:59:24 -04:00
ASSERT( fPeriod > 0 );
// todo: account for SSC_FUTURES -aj
m_Effect = pulse;
SetEffectPeriod( fPeriod );
m_vEffectMagnitude[0] = fMinZoom;
m_vEffectMagnitude[1] = fMaxZoom;
}
void Actor::AddRotationH( float rot )
{
RageQuatMultiply( &DestTweenState().quat, DestTweenState().quat, RageQuatFromH(rot) );
}
void Actor::AddRotationP( float rot )
{
RageQuatMultiply( &DestTweenState().quat, DestTweenState().quat, RageQuatFromP(rot) );
}
void Actor::AddRotationR( float rot )
{
RageQuatMultiply( &DestTweenState().quat, DestTweenState().quat, RageQuatFromR(rot) );
}
void Actor::RunCommands( const LuaReference& cmds, const LuaReference *pParamTable )
{
2013-10-20 23:05:26 -04:00
if( !cmds.IsSet() || cmds.IsNil() )
{
LuaHelpers::ReportScriptErrorFmt("RunCommands: commands for %s are unset or nil", GetLineage().c_str());
2013-10-20 23:05:26 -04:00
return;
}
Lua *L = LUA->Get();
// function
cmds.PushSelf( L );
2013-10-13 14:35:48 -04:00
if( lua_isnil(L, -1) )
{
LuaHelpers::ReportScriptErrorFmt("RunCommands: Error compiling commands for %s", GetLineage().c_str());
2013-10-13 14:35:48 -04:00
LUA->Release(L);
return;
}
// 1st parameter
this->PushSelf( L );
// 2nd parameter
2019-06-22 12:35:38 -07:00
if( pParamTable == nullptr )
lua_pushnil( L );
else
pParamTable->PushSelf( L );
// call function with 2 arguments and 0 results
RString Error= "Error playing command:";
LuaHelpers::RunScriptOnStack(L, Error, 2, 0, true);
LUA->Release(L);
}
float Actor::GetTweenTimeLeft() const
{
float tot = 0;
tot += m_fHibernateSecondsLeft;
for( unsigned i=0; i<m_Tweens.size(); ++i )
tot += m_Tweens[i]->info.m_fTimeLeftInTween;
return tot;
}
/* This is a hack to change all tween states while leaving existing tweens alone.
*
* Hmm. Most commands actually act on a TweenStateAndInfo, not the Actor itself.
* Conceptually, it wouldn't be hard to give TweenState a presence in Lua, so
* we can simply say eg. "for x in states(Actor) do x.SetDiffuseColor(c) end".
* However, we'd then have to give every TweenState a userdata in Lua while it's
* being manipulated, which would add overhead ... */
void Actor::SetGlobalDiffuseColor( RageColor c )
{
for( int i=0; i<NUM_DIFFUSE_COLORS; i++ ) // color, not alpha
{
for( unsigned ts = 0; ts < m_Tweens.size(); ++ts )
{
2023-04-19 14:22:59 +02:00
m_Tweens[ts]->state.diffuse[i].r = c.r;
m_Tweens[ts]->state.diffuse[i].g = c.g;
m_Tweens[ts]->state.diffuse[i].b = c.b;
}
m_current.diffuse[i].r = c.r;
m_current.diffuse[i].g = c.g;
m_current.diffuse[i].b = c.b;
m_start.diffuse[i].r = c.r;
m_start.diffuse[i].g = c.g;
m_start.diffuse[i].b = c.b;
}
}
void Actor::SetDiffuseColor( RageColor c )
{
for( int i=0; i<NUM_DIFFUSE_COLORS; i++ )
{
DestTweenState().diffuse[i].r = c.r;
DestTweenState().diffuse[i].g = c.g;
DestTweenState().diffuse[i].b = c.b;
}
}
void Actor::TweenState::Init()
{
pos = RageVector3( 0, 0, 0 );
rotation = RageVector3( 0, 0, 0 );
quat = RageVector4( 0, 0, 0, 1 );
scale = RageVector3( 1, 1, 1 );
fSkewX = 0;
fSkewY = 0;
crop = RectF( 0,0,0,0 );
fade = RectF( 0,0,0,0 );
for( int i=0; i<NUM_DIFFUSE_COLORS; i++ )
diffuse[i] = RageColor( 1, 1, 1, 1 );
glow = RageColor( 1, 1, 1, 0 );
aux = 0;
}
bool Actor::TweenState::operator==(const TweenState& other) const {
if (pos != other.pos) return false;
if (rotation != other.rotation) return false;
if (quat != other.quat) return false;
if (scale != other.scale) return false;
if (fSkewX != other.fSkewX) return false;
if (fSkewY != other.fSkewY) return false;
if (crop != other.crop) return false;
if (fade != other.fade) return false;
for (unsigned i = 0; i < ARRAYLEN(diffuse); i++) {
if (diffuse[i] != other.diffuse[i]) return false;
}
if (glow != other.glow) return false;
if (aux != other.aux) return false;
return true;
}
void Actor::TweenState::MakeWeightedAverage( TweenState& average_out, const TweenState& ts1, const TweenState& ts2, float fPercentBetween )
{
average_out.pos = lerp( fPercentBetween, ts1.pos, ts2.pos );
average_out.scale = lerp( fPercentBetween, ts1.scale, ts2.scale );
average_out.rotation = lerp( fPercentBetween, ts1.rotation, ts2.rotation );
RageQuatSlerp( &average_out.quat, ts1.quat, ts2.quat, fPercentBetween );
average_out.fSkewX = lerp( fPercentBetween, ts1.fSkewX, ts2.fSkewX );
average_out.fSkewY = lerp( fPercentBetween, ts1.fSkewY, ts2.fSkewY );
average_out.crop.left = lerp( fPercentBetween, ts1.crop.left, ts2.crop.left );
average_out.crop.top = lerp( fPercentBetween, ts1.crop.top, ts2.crop.top );
average_out.crop.right = lerp( fPercentBetween, ts1.crop.right, ts2.crop.right );
average_out.crop.bottom = lerp( fPercentBetween, ts1.crop.bottom, ts2.crop.bottom );
average_out.fade.left = lerp( fPercentBetween, ts1.fade.left, ts2.fade.left );
average_out.fade.top = lerp( fPercentBetween, ts1.fade.top, ts2.fade.top );
average_out.fade.right = lerp( fPercentBetween, ts1.fade.right, ts2.fade.right );
average_out.fade.bottom = lerp( fPercentBetween, ts1.fade.bottom, ts2.fade.bottom );
for( int i=0; i<NUM_DIFFUSE_COLORS; ++i )
average_out.diffuse[i] = lerp( fPercentBetween, ts1.diffuse[i], ts2.diffuse[i] );
average_out.glow = lerp( fPercentBetween, ts1.glow, ts2.glow );
average_out.aux = lerp( fPercentBetween, ts1.aux, ts2.aux );
}
void Actor::Sleep( float time )
{
2013-10-20 18:02:13 -04:00
ASSERT( time >= 0 );
BeginTweening( time, TWEEN_LINEAR );
2023-04-19 14:22:59 +02:00
BeginTweening( 0, TWEEN_LINEAR );
}
void Actor::QueueCommand( const RString& sCommandName )
{
BeginTweening( 0, TWEEN_LINEAR );
TweenInfo &TI = m_Tweens.back()->info;
TI.m_sCommandName = sCommandName;
}
void Actor::QueueMessage( const RString& sMessageName )
{
// Hack: use "!" as a marker to broadcast a command, instead of playing a
// command, so we don't have to add yet another element to every tween
// state for this rarely-used command.
BeginTweening( 0, TWEEN_LINEAR );
TweenInfo &TI = m_Tweens.back()->info;
TI.m_sCommandName = "!" + sMessageName;
}
void Actor::AddCommand( const RString &sCmdName, apActorCommands apac, bool warn )
{
if( HasCommand(sCmdName) && warn)
{
RString sWarning = GetLineage()+"'s command '"+sCmdName+"' defined twice";
LuaHelpers::ReportScriptError(sWarning, "COMMAND_DEFINED_TWICE");
}
RString sMessage;
if( GetMessageNameFromCommandName(sCmdName, sMessage) )
{
SubscribeToMessage( sMessage );
m_mapNameToCommands[sMessage] = apac; // sCmdName w/o "Message" at the end
}
else
{
m_mapNameToCommands[sCmdName] = apac;
}
}
bool Actor::HasCommand( const RString &sCmdName ) const
{
2019-06-22 12:35:38 -07:00
return GetCommand(sCmdName) != nullptr;
}
const apActorCommands *Actor::GetCommand( const RString &sCommandName ) const
{
std::map<RString, apActorCommands>::const_iterator it = m_mapNameToCommands.find( sCommandName );
if( it == m_mapNameToCommands.end() )
2019-06-22 12:35:38 -07:00
return nullptr;
return &it->second;
}
void Actor::HandleMessage( const Message &msg )
{
PlayCommandNoRecurse( msg );
}
void Actor::PlayCommandNoRecurse( const Message &msg )
{
const apActorCommands *pCmd = GetCommand( msg.GetName() );
2019-06-22 12:35:38 -07:00
if(pCmd != nullptr && (*pCmd)->IsSet() && !(*pCmd)->IsNil())
{
RunCommands( *pCmd, &msg.GetParamTable() );
}
}
void Actor::PushContext( lua_State *L )
{
// self.ctx should already exist
m_pLuaInstance->PushSelf( L );
lua_getfield( L, -1, "ctx" );
lua_remove( L, -2 );
}
void Actor::SetParent( Actor *pParent )
{
m_pParent = pParent;
Lua *L = LUA->Get();
int iTop = lua_gettop( L );
this->PushContext( L );
lua_pushstring( L, "__index" );
pParent->PushContext( L );
lua_settable( L, -3 );
lua_settop( L, iTop );
LUA->Release( L );
}
Actor::TweenInfo::TweenInfo()
{
2019-06-22 12:35:38 -07:00
m_pTween = nullptr;
}
Actor::TweenInfo::~TweenInfo()
{
delete m_pTween;
}
Actor::TweenInfo::TweenInfo( const TweenInfo &cpy )
{
2019-06-22 12:35:38 -07:00
m_pTween = nullptr;
*this = cpy;
}
Actor::TweenInfo &Actor::TweenInfo::operator=( const TweenInfo &rhs )
{
delete m_pTween;
2019-06-22 12:35:38 -07:00
m_pTween = (rhs.m_pTween? rhs.m_pTween->Copy():nullptr);
m_fTimeLeftInTween = rhs.m_fTimeLeftInTween;
m_fTweenTime = rhs.m_fTweenTime;
m_sCommandName = rhs.m_sCommandName;
return *this;
}
// lua start
#include "LuaBinding.h"
2023-04-19 14:22:59 +02:00
/** @brief Allow Lua to have access to the Actor. */
class LunaActor : public Luna<Actor>
{
public:
static int name( T* p, lua_State *L ) { p->SetName(SArg(1)); COMMON_RETURN_SELF; }
2013-10-20 18:02:13 -04:00
static int sleep( T* p, lua_State *L )
{
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: sleep(%f): time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
p->Sleep(fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
static int linear( T* p, lua_State *L )
{
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: linear(%f): tween time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
p->BeginTweening(fTime, TWEEN_LINEAR);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
static int accelerate( T* p, lua_State *L )
{
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: accelerate(%f): tween time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
p->BeginTweening(fTime, TWEEN_ACCELERATE);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
static int decelerate( T* p, lua_State *L )
{
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: decelerate(%f): tween time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
p->BeginTweening(fTime, TWEEN_DECELERATE);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
static int spring( T* p, lua_State *L )
{
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: spring(%f): tween time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
p->BeginTweening(fTime, TWEEN_SPRING);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
static int tween( T* p, lua_State *L )
{
2013-10-20 18:02:13 -04:00
float fTime = FArg(1);
if (fTime < 0)
{
LuaHelpers::ReportScriptErrorFmt("Lua: tween(%f): tween time must not be negative", fTime);
COMMON_RETURN_SELF;
2013-10-20 18:02:13 -04:00
}
ITween *pTween = ITween::CreateFromStack( L, 2 );
2019-06-22 12:35:38 -07:00
if(pTween != nullptr)
{
p->BeginTweening(fTime, pTween);
}
COMMON_RETURN_SELF;
}
static int stoptweening( T* p, lua_State *L ) { p->StopTweening(); COMMON_RETURN_SELF; }
static int finishtweening( T* p, lua_State *L ) { p->FinishTweening(); COMMON_RETURN_SELF; }
static int hurrytweening( T* p, lua_State *L )
{
float time= FArg(1);
if(time < 0.0f)
{
luaL_error(L, "Tweening hurry factor cannot be negative. %f", time);
}
p->HurryTweening(time);
COMMON_RETURN_SELF;
}
static int GetTweenTimeLeft( T* p, lua_State *L ) { lua_pushnumber( L, p->GetTweenTimeLeft() ); return 1; }
static int x( T* p, lua_State *L ) { p->SetX(FArg(1)); COMMON_RETURN_SELF; }
static int y( T* p, lua_State *L ) { p->SetY(FArg(1)); COMMON_RETURN_SELF; }
static int z( T* p, lua_State *L ) { p->SetZ(FArg(1)); COMMON_RETURN_SELF; }
static int xy( T* p, lua_State *L ) { p->SetXY(FArg(1),FArg(2)); COMMON_RETURN_SELF; }
static int addx( T* p, lua_State *L ) { p->AddX(FArg(1)); COMMON_RETURN_SELF; }
static int addy( T* p, lua_State *L ) { p->AddY(FArg(1)); COMMON_RETURN_SELF; }
static int addz( T* p, lua_State *L ) { p->AddZ(FArg(1)); COMMON_RETURN_SELF; }
static int zoom( T* p, lua_State *L ) { p->SetZoom(FArg(1)); COMMON_RETURN_SELF; }
static int zoomx( T* p, lua_State *L ) { p->SetZoomX(FArg(1)); COMMON_RETURN_SELF; }
static int zoomy( T* p, lua_State *L ) { p->SetZoomY(FArg(1)); COMMON_RETURN_SELF; }
static int zoomz( T* p, lua_State *L ) { p->SetZoomZ(FArg(1)); COMMON_RETURN_SELF; }
static int zoomto( T* p, lua_State *L ) { p->ZoomTo(FArg(1), FArg(2)); COMMON_RETURN_SELF; }
static int zoomtowidth( T* p, lua_State *L ) { p->ZoomToWidth(FArg(1)); COMMON_RETURN_SELF; }
static int zoomtoheight( T* p, lua_State *L ) { p->ZoomToHeight(FArg(1)); COMMON_RETURN_SELF; }
static int setsize( T* p, lua_State *L ) { p->SetWidth(FArg(1)); p->SetHeight(FArg(2)); COMMON_RETURN_SELF; }
static int SetWidth( T* p, lua_State *L ) { p->SetWidth(FArg(1)); COMMON_RETURN_SELF; }
static int SetHeight( T* p, lua_State *L ) { p->SetHeight(FArg(1)); COMMON_RETURN_SELF; }
static int basealpha( T* p, lua_State *L ) { p->SetBaseAlpha(FArg(1)); COMMON_RETURN_SELF; }
static int basezoom( T* p, lua_State *L ) { p->SetBaseZoom(FArg(1)); COMMON_RETURN_SELF; }
static int basezoomx( T* p, lua_State *L ) { p->SetBaseZoomX(FArg(1)); COMMON_RETURN_SELF; }
static int basezoomy( T* p, lua_State *L ) { p->SetBaseZoomY(FArg(1)); COMMON_RETURN_SELF; }
static int basezoomz( T* p, lua_State *L ) { p->SetBaseZoomZ(FArg(1)); COMMON_RETURN_SELF; }
static int stretchto( T* p, lua_State *L ) { p->StretchTo( RectF(FArg(1),FArg(2),FArg(3),FArg(4)) ); COMMON_RETURN_SELF; }
static int cropleft( T* p, lua_State *L ) { p->SetCropLeft(FArg(1)); COMMON_RETURN_SELF; }
static int croptop( T* p, lua_State *L ) { p->SetCropTop(FArg(1)); COMMON_RETURN_SELF; }
static int cropright( T* p, lua_State *L ) { p->SetCropRight(FArg(1)); COMMON_RETURN_SELF; }
static int cropbottom( T* p, lua_State *L ) { p->SetCropBottom(FArg(1)); COMMON_RETURN_SELF; }
static int fadeleft( T* p, lua_State *L ) { p->SetFadeLeft(FArg(1)); COMMON_RETURN_SELF; }
static int fadetop( T* p, lua_State *L ) { p->SetFadeTop(FArg(1)); COMMON_RETURN_SELF; }
static int faderight( T* p, lua_State *L ) { p->SetFadeRight(FArg(1)); COMMON_RETURN_SELF; }
static int fadebottom( T* p, lua_State *L ) { p->SetFadeBottom(FArg(1)); COMMON_RETURN_SELF; }
static int diffuse( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuse( c ); COMMON_RETURN_SELF; }
static int diffuseupperleft( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseUpperLeft( c ); COMMON_RETURN_SELF; }
static int diffuseupperright( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseUpperRight( c ); COMMON_RETURN_SELF; }
static int diffuselowerleft( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseLowerLeft( c ); COMMON_RETURN_SELF; }
static int diffuselowerright( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseLowerRight( c ); COMMON_RETURN_SELF; }
static int diffuseleftedge( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseLeftEdge( c ); COMMON_RETURN_SELF; }
static int diffuserightedge( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseRightEdge( c ); COMMON_RETURN_SELF; }
static int diffusetopedge( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseTopEdge( c ); COMMON_RETURN_SELF; }
static int diffusebottomedge( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseBottomEdge( c ); COMMON_RETURN_SELF; }
static int diffusealpha( T* p, lua_State *L ) { p->SetDiffuseAlpha(FArg(1)); COMMON_RETURN_SELF; }
static int diffusecolor( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetDiffuseColor( c ); COMMON_RETURN_SELF; }
static int glow( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetGlow( c ); COMMON_RETURN_SELF; }
static int aux( T* p, lua_State *L ) { p->SetAux( FArg(1) ); COMMON_RETURN_SELF; }
static int getaux( T* p, lua_State *L ) { lua_pushnumber( L, p->GetAux() ); return 1; }
static int rotationx( T* p, lua_State *L ) { p->SetRotationX(FArg(1)); COMMON_RETURN_SELF; }
static int rotationy( T* p, lua_State *L ) { p->SetRotationY(FArg(1)); COMMON_RETURN_SELF; }
static int rotationz( T* p, lua_State *L ) { p->SetRotationZ(FArg(1)); COMMON_RETURN_SELF; }
static int addrotationx( T* p, lua_State *L ) { p->AddRotationX(FArg(1)); COMMON_RETURN_SELF; }
static int addrotationy( T* p, lua_State *L ) { p->AddRotationY(FArg(1)); COMMON_RETURN_SELF; }
static int addrotationz( T* p, lua_State *L ) { p->AddRotationZ(FArg(1)); COMMON_RETURN_SELF; }
static int getrotation( T* p, lua_State *L ) { lua_pushnumber(L, p->GetRotationX()); lua_pushnumber(L, p->GetRotationY()); lua_pushnumber(L, p->GetRotationZ()); return 3; }
static int baserotationx( T* p, lua_State *L ) { p->SetBaseRotationX(FArg(1)); COMMON_RETURN_SELF; }
static int baserotationy( T* p, lua_State *L ) { p->SetBaseRotationY(FArg(1)); COMMON_RETURN_SELF; }
static int baserotationz( T* p, lua_State *L ) { p->SetBaseRotationZ(FArg(1)); COMMON_RETURN_SELF; }
static int skewx( T* p, lua_State *L ) { p->SetSkewX(FArg(1)); COMMON_RETURN_SELF; }
static int skewy( T* p, lua_State *L ) { p->SetSkewY(FArg(1)); COMMON_RETURN_SELF; }
static int heading( T* p, lua_State *L ) { p->AddRotationH(FArg(1)); COMMON_RETURN_SELF; }
static int pitch( T* p, lua_State *L ) { p->AddRotationP(FArg(1)); COMMON_RETURN_SELF; }
static int roll( T* p, lua_State *L ) { p->AddRotationR(FArg(1)); COMMON_RETURN_SELF; }
static int shadowlength( T* p, lua_State *L ) { p->SetShadowLength(FArg(1)); COMMON_RETURN_SELF; }
static int shadowlengthx( T* p, lua_State *L ) { p->SetShadowLengthX(FArg(1)); COMMON_RETURN_SELF; }
static int shadowlengthy( T* p, lua_State *L ) { p->SetShadowLengthY(FArg(1)); COMMON_RETURN_SELF; }
static int shadowcolor( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetShadowColor( c ); COMMON_RETURN_SELF; }
static int horizalign( T* p, lua_State *L ) { p->SetHorizAlign(Enum::Check<HorizAlign>(L, 1)); COMMON_RETURN_SELF; }
static int vertalign( T* p, lua_State *L ) { p->SetVertAlign(Enum::Check<VertAlign>(L, 1)); COMMON_RETURN_SELF; }
static int halign( T* p, lua_State *L ) { p->SetHorizAlign(FArg(1)); COMMON_RETURN_SELF; }
static int valign( T* p, lua_State *L ) { p->SetVertAlign(FArg(1)); COMMON_RETURN_SELF; }
static int diffuseblink( T* p, lua_State *L ) { p->SetEffectDiffuseBlink(1.0f, RageColor(0.5f,0.5f,0.5f,0.5f), RageColor(1,1,1,1)); COMMON_RETURN_SELF; }
static int diffuseshift( T* p, lua_State *L ) { p->SetEffectDiffuseShift(1.0f, RageColor(0,0,0,1), RageColor(1,1,1,1)); COMMON_RETURN_SELF; }
static int diffuseramp( T* p, lua_State *L ) { p->SetEffectDiffuseRamp(1.0f, RageColor(0,0,0,1), RageColor(1,1,1,1)); COMMON_RETURN_SELF; }
static int glowblink( T* p, lua_State *L ) { p->SetEffectGlowBlink(1.0f, RageColor(1,1,1,0.2f), RageColor(1,1,1,0.8f)); COMMON_RETURN_SELF; }
static int glowshift( T* p, lua_State *L ) { p->SetEffectGlowShift(1.0f, RageColor(1,1,1,0.2f), RageColor(1,1,1,0.8f)); COMMON_RETURN_SELF; }
static int glowramp( T* p, lua_State *L ) { p->SetEffectGlowRamp(1.0f, RageColor(1,1,1,0.2f), RageColor(1,1,1,0.8f)); COMMON_RETURN_SELF; }
static int rainbow( T* p, lua_State *L ) { p->SetEffectRainbow(2.0f); COMMON_RETURN_SELF; }
static int wag( T* p, lua_State *L ) { p->SetEffectWag(2.0f, RageVector3(0,0,20)); COMMON_RETURN_SELF; }
static int bounce( T* p, lua_State *L ) { p->SetEffectBounce(2.0f, RageVector3(0,20,0)); COMMON_RETURN_SELF; }
static int bob( T* p, lua_State *L ) { p->SetEffectBob(2.0f, RageVector3(0,20,0)); COMMON_RETURN_SELF; }
static int pulse( T* p, lua_State *L ) { p->SetEffectPulse(2.0f, 0.5f, 1.0f); COMMON_RETURN_SELF; }
static int spin( T* p, lua_State *L ) { p->SetEffectSpin(RageVector3(0,0,180)); COMMON_RETURN_SELF; }
static int vibrate( T* p, lua_State *L ) { p->SetEffectVibrate(RageVector3(10,10,10)); COMMON_RETURN_SELF; }
static int stopeffect( T* p, lua_State *L ) { p->StopEffect(); COMMON_RETURN_SELF; }
static int effectcolor1( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetEffectColor1( c ); COMMON_RETURN_SELF; }
static int effectcolor2( T* p, lua_State *L ) { RageColor c; c.FromStackCompat( L, 1 ); p->SetEffectColor2( c ); COMMON_RETURN_SELF; }
2013-10-20 16:59:24 -04:00
static int effectperiod( T* p, lua_State *L )
{
float fPeriod = FArg(1);
if (fPeriod <= 0)
{
LuaHelpers::ReportScriptErrorFmt("Effect period (%f) must be positive; ignoring", fPeriod);
COMMON_RETURN_SELF;
2013-10-20 16:59:24 -04:00
}
p->SetEffectPeriod(FArg(1));
COMMON_RETURN_SELF;
2013-10-20 16:59:24 -04:00
}
static int effecttiming( T* p, lua_State *L )
{
float rth= FArg(1);
float hah= FArg(2);
float rtf= FArg(3);
float haz= FArg(4);
// Compatibility: hold_at_full is optional.
float haf= 0;
if(lua_isnumber(L, 5))
{
haf= FArg(5);
}
RString err;
if(!p->SetEffectTiming(rth, hah, rtf, haf, haz, err))
{
luaL_error(L, err.c_str());
}
COMMON_RETURN_SELF;
}
static int effect_hold_at_full(T* p, lua_State*L)
{
RString err;
if(!p->SetEffectHoldAtFull(FArg(1), err))
{
luaL_error(L, err.c_str());
}
COMMON_RETURN_SELF;
}
static int effectoffset( T* p, lua_State *L ) { p->SetEffectOffset(FArg(1)); COMMON_RETURN_SELF; }
static int effectclock( T* p, lua_State *L ) { p->SetEffectClockString(SArg(1)); COMMON_RETURN_SELF; }
static int effectmagnitude( T* p, lua_State *L ) { p->SetEffectMagnitude( RageVector3(FArg(1),FArg(2),FArg(3)) ); COMMON_RETURN_SELF; }
static int geteffectmagnitude( T* p, lua_State *L ) { RageVector3 v = p->GetEffectMagnitude(); lua_pushnumber(L, v[0]); lua_pushnumber(L, v[1]); lua_pushnumber(L, v[2]); return 3; }
GETTER_SETTER_BOOL_METHOD(tween_uses_effect_delta);
static int scaletocover( T* p, lua_State *L ) { p->ScaleToCover( RectF(FArg(1), FArg(2), FArg(3), FArg(4)) ); COMMON_RETURN_SELF; }
static int scaletofit( T* p, lua_State *L ) { p->ScaleToFitInside( RectF(FArg(1), FArg(2), FArg(3), FArg(4)) ); COMMON_RETURN_SELF; }
static int animate( T* p, lua_State *L ) { p->EnableAnimation(BIArg(1)); COMMON_RETURN_SELF; }
static int play( T* p, lua_State *L ) { p->EnableAnimation(true); COMMON_RETURN_SELF; }
static int pause( T* p, lua_State *L ) { p->EnableAnimation(false); COMMON_RETURN_SELF; }
static int setstate( T* p, lua_State *L ) { p->SetState(IArg(1)); COMMON_RETURN_SELF; }
static int GetNumStates( T* p, lua_State *L ) { LuaHelpers::Push( L, p->GetNumStates() ); return 1; }
static int texturetranslate( T* p, lua_State *L ) { p->SetTextureTranslate(FArg(1),FArg(2)); COMMON_RETURN_SELF; }
static int texturewrapping( T* p, lua_State *L ) { p->SetTextureWrapping(BIArg(1)); COMMON_RETURN_SELF; }
static int SetTextureFiltering( T* p, lua_State *L ) { p->SetTextureFiltering(BArg(1)); COMMON_RETURN_SELF; }
static int blend( T* p, lua_State *L ) { p->SetBlendMode( Enum::Check<BlendMode>(L, 1) ); COMMON_RETURN_SELF; }
static int zbuffer( T* p, lua_State *L ) { p->SetUseZBuffer(BIArg(1)); COMMON_RETURN_SELF; }
static int ztest( T* p, lua_State *L ) { p->SetZTestMode((BIArg(1))?ZTEST_WRITE_ON_PASS:ZTEST_OFF); COMMON_RETURN_SELF; }
static int ztestmode( T* p, lua_State *L ) { p->SetZTestMode( Enum::Check<ZTestMode>(L, 1) ); COMMON_RETURN_SELF; }
static int zwrite( T* p, lua_State *L ) { p->SetZWrite(BIArg(1)); COMMON_RETURN_SELF; }
static int zbias( T* p, lua_State *L ) { p->SetZBias(FArg(1)); COMMON_RETURN_SELF; }
static int clearzbuffer( T* p, lua_State *L ) { p->SetClearZBuffer(BIArg(1)); COMMON_RETURN_SELF; }
static int backfacecull( T* p, lua_State *L ) { p->SetCullMode((BIArg(1)) ? CULL_BACK : CULL_NONE); COMMON_RETURN_SELF; }
static int cullmode( T* p, lua_State *L ) { p->SetCullMode( Enum::Check<CullMode>(L, 1)); COMMON_RETURN_SELF; }
static int visible( T* p, lua_State *L ) { p->SetVisible(BIArg(1)); COMMON_RETURN_SELF; }
static int hibernate( T* p, lua_State *L ) { p->SetHibernate(FArg(1)); COMMON_RETURN_SELF; }
static int draworder( T* p, lua_State *L ) { p->SetDrawOrder(IArg(1)); COMMON_RETURN_SELF; }
static int playcommand( T* p, lua_State *L )
{
if( !lua_istable(L, 2) && !lua_isnoneornil(L, 2) )
luaL_typerror( L, 2, "table or nil" );
LuaReference ParamTable;
lua_pushvalue( L, 2 );
ParamTable.SetFromStack( L );
Message msg( SArg(1), ParamTable );
p->HandleMessage( msg );
COMMON_RETURN_SELF;
}
static int queuecommand( T* p, lua_State *L ) { p->QueueCommand(SArg(1)); COMMON_RETURN_SELF; }
static int queuemessage( T* p, lua_State *L ) { p->QueueMessage(SArg(1)); COMMON_RETURN_SELF; }
static int addcommand( T* p, lua_State *L )
{
LuaReference *pRef = new LuaReference;
pRef->SetFromStack( L );
p->AddCommand( SArg(1), apActorCommands(pRef) );
COMMON_RETURN_SELF;
}
static int GetCommand( T* p, lua_State *L )
{
const apActorCommands *pCommand = p->GetCommand(SArg(1));
2019-06-22 12:35:38 -07:00
if( pCommand == nullptr )
lua_pushnil( L );
else
(*pCommand)->PushSelf(L);
return 1;
}
static int RunCommandsRecursively( T* p, lua_State *L )
{
luaL_checktype( L, 1, LUA_TFUNCTION );
if( !lua_istable(L, 2) && !lua_isnoneornil(L, 2) )
luaL_typerror( L, 2, "table or nil" );
LuaReference ref;
lua_pushvalue( L, 1 );
ref.SetFromStack( L );
LuaReference ParamTable;
lua_pushvalue( L, 2 );
ParamTable.SetFromStack( L );
p->RunCommandsRecursively( ref, &ParamTable );
COMMON_RETURN_SELF;
}
static int GetX( T* p, lua_State *L ) { lua_pushnumber( L, p->GetX() ); return 1; }
static int GetY( T* p, lua_State *L ) { lua_pushnumber( L, p->GetY() ); return 1; }
static int GetZ( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZ() ); return 1; }
2014-07-04 02:03:05 -06:00
static int GetDestX( T* p, lua_State *L ) { lua_pushnumber( L, p->GetDestX() ); return 1; }
static int GetDestY( T* p, lua_State *L ) { lua_pushnumber( L, p->GetDestY() ); return 1; }
static int GetDestZ( T* p, lua_State *L ) { lua_pushnumber( L, p->GetDestZ() ); return 1; }
static int GetWidth( T* p, lua_State *L ) { lua_pushnumber( L, p->GetUnzoomedWidth() ); return 1; }
static int GetHeight( T* p, lua_State *L ) { lua_pushnumber( L, p->GetUnzoomedHeight() ); return 1; }
static int GetZoomedWidth( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoomedWidth() ); return 1; }
static int GetZoomedHeight( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoomedHeight() ); return 1; }
static int GetZoom( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoom() ); return 1; }
static int GetZoomX( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoomX() ); return 1; }
static int GetZoomY( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoomY() ); return 1; }
static int GetZoomZ( T* p, lua_State *L ) { lua_pushnumber( L, p->GetZoomZ() ); return 1; }
static int GetBaseZoomX( T* p, lua_State *L ) { lua_pushnumber( L, p->GetBaseZoomX() ); return 1; }
static int GetBaseZoomY( T* p, lua_State *L ) { lua_pushnumber( L, p->GetBaseZoomY() ); return 1; }
static int GetBaseZoomZ( T* p, lua_State *L ) { lua_pushnumber( L, p->GetBaseZoomZ() ); return 1; }
static int GetRotationX( T* p, lua_State *L ) { lua_pushnumber( L, p->GetRotationX() ); return 1; }
static int GetRotationY( T* p, lua_State *L ) { lua_pushnumber( L, p->GetRotationY() ); return 1; }
static int GetRotationZ( T* p, lua_State *L ) { lua_pushnumber( L, p->GetRotationZ() ); return 1; }
static int GetSecsIntoEffect( T* p, lua_State *L ) { lua_pushnumber( L, p->GetSecsIntoEffect() ); return 1; }
static int GetEffectDelta( T* p, lua_State *L ) { lua_pushnumber( L, p->GetEffectDelta() ); return 1; }
DEFINE_METHOD( GetDiffuse, GetDiffuse() )
DEFINE_METHOD( GetGlow, GetGlow() )
static int GetDiffuseAlpha( T* p, lua_State *L ) { lua_pushnumber( L, p->GetDiffuseAlpha() ); return 1; }
static int GetVisible( T* p, lua_State *L ) { lua_pushboolean( L, p->GetVisible() ); return 1; }
static int GetHAlign( T* p, lua_State *L ) { lua_pushnumber( L, p->GetHorizAlign() ); return 1; }
static int GetVAlign( T* p, lua_State *L ) { lua_pushnumber( L, p->GetVertAlign() ); return 1; }
static int GetName( T* p, lua_State *L ) { lua_pushstring( L, p->GetName().c_str() ); return 1; }
static int GetParent( T* p, lua_State *L )
{
Actor *pParent = p->GetParent();
2019-06-22 12:35:38 -07:00
if( pParent == nullptr )
lua_pushnil( L );
else
pParent->PushSelf(L);
return 1;
}
static int GetFakeParent(T* p, lua_State *L)
{
Actor* fake= p->GetFakeParent();
2019-06-22 12:35:38 -07:00
if(fake == nullptr)
{
lua_pushnil(L);
}
else
{
fake->PushSelf(L);
}
return 1;
}
static int SetFakeParent(T* p, lua_State* L)
{
if(lua_isnoneornil(L, 1))
{
2019-06-22 12:35:38 -07:00
p->SetFakeParent(nullptr);
}
else
{
Actor* fake= Luna<Actor>::check(L, 1);
p->SetFakeParent(fake);
}
COMMON_RETURN_SELF;
}
static int AddWrapperState(T* p, lua_State* L)
{
p->AddWrapperState();
p->GetWrapperState(p->GetNumWrapperStates()-1)->PushSelf(L);
return 1;
}
2024-10-01 00:43:07 -07:00
static size_t get_state_index(T* p, lua_State* L, int stack_index)
{
// Lua is one indexed.
int i= IArg(stack_index)-1;
2024-10-01 00:43:07 -07:00
const size_t si= static_cast<size_t>(i);
if(i < 0 || si >= p->GetNumWrapperStates())
{
luaL_error(L, "%d is not a valid wrapper state index.", i+1);
}
return si;
}
static int RemoveWrapperState(T* p, lua_State* L)
{
2024-10-01 00:43:07 -07:00
size_t si= get_state_index(p, L, 1);
p->RemoveWrapperState(si);
COMMON_RETURN_SELF;
}
static int GetNumWrapperStates(T* p, lua_State* L)
{
lua_pushnumber(L, p->GetNumWrapperStates());
return 1;
}
static int GetWrapperState(T* p, lua_State* L)
{
2024-10-01 00:43:07 -07:00
size_t si= get_state_index(p, L, 1);
p->GetWrapperState(si)->PushSelf(L);
return 1;
}
static int Draw( T* p, lua_State *L )
{
LUA->YieldLua();
p->Draw();
LUA->UnyieldLua();
COMMON_RETURN_SELF;
}
static int SetRateScalingEnabled( T* p, lua_State *L ) { p->SetRateScalingEnabled(BArg(1)); COMMON_RETURN_SELF; }
static int GetRateScalingEnabled( T* p, lua_State *L ) { lua_pushboolean( L, p->GetRateScalingEnabled() ); return 1; }
LunaActor()
{
ADD_METHOD( name );
ADD_METHOD( sleep );
ADD_METHOD( linear );
ADD_METHOD( accelerate );
ADD_METHOD( decelerate );
ADD_METHOD( spring );
ADD_METHOD( tween );
ADD_METHOD( stoptweening );
ADD_METHOD( finishtweening );
ADD_METHOD( hurrytweening );
ADD_METHOD( GetTweenTimeLeft );
ADD_METHOD( x );
ADD_METHOD( y );
ADD_METHOD( z );
ADD_METHOD( xy );
ADD_METHOD( addx );
ADD_METHOD( addy );
ADD_METHOD( addz );
ADD_METHOD( zoom );
ADD_METHOD( zoomx );
ADD_METHOD( zoomy );
ADD_METHOD( zoomz );
ADD_METHOD( zoomto );
ADD_METHOD( zoomtowidth );
ADD_METHOD( zoomtoheight );
ADD_METHOD( setsize );
ADD_METHOD( SetWidth );
ADD_METHOD( SetHeight );
ADD_METHOD( basealpha );
ADD_METHOD( basezoom );
ADD_METHOD( basezoomx );
ADD_METHOD( basezoomy );
ADD_METHOD( basezoomz );
ADD_METHOD( stretchto );
ADD_METHOD( cropleft );
ADD_METHOD( croptop );
ADD_METHOD( cropright );
ADD_METHOD( cropbottom );
ADD_METHOD( fadeleft );
ADD_METHOD( fadetop );
ADD_METHOD( faderight );
ADD_METHOD( fadebottom );
ADD_METHOD( diffuse );
ADD_METHOD( diffuseupperleft );
ADD_METHOD( diffuseupperright );
ADD_METHOD( diffuselowerleft );
ADD_METHOD( diffuselowerright );
ADD_METHOD( diffuseleftedge );
ADD_METHOD( diffuserightedge );
ADD_METHOD( diffusetopedge );
ADD_METHOD( diffusebottomedge );
ADD_METHOD( diffusealpha );
ADD_METHOD( diffusecolor );
ADD_METHOD( glow );
ADD_METHOD( aux );
ADD_METHOD( getaux );
ADD_METHOD( rotationx );
ADD_METHOD( rotationy );
ADD_METHOD( rotationz );
ADD_METHOD( addrotationx );
ADD_METHOD( addrotationy );
ADD_METHOD( addrotationz );
ADD_METHOD( getrotation );
ADD_METHOD( baserotationx );
ADD_METHOD( baserotationy );
ADD_METHOD( baserotationz );
ADD_METHOD( skewx );
ADD_METHOD( skewy );
ADD_METHOD( heading );
ADD_METHOD( pitch );
ADD_METHOD( roll );
ADD_METHOD( shadowlength );
ADD_METHOD( shadowlengthx );
ADD_METHOD( shadowlengthy );
ADD_METHOD( shadowcolor );
ADD_METHOD( horizalign );
ADD_METHOD( vertalign );
ADD_METHOD( halign );
ADD_METHOD( valign );
ADD_METHOD( diffuseblink );
ADD_METHOD( diffuseshift );
ADD_METHOD( diffuseramp );
ADD_METHOD( glowblink );
ADD_METHOD( glowshift );
ADD_METHOD( glowramp );
ADD_METHOD( rainbow );
ADD_METHOD( wag );
ADD_METHOD( bounce );
ADD_METHOD( bob );
ADD_METHOD( pulse );
ADD_METHOD( spin );
ADD_METHOD( vibrate );
ADD_METHOD( stopeffect );
ADD_METHOD( effectcolor1 );
ADD_METHOD( effectcolor2 );
ADD_METHOD( effectperiod );
ADD_METHOD( effecttiming );
ADD_METHOD(effect_hold_at_full);
ADD_METHOD( effectoffset );
ADD_METHOD( effectclock );
ADD_METHOD( effectmagnitude );
ADD_METHOD( geteffectmagnitude );
ADD_GET_SET_METHODS(tween_uses_effect_delta);
ADD_METHOD( scaletocover );
ADD_METHOD( scaletofit );
ADD_METHOD( animate );
ADD_METHOD( play );
ADD_METHOD( pause );
ADD_METHOD( setstate );
ADD_METHOD( GetNumStates );
2014-02-20 21:17:10 -06:00
ADD_METHOD( texturetranslate );
ADD_METHOD( texturewrapping );
ADD_METHOD( SetTextureFiltering );
ADD_METHOD( blend );
ADD_METHOD( zbuffer );
ADD_METHOD( ztest );
ADD_METHOD( ztestmode );
ADD_METHOD( zwrite );
ADD_METHOD( zbias );
ADD_METHOD( clearzbuffer );
ADD_METHOD( backfacecull );
ADD_METHOD( cullmode );
ADD_METHOD( visible );
ADD_METHOD( hibernate );
ADD_METHOD( draworder );
ADD_METHOD( playcommand );
ADD_METHOD( queuecommand );
ADD_METHOD( queuemessage );
ADD_METHOD( addcommand );
ADD_METHOD( GetCommand );
ADD_METHOD( RunCommandsRecursively );
ADD_METHOD( GetX );
ADD_METHOD( GetY );
ADD_METHOD( GetZ );
2014-07-04 02:03:05 -06:00
ADD_METHOD( GetDestX );
ADD_METHOD( GetDestY );
ADD_METHOD( GetDestZ );
ADD_METHOD( GetWidth );
ADD_METHOD( GetHeight );
ADD_METHOD( GetZoomedWidth );
ADD_METHOD( GetZoomedHeight );
ADD_METHOD( GetZoom );
ADD_METHOD( GetZoomX );
ADD_METHOD( GetZoomY );
ADD_METHOD( GetZoomZ );
ADD_METHOD( GetRotationX );
ADD_METHOD( GetRotationY );
ADD_METHOD( GetRotationZ );
ADD_METHOD( GetBaseZoomX );
ADD_METHOD( GetBaseZoomY );
ADD_METHOD( GetBaseZoomZ );
ADD_METHOD( GetSecsIntoEffect );
ADD_METHOD( GetEffectDelta );
ADD_METHOD( GetDiffuse );
ADD_METHOD( GetDiffuseAlpha );
ADD_METHOD( GetGlow );
ADD_METHOD( GetVisible );
ADD_METHOD( GetHAlign );
ADD_METHOD( GetVAlign );
ADD_METHOD( GetName );
ADD_METHOD( GetParent );
ADD_METHOD( GetFakeParent );
ADD_METHOD( SetFakeParent );
ADD_METHOD( AddWrapperState );
ADD_METHOD( RemoveWrapperState );
ADD_METHOD( GetNumWrapperStates );
ADD_METHOD( GetWrapperState );
ADD_METHOD( SetRateScalingEnabled );
ADD_METHOD( GetRateScalingEnabled );
ADD_METHOD( Draw );
}
};
LUA_REGISTER_INSTANCED_BASE_CLASS( Actor )
// lua end
/*
* (c) 2001-2004 Chris Danford
* All rights reserved.
2023-04-19 14:22:59 +02:00
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, and/or sell copies of the Software, and to permit persons to
* whom the Software is furnished to do so, provided that the above
* copyright notice(s) and this permission notice appear in all copies of
* the Software and that both the above copyright notice(s) and this
* permission notice appear in supporting documentation.
2023-04-19 14:22:59 +02:00
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF
* THIRD PARTY RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS
* INCLUDED IN THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT
* OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
* OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
* OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
* PERFORMANCE OF THIS SOFTWARE.
*/