|
|
//========= Copyright � 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//=============================================================================//
#include "tier1/strtools.h"
#include "macro_texture.h"
#include "bsplib.h"
#include "cmdlib.h"
#include "vtf/vtf.h"
#include "tier1/utldict.h"
#include "tier1/utlbuffer.h"
#include "bitmap/imageformat.h"
class CMacroTextureData { public: int m_Width, m_Height; CUtlMemory<unsigned char> m_ImageData; };
CMacroTextureData *g_pGlobalMacroTextureData = NULL;
// Which macro texture each map face uses.
static CUtlDict<CMacroTextureData*, int> g_MacroTextureLookup; // Stores a list of unique macro textures.
static CUtlVector<CMacroTextureData*> g_FaceMacroTextures; // Which macro texture each face wants to use.
static Vector g_MacroWorldMins, g_MacroWorldMaxs;
CMacroTextureData* FindMacroTexture( const char *pFilename ) { int index = g_MacroTextureLookup.Find( pFilename ); if ( g_MacroTextureLookup.IsValidIndex( index ) ) return g_MacroTextureLookup[index]; else return NULL; }
CMacroTextureData* LoadMacroTextureFile( const char *pFilename ) { FileHandle_t hFile = g_pFileSystem->Open( pFilename, "rb" ); if ( hFile == FILESYSTEM_INVALID_HANDLE ) return NULL;
// Read the file in.
CUtlVector<char> tempData; tempData.SetSize( g_pFileSystem->Size( hFile ) ); g_pFileSystem->Read( tempData.Base(), tempData.Count(), hFile ); g_pFileSystem->Close( hFile ); // Now feed the data into a CUtlBuffer (great...)
CUtlBuffer buf; buf.Put( tempData.Base(), tempData.Count() );
// Now make a texture out of it.
IVTFTexture *pTex = CreateVTFTexture(); if ( !pTex->Unserialize( buf ) ) Error( "IVTFTexture::Unserialize( %s ) failed.", pFilename );
pTex->ConvertImageFormat( IMAGE_FORMAT_RGBA8888, false ); // Get it in a format we like.
// Now convert to a CMacroTextureData.
CMacroTextureData *pData = new CMacroTextureData; pData->m_Width = pTex->Width(); pData->m_Height = pTex->Height(); pData->m_ImageData.EnsureCapacity( pData->m_Width * pData->m_Height * 4 ); memcpy( pData->m_ImageData.Base(), pTex->ImageData(), pData->m_Width * pData->m_Height * 4 );
DestroyVTFTexture( pTex );
Msg( "-- LoadMacroTextureFile: %s\n", pFilename ); return pData; }
void InitMacroTexture( const char *pBSPFilename ) { // Get the world bounds (same ones used by minimaps and level designers know how to use).
int i = 0; for (i; i < num_entities; ++i) { char* pEntity = ValueForKey(&entities[i], "classname"); if( !strcmp(pEntity, "worldspawn") ) { GetVectorForKey( &entities[i], "world_mins", g_MacroWorldMins ); GetVectorForKey( &entities[i], "world_maxs", g_MacroWorldMaxs ); break; } }
if ( i == num_entities ) { Warning( "MaskOnMacroTexture: can't find worldspawn" ); return; }
// Load the macro texture that is mapped onto everything.
char mapName[512], vtfFilename[512]; Q_FileBase( pBSPFilename, mapName, sizeof( mapName ) ); Q_snprintf( vtfFilename, sizeof( vtfFilename ), "materials/macro/%s/base.vtf", mapName ); g_pGlobalMacroTextureData = LoadMacroTextureFile( vtfFilename );
// Now load the macro texture for each face.
g_FaceMacroTextures.SetSize( numfaces ); for ( int iFace=0; iFace < numfaces; iFace++ ) { g_FaceMacroTextures[iFace] = NULL;
if ( iFace < g_FaceMacroTextureInfos.Count() ) { unsigned short stringID = g_FaceMacroTextureInfos[iFace].m_MacroTextureNameID; if ( stringID != 0xFFFF ) { const char *pMacroTextureName = &g_TexDataStringData[ g_TexDataStringTable[stringID] ]; Q_snprintf( vtfFilename, sizeof( vtfFilename ), "%smaterials/%s.vtf", gamedir, pMacroTextureName ); g_FaceMacroTextures[iFace] = FindMacroTexture( vtfFilename ); if ( !g_FaceMacroTextures[iFace] ) { g_FaceMacroTextures[iFace] = LoadMacroTextureFile( vtfFilename ); if ( g_FaceMacroTextures[iFace] ) { g_MacroTextureLookup.Insert( vtfFilename, g_FaceMacroTextures[iFace] ); } } } } } }
inline Vector SampleMacroTexture( const CMacroTextureData *t, const Vector &vWorldPos ) { int ix = (int)RemapVal( vWorldPos.x, g_MacroWorldMins.x, g_MacroWorldMaxs.x, 0, t->m_Width-0.00001 ); int iy = (int)RemapVal( vWorldPos.y, g_MacroWorldMins.y, g_MacroWorldMaxs.y, 0, t->m_Height-0.00001 ); ix = clamp( ix, 0, t->m_Width-1 ); iy = t->m_Height - 1 - clamp( iy, 0, t->m_Height-1 );
const unsigned char *pInputColor = &t->m_ImageData[(iy*t->m_Width + ix) * 4]; return Vector( pInputColor[0] / 255.0, pInputColor[1] / 255.0, pInputColor[2] / 255.0 ); }
void ApplyMacroTextures( int iFace, const Vector &vWorldPos, Vector &outLuxel ) { // Add the global macro texture.
Vector vGlobal; if ( g_pGlobalMacroTextureData ) outLuxel *= SampleMacroTexture( g_pGlobalMacroTextureData, vWorldPos );
// Now add the per-material macro texture.
if ( g_FaceMacroTextures[iFace] ) outLuxel *= SampleMacroTexture( g_FaceMacroTextures[iFace], vWorldPos ); }
|