//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Decorator class to make a DME renderable as a MDL
//													 
//===========================================================================//

#ifndef DMEMDLRENDERABLE_H
#define DMEMDLRENDERABLE_H
#ifdef _WIN32
#pragma once
#endif

#include "toolutils/dmerenderable.h"
#include "movieobjects/dmemdl.h"
#include "movieobjects/dmetransform.h"
#include "datacache/imdlcache.h"
#include "mathlib/mathlib.h"
#include "datamodel/dmehandle.h"
#include "toolutils/enginetools_int.h"
#include "materialsystem/imaterialsystemhardwareconfig.h"
#include "tier3/tier3.h"


//-----------------------------------------------------------------------------
// Deals with the base implementation for turning a Dme into a renderable
//-----------------------------------------------------------------------------
template < class T >
class CDmeMdlRenderable : public CDmeRenderable< T >
{
	DEFINE_UNINSTANCEABLE_ELEMENT( CDmeMdlRenderable, CDmeRenderable< T > );

// IClientUnknown implementation.
public:
	virtual int					GetBody();
	virtual int					GetSkin();
	virtual int					DrawModel( int flags );
	virtual void				GetRenderBounds( Vector& mins, Vector& maxs );

	void SetModelName( const char *pMDLName );

protected:
	CDmeMDL *GetMDL() { return m_hMDL; }

private:
	void SetUpLighting( const Vector &vecCenter );

	CDmeHandle<CDmeMDL> m_hMDL;
	CDmeHandle<CDmeTransform> m_hTransform;
};


//-----------------------------------------------------------------------------
// Construction, destruction
//-----------------------------------------------------------------------------
template < class T >
void CDmeMdlRenderable<T>::OnConstruction() 
{
	m_hMDL = CreateElement<CDmeMDL>( "MDLRenderable", GetFileId() );
	m_hTransform = CreateElement<CDmeTransform>( "MDLTransform", GetFileId() );
}

template < class T >
void CDmeMdlRenderable<T>::OnDestruction() 
{
	g_pDataModel->DestroyElement( m_hMDL );
	g_pDataModel->DestroyElement( m_hTransform );
}

template < class T >
int	CDmeMdlRenderable<T>::GetBody()
{
	return m_hMDL->m_nBody;
}

template < class T >
int	CDmeMdlRenderable<T>::GetSkin()
{
	return m_hMDL->m_nSkin;
}

template < class T >
int	CDmeMdlRenderable<T>::DrawModel( int flags )
{
	matrix3x4_t mat;
	AngleMatrix( GetRenderAngles(), GetRenderOrigin(), mat );
	m_hTransform->SetTransform( mat );
	m_hMDL->m_flTime = Plat_FloatTime();
	SetUpLighting( GetRenderOrigin() );
	bool bIsDrawingInEngine = m_hMDL->IsDrawingInEngine();
	m_hMDL->DrawInEngine( true );
	m_hMDL->Draw( mat );
	m_hMDL->DrawInEngine( bIsDrawingInEngine );
	return 1;
}

template < class T >
void CDmeMdlRenderable<T>::GetRenderBounds( Vector& mins, Vector& maxs )
{
	m_hMDL->GetBoundingBox( &mins, &maxs );
}

template < class T >
void CDmeMdlRenderable<T>::SetModelName( const char *pMDLRelativePath )
{
	if ( pMDLRelativePath )
	{
		MDLHandle_t hMdl = g_pMDLCache->FindMDL( pMDLRelativePath );
		m_hMDL->SetMDL( hMdl );
	}
	else
	{
		m_hMDL->SetMDL( MDLHANDLE_INVALID );
	}
}

//-----------------------------------------------------------------------------
// Set up lighting conditions
//-----------------------------------------------------------------------------
template < class T >
void CDmeMdlRenderable<T>::SetUpLighting( const Vector &vecCenter )
{
	// Set up lighting conditions
	Vector vecAmbient[6];
	Vector4D vecAmbient4D[6];
	LightDesc_t desc[2];
	int nLightCount = enginetools->GetLightingConditions( vecCenter, vecAmbient, 2, desc );
	int nMaxLights = g_pMaterialSystemHardwareConfig->MaxNumLights();
	if( nLightCount > nMaxLights )
	{
		nLightCount = nMaxLights;
	}

	int i;
	for( i = 0; i < 6; i++ )
	{
		VectorCopy( vecAmbient[i], vecAmbient4D[i].AsVector3D() );
		vecAmbient4D[i][3] = 1.0f;
	}
	CMatRenderContextPtr pRenderContext( g_pMaterialSystem );
	pRenderContext->SetAmbientLightCube( vecAmbient4D );

	for( i = 0; i < nLightCount; i++ )
	{
		LightDesc_t *pLight = &desc[i];
		pLight->m_Flags = 0;
		if( pLight->m_Attenuation0 != 0.0f )
		{
			pLight->m_Flags |= LIGHTTYPE_OPTIMIZATIONFLAGS_HAS_ATTENUATION0;
		}
		if( pLight->m_Attenuation1 != 0.0f )
		{
			pLight->m_Flags |= LIGHTTYPE_OPTIMIZATIONFLAGS_HAS_ATTENUATION1;
		}
		if( pLight->m_Attenuation2 != 0.0f )
		{
			pLight->m_Flags |= LIGHTTYPE_OPTIMIZATIONFLAGS_HAS_ATTENUATION2;
		}

		pRenderContext->SetLight( i, desc[i] );
	}

	for( ; i < nMaxLights; i++ )
	{
		LightDesc_t disableDesc;
		disableDesc.m_Type = MATERIAL_LIGHT_DISABLE;
		pRenderContext->SetLight( i, disableDesc );
	}
}


#endif // DMEMDLRENDERABLE_H