mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2025-01-14 23:35:22 +01:00
271 lines
7.4 KiB
C++
271 lines
7.4 KiB
C++
|
/* Copyright (c) 2002-2012 Croteam Ltd. All rights reserved. */
|
||
|
|
||
|
#include "stdh.h"
|
||
|
|
||
|
#include <Engine/Brushes/Brush.h>
|
||
|
#include <Engine/World/World.h>
|
||
|
#include <Engine/World/WorldEditingProfile.h>
|
||
|
#include <Engine/Math/Object3D.h>
|
||
|
#include <Engine/Base/ListIterator.inl>
|
||
|
#include <Engine/Math/Projection_DOUBLE.h>
|
||
|
#include <Engine/Templates/StaticArray.cpp>
|
||
|
#include <Engine/Templates/DynamicArray.cpp>
|
||
|
#include <Engine/Math/Float.h>
|
||
|
#include <Engine/Entities/Entity.h>
|
||
|
|
||
|
|
||
|
// constructor
|
||
|
CBrush3D::CBrush3D(void)
|
||
|
{
|
||
|
br_penEntity = NULL;
|
||
|
br_pfsFieldSettings = NULL;
|
||
|
br_ulFlags = 0;
|
||
|
}
|
||
|
|
||
|
// destructor
|
||
|
CBrush3D::~CBrush3D(void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/* Delete a brush mip with given factor. */
|
||
|
void CBrush3D::DeleteBrushMip(CBrushMip *pbmToDelete)
|
||
|
{
|
||
|
ASSERT(pbmToDelete!=NULL);
|
||
|
ASSERT(pbmToDelete->bm_pbrBrush == this);
|
||
|
// if there is only one brush mip
|
||
|
if (br_lhBrushMips.Count()<=1) {
|
||
|
// do nothing;
|
||
|
return;
|
||
|
}
|
||
|
// remove it from list
|
||
|
pbmToDelete->bm_lnInBrush.Remove();
|
||
|
// destroy it
|
||
|
delete pbmToDelete;
|
||
|
}
|
||
|
|
||
|
/* Create a new brush mip. */
|
||
|
CBrushMip *CBrush3D::NewBrushMipAfter(CBrushMip *pbmOld, BOOL bCopy)
|
||
|
{
|
||
|
ASSERT(pbmOld!=NULL);
|
||
|
ASSERT(pbmOld->bm_pbrBrush == this);
|
||
|
|
||
|
// create one brush mip
|
||
|
CBrushMip *pbmNew = new CBrushMip;
|
||
|
pbmNew->bm_pbrBrush = this;
|
||
|
// add it to the brush
|
||
|
pbmOld->bm_lnInBrush.AddAfter(pbmNew->bm_lnInBrush);
|
||
|
// copy the original to it
|
||
|
if (bCopy) {
|
||
|
pbmNew->Copy(*pbmOld, 1.0f, FALSE);
|
||
|
}
|
||
|
// respread the mips after old one
|
||
|
pbmOld->SpreadFurtherMips();
|
||
|
|
||
|
return pbmNew;
|
||
|
}
|
||
|
CBrushMip *CBrush3D::NewBrushMipBefore(CBrushMip *pbmOld, BOOL bCopy)
|
||
|
{
|
||
|
ASSERT(pbmOld!=NULL);
|
||
|
ASSERT(pbmOld->bm_pbrBrush == this);
|
||
|
|
||
|
// create one brush mip
|
||
|
CBrushMip *pbmNew = new CBrushMip;
|
||
|
pbmNew->bm_pbrBrush = this;
|
||
|
// add it to the brush
|
||
|
pbmOld->bm_lnInBrush.AddBefore(pbmNew->bm_lnInBrush);
|
||
|
// copy the original to it
|
||
|
if (bCopy) {
|
||
|
// copy the original to it
|
||
|
pbmNew->Copy(*pbmOld, 1.0f, FALSE);
|
||
|
}
|
||
|
|
||
|
// get factor of mip before the new one
|
||
|
FLOAT fFactorBefore = 0.0f;
|
||
|
CBrushMip *pbmBefore = pbmNew->GetPrev();
|
||
|
if (pbmBefore!=NULL) {
|
||
|
fFactorBefore = pbmBefore->bm_fMaxDistance;
|
||
|
}
|
||
|
|
||
|
// calculate factor for new one to be between those two
|
||
|
pbmNew->bm_fMaxDistance = (fFactorBefore+pbmOld->bm_fMaxDistance)/2.0f;
|
||
|
|
||
|
return pbmNew;
|
||
|
}
|
||
|
|
||
|
// make 'for' construct for walking a list reversely
|
||
|
#define FOREACHINLIST_R(baseclass, member, head, iter) \
|
||
|
for ( LISTITER(baseclass, member) iter(head.IterationTail()); \
|
||
|
!iter->member.IsHeadMarker(); iter.MoveToPrev() )
|
||
|
|
||
|
/*
|
||
|
* Get a brush mip for given mip-factor.
|
||
|
*/
|
||
|
CBrushMip *CBrush3D::GetBrushMipByDistance(FLOAT fMipDistance)
|
||
|
{
|
||
|
// initially there is no brush mip
|
||
|
CBrushMip *pbmLastGood=NULL;
|
||
|
// for all brush mips in brush reversely
|
||
|
FOREACHINLIST_R(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
CBrushMip &bm = *itbm;
|
||
|
// if this mip cannot be of given factor
|
||
|
if (bm.bm_fMaxDistance<fMipDistance) {
|
||
|
// return last mip found
|
||
|
return pbmLastGood;
|
||
|
}
|
||
|
// remember this mip
|
||
|
pbmLastGood = itbm;
|
||
|
}
|
||
|
// return last mip found
|
||
|
return pbmLastGood;
|
||
|
}
|
||
|
/* Get a brush mip by its given index. */
|
||
|
CBrushMip *CBrush3D::GetBrushMipByIndex(INDEX iMip)
|
||
|
{
|
||
|
INDEX iCurrentMip = 0;
|
||
|
// for all brush mips in brush
|
||
|
FOREACHINLIST(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
iCurrentMip++;
|
||
|
// if this is the mip
|
||
|
if (iCurrentMip == iMip) {
|
||
|
// return the mip found
|
||
|
return &*itbm;
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
// get first brush mip
|
||
|
CBrushMip *CBrush3D::GetFirstMip(void)
|
||
|
{
|
||
|
return LIST_HEAD(br_lhBrushMips, CBrushMip, bm_lnInBrush);
|
||
|
}
|
||
|
|
||
|
// get last brush mip
|
||
|
CBrushMip *CBrush3D::GetLastMip(void)
|
||
|
{
|
||
|
return LIST_TAIL(br_lhBrushMips, CBrushMip, bm_lnInBrush);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Wrapper for CObject3D::Optimize(), updates profiling information.
|
||
|
*/
|
||
|
void CBrush3D::OptimizeObject3D(CObject3D &ob)
|
||
|
{
|
||
|
_pfWorldEditingProfile.StartTimer(CWorldEditingProfile::PTI_OBJECTOPTIMIZE);
|
||
|
_pfWorldEditingProfile.IncrementCounter(CWorldEditingProfile::PCI_SECTORSOPTIMIZED,
|
||
|
ob.ob_aoscSectors.Count());
|
||
|
ob.Optimize();
|
||
|
_pfWorldEditingProfile.StopTimer(CWorldEditingProfile::PTI_OBJECTOPTIMIZE);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Free all memory and leave empty brush.
|
||
|
*/
|
||
|
void CBrush3D::Clear(void) {
|
||
|
// delete all brush mips
|
||
|
FORDELETELIST(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
delete &*itbm;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Copy brush from another brush with possible mirror and stretch. */
|
||
|
void CBrush3D::Copy(CBrush3D &brOther, FLOAT fStretch, BOOL bMirrorX)
|
||
|
{
|
||
|
// clear this brush in case there is something in it
|
||
|
Clear();
|
||
|
|
||
|
// for all brush mips in other brush
|
||
|
FOREACHINLIST(CBrushMip, bm_lnInBrush, brOther.br_lhBrushMips, itbmOther) {
|
||
|
// create one brush mip
|
||
|
CBrushMip *pbmBrushMip = new CBrushMip;
|
||
|
// add it to the brush
|
||
|
br_lhBrushMips.AddTail(pbmBrushMip->bm_lnInBrush);
|
||
|
pbmBrushMip->bm_pbrBrush = this;
|
||
|
|
||
|
// copy the brush mip from original
|
||
|
pbmBrushMip->Copy(*itbmOther, fStretch, bMirrorX);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Prepare a projection from brush space to absolute space.
|
||
|
*/
|
||
|
void CBrush3D::PrepareRelativeToAbsoluteProjection(
|
||
|
CSimpleProjection3D_DOUBLE &prRelativeToAbsolute)
|
||
|
{
|
||
|
// brush that does not have an entity is initialized at origin
|
||
|
if(br_penEntity==NULL) {
|
||
|
prRelativeToAbsolute.ObjectPlacementL().pl_PositionVector = FLOAT3D(0.0f, 0.0f, 0.0f);
|
||
|
prRelativeToAbsolute.ObjectPlacementL().pl_OrientationAngle = ANGLE3D(0, 0, 0);
|
||
|
} else {
|
||
|
prRelativeToAbsolute.ObjectPlacementL() = br_penEntity->en_plPlacement;
|
||
|
}
|
||
|
prRelativeToAbsolute.ViewerPlacementL().pl_PositionVector = FLOAT3D(0.0f, 0.0f, 0.0f);
|
||
|
prRelativeToAbsolute.ViewerPlacementL().pl_OrientationAngle = ANGLE3D(0, 0, 0);
|
||
|
prRelativeToAbsolute.Prepare();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Calculate bounding boxes in all brush mips.
|
||
|
*/
|
||
|
void CBrush3D::CalculateBoundingBoxes(void)
|
||
|
{
|
||
|
CalculateBoundingBoxesForOneMip(NULL);
|
||
|
}
|
||
|
void CBrush3D::CalculateBoundingBoxesForOneMip(CBrushMip *pbmOnly) // for only one mip
|
||
|
{
|
||
|
// set FPU to double precision
|
||
|
CSetFPUPrecision FPUPrecision(FPT_53BIT);
|
||
|
|
||
|
// prepare a projection from brush space to absolute space
|
||
|
CSimpleProjection3D_DOUBLE prBrushToAbsolute;
|
||
|
PrepareRelativeToAbsoluteProjection(prBrushToAbsolute);
|
||
|
// for all brush mips
|
||
|
FOREACHINLIST(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
CBrushMip *pbm = itbm;
|
||
|
if (pbmOnly==NULL || pbm==pbmOnly) {
|
||
|
// calculate its boxes
|
||
|
pbm->CalculateBoundingBoxes(prBrushToAbsolute);
|
||
|
}
|
||
|
}
|
||
|
// if this brush is zoning
|
||
|
if (br_penEntity!=NULL && (br_penEntity->en_ulFlags&ENF_ZONING)) {
|
||
|
// portal links must be updated also
|
||
|
extern BOOL _bPortalSectorLinksPreLoaded;
|
||
|
extern BOOL _bDontDiscardLinks;
|
||
|
br_penEntity->en_pwoWorld->wo_bPortalLinksUpToDate = _bPortalSectorLinksPreLoaded||_bDontDiscardLinks;
|
||
|
}
|
||
|
|
||
|
br_penEntity->UpdateSpatialRange();
|
||
|
}
|
||
|
|
||
|
// switch from zoning to non-zoning
|
||
|
void CBrush3D::SwitchToNonZoning(void)
|
||
|
{
|
||
|
CalculateBoundingBoxes();
|
||
|
|
||
|
// for all brush mips
|
||
|
FOREACHINLIST(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
// for all sectors in the mip
|
||
|
{FOREACHINDYNAMICARRAY(itbm->bm_abscSectors, CBrushSector, itbsc) {
|
||
|
// unset spatial clasification
|
||
|
itbsc->bsc_rsEntities.Clear();
|
||
|
}}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// switch from non-zoning to zoning
|
||
|
void CBrush3D::SwitchToZoning(void)
|
||
|
{
|
||
|
CalculateBoundingBoxes();
|
||
|
|
||
|
// for all brush mips
|
||
|
FOREACHINLIST(CBrushMip, bm_lnInBrush, br_lhBrushMips, itbm) {
|
||
|
// for all sectors in the mip
|
||
|
{FOREACHINDYNAMICARRAY(itbm->bm_abscSectors, CBrushSector, itbsc) {
|
||
|
// find entities in sector
|
||
|
itbsc->FindEntitiesInSector();
|
||
|
}}
|
||
|
}
|
||
|
}
|