1
0
forked from metin2/client
client/GameLib/Area.cpp

1449 lines
42 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "StdAfx.h"
#include "../eterLib/ResourceManager.h"
#include "../eterLib/StateManager.h"
#include "../effectLib/EffectManager.h"
#include "../SpeedTreeLib/SpeedTreeForestDirectX8.h"
#include "../eterBase/Timer.h"
#include "Area.h"
#include "PropertyManager.h"
#include "Property.h"
#include <boost/algorithm/string.hpp>
CDynamicPool<CArea::TObjectInstance> CArea::ms_ObjectInstancePool;
CDynamicPool<CAttributeInstance> CArea::ms_AttributeInstancePool;
CDynamicPool<CArea::TAmbienceInstance> CArea::ms_AmbienceInstancePool;
CDynamicPool<CDungeonBlock> CArea::ms_DungeonBlockInstancePool;
CDynamicPool<CArea> CArea::ms_kPool;
void CArea::TObjectData::InitializeRotation()
{
m_fYaw=m_fPitch=m_fRoll=0.0f;
}
CArea* CArea::New()
{
return ms_kPool.Alloc();
}
void CArea::Delete(CArea* pkArea)
{
pkArea->Clear();
ms_kPool.Free(pkArea);
}
void CArea::DestroySystem()
{
ms_kPool.Destroy();
ms_ObjectInstancePool.Destroy();
ms_AttributeInstancePool.Destroy();
ms_AmbienceInstancePool.Destroy();
ms_DungeonBlockInstancePool.Destroy();
}
void CArea::__UpdateAniThingList()
{
{
CGraphicThingInstance* pkThingInst;
TThingInstanceVector::iterator i=m_ThingCloneInstaceVector.begin();
while (i!=m_ThingCloneInstaceVector.end())
{
pkThingInst=*i++;
if (pkThingInst->isShow())
{
pkThingInst->UpdateLODLevel();
}
}
}
{
CGraphicThingInstance* pkThingInst;
TThingInstanceVector::iterator i=m_AniThingCloneInstanceVector.begin();
while (i!=m_AniThingCloneInstanceVector.end())
{
pkThingInst=*i++;
pkThingInst->Update();
}
}
}
void CArea::__UpdateEffectList()
{
//if (!CEffectManager::InstancePtr())
// return;
CEffectManager& rkEftMgr=CEffectManager::Instance();
// Effect
TEffectInstanceIterator i;
for (i = m_EffectInstanceMap.begin(); i != m_EffectInstanceMap.end();)
{
CEffectInstance * pEffectInstance = i->second;
pEffectInstance->Update();
if (!pEffectInstance->isAlive())
{
i = m_EffectInstanceMap.erase(i);
rkEftMgr.DestroyUnsafeEffectInstance(pEffectInstance);
}
else
++i;
}
}
void CArea::Update()
{
__UpdateAniThingList();
}
void CArea::UpdateAroundAmbience(float fX, float fY, float fZ)
{
// Ambience
TAmbienceInstanceVector::iterator i;
for (i = m_AmbienceCloneInstanceVector.begin(); i != m_AmbienceCloneInstanceVector.end(); ++i)
{
TAmbienceInstance * pInstance = *i;
pInstance->__Update(fX, fY, fZ);
}
}
struct CArea_LessEffectInstancePtrRenderOrder
{
bool operator() (CEffectInstance* pkLeft, CEffectInstance* pkRight)
{
return pkLeft->LessRenderOrder(pkRight);
}
};
struct CArea_FEffectInstanceRender
{
inline void operator () (CEffectInstance * pkEftInst)
{
pkEftInst->Render();
}
};
void CArea::RenderEffect()
{
__UpdateEffectList();
// Effect
STATEMANAGER.SetTexture(0, NULL);
STATEMANAGER.SetTexture(1, NULL);
bool m_isDisableSortRendering=false;
if (m_isDisableSortRendering)
{
TEffectInstanceIterator i;
for (i = m_EffectInstanceMap.begin(); i != m_EffectInstanceMap.end();)
{
CEffectInstance * pEffectInstance = i->second;
pEffectInstance->Render();
++i;
}
}
else
{
static std::vector<CEffectInstance*> s_kVct_pkEftInstSort;
s_kVct_pkEftInstSort.clear();
TEffectInstanceMap& rkMap_pkEftInstSrc=m_EffectInstanceMap;
TEffectInstanceMap::iterator i;
for (i=rkMap_pkEftInstSrc.begin(); i!=rkMap_pkEftInstSrc.end(); ++i)
s_kVct_pkEftInstSort.push_back(i->second);
std::sort(s_kVct_pkEftInstSort.begin(), s_kVct_pkEftInstSort.end(), CArea_LessEffectInstancePtrRenderOrder());
std::for_each(s_kVct_pkEftInstSort.begin(), s_kVct_pkEftInstSort.end(), CArea_FEffectInstanceRender());
}
}
DWORD CArea::DEBUG_GetRenderedCRCNum()
{ return m_kRenderedThingInstanceCRCWithNumberVector.size(); }
CArea::TCRCWithNumberVector & CArea::DEBUG_GetRenderedCRCWithNumVector()
{ return m_kRenderedThingInstanceCRCWithNumberVector; }
DWORD CArea::DEBUG_GetRenderedGrapphicThingInstanceNum()
{ return m_kRenderedGrapphicThingInstanceVector.size(); }
void CArea::CollectRenderingObject(std::vector<CGraphicThingInstance*>& rkVct_pkOpaqueThingInst)
{
TThingInstanceVector::iterator i;
for (i=m_ThingCloneInstaceVector.begin(); i!=m_ThingCloneInstaceVector.end(); ++i)
{
CGraphicThingInstance* pkThingInst=*i;
if (pkThingInst->isShow())
{
if (!pkThingInst->HaveBlendThing())
rkVct_pkOpaqueThingInst.push_back(*i);
}
}
}
void CArea::CollectBlendRenderingObject(std::vector<CGraphicThingInstance*>& rkVct_pkBlendThingInst)
{
TThingInstanceVector::iterator i;
for (i=m_ThingCloneInstaceVector.begin(); i!=m_ThingCloneInstaceVector.end(); ++i)
{
CGraphicThingInstance* pkThingInst=*i;
if (pkThingInst->isShow())
{
if (pkThingInst->HaveBlendThing())
rkVct_pkBlendThingInst.push_back(*i);
}
}
}
void CArea::Render()
{
{
CGraphicThingInstance* pkThingInst;
TThingInstanceVector::iterator i=m_AniThingCloneInstanceVector.begin();
while (i!=m_AniThingCloneInstanceVector.end())
{
pkThingInst=*i++;
pkThingInst->Deform();
}
}
CGraphicThingInstance* pkThingInst;
TThingInstanceVector::iterator i=m_ThingCloneInstaceVector.begin();
m_kRenderedThingInstanceCRCWithNumberVector.clear();
m_kRenderedGrapphicThingInstanceVector.clear();
TGraphicThingInstanceCRCMap::iterator aGraphicThingInstanceCRCMapIterator;
while (i!=m_ThingCloneInstaceVector.end())
{
pkThingInst=*i++;
if (pkThingInst->Render())
{
aGraphicThingInstanceCRCMapIterator = m_GraphicThingInstanceCRCMap.find(pkThingInst);
DWORD dwCRC = (*aGraphicThingInstanceCRCMapIterator).second;
m_kRenderedGrapphicThingInstanceVector.push_back(pkThingInst);
TCRCWithNumberVector::iterator aCRCWithNumberVectorIterator = std::find_if(m_kRenderedThingInstanceCRCWithNumberVector.begin(), m_kRenderedThingInstanceCRCWithNumberVector.end(), FFindIfCRC(dwCRC));
if ( m_kRenderedThingInstanceCRCWithNumberVector.end() == aCRCWithNumberVectorIterator)
{
TCRCWithNumber aCRCWithNumber;
aCRCWithNumber.dwCRC = dwCRC;
aCRCWithNumber.dwNumber = 1;
m_kRenderedThingInstanceCRCWithNumberVector.push_back(aCRCWithNumber);
}
else
{
TCRCWithNumber & rCRCWithNumber = *aCRCWithNumberVectorIterator;
rCRCWithNumber.dwNumber += 1;
}
}
}
std::sort(m_kRenderedThingInstanceCRCWithNumberVector.begin(), m_kRenderedThingInstanceCRCWithNumberVector.end(), CRCNumComp());
}
void CArea::RenderCollision()
{
DWORD i;
STATEMANAGER.SetTexture(0, NULL);
STATEMANAGER.SetTexture(1, NULL);
STATEMANAGER.SaveRenderState(D3DRS_ALPHABLENDENABLE, FALSE);
STATEMANAGER.SaveRenderState(D3DRS_CULLMODE, D3DCULL_NONE);
STATEMANAGER.SetRenderState(D3DRS_LIGHTING, FALSE);
STATEMANAGER.SetRenderState(D3DRS_TEXTUREFACTOR, 0xff000000);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLORARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_SELECTARG1);
STATEMANAGER.SetTextureStageState(0, D3DTSS_ALPHAOP, D3DTOP_DISABLE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_COLOROP, D3DTOP_DISABLE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAOP, D3DTOP_DISABLE);
for(i=0;i<GetObjectInstanceCount();i++)
{
const TObjectInstance * po;
if (GetObjectInstancePointer(i,&po))
{
if (po->pTree && po->pTree->isShow())
{
DWORD j;
for(j=0;j<po->pTree->GetCollisionInstanceCount();j++)
{
po->pTree->GetCollisionInstanceData(j)->Render();
}
}
if (po->pThingInstance && po->pThingInstance->isShow())
{
DWORD j;
for(j=0;j<po->pThingInstance->GetCollisionInstanceCount();j++)
{
po->pThingInstance->GetCollisionInstanceData(j)->Render();
}
}
if (po->pDungeonBlock && po->pDungeonBlock->isShow())
{
DWORD j;
for(j=0;j<po->pDungeonBlock->GetCollisionInstanceCount();j++)
{
po->pDungeonBlock->GetCollisionInstanceData(j)->Render();
}
}
}
}
STATEMANAGER.RestoreRenderState(D3DRS_ALPHABLENDENABLE);
STATEMANAGER.RestoreRenderState(D3DRS_CULLMODE);
STATEMANAGER.SetRenderState(D3DRS_LIGHTING, TRUE);
}
void CArea::RenderAmbience()
{
DWORD dwColorArg1, dwColorOp;
STATEMANAGER.GetTextureStageState(0, D3DTSS_COLORARG1, &dwColorArg1);
STATEMANAGER.GetTextureStageState(0, D3DTSS_COLOROP, &dwColorOp);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLORARG1, D3DTA_TFACTOR);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_SELECTARG1);
TAmbienceInstanceVector::iterator itor = m_AmbienceCloneInstanceVector.begin();
for (; itor != m_AmbienceCloneInstanceVector.end(); ++itor)
{
TAmbienceInstance * pInstance = *itor;
pInstance->Render();
}
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLORARG1, dwColorArg1);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLOROP, dwColorOp);
}
void CArea::RenderDungeon()
{
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLORARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_SELECTARG1);
STATEMANAGER.SetTextureStageState(0, D3DTSS_ALPHAARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_SELECTARG1);
STATEMANAGER.SetTextureStageState(1, D3DTSS_COLORARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_COLORARG2, D3DTA_CURRENT);
STATEMANAGER.SetTextureStageState(1, D3DTSS_COLOROP, D3DTOP_MODULATE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAARG2, D3DTA_CURRENT);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
#ifdef WORLD_EDITOR
bool bRenderTransparent = false;
DWORD oldAlphaBlendState = 0;
DWORD oldZWriteenableState = 0;
if (GetAsyncKeyState(VK_LSHIFT) & 0x8001)
{
bRenderTransparent = true;
oldAlphaBlendState = STATEMANAGER.GetRenderState(D3DRS_ALPHABLENDENABLE);
oldZWriteenableState = STATEMANAGER.GetRenderState(D3DRS_ZWRITEENABLE);
STATEMANAGER.SetRenderState(D3DRS_ALPHABLENDENABLE, TRUE);
STATEMANAGER.SetRenderState(D3DRS_ZWRITEENABLE, FALSE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_ALPHAARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(0, D3DTSS_ALPHAARG2, D3DTA_TFACTOR);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAARG1, D3DTA_TEXTURE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAARG2, D3DTA_TFACTOR);
STATEMANAGER.SetRenderState(D3DRS_TEXTUREFACTOR, D3DCOLOR_ARGB( 128, 255, 0, 0) );
}
#endif
TDungeonBlockInstanceVector::iterator itor = m_DungeonBlockCloneInstanceVector.begin();
for (; itor != m_DungeonBlockCloneInstanceVector.end(); ++itor)
{
(*itor)->Render();
}
#ifdef WORLD_EDITOR
if (bRenderTransparent)
{
STATEMANAGER.SetRenderState(D3DRS_ZWRITEENABLE, oldZWriteenableState);
STATEMANAGER.SetRenderState(D3DRS_ALPHABLENDENABLE, oldAlphaBlendState);
}
#endif
STATEMANAGER.SetTextureStageState(1, D3DTSS_COLOROP, D3DTOP_DISABLE);
STATEMANAGER.SetTextureStageState(1, D3DTSS_ALPHAOP, D3DTOP_DISABLE);
}
void CArea::Refresh()
{
m_TreeCloneInstaceVector.clear();
m_ThingCloneInstaceVector.clear();
m_DungeonBlockCloneInstanceVector.clear();
m_AniThingCloneInstanceVector.clear();
m_ShadowThingCloneInstaceVector.clear();
m_AmbienceCloneInstanceVector.clear();
TObjectInstanceVector::iterator it;
for(it = m_ObjectInstanceVector.begin();it!=m_ObjectInstanceVector.end();++it)
{
TObjectInstance * pObjectInstance = *it;
if (prt::PROPERTY_TYPE_TREE == pObjectInstance->dwType)
{
if (pObjectInstance->pTree)
{
m_TreeCloneInstaceVector.push_back(pObjectInstance->pTree);
const float * pfPosition;
pfPosition = pObjectInstance->pTree->GetPosition();
pObjectInstance->pTree->UpdateBoundingSphere();
pObjectInstance->pTree->UpdateCollisionData();
}
}
else if (prt::PROPERTY_TYPE_BUILDING == pObjectInstance->dwType)
{
pObjectInstance->pThingInstance->Update();
pObjectInstance->pThingInstance->Transform();
pObjectInstance->pThingInstance->Show();
pObjectInstance->pThingInstance->DeformAll();
m_ThingCloneInstaceVector.push_back(pObjectInstance->pThingInstance);
pObjectInstance->pThingInstance->BuildBoundingSphere();
pObjectInstance->pThingInstance->UpdateBoundingSphere();
if (pObjectInstance->pThingInstance->IsMotionThing())
{
m_AniThingCloneInstanceVector.push_back(pObjectInstance->pThingInstance);
pObjectInstance->pThingInstance->SetMotion(0);
}
if (pObjectInstance->isShadowFlag)
{
m_ShadowThingCloneInstaceVector.push_back(pObjectInstance->pThingInstance);
}
if (pObjectInstance->pAttributeInstance)
{
pObjectInstance->pThingInstance->UpdateCollisionData(&pObjectInstance->pAttributeInstance->GetObjectPointer()->GetCollisionDataVector());
pObjectInstance->pAttributeInstance->RefreshObject(pObjectInstance->pThingInstance->GetTransform());
pObjectInstance->pThingInstance->UpdateHeightInstance(pObjectInstance->pAttributeInstance);
}
}
else if (prt::PROPERTY_TYPE_EFFECT == pObjectInstance->dwType)
{
}
else if (prt::PROPERTY_TYPE_AMBIENCE == pObjectInstance->dwType)
{
m_AmbienceCloneInstanceVector.push_back(pObjectInstance->pAmbienceInstance);
}
else if (prt::PROPERTY_TYPE_DUNGEON_BLOCK == pObjectInstance->dwType)
{
pObjectInstance->pDungeonBlock->Update();
pObjectInstance->pDungeonBlock->Deform();
pObjectInstance->pDungeonBlock->UpdateBoundingSphere();
m_DungeonBlockCloneInstanceVector.push_back(pObjectInstance->pDungeonBlock);
if (pObjectInstance->pAttributeInstance)
{
pObjectInstance->pDungeonBlock->UpdateCollisionData(&pObjectInstance->pAttributeInstance->GetObjectPointer()->GetCollisionDataVector());
pObjectInstance->pAttributeInstance->RefreshObject(pObjectInstance->pDungeonBlock->GetTransform());
pObjectInstance->pDungeonBlock->UpdateHeightInstance(pObjectInstance->pAttributeInstance);
}
}
}
}
void CArea::__Load_BuildObjectInstances()
{
m_ObjectInstanceVector.clear();
m_ObjectInstanceVector.resize(GetObjectDataCount());
m_GraphicThingInstanceCRCMap.clear();
std::sort(m_ObjectDataVector.begin(), m_ObjectDataVector.end(), ObjectDataComp());
DWORD i=0;
TObjectInstanceVector::iterator it;
for (it = m_ObjectInstanceVector.begin(); it!=m_ObjectInstanceVector.end();++it,++i)
{
*it = ms_ObjectInstancePool.Alloc();
(*it)->Clear();
const TObjectData * c_pObjectData;
if (!GetObjectDataPointer(i, &c_pObjectData))
continue;
__SetObjectInstance(*it, c_pObjectData);
// <20><><EFBFBD><EFBFBD>ȭ<EFBFBD><C8AD>
if ((*it)->dwType == prt::PROPERTY_TYPE_BUILDING)
m_GraphicThingInstanceCRCMap.insert(TGraphicThingInstanceCRCMap::value_type( (*it)->pThingInstance, c_pObjectData->dwCRC ) );
}
//////////
Refresh();
//////////
}
void CArea::__SetObjectInstance(TObjectInstance * pObjectInstance, const TObjectData * c_pData)
{
CProperty * pProperty;
if (!CPropertyManager::Instance().Get(c_pData->dwCRC, &pProperty))
return;
const char * c_szPropertyType;
if (!pProperty->GetString("PropertyType", &c_szPropertyType))
return;
switch (prt::GetPropertyType(c_szPropertyType))
{
case prt::PROPERTY_TYPE_TREE:
__SetObjectInstance_SetTree(pObjectInstance, c_pData, pProperty);
break;
case prt::PROPERTY_TYPE_BUILDING:
__SetObjectInstance_SetBuilding(pObjectInstance, c_pData, pProperty);
break;
case prt::PROPERTY_TYPE_EFFECT:
__SetObjectInstance_SetEffect(pObjectInstance, c_pData, pProperty);
break;
case prt::PROPERTY_TYPE_AMBIENCE:
__SetObjectInstance_SetAmbience(pObjectInstance, c_pData, pProperty);
break;
case prt::PROPERTY_TYPE_DUNGEON_BLOCK:
__SetObjectInstance_SetDungeonBlock(pObjectInstance, c_pData, pProperty);
break;
}
}
void CArea::__SetObjectInstance_SetEffect(TObjectInstance * pObjectInstance, const TObjectData * c_pData, CProperty * pProperty)
{
prt::TPropertyEffect Data;
if (!prt::PropertyEffectStringToData(pProperty, &Data))
return;
pObjectInstance->dwType = prt::PROPERTY_TYPE_EFFECT;
pObjectInstance->dwEffectID = GetCaseCRC32(Data.strFileName.c_str(),Data.strFileName.size());
CEffectManager & rem = CEffectManager::Instance();
CEffectData * pData;
if (!rem.GetEffectData(pObjectInstance->dwEffectID,&pData))
{
if (!rem.RegisterEffect(Data.strFileName.c_str()))
{
pObjectInstance->dwEffectID = 0xffffffff;
TraceError("CArea::SetEffect effect register error %s\n",Data.strFileName.c_str());
return;
}
}
CEffectInstance * pEffectInstance;
rem.CreateUnsafeEffectInstance(pObjectInstance->dwEffectID, &pEffectInstance);
D3DXMATRIX mat;
D3DXMatrixRotationYawPitchRoll(&mat,
D3DXToRadian(c_pData->m_fYaw),
D3DXToRadian(c_pData->m_fPitch),
D3DXToRadian(c_pData->m_fRoll)
);
mat._41 = c_pData->Position.x;
mat._42 = c_pData->Position.y;
mat._43 = c_pData->Position.z + c_pData->m_fHeightBias;
pEffectInstance->SetGlobalMatrix(mat);
pObjectInstance->dwEffectInstanceIndex = m_EffectInstanceMap.size();
m_EffectInstanceMap.insert(TEffectInstanceMap::value_type(pObjectInstance->dwEffectInstanceIndex, pEffectInstance));
}
void CArea::__SetObjectInstance_SetTree(TObjectInstance * pObjectInstance, const TObjectData * c_pData, CProperty * pProperty)
{
const char * c_szTreeName;
if (!pProperty->GetString("TreeFile", &c_szTreeName))
return;
pObjectInstance->SetTree(
c_pData->Position.x,
c_pData->Position.y,
c_pData->Position.z + c_pData->m_fHeightBias,
c_pData->dwCRC,
c_szTreeName
);
}
void CArea::TObjectInstance::SetTree(float x, float y, float z, DWORD dwTreeCRC, const char* c_szTreeName)
{
CSpeedTreeForestDirectX8& rkForest=CSpeedTreeForestDirectX8::Instance();
pTree=rkForest.CreateInstance(x, y, z, dwTreeCRC, c_szTreeName);
dwType = prt::PROPERTY_TYPE_TREE;
}
void CArea::__SetObjectInstance_SetBuilding(TObjectInstance * pObjectInstance, const TObjectData * c_pData, CProperty * pProperty)
{
prt::TPropertyBuilding Data;
if (!prt::PropertyBuildingStringToData(pProperty, &Data))
return;
CResourceManager& rkResMgr=CResourceManager::Instance();
CGraphicThing * pThing = (CGraphicThing *)rkResMgr.GetResourcePointer(Data.strFileName.c_str());
pThing->AddReference();
if (pThing->IsEmpty())
{
#ifdef _DEBUG
TraceError("CArea::SetBuilding: There is no data: %s", Data.strFileName.c_str());
#endif
return;
}
int iModelCount = pThing->GetModelCount();
int iMotionCount = pThing->GetMotionCount();
pObjectInstance->dwType = prt::PROPERTY_TYPE_BUILDING;
pObjectInstance->pThingInstance = CGraphicThingInstance::New();
pObjectInstance->pThingInstance->Initialize();
pObjectInstance->pThingInstance->ReserveModelThing(iModelCount);
pObjectInstance->pThingInstance->ReserveModelInstance(iModelCount);
pObjectInstance->pThingInstance->RegisterModelThing(0, pThing);
for (int j = 0; j < PORTAL_ID_MAX_NUM; ++j)
if (0 != c_pData->abyPortalID[j])
pObjectInstance->pThingInstance->SetPortal(j, c_pData->abyPortalID[j]);
{
std::string stSrcModelFileName=Data.strFileName;
std::string stLODModelFileName;
char szLODModelFileNameEnd[256];
for (UINT uLODIndex=1; uLODIndex<=3; ++uLODIndex)
{
sprintf(szLODModelFileNameEnd, "_lod_%.2d.gr2", uLODIndex);
stLODModelFileName = CFileNameHelper::NoExtension(stSrcModelFileName) + szLODModelFileNameEnd;
if (!rkResMgr.IsFileExist(stLODModelFileName.c_str()))
break;
CGraphicThing* pLODModelThing = (CGraphicThing *)rkResMgr.GetResourcePointer(stLODModelFileName.c_str());
if (!pLODModelThing)
break;
pObjectInstance->pThingInstance->RegisterLODThing(0, pLODModelThing);
}
}
for (int i = 0; i < iModelCount; ++i)
pObjectInstance->pThingInstance->SetModelInstance(i, 0, i);
if (iMotionCount)
{
pObjectInstance->pThingInstance->RegisterMotionThing(0, pThing);
}
pObjectInstance->pThingInstance->SetPosition(c_pData->Position.x, c_pData->Position.y, c_pData->Position.z + c_pData->m_fHeightBias);
pObjectInstance->pThingInstance->SetRotation(
c_pData->m_fYaw,
c_pData->m_fPitch,
c_pData->m_fRoll
);
pObjectInstance->isShadowFlag = Data.isShadowFlag;
pObjectInstance->pThingInstance->RegisterBoundingSphere();
__LoadAttribute(pObjectInstance, Data.strAttributeDataFileName.c_str());
pThing->Release();
}
void CArea::__SetObjectInstance_SetAmbience(TObjectInstance * pObjectInstance, const TObjectData * c_pData, CProperty * pProperty)
{
pObjectInstance->pAmbienceInstance = ms_AmbienceInstancePool.Alloc();
if (!prt::PropertyAmbienceStringToData(pProperty, &pObjectInstance->pAmbienceInstance->AmbienceData))
return;
pObjectInstance->dwType = prt::PROPERTY_TYPE_AMBIENCE;
TAmbienceInstance * pAmbienceInstance = pObjectInstance->pAmbienceInstance;
pAmbienceInstance->fx = c_pData->Position.x;
pAmbienceInstance->fy = c_pData->Position.y;
pAmbienceInstance->fz = c_pData->Position.z + c_pData->m_fHeightBias;
pAmbienceInstance->dwRange = c_pData->dwRange;
pAmbienceInstance->fMaxVolumeAreaPercentage = c_pData->fMaxVolumeAreaPercentage;
if (0 == pAmbienceInstance->AmbienceData.strPlayType.compare("ONCE"))
{
pAmbienceInstance->Update = &TAmbienceInstance::UpdateOnceSound;
}
else if (0 == pAmbienceInstance->AmbienceData.strPlayType.compare("STEP"))
{
pAmbienceInstance->Update = &TAmbienceInstance::UpdateStepSound;
}
else if (0 == pAmbienceInstance->AmbienceData.strPlayType.compare("LOOP"))
{
pAmbienceInstance->Update = &TAmbienceInstance::UpdateLoopSound;
}
}
void CArea::__SetObjectInstance_SetDungeonBlock(TObjectInstance * pObjectInstance, const TObjectData * c_pData, CProperty * pProperty)
{
prt::TPropertyDungeonBlock Data;
if (!prt::PropertyDungeonBlockStringToData(pProperty, &Data))
return;
pObjectInstance->dwType = prt::PROPERTY_TYPE_DUNGEON_BLOCK;
pObjectInstance->pDungeonBlock = ms_DungeonBlockInstancePool.Alloc();
pObjectInstance->pDungeonBlock->Load(Data.strFileName.c_str());
pObjectInstance->pDungeonBlock->SetPosition(c_pData->Position.x, c_pData->Position.y, c_pData->Position.z + c_pData->m_fHeightBias);
pObjectInstance->pDungeonBlock->SetRotation(
c_pData->m_fYaw,
c_pData->m_fPitch,
c_pData->m_fRoll
);
pObjectInstance->pDungeonBlock->Update();
pObjectInstance->pDungeonBlock->BuildBoundingSphere();
pObjectInstance->pDungeonBlock->RegisterBoundingSphere();
for (int j = 0; j < PORTAL_ID_MAX_NUM; ++j)
if (0 != c_pData->abyPortalID[j])
pObjectInstance->pDungeonBlock->SetPortal(j, c_pData->abyPortalID[j]);
__LoadAttribute(pObjectInstance, Data.strAttributeDataFileName.c_str());
}
void CArea::__LoadAttribute(TObjectInstance * pObjectInstance, const char * c_szAttributeFileName)
{
// OBB<42><42> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><20><><EFBFBD><EFBFBD> <20>ڵ<EFBFBD> <20><><EFBFBD><EFBFBD>.
const bool bFileExist = CResourceManager::Instance().IsFileExist(c_szAttributeFileName);
CAttributeData * pAttributeData = (CAttributeData *) CResourceManager::Instance().GetResourcePointer(c_szAttributeFileName);
CAttributeInstance * pAttrInstance = ms_AttributeInstancePool.Alloc();
pAttrInstance->Clear();
pAttrInstance->SetObjectPointer(pAttributeData);
if (false == bFileExist)
{
std::string attrFileName(c_szAttributeFileName);
boost::algorithm::to_lower(attrFileName);
const bool bIsDungeonObject = (std::string::npos != attrFileName.find("/dungeon/")) || (std::string::npos != attrFileName.find("\\dungeon\\"));
// NOTE: dungeon <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ʈ<EFBFBD><C6AE> Dummy Collision<6F><6E> <20>ڵ<EFBFBD><DAB5><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20>ʵ<EFBFBD><CAB5><EFBFBD> <20><> (<28><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD> <20>ø<EFBFBD><C3B8><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><> <20><><EFBFBD><20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20>־<EFBFBD><D6BE><EFBFBD>. <20>̷<EFBFBD><CCB7><EFBFBD> <20>ϱ<EFBFBD><CFB1><EFBFBD> <20>׷<EFBFBD><D7B7><EFBFBD> <20><><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD> <20>Ϸ<EFBFBD>)
if (pAttributeData->IsEmpty() && false == bIsDungeonObject)
{
if (NULL != pObjectInstance && NULL != pObjectInstance->pThingInstance)
{
CGraphicThingInstance* object = pObjectInstance->pThingInstance;
D3DXVECTOR3 v3Min, v3Max;
object->GetBoundingAABB(v3Min, v3Max);
CStaticCollisionData collision;
collision.dwType = COLLISION_TYPE_OBB;
D3DXQuaternionRotationYawPitchRoll(&collision.quatRotation, object->GetYaw(), object->GetPitch(), object->GetRoll());
strcpy(collision.szName, "DummyCollisionOBB");
collision.v3Position = (v3Min + v3Max) * 0.5f;
D3DXVECTOR3 vDelta = (v3Max - v3Min);
collision.fDimensions[0] = vDelta.x * 0.5f;
collision.fDimensions[1] = vDelta.y * 0.5f;
collision.fDimensions[2] = vDelta.z * 0.5f;
pAttributeData->AddCollisionData(collision);
}
}
}
if (!pAttributeData->IsEmpty())
{
pObjectInstance->pAttributeInstance = pAttrInstance;
}
else
{
pAttrInstance->Clear();
ms_AttributeInstancePool.Free(pAttrInstance);
}
}
/*
void CArea::__LoadAttribute(TObjectInstance * pObjectInstance, const char * c_szAttributeFileName)
{
// AABB<42><42> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><20><><EFBFBD><EFBFBD> <20>ڵ<EFBFBD> <20><><EFBFBD><EFBFBD>.
const bool bFileExist = CResourceManager::Instance().IsFileExist(c_szAttributeFileName);
CAttributeData * pAttributeData = (CAttributeData *) CResourceManager::Instance().GetResourcePointer(c_szAttributeFileName);
CAttributeInstance * pAttrInstance = ms_AttributeInstancePool.Alloc();
pAttrInstance->Clear();
pAttrInstance->SetObjectPointer(pAttributeData);
if (false == bFileExist)
{
if (pAttributeData->IsEmpty())
{
if (NULL != pObjectInstance && NULL != pObjectInstance->pThingInstance)
{
CGraphicThingInstance* object = pObjectInstance->pThingInstance;
D3DXVECTOR3 v3Min, v3Max;
object->GetBoundingAABB(v3Min, v3Max);
CStaticCollisionData collision;
collision.dwType = COLLISION_TYPE_AABB;
collision.quatRotation = D3DXQUATERNION(0.0f, 0.0f, 0.0f, 1.0f);
strcpy(collision.szName, "DummyCollisionAABB");
collision.v3Position = (v3Min + v3Max) * 0.5f;
D3DXVECTOR3 vDelta = (v3Max - v3Min);
collision.fDimensions[0] = vDelta.x * 0.5f; // v3Min, v3Max<61><78> <20><><EFBFBD>ϱ<EFBFBD> <20><><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD>, <20><><EFBFBD><EFBFBD>, <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD>ݰ<EFBFBD> <20><><EFBFBD><EFBFBD>
collision.fDimensions[1] = vDelta.y * 0.5f;
collision.fDimensions[2] = vDelta.z * 0.5f;
pAttributeData->AddCollisionData(collision);
}
}
}
if (!pAttributeData->IsEmpty())
{
pObjectInstance->pAttributeInstance = pAttrInstance;
}
else
{
pAttrInstance->Clear();
ms_AttributeInstancePool.Free(pAttrInstance);
}
}
*/
/*
void CArea::__LoadAttribute(TObjectInstance * pObjectInstance, const char * c_szAttributeFileName)
{
// Sphere<72><65> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><20><><EFBFBD><EFBFBD> <20>ڵ<EFBFBD> <20><><EFBFBD><EFBFBD>.
const bool bFileExist = CResourceManager::Instance().IsFileExist(c_szAttributeFileName);
CAttributeData * pAttributeData = (CAttributeData *) CResourceManager::Instance().GetResourcePointer(c_szAttributeFileName);
CAttributeInstance * pAttrInstance = ms_AttributeInstancePool.Alloc();
pAttrInstance->Clear();
pAttrInstance->SetObjectPointer(pAttributeData);
if (false == bFileExist)
{
if (pAttributeData->IsEmpty())
{
if (NULL != pObjectInstance && NULL != pObjectInstance->pThingInstance)
{
CGraphicThingInstance* object = pObjectInstance->pThingInstance;
D3DXVECTOR3 v3Center;
float fRadius = 0.0f;
object->GetBoundingSphere(v3Center, fRadius);
CStaticCollisionData collision;
collision.dwType = COLLISION_TYPE_SPHERE;
collision.quatRotation = D3DXQUATERNION(0.0f, 0.0f, 0.0f, 1.0f);
strcpy(collision.szName, "DummyCollisionSphere");
collision.fDimensions[0] = fRadius * 0.25;
collision.v3Position = v3Center;
pAttributeData->AddCollisionData(collision);
}
}
}
if (!pAttributeData->IsEmpty())
{
pObjectInstance->pAttributeInstance = pAttrInstance;
}
else
{
pAttrInstance->Clear();
ms_AttributeInstancePool.Free(pAttrInstance);
}
}
*/
bool CArea::Load(const char * c_szPathName)
{
Clear();
std::string strObjectDataFileName = c_szPathName + std::string("AreaData.txt");
std::string strAmbienceDataFileName = c_szPathName + std::string("AreaAmbienceData.txt");
__Load_LoadObject(strObjectDataFileName.c_str());
__Load_LoadAmbience(strAmbienceDataFileName.c_str());
__Load_BuildObjectInstances();
return true;
}
bool CArea::__Load_LoadObject(const char * c_szFileName)
{
CTokenVectorMap stTokenVectorMap;
if (!LoadMultipleTextData(c_szFileName, stTokenVectorMap))
{
TraceError(" CArea::Load File Load %s ERROR", c_szFileName);
return false;
}
if (stTokenVectorMap.end() == stTokenVectorMap.find("areadatafile"))
{
TraceError(" CArea::__LoadObject File Format %s ERROR 1", c_szFileName);
return false;
}
if (stTokenVectorMap.end() == stTokenVectorMap.find("objectcount"))
{
TraceError(" CArea::__LoadObject File Format %s ERROR 2", c_szFileName);
return false;
}
const std::string & c_rstrCount = stTokenVectorMap["objectcount"][0];
DWORD dwCount = atoi(c_rstrCount.c_str());
char szObjectName[32+1];
for (DWORD i = 0; i < dwCount; ++i)
{
_snprintf(szObjectName, sizeof(szObjectName), "object%03d", i);
if (stTokenVectorMap.end() == stTokenVectorMap.find(szObjectName))
continue;
const CTokenVector & rVector = stTokenVectorMap[szObjectName];
const std::string & c_rstrxPosition = rVector[0].c_str();
const std::string & c_rstryPosition = rVector[1].c_str();
const std::string & c_rstrzPosition = rVector[2].c_str();
const std::string & c_rstrCRC = rVector[3].c_str();
TObjectData ObjectData;
ZeroMemory(&ObjectData, sizeof(ObjectData));
ObjectData.Position.x = atof(c_rstrxPosition.c_str());
ObjectData.Position.y = atof(c_rstryPosition.c_str());
ObjectData.Position.z = atof(c_rstrzPosition.c_str());
ObjectData.dwCRC = atoi (c_rstrCRC.c_str());
// 20041217.myevan.<2E><><EFBFBD><EFBFBD><EFBFBD>̼<EFBFBD> <20><><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD>
ObjectData.InitializeRotation(); //ObjectData.m_fYaw = ObjectData.m_fPitch = ObjectData.m_fRoll = 0;
if (rVector.size() > 4)
{
std::string::size_type s=rVector[4].find('#');
if (s!=rVector[4].npos)
{
ObjectData.m_fYaw = atoi(rVector[4].substr(0,s-1).c_str());
int p = s+1;
s = rVector[4].find('#',p);
ObjectData.m_fPitch = atoi(rVector[4].substr(p,s-1-p+1).c_str());
ObjectData.m_fRoll = atoi(rVector[4].substr(s+1).c_str());
}
else
{
ObjectData.m_fYaw = 0.0f;
ObjectData.m_fPitch = 0.0f;
ObjectData.m_fRoll = atoi(rVector[4].c_str());
}
}
ObjectData.m_fHeightBias = 0.0f;
if (rVector.size() > 5)
{
ObjectData.m_fHeightBias = atof(rVector[5].c_str());
}
if (rVector.size() > 6)
{
for (int portalIdx = 0; portalIdx < min(rVector.size()-6, PORTAL_ID_MAX_NUM); ++portalIdx)
{
ObjectData.abyPortalID[portalIdx] = atoi(rVector[6+portalIdx].c_str());
}
}
// If data is not inside property, then delete it.
CProperty * pProperty;
if (!CPropertyManager::Instance().Get(ObjectData.dwCRC, &pProperty))
{
TraceError(" CArea::LoadObject Property(%u) Load ERROR", ObjectData.dwCRC);
continue;
}
m_ObjectDataVector.push_back(ObjectData);
}
return true;
}
bool CArea::__Load_LoadAmbience(const char * c_szFileName)
{
CTokenVectorMap stTokenVectorMap;
if (!LoadMultipleTextData(c_szFileName, stTokenVectorMap))
{
TraceError(" CArea::Load File Load %s ERROR", c_szFileName);
return false;
}
if (stTokenVectorMap.end() == stTokenVectorMap.find("areaambiencedatafile"))
{
TraceError(" CArea::__LoadAmbience File Format %s ERROR 1", c_szFileName);
return false;
}
if (stTokenVectorMap.end() == stTokenVectorMap.find("objectcount"))
{
TraceError(" CArea::__LoadAmbience File Format %s ERROR 2", c_szFileName);
return false;
}
const std::string & c_rstrCount = stTokenVectorMap["objectcount"][0];
DWORD dwCount = atoi(c_rstrCount.c_str());
char szObjectName[32+1];
for (DWORD i = 0; i < dwCount; ++i)
{
_snprintf(szObjectName, sizeof(szObjectName), "object%03d", i);
if (stTokenVectorMap.end() == stTokenVectorMap.find(szObjectName))
continue;
const CTokenVector & rVector = stTokenVectorMap[szObjectName];
const std::string & c_rstrxPosition = rVector[0].c_str();
const std::string & c_rstryPosition = rVector[1].c_str();
const std::string & c_rstrzPosition = rVector[2].c_str();
const std::string & c_rstrCRC = rVector[3].c_str();
const std::string & c_rstrRange = rVector[4].c_str();
TObjectData ObjectData;
ZeroMemory(&ObjectData, sizeof(ObjectData));
ObjectData.Position.x = atof(c_rstrxPosition.c_str());
ObjectData.Position.y = atof(c_rstryPosition.c_str());
ObjectData.Position.z = atof(c_rstrzPosition.c_str());
ObjectData.dwCRC = atoi (c_rstrCRC.c_str());
ObjectData.dwRange = atoi(c_rstrRange.c_str());
// 20041217.myevan.<2E><><EFBFBD><EFBFBD><EFBFBD>̼<EFBFBD> <20>ʱ<EFBFBD>ȭ
ObjectData.InitializeRotation();
ObjectData.m_fHeightBias = 0.0f;
ObjectData.fMaxVolumeAreaPercentage = 0.0f;
if (rVector.size() >= 6)
{
const std::string & c_rstrPercentage = rVector[5].c_str();
ObjectData.fMaxVolumeAreaPercentage = atof(c_rstrPercentage.c_str());
}
// If data is not inside property, then delete it.
CProperty * pProperty;
if (!CPropertyManager::Instance().Get(ObjectData.dwCRC, &pProperty))
{
TraceError(" CArea::LoadAmbience Property(%d) Load ERROR", ObjectData.dwCRC);
continue;
}
m_ObjectDataVector.push_back(ObjectData);
}
return true;
}
//////////////////////////////////////////////////////////////////////////
bool CArea::CheckObjectIndex(DWORD dwIndex) const
{
if (dwIndex >= m_ObjectDataVector.size())
return false;
return true;
}
DWORD CArea::GetObjectDataCount()
{
return m_ObjectDataVector.size();
}
bool CArea::GetObjectDataPointer(DWORD dwIndex, const TObjectData ** ppObjectData) const
{
if (!CheckObjectIndex(dwIndex))
{
assert(!"Setting Object Index is corrupted!");
return false;
}
*ppObjectData = &m_ObjectDataVector[dwIndex];
return true;
}
const DWORD CArea::GetObjectInstanceCount() const
{
return m_ObjectInstanceVector.size();
}
const bool CArea::GetObjectInstancePointer(const DWORD & dwIndex, const TObjectInstance ** ppObjectInstance) const
{
if (dwIndex >= m_ObjectInstanceVector.size())
return false;
*ppObjectInstance = m_ObjectInstanceVector[dwIndex];
return true;
}
void CArea::EnablePortal(BOOL bFlag)
{
if (m_bPortalEnable == bFlag)
return;
m_bPortalEnable = bFlag;
}
void CArea::ClearPortal()
{
m_kSet_ShowingPortalID.clear();
}
void CArea::AddShowingPortalID(int iNum)
{
m_kSet_ShowingPortalID.insert(iNum);
}
void CArea::RefreshPortal()
{
std::set<TObjectInstance *> kSet_ShowingObjectInstance;
kSet_ShowingObjectInstance.clear();
for (DWORD i = 0; i < m_ObjectDataVector.size(); ++i)
{
TObjectData & rData = m_ObjectDataVector[i];
TObjectInstance * pInstance = m_ObjectInstanceVector[i];
for (int j = 0; j < PORTAL_ID_MAX_NUM; ++j)
{
BYTE byPortalID = rData.abyPortalID[j];
if (0 == byPortalID)
break;
if (m_kSet_ShowingPortalID.end() == m_kSet_ShowingPortalID.find(byPortalID))
continue;
kSet_ShowingObjectInstance.insert(pInstance);
break;
}
}
///////////////////////////////////////////////////////////////////////////////////////////////
m_TreeCloneInstaceVector.clear();
m_ThingCloneInstaceVector.clear();
m_DungeonBlockCloneInstanceVector.clear();
TObjectInstanceVector::iterator it;
for(it = m_ObjectInstanceVector.begin(); it != m_ObjectInstanceVector.end(); ++it)
{
TObjectInstance * pObjectInstance = *it;
if (m_bPortalEnable)
{
if (kSet_ShowingObjectInstance.end() == kSet_ShowingObjectInstance.find(pObjectInstance))
continue;
}
if (prt::PROPERTY_TYPE_TREE == pObjectInstance->dwType)
{
assert(pObjectInstance->pTree);
m_TreeCloneInstaceVector.push_back(pObjectInstance->pTree);
}
else if (prt::PROPERTY_TYPE_BUILDING == pObjectInstance->dwType)
{
assert(pObjectInstance->pThingInstance);
m_ThingCloneInstaceVector.push_back(pObjectInstance->pThingInstance);
}
else if (prt::PROPERTY_TYPE_DUNGEON_BLOCK == pObjectInstance->dwType)
{
assert(pObjectInstance->pDungeonBlock);
m_DungeonBlockCloneInstanceVector.push_back(pObjectInstance->pDungeonBlock);
}
}
}
//////////////////////////////////////////////////////////////////////////
void CArea::Clear()
{
// Real Instances
TObjectInstanceVector::iterator it;
for (it = m_ObjectInstanceVector.begin();it!=m_ObjectInstanceVector.end();++it)
__Clear_DestroyObjectInstance(*it);
m_ObjectDataVector.clear();
m_ObjectInstanceVector.clear();
// Clones
m_TreeCloneInstaceVector.clear();
m_ThingCloneInstaceVector.clear();
m_DungeonBlockCloneInstanceVector.clear();
m_AniThingCloneInstanceVector.clear();
m_ShadowThingCloneInstaceVector.clear();
m_AmbienceCloneInstanceVector.clear();
m_GraphicThingInstanceCRCMap.clear();
m_kRenderedThingInstanceCRCWithNumberVector.clear();
m_kRenderedGrapphicThingInstanceVector.clear();
m_bPortalEnable = FALSE;
ClearPortal();
CEffectManager& rkEftMgr=CEffectManager::Instance();
TEffectInstanceIterator i;
for (i = m_EffectInstanceMap.begin(); i != m_EffectInstanceMap.end(); ++i)
{
CEffectInstance * pEffectInstance = i->second;
rkEftMgr.DestroyUnsafeEffectInstance(pEffectInstance);
}
m_EffectInstanceMap.clear();
}
void CArea::__Clear_DestroyObjectInstance(TObjectInstance * pObjectInstance)
{
if (pObjectInstance->dwEffectInstanceIndex!=0xffffffff)
{
TEffectInstanceIterator f= m_EffectInstanceMap.find(pObjectInstance->dwEffectInstanceIndex);
if (m_EffectInstanceMap.end()!=f)
{
CEffectInstance * pEffectInstance = f->second;
m_EffectInstanceMap.erase(f);
if (CEffectManager::InstancePtr())
CEffectManager::Instance().DestroyUnsafeEffectInstance(pEffectInstance);
}
pObjectInstance->dwEffectInstanceIndex = 0xffffffff;
}
if (pObjectInstance->pAttributeInstance)
{
pObjectInstance->pAttributeInstance->Clear();
ms_AttributeInstancePool.Free(pObjectInstance->pAttributeInstance);
pObjectInstance->pAttributeInstance = NULL;
}
if (pObjectInstance->pTree)
{
pObjectInstance->pTree->Clear();
CSpeedTreeForestDirectX8::Instance().DeleteInstance(pObjectInstance->pTree);
pObjectInstance->pTree = NULL;
}
if (pObjectInstance->pThingInstance)
{
CGraphicThingInstance::Delete(pObjectInstance->pThingInstance);
pObjectInstance->pThingInstance = NULL;
}
if (pObjectInstance->pAmbienceInstance)
{
ms_AmbienceInstancePool.Free(pObjectInstance->pAmbienceInstance);
pObjectInstance->pAmbienceInstance = NULL;
}
if (pObjectInstance->pDungeonBlock)
{
ms_DungeonBlockInstancePool.Free(pObjectInstance->pDungeonBlock);
pObjectInstance->pDungeonBlock = NULL;
}
pObjectInstance->Clear();
ms_ObjectInstancePool.Free(pObjectInstance);
}
//////////////////////////////////////////////////////////////////////////
// Coordination <20><><EFBFBD><EFBFBD>
void CArea::GetCoordinate(unsigned short * usCoordX, unsigned short * usCoordY)
{
*usCoordX = m_wX;
*usCoordY = m_wY;
}
void CArea::SetCoordinate(const unsigned short & usCoordX, const unsigned short & usCoordY)
{
m_wX = usCoordX;
m_wY = usCoordY;
}
//////////////////////////////////////////////////////////////////////////
void CArea::SetMapOutDoor(CMapOutdoor * pOwnerOutdoorMap)
{
m_pOwnerOutdoorMap=pOwnerOutdoorMap;
}
CArea::CArea()
{
m_wX = m_wY = 0xFF;
}
CArea::~CArea()
{
Clear();
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
void CArea::TAmbienceInstance::__Update(float fxCenter, float fyCenter, float fzCenter)
{
if (0 == dwRange)
return;
(this->*Update)(fxCenter, fyCenter, fzCenter);
}
void CArea::TAmbienceInstance::UpdateOnceSound(float fxCenter, float fyCenter, float fzCenter)
{
float fDistance = sqrtf((fx - fxCenter)*(fx - fxCenter) + (fy - fyCenter)*(fy - fyCenter) + (fz - fzCenter)*(fz - fzCenter));
if (DWORD(fDistance) < dwRange)
{
if (-1 == iPlaySoundIndex)
{
if (AmbienceData.AmbienceSoundVector.empty())
return;
const char * c_szFileName = AmbienceData.AmbienceSoundVector[0].c_str();
iPlaySoundIndex = CSoundManager::Instance().PlayAmbienceSound3D(fx, fy, fz, c_szFileName);
// Tracef(" %d : OncePlay [%f] : %s\n", iPlaySoundIndex, fDistance, c_szFileName);
}
}
else
{
iPlaySoundIndex = -1;
}
}
void CArea::TAmbienceInstance::UpdateStepSound(float fxCenter, float fyCenter, float fzCenter)
{
float fDistance = sqrtf((fx - fxCenter)*(fx - fxCenter) + (fy - fyCenter)*(fy - fyCenter) + (fz - fzCenter)*(fz - fzCenter));
if (DWORD(fDistance) < dwRange)
{
float fcurTime = CTimer::Instance().GetCurrentSecond();
if (fcurTime > fNextPlayTime)
{
if (AmbienceData.AmbienceSoundVector.empty())
return;
const char * c_szFileName = AmbienceData.AmbienceSoundVector[0].c_str();
iPlaySoundIndex = CSoundManager::Instance().PlayAmbienceSound3D(fx, fy, fz, c_szFileName);
// Tracef(" %d : StepPlay [%f] : %s\n", iPlaySoundIndex, fDistance, c_szFileName);
fNextPlayTime = CTimer::Instance().GetCurrentSecond();
fNextPlayTime += AmbienceData.fPlayInterval + frandom(0.0f, AmbienceData.fPlayIntervalVariation);
}
}
else
{
iPlaySoundIndex = -1;
fNextPlayTime = 0.0f;
}
}
void CArea::TAmbienceInstance::UpdateLoopSound(float fxCenter, float fyCenter, float fzCenter)
{
float fDistance = sqrtf((fx - fxCenter)*(fx - fxCenter) + (fy - fyCenter)*(fy - fyCenter) + (fz - fzCenter)*(fz - fzCenter));
if (DWORD(fDistance) < dwRange)
{
if (-1 == iPlaySoundIndex)
{
if (AmbienceData.AmbienceSoundVector.empty())
return;
const char * c_szFileName = AmbienceData.AmbienceSoundVector[0].c_str();
iPlaySoundIndex = CSoundManager::Instance().PlayAmbienceSound3D(fx, fy, fz, c_szFileName, 0);
// Tracef(" %d : LoopPlay [%f] : %s\n", iPlaySoundIndex, fDistance, c_szFileName);
}
if (-1 != iPlaySoundIndex)
{
// Tracef("%d : %f\n", iPlaySoundIndex, __GetVolumeFromDistance(fDistance));
CSoundManager::Instance().SetSoundVolume3D(iPlaySoundIndex, __GetVolumeFromDistance(fDistance));
}
}
else
{
if (-1 != iPlaySoundIndex)
{
// Tracef(" %d : LoopStop\n", iPlaySoundIndex);
CSoundManager::Instance().StopSound3D(iPlaySoundIndex);
iPlaySoundIndex = -1;
}
}
}
float CArea::TAmbienceInstance::__GetVolumeFromDistance(float fDistance)
{
float fMaxVolumeAreaRadius = float(dwRange) * fMaxVolumeAreaPercentage;
if (fMaxVolumeAreaRadius <= 0.0f)
return 1.0f;
if (fDistance <= fMaxVolumeAreaRadius)
return 1.0f;
return 1.0f - ((fDistance - fMaxVolumeAreaRadius) / (dwRange - fMaxVolumeAreaRadius));
}
void CArea::TAmbienceInstance::Render()
{
float fBoxSize = 10.0f;
STATEMANAGER.SetRenderState(D3DRS_TEXTUREFACTOR, 0xff00ff00);
RenderCube(fx-fBoxSize, fy-fBoxSize, fz-fBoxSize, fx+fBoxSize, fy+fBoxSize, fz+fBoxSize);
STATEMANAGER.SetRenderState(D3DRS_TEXTUREFACTOR, 0xffffffff);
RenderSphere(NULL, fx, fy, fz, float(dwRange) * fMaxVolumeAreaPercentage, D3DFILL_POINT);
RenderSphere(NULL, fx, fy, fz, float(dwRange), D3DFILL_POINT);
RenderCircle2d(fx, fy, fz, float(dwRange) * fMaxVolumeAreaPercentage);
RenderCircle2d(fx, fy, fz, float(dwRange));
for (int i = 0; i < 4; ++i)
{
float fxAdd = cosf(float(i) * D3DX_PI/4.0f) * float(dwRange) / 2.0f;
float fyAdd = sinf(float(i) * D3DX_PI/4.0f) * float(dwRange) / 2.0f;
if (i%2)
{
fxAdd /= 2.0f;
fyAdd /= 2.0f;
}
RenderLine2d(fx + fxAdd, fy + fyAdd, fx - fxAdd, fy - fyAdd, fz);
}
}
bool CArea::SAmbienceInstance::Picking()
{
return CGraphicCollisionObject::IntersectSphere(D3DXVECTOR3(fx, fy, fz), dwRange);
}
CArea::SAmbienceInstance::SAmbienceInstance()
{
fx = 0.0f;
fy = 0.0f;
fz = 0.0f;
dwRange = 0;
iPlaySoundIndex = -1;
fNextPlayTime = 0.0f;
}