2021-07-24 21:11:47 -07:00

397 lines
12 KiB
C++

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "graphicgroup.h"
#include "gamegraphic.h"
#include "gameuidefinition.h"
#include "animdata.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
BEGIN_DMXELEMENT_UNPACK ( CGraphicGroup )
DMXELEMENT_UNPACK_FIELD_UTLSTRING( "name", "", m_pName )
DMXELEMENT_UNPACK_FIELD( "center", "0 0", Vector2D, m_Geometry.m_Center )
DMXELEMENT_UNPACK_FIELD( "scale", "1 1", Vector2D, m_Geometry.m_Scale )
DMXELEMENT_UNPACK_FIELD( "rotation", "0", float, m_Geometry.m_Rotation )
DMXELEMENT_UNPACK_FIELD( "maintainaspectratio", "0", bool, m_Geometry.m_bMaintainAspectRatio )
DMXELEMENT_UNPACK_FIELD( "sublayertype", "0", int, m_Geometry.m_Sublayer )
DMXELEMENT_UNPACK_FIELD( "visible", "1", bool, m_Geometry.m_bVisible )
DMXELEMENT_UNPACK_FIELD( "initialstate", "-1", int, m_CurrentState )
DMXELEMENT_UNPACK_FIELD( "horizgradient", "0", bool, m_Geometry.m_bHorizontalGradient )
DMXELEMENT_UNPACK_FIELD( "color", "255 255 255 255", Color, m_Geometry.m_Color )
DMXELEMENT_UNPACK_FIELD( "topcolor", "255 255 255 255", Color, m_Geometry.m_TopColor )
DMXELEMENT_UNPACK_FIELD( "bottomcolor", "255 255 255 255", Color, m_Geometry.m_BottomColor )
END_DMXELEMENT_UNPACK( CGraphicGroup, s_GraphicGroupUnpack )
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
CGraphicGroup::CGraphicGroup()
{
m_ResultantColor.r = 0;
m_ResultantColor.g = 0;
m_ResultantColor.b = 0;
m_ResultantColor.a = 0;
}
CGraphicGroup::~CGraphicGroup()
{
}
//-----------------------------------------------------------------------------
// Load data from file.
//-----------------------------------------------------------------------------
bool CGraphicGroup::Unserialize( CDmxElement *pElement, CUtlDict< CGameGraphic *, int > &unserializedGraphicMapping )
{
pElement->UnpackIntoStructure( this, s_GraphicGroupUnpack );
CDmxAttribute *pGroupElements = pElement->GetAttribute( "groupElements" );
if ( !pGroupElements || pGroupElements->GetType() != AT_ELEMENT_ARRAY )
{
return false;
}
const CUtlVector< CDmxElement * > &elements = pGroupElements->GetArray< CDmxElement * >( );
int nCount = elements.Count();
for ( int i = 0; i < nCount; ++i )
{
// Find the element in the map.
char pBuf[255];
UniqueIdToString( elements[i]->GetId(), pBuf, 255 );
int index = unserializedGraphicMapping.Find( pBuf );
Assert( unserializedGraphicMapping.IsValidIndex( index ) );
CGameGraphic *pGraphic = unserializedGraphicMapping.Element(index);
pGraphic->SetGroup( this );
m_MemberList.AddToTail( pGraphic );
}
// ANIMSTATES
CDmxAttribute *pImageAnims = pElement->GetAttribute( "imageanims" );
if ( !pImageAnims || pImageAnims->GetType() != AT_ELEMENT_ARRAY )
{
return false;
}
const CUtlVector< CDmxElement * > &imageanims = pImageAnims->GetArray< CDmxElement * >( );
nCount = imageanims.Count();
for ( int i = 0; i < nCount; ++i )
{
CAnimData *pAnimData = new CAnimData;
if ( !pAnimData->Unserialize( imageanims[i] ) )
{
delete pAnimData;
return false;
}
m_Anims.AddToTail( pAnimData );
}
char pBuf[255];
UniqueIdToString( pElement->GetId(), pBuf, 255 );
unserializedGraphicMapping.Insert( pBuf, this );
// Ok the initial state is 0, which is (usually ) default.
// default could be aliased to another state though so if it is fix the initial state here.
// default might also not be the state that is 0 so this sets the graphic's initial
// state to be the default one.
SetState( "default" );
return true;
}
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
void CGraphicGroup::UpdateGeometry()
{
if ( m_CurrentState == -1 )
return;
DmeTime_t flAnimTime = GetAnimationTimePassed();
// Update color
m_Anims[ m_CurrentState ]->m_ColorAnim.GetValue( flAnimTime, &m_Geometry.m_Color );
// Update center location
m_Anims[ m_CurrentState ]->m_CenterPosAnim.GetValue( flAnimTime, &m_Geometry.m_Center );
// Update scale
m_Anims[ m_CurrentState ]->m_ScaleAnim.GetValue( flAnimTime, &m_Geometry.m_Scale );
// Update rotation
m_Anims[ m_CurrentState ]->m_RotationAnim.GetValue( flAnimTime, &m_Geometry.m_Rotation );
for ( int i = 0; i < m_MemberList.Count(); ++i )
{
m_MemberList[i]->UpdateGeometry();
}
}
//-----------------------------------------------------------------------------
// Store the resultant color for groups so kids can just get that.
//-----------------------------------------------------------------------------
void CGraphicGroup::UpdateRenderData( color32 parentColor )
{
if ( !m_Geometry.m_bVisible )
return;
m_ResultantColor.r = (int)( (float)m_Geometry.m_Color.r * (float)(parentColor.r/255.0) );
m_ResultantColor.g = (int)( (float)m_Geometry.m_Color.g * (float)(parentColor.g/255.0) );
m_ResultantColor.b = (int)( (float)m_Geometry.m_Color.b * (float)(parentColor.b/255.0) );
m_ResultantColor.a = (int)( (float)m_Geometry.m_Color.a * (float)(parentColor.a/255.0) );
// Update any children that are groups
// Graphic members are not done because the colors are calculated when we get the render data out.
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
if ( m_MemberList[i]->IsGroup() )
{
CGraphicGroup *pGroup = (CGraphicGroup *)m_MemberList[i];
pGroup->UpdateRenderData( parentColor );
}
}
}
//-----------------------------------------------------------------------------
// Populate lists for rendering
//-----------------------------------------------------------------------------
void CGraphicGroup::UpdateRenderTransforms( const StageRenderInfo_t &stageRenderInfo )
{
m_Geometry.UpdateRenderTransforms( stageRenderInfo, GetGroup() );
// Create a matrix that ensures no aspect ratio changes.
// Update positions relative to the center, texture coords, and vertex colors
// If the group maintains aspect ratio it will already have handled this in its transform update.
Vector2D center;
// If this is the case we transform the center to screen coords first.
// Then take into account any size scaling in the scalemat
matrix3x4_t screenScalemat;
SetScaleMatrix( stageRenderInfo.parentScale.x, stageRenderInfo.parentScale.y, 1, screenScalemat );
Vector centerVec( m_Geometry.m_Center.x, m_Geometry.m_Center.y, 0 );
Vector centerInScreen;
VectorTransform( centerVec, screenScalemat, centerInScreen );
center.x = centerInScreen.x;
center.y = centerInScreen.y;
matrix3x4_t transmat;
Vector position( center.x, center.y, 0 );
SetIdentityMatrix( transmat );
PositionMatrix( position, transmat );
matrix3x4_t scalemat;
SetScaleMatrix( m_Geometry.m_Scale.x, m_Geometry.m_Scale.y, 1, scalemat );
matrix3x4_t rotmat;
Vector axis( 0, 0, 1 );
MatrixBuildRotationAboutAxis( axis, m_Geometry.m_Rotation, rotmat );
matrix3x4_t temp;
MatrixMultiply( rotmat, scalemat, temp );
matrix3x4_t rawToLocal;
MatrixMultiply( transmat, temp, rawToLocal );
matrix3x4_t groupToScreen;
// Use the matrix that doesn't contain any scale changes if we should
m_pGroup->GetRenderTransform( groupToScreen, true );
MatrixMultiply( groupToScreen, rawToLocal, m_RelToScreenHoldAspectRatio );
// Update all children
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
m_MemberList[i]->UpdateRenderTransforms( stageRenderInfo );
}
}
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
void CGraphicGroup::AddToGroup( CGameGraphic *pGraphic )
{
m_MemberList.AddToTail( pGraphic );
}
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
void CGraphicGroup::RemoveFromGroup( CGameGraphic *pGraphic )
{
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
// TODO, what if the graphic is a group?
if ( m_MemberList[i] == pGraphic )
{
m_MemberList.Remove( i );
return;
}
}
}
//-----------------------------------------------------------------------------
// Returns true if any graphic in this group has the state.
//-----------------------------------------------------------------------------
bool CGraphicGroup::HasState( const char *pStateName )
{
if ( CGameGraphic::HasState( pStateName ) )
return true;
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
if ( m_MemberList[i]->HasState( pStateName ) )
return true;
}
return false;
}
//-----------------------------------------------------------------------------
// Set the state of all members to this state
//-----------------------------------------------------------------------------
void CGraphicGroup::SetState( const char *pStateName )
{
CGameGraphic::SetState( pStateName );
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
m_MemberList[i]->SetState( pStateName );
}
}
//-----------------------------------------------------------------------------
// Start playing animations
//-----------------------------------------------------------------------------
void CGraphicGroup::StartPlaying()
{
CGameGraphic::StartPlaying();
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
m_MemberList[i]->StartPlaying();
}
}
//-----------------------------------------------------------------------------
// Stop playing animations
//-----------------------------------------------------------------------------
void CGraphicGroup::StopPlaying()
{
CGameGraphic::StopPlaying();
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
m_MemberList[i]->StopPlaying();
}
}
//-----------------------------------------------------------------------------
// Move all members to the next available state
// Note this could put all of them into different states
//-----------------------------------------------------------------------------
void CGraphicGroup::AdvanceState()
{
CGameGraphic::AdvanceState();
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
m_MemberList[i]->AdvanceState();
}
}
//-----------------------------------------------------------------------------
// Return the first member of this group that can have keyfocus.
//-----------------------------------------------------------------------------
CHitArea *CGraphicGroup::GetKeyFocusRequestGraphic()
{
for ( int i = 0; i < m_MemberList.Count(); i++ )
{
if ( m_MemberList[i]->CanAcceptInput() )
{
return ( CHitArea * )m_MemberList[i];
}
}
return NULL;
}
//-----------------------------------------------------------------------------
// Does this graphic own a graphic with this name?
//-----------------------------------------------------------------------------
CGameGraphic *CGraphicGroup::FindGraphicByName( const char *pName ) const
{
int nGraphicCount = m_MemberList.Count();
for ( int i = 0; i < nGraphicCount; ++i )
{
CGameGraphic *pMember = m_MemberList[i];
if ( pMember->IsGraphicNamed( pName ) )
{
// Match.
return pMember;
}
}
return NULL;
}
//-----------------------------------------------------------------------------
// Group visibility affects all children.
//-----------------------------------------------------------------------------
void CGraphicGroup::SetVisible( bool bVisible )
{
CGameGraphic::SetVisible( bVisible );
int nGraphicCount = m_MemberList.Count();
for ( int i = 0; i < nGraphicCount; ++i )
{
CGameGraphic *pMember = m_MemberList[i];
pMember->SetVisible( bVisible );
}
}
//-----------------------------------------------------------------------------
// Return the appropriate render transform.
// m_RelToScreenHoldAspectRatio is calculated for a stage aspect ratio that has not changed.
//-----------------------------------------------------------------------------
void CGraphicGroup::GetRenderTransform( matrix3x4_t &relToScreen, bool bMaintainAspectRatio ) const
{
if ( bMaintainAspectRatio )
{
relToScreen = m_RelToScreenHoldAspectRatio;
}
else
{
relToScreen = m_Geometry.m_RenderToScreen;
}
}
//-----------------------------------------------------------------------------
// If any parent of this group should maintain aspect ratio, this group should.
//-----------------------------------------------------------------------------
bool CGraphicGroup::MaintainAspectRatio() const
{
if ( m_pGroup && !m_Geometry.m_bMaintainAspectRatio )
{
return m_pGroup->MaintainAspectRatio();
}
return m_Geometry.m_bMaintainAspectRatio;
}