/* =========================================================================== Doom 3 GPL Source Code Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company. This file is part of the Doom 3 GPL Source Code ("Doom 3 Source Code"). Doom 3 Source Code is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Doom 3 Source Code is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Doom 3 Source Code. If not, see . In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below. If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA. =========================================================================== */ #include "../../../idlib/precompiled.h" #pragma hdrstop #include "Brush.h" #define BRUSH_EPSILON 0.1f #define BRUSH_PLANE_NORMAL_EPSILON 0.00001f #define BRUSH_PLANE_DIST_EPSILON 0.01f #define OUTPUT_UPDATE_TIME 500 // update every 500 msec //#define OUTPUT_CHOP_STATS /* ============ DisplayRealTimeString ============ */ void DisplayRealTimeString( char *string, ... ) { va_list argPtr; char buf[MAX_STRING_CHARS]; static int lastUpdateTime; int time; time = Sys_Milliseconds(); if ( time > lastUpdateTime + OUTPUT_UPDATE_TIME ) { va_start( argPtr, string ); vsprintf( buf, string, argPtr ); va_end( argPtr ); common->Printf( buf ); lastUpdateTime = time; } } //=============================================================== // // idBrushSide // //=============================================================== /* ============ idBrushSide::idBrushSide ============ */ idBrushSide::idBrushSide( void ) { flags = 0; planeNum = -1; winding = NULL; } /* ============ idBrushSide::idBrushSide ============ */ idBrushSide::idBrushSide( const idPlane &plane, int planeNum ) { this->flags = 0; this->plane = plane; this->planeNum = planeNum; this->winding = NULL; } /* ============ idBrushSide::~idBrushSide ============ */ idBrushSide::~idBrushSide( void ) { if ( winding ) { delete winding; } } /* ============ idBrushSide::Copy ============ */ idBrushSide *idBrushSide::Copy( void ) const { idBrushSide *side; side = new idBrushSide( plane, planeNum ); side->flags = flags; if ( winding ) { side->winding = winding->Copy(); } else { side->winding = NULL; } return side; } /* ============ idBrushSide::Split ============ */ int idBrushSide::Split( const idPlane &splitPlane, idBrushSide **front, idBrushSide **back ) const { idWinding *frontWinding, *backWinding; assert( winding ); *front = *back = NULL; winding->Split( splitPlane, 0.0f, &frontWinding, &backWinding ); if ( frontWinding ) { (*front) = new idBrushSide( plane, planeNum ); (*front)->winding = frontWinding; (*front)->flags = flags; } if ( backWinding ) { (*back) = new idBrushSide( plane, planeNum ); (*back)->winding = backWinding; (*back)->flags = flags; } if ( frontWinding && backWinding ) { return PLANESIDE_CROSS; } else if ( frontWinding ) { return PLANESIDE_FRONT; } else { return PLANESIDE_BACK; } } //=============================================================== // // idBrushSide // //=============================================================== /* ============ idBrush::idBrush ============ */ idBrush::idBrush( void ) { contents = flags = 0; bounds.Clear(); sides.Clear(); windingsValid = false; } /* ============ idBrush::~idBrush ============ */ idBrush::~idBrush( void ) { for ( int i = 0; i < sides.Num(); i++ ) { delete sides[i]; } } /* ============ idBrush::RemoveSidesWithoutWinding ============ */ bool idBrush::RemoveSidesWithoutWinding( void ) { int i; for ( i = 0; i < sides.Num(); i++ ) { if ( sides[i]->winding ) { continue; } sides.RemoveIndex( i ); i--; } return ( sides.Num() >= 4 ); } /* ============ idBrush::CreateWindings ============ */ bool idBrush::CreateWindings( void ) { int i, j; idBrushSide *side; bounds.Clear(); for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; if ( side->winding ) { delete side->winding; } side->winding = new idWinding( side->plane.Normal(), side->plane.Dist() ); for ( j = 0; j < sides.Num() && side->winding; j++ ) { if ( i == j ) { continue; } // keep the winding if on the clip plane side->winding = side->winding->Clip( -sides[j]->plane, BRUSH_EPSILON, true ); } if ( side->winding ) { for ( j = 0; j < side->winding->GetNumPoints(); j++ ) { bounds.AddPoint( (*side->winding)[j].ToVec3() ); } } } if ( bounds[0][0] > bounds[1][0] ) { return false; } for ( i = 0; i < 3; i++ ) { if ( bounds[0][i] < MIN_WORLD_COORD || bounds[1][i] > MAX_WORLD_COORD ) { return false; } } windingsValid = true; return true; } /* ============ idBrush::BoundBrush ============ */ void idBrush::BoundBrush( const idBrush *original ) { int i, j; idBrushSide *side; idWinding *w; assert( windingsValid ); bounds.Clear(); for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; w = side->winding; if ( !w ) { continue; } for ( j = 0; j < w->GetNumPoints(); j++ ) { bounds.AddPoint( (*w)[j].ToVec3() ); } } if ( bounds[0][0] > bounds[1][0] ) { if ( original ) { idBrushMap *bm = new idBrushMap( "error_brush", "_original" ); bm->WriteBrush( original ); delete bm; } common->Error( "idBrush::BoundBrush: brush %d on entity %d without windings", primitiveNum, entityNum ); } for ( i = 0; i < 3; i++ ) { if ( bounds[0][i] < MIN_WORLD_COORD || bounds[1][i] > MAX_WORLD_COORD ) { if ( original ) { idBrushMap *bm = new idBrushMap( "error_brush", "_original" ); bm->WriteBrush( original ); delete bm; } common->Error( "idBrush::BoundBrush: brush %d on entity %d is unbounded", primitiveNum, entityNum ); } } } /* ============ idBrush::FromSides ============ */ bool idBrush::FromSides( idList &sideList ) { int i; for ( i = 0; i < sideList.Num(); i++ ) { sides.Append( sideList[i] ); } sideList.Clear(); return CreateWindings(); } /* ============ idBrush::FromWinding ============ */ bool idBrush::FromWinding( const idWinding &w, const idPlane &windingPlane ) { int i, j, bestAxis; idPlane plane; idVec3 normal, axialNormal; sides.Append( new idBrushSide( windingPlane, -1 ) ); sides.Append( new idBrushSide( -windingPlane, -1 ) ); bestAxis = 0; for ( i = 1; i < 3; i++ ) { if ( idMath::Fabs( windingPlane.Normal()[i] ) > idMath::Fabs( windingPlane.Normal()[bestAxis] ) ) { bestAxis = i; } } axialNormal = vec3_origin; if ( windingPlane.Normal()[bestAxis] > 0.0f ) { axialNormal[bestAxis] = 1.0f; } else { axialNormal[bestAxis] = -1.0f; } for ( i = 0; i < w.GetNumPoints(); i++ ) { j = (i+1) % w.GetNumPoints(); normal = ( w[j].ToVec3() - w[i].ToVec3() ).Cross( axialNormal ); if ( normal.Normalize() < 0.5f ) { continue; } plane.SetNormal( normal ); plane.FitThroughPoint( w[j].ToVec3() ); sides.Append( new idBrushSide( plane, -1 ) ); } if ( sides.Num() < 4 ) { for ( i = 0; i < sides.Num(); i++ ) { delete sides[i]; } sides.Clear(); return false; } sides[0]->winding = w.Copy(); windingsValid = true; BoundBrush(); return true; } /* ============ idBrush::FromBounds ============ */ bool idBrush::FromBounds( const idBounds &bounds ) { int axis, dir; idVec3 normal; idPlane plane; for ( axis = 0; axis < 3; axis++ ) { for ( dir = -1; dir <= 1; dir += 2 ) { normal = vec3_origin; normal[axis] = dir; plane.SetNormal( normal ); plane.SetDist( dir * bounds[(dir == 1)][axis] ); sides.Append( new idBrushSide( plane, -1 ) ); } } return CreateWindings(); } /* ============ idBrush::Transform ============ */ void idBrush::Transform( const idVec3 &origin, const idMat3 &axis ) { int i; bool transformed = false; if ( axis.IsRotated() ) { for ( i = 0; i < sides.Num(); i++ ) { sides[i]->plane.RotateSelf( vec3_origin, axis ); } transformed = true; } if ( origin != vec3_origin ) { for ( i = 0; i < sides.Num(); i++ ) { sides[i]->plane.TranslateSelf( origin ); } transformed = true; } if ( transformed ) { CreateWindings(); } } /* ============ idBrush::GetVolume ============ */ float idBrush::GetVolume( void ) const { int i; idWinding *w; idVec3 corner; float d, area, volume; // grab the first valid point as a corner w = NULL; for ( i = 0; i < sides.Num(); i++ ) { w = sides[i]->winding; if ( w ) { break; } } if ( !w ) { return 0.0f; } corner = (*w)[0].ToVec3(); // create tetrahedrons to all other sides volume = 0.0f; for ( ; i < sides.Num(); i++) { w = sides[i]->winding; if ( !w ) { continue; } d = -( corner * sides[i]->plane.Normal() - sides[i]->plane.Dist() ); area = w->GetArea(); volume += d * area; } return ( volume * ( 1.0f / 3.0f ) ); } /* ============ idBrush::Subtract ============ */ bool idBrush::Subtract( const idBrush *b, idBrushList &list ) const { int i; idBrush *front, *back; const idBrush *in; list.Clear(); in = this; for ( i = 0; i < b->sides.Num() && in; i++ ) { in->Split( b->sides[i]->plane, b->sides[i]->planeNum, &front, &back ); if ( in != this ) { delete in; } if ( front ) { list.AddToTail( front ); } in = back; } // if didn't really intersect if ( !in ) { list.Free(); return false; } delete in; return true; } /* ============ idBrush::TryMerge ============ */ bool idBrush::TryMerge( const idBrush *brush, const idPlaneSet &planeList ) { int i, j, k, l, m, seperatingPlane; const idBrush *brushes[2]; const idWinding *w; const idPlane *plane; // brush bounds should overlap for ( i = 0; i < 3; i++ ) { if ( bounds[0][i] > brush->bounds[1][i] + 0.1f ) { return false; } if ( bounds[1][i] < brush->bounds[0][i] - 0.1f ) { return false; } } // the brushes should share an opposite plane seperatingPlane = -1; for ( i = 0; i < GetNumSides(); i++ ) { for ( j = 0; j < brush->GetNumSides(); j++ ) { if ( GetSide(i)->GetPlaneNum() == (brush->GetSide(j)->GetPlaneNum() ^ 1) ) { // may only have one seperating plane if ( seperatingPlane != -1 ) { return false; } seperatingPlane = GetSide(i)->GetPlaneNum(); break; } } } if ( seperatingPlane == -1 ) { return false; } brushes[0] = this; brushes[1] = brush; for ( i = 0; i < 2; i++ ) { j = !i; for ( k = 0; k < brushes[i]->GetNumSides(); k++ ) { // if the brush side plane is the seprating plane if ( !( ( brushes[i]->GetSide(k)->GetPlaneNum() ^ seperatingPlane ) >> 1 ) ) { continue; } plane = &brushes[i]->GetSide(k)->GetPlane(); // all the non seperating brush sides of the other brush should be at the back or on the plane for ( l = 0; l < brushes[j]->GetNumSides(); l++ ) { w = brushes[j]->GetSide(l)->GetWinding(); if ( !w ) { continue; } if ( !( ( brushes[j]->GetSide(l)->GetPlaneNum() ^ seperatingPlane ) >> 1 ) ) { continue; } for ( m = 0; m < w->GetNumPoints(); m++ ) { if ( plane->Distance( (*w)[m].ToVec3() ) > 0.1f ) { return false; } } } } } // add any sides from the other brush to this brush for ( i = 0; i < brush->GetNumSides(); i++ ) { for ( j = 0; j < GetNumSides(); j++ ) { if ( !( ( brush->GetSide(i)->GetPlaneNum() ^ GetSide(j)->GetPlaneNum() ) >> 1 ) ) { break; } } if ( j < GetNumSides() ) { sides[j]->flags &= brush->GetSide(i)->GetFlags(); continue; } sides.Append( brush->GetSide(i)->Copy() ); } // remove any side from this brush that is the opposite of a side of the other brush for ( i = 0; i < GetNumSides(); i++ ) { for ( j = 0; j < brush->GetNumSides(); j++ ) { if ( GetSide(i)->GetPlaneNum() == ( brush->GetSide(j)->GetPlaneNum() ^ 1 ) ) { break; } } if ( j < brush->GetNumSides() ) { delete sides[i]; sides.RemoveIndex(i); i--; continue; } } contents |= brush->contents; CreateWindings(); BoundBrush(); return true; } /* ============ idBrush::Split ============ */ int idBrush::Split( const idPlane &plane, int planeNum, idBrush **front, idBrush **back ) const { int res, i, j; idBrushSide *side, *frontSide, *backSide; float dist, maxBack, maxFront, *maxBackWinding, *maxFrontWinding; idWinding *w, *mid; assert( windingsValid ); if ( front ) { *front = NULL; } if ( back ) { *back = NULL; } res = bounds.PlaneSide( plane, -BRUSH_EPSILON ); if ( res == PLANESIDE_FRONT ) { if ( front ) { *front = Copy(); } return res; } if ( res == PLANESIDE_BACK ) { if ( back ) { *back = Copy(); } return res; } maxBackWinding = (float *) _alloca16( sides.Num() * sizeof(float) ); maxFrontWinding = (float *) _alloca16( sides.Num() * sizeof(float) ); maxFront = maxBack = 0.0f; for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; w = side->winding; if ( !w ) { continue; } maxBackWinding[i] = 10.0f; maxFrontWinding[i] = -10.0f; for ( j = 0; j < w->GetNumPoints(); j++ ) { dist = plane.Distance( (*w)[j].ToVec3() ); if ( dist > maxFrontWinding[i] ) { maxFrontWinding[i] = dist; } if ( dist < maxBackWinding[i] ) { maxBackWinding[i] = dist; } } if ( maxFrontWinding[i] > maxFront ) { maxFront = maxFrontWinding[i]; } if ( maxBackWinding[i] < maxBack ) { maxBack = maxBackWinding[i]; } } if ( maxFront < BRUSH_EPSILON ) { if ( back ) { *back = Copy(); } return PLANESIDE_BACK; } if ( maxBack > -BRUSH_EPSILON ) { if ( front ) { *front = Copy(); } return PLANESIDE_FRONT; } mid = new idWinding( plane.Normal(), plane.Dist() ); for ( i = 0; i < sides.Num() && mid; i++ ) { mid = mid->Clip( -sides[i]->plane, BRUSH_EPSILON, false ); } if ( mid ) { if ( mid->IsTiny() ) { delete mid; mid = NULL; } else if ( mid->IsHuge() ) { // if the winding is huge then the brush is unbounded common->Warning( "brush %d on entity %d is unbounded" "( %1.2f %1.2f %1.2f )-( %1.2f %1.2f %1.2f )-( %1.2f %1.2f %1.2f )", primitiveNum, entityNum, bounds[0][0], bounds[0][1], bounds[0][2], bounds[1][0], bounds[1][1], bounds[1][2], bounds[1][0]-bounds[0][0], bounds[1][1]-bounds[0][1], bounds[1][2]-bounds[0][2] ); delete mid; mid = NULL; } } if ( !mid ) { if ( maxFront > - maxBack ) { if ( front ) { *front = Copy(); } return PLANESIDE_FRONT; } else { if ( back ) { *back = Copy(); } return PLANESIDE_BACK; } } if ( !front && !back ) { delete mid; return PLANESIDE_CROSS; } *front = new idBrush(); (*front)->SetContents( contents ); (*front)->SetEntityNum( entityNum ); (*front)->SetPrimitiveNum( primitiveNum ); *back = new idBrush(); (*back)->SetContents( contents ); (*back)->SetEntityNum( entityNum ); (*back)->SetPrimitiveNum( primitiveNum ); for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; if ( !side->winding ) { continue; } // if completely at the front if ( maxBackWinding[i] >= BRUSH_EPSILON ) { (*front)->sides.Append( side->Copy() ); } // if completely at the back else if ( maxFrontWinding[i] <= -BRUSH_EPSILON ) { (*back)->sides.Append( side->Copy() ); } else { // split the side side->Split( plane, &frontSide, &backSide ); if ( frontSide ) { (*front)->sides.Append( frontSide ); } else if ( maxFrontWinding[i] > -BRUSH_EPSILON ) { // favor an overconstrained brush side = side->Copy(); side->winding = side->winding->Clip( idPlane( plane.Normal(), (plane.Dist() - (BRUSH_EPSILON+0.02f)) ), 0.01f, true ); assert( side->winding ); (*front)->sides.Append( side ); } if ( backSide ) { (*back)->sides.Append( backSide ); } else if ( maxBackWinding[i] < BRUSH_EPSILON ) { // favor an overconstrained brush side = side->Copy(); side->winding = side->winding->Clip( idPlane( -plane.Normal(), -(plane.Dist() + (BRUSH_EPSILON+0.02f)) ), 0.01f, true ); assert( side->winding ); (*back)->sides.Append( side ); } } } side = new idBrushSide( -plane, planeNum^1 ); side->winding = mid->Reverse(); side->flags |= SFL_SPLIT; (*front)->sides.Append( side ); (*front)->windingsValid = true; (*front)->BoundBrush( this ); side = new idBrushSide( plane, planeNum ); side->winding = mid; side->flags |= SFL_SPLIT; (*back)->sides.Append( side ); (*back)->windingsValid = true; (*back)->BoundBrush( this ); return PLANESIDE_CROSS; } /* ============ idBrush::AddBevelsForAxialBox ============ */ #define BRUSH_BEVEL_EPSILON 0.1f void idBrush::AddBevelsForAxialBox( void ) { int axis, dir, i, j, k, l, order; idBrushSide *side, *newSide; idPlane plane; idVec3 normal, vec; idWinding *w, *w2; float d, minBack; assert( windingsValid ); // add the axial planes order = 0; for ( axis = 0; axis < 3; axis++ ) { for ( dir = -1; dir <= 1; dir += 2, order++ ) { // see if the plane is already present for ( i = 0; i < sides.Num(); i++ ) { if ( dir > 0 ) { if ( sides[i]->plane.Normal()[axis] >= 0.9999f ) { break; } } else { if ( sides[i]->plane.Normal()[axis] <= -0.9999f ) { break; } } } if ( i >= sides.Num() ) { normal = vec3_origin; normal[axis] = dir; plane.SetNormal( normal ); plane.SetDist( dir * bounds[(dir == 1)][axis] ); newSide = new idBrushSide( plane, -1 ); newSide->SetFlag( SFL_BEVEL ); sides.Append( newSide ); } } } // if the brush is pure axial we're done if ( sides.Num() == 6 ) { return; } // test the non-axial plane edges for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; w = side->winding; if ( !w ) { continue; } for ( j = 0; j < w->GetNumPoints(); j++) { k = (j+1) % w->GetNumPoints(); vec = (*w)[j].ToVec3() - (*w)[k].ToVec3(); if ( vec.Normalize() < 0.5f ) { continue; } for ( k = 0; k < 3; k++ ) { if ( vec[k] == 1.0f || vec[k] == -1.0f || (vec[k] == 0.0f && vec[(k+1)%3] == 0.0f) ) { break; // axial } } if ( k < 3 ) { continue; // only test non-axial edges } // try the six possible slanted axials from this edge for ( axis = 0; axis < 3; axis++ ) { for ( dir = -1; dir <= 1; dir += 2 ) { // construct a plane normal = vec3_origin; normal[axis] = dir; normal = vec.Cross( normal ); if ( normal.Normalize() < 0.5f ) { continue; } plane.SetNormal( normal ); plane.FitThroughPoint( (*w)[j].ToVec3() ); // if all the points on all the sides are // behind this plane, it is a proper edge bevel for ( k = 0; k < sides.Num(); k++ ) { // if this plane has allready been used, skip it if ( plane.Compare( sides[k]->plane, 0.001f, 0.1f ) ) { break; } w2 = sides[k]->winding; if ( !w2 ) { continue; } minBack = 0.0f; for ( l = 0; l < w2->GetNumPoints(); l++ ) { d = plane.Distance( (*w2)[l].ToVec3() ); if ( d > BRUSH_BEVEL_EPSILON ) { break; // point at the front } if ( d < minBack ) { minBack = d; } } // if some point was at the front if ( l < w2->GetNumPoints() ) { break; } // if no points at the back then the winding is on the bevel plane if ( minBack > -BRUSH_BEVEL_EPSILON ) { break; } } if ( k < sides.Num() ) { continue; // wasn't part of the outer hull } // add this plane newSide = new idBrushSide( plane, -1 ); newSide->SetFlag( SFL_BEVEL ); sides.Append( newSide ); } } } } } /* ============ idBrush::ExpandForAxialBox ============ */ void idBrush::ExpandForAxialBox( const idBounds &bounds ) { int i, j; idBrushSide *side; idVec3 v; AddBevelsForAxialBox(); for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; for ( j = 0; j < 3; j++ ) { if ( side->plane.Normal()[j] > 0.0f ) { v[j] = bounds[0][j]; } else { v[j] = bounds[1][j]; } } side->plane.SetDist( side->plane.Dist() + v * -side->plane.Normal() ); } if ( !CreateWindings() ) { common->Error( "idBrush::ExpandForAxialBox: brush %d on entity %d imploded", primitiveNum, entityNum ); } /* // after expansion at least all non bevel sides should have a winding for ( i = 0; i < sides.Num(); i++ ) { side = sides[i]; if ( !side->winding ) { if ( !( side->flags & SFL_BEVEL ) ) { int shit = 1; } } } */ } /* ============ idBrush::Copy ============ */ idBrush *idBrush::Copy( void ) const { int i; idBrush *b; b = new idBrush(); b->entityNum = entityNum; b->primitiveNum = primitiveNum; b->contents = contents; b->windingsValid = windingsValid; b->bounds = bounds; for ( i = 0; i < sides.Num(); i++ ) { b->sides.Append( sides[i]->Copy() ); } return b; } //=============================================================== // // idBrushList // //=============================================================== /* ============ idBrushList::idBrushList ============ */ idBrushList::idBrushList( void ) { numBrushes = numBrushSides = 0; head = tail = NULL; } /* ============ idBrushList::~idBrushList ============ */ idBrushList::~idBrushList( void ) { } /* ============ idBrushList::GetBounds ============ */ idBounds idBrushList::GetBounds( void ) const { idBounds bounds; idBrush *b; bounds.Clear(); for ( b = Head(); b; b = b->Next() ) { bounds += b->GetBounds(); } return bounds; } /* ============ idBrushList::AddToTail ============ */ void idBrushList::AddToTail( idBrush *brush ) { brush->next = NULL; if ( tail ) { tail->next = brush; } tail = brush; if ( !head ) { head = brush; } numBrushes++; numBrushSides += brush->sides.Num(); } /* ============ idBrushList::AddToTail ============ */ void idBrushList::AddToTail( idBrushList &list ) { idBrush *brush, *next; for ( brush = list.head; brush; brush = next ) { next = brush->next; brush->next = NULL; if ( tail ) { tail->next = brush; } tail = brush; if ( !head ) { head = brush; } numBrushes++; numBrushSides += brush->sides.Num(); } list.head = list.tail = NULL; list.numBrushes = 0; } /* ============ idBrushList::AddToFront ============ */ void idBrushList::AddToFront( idBrush *brush ) { brush->next = head; head = brush; if ( !tail ) { tail = brush; } numBrushes++; numBrushSides += brush->sides.Num(); } /* ============ idBrushList::AddToFront ============ */ void idBrushList::AddToFront( idBrushList &list ) { idBrush *brush, *next; for ( brush = list.head; brush; brush = next ) { next = brush->next; brush->next = head; head = brush; if ( !tail ) { tail = brush; } numBrushes++; numBrushSides += brush->sides.Num(); } list.head = list.tail = NULL; list.numBrushes = 0; } /* ============ idBrushList::Remove ============ */ void idBrushList::Remove( idBrush *brush ) { idBrush *b, *last; last = NULL; for ( b = head; b; b = b->next ) { if ( b == brush ) { if ( last ) { last->next = b->next; } else { head = b->next; } if ( b == tail ) { tail = last; } numBrushes--; numBrushSides -= brush->sides.Num(); return; } last = b; } } /* ============ idBrushList::Delete ============ */ void idBrushList::Delete( idBrush *brush ) { idBrush *b, *last; last = NULL; for ( b = head; b; b = b->next ) { if ( b == brush ) { if ( last ) { last->next = b->next; } else { head = b->next; } if ( b == tail ) { tail = last; } numBrushes--; numBrushSides -= b->sides.Num(); delete b; return; } last = b; } } /* ============ idBrushList::Copy ============ */ idBrushList *idBrushList::Copy( void ) const { idBrush *brush; idBrushList *list; list = new idBrushList; for ( brush = head; brush; brush = brush->next ) { list->AddToTail( brush->Copy() ); } return list; } /* ============ idBrushList::Free ============ */ void idBrushList::Free( void ) { idBrush *brush, *next; for ( brush = head; brush; brush = next ) { next = brush->next; delete brush; } head = tail = NULL; numBrushes = numBrushSides = 0; } /* ============ idBrushList::Split ============ */ void idBrushList::Split( const idPlane &plane, int planeNum, idBrushList &frontList, idBrushList &backList, bool useBrushSavedPlaneSide ) { idBrush *b, *front, *back; frontList.Clear(); backList.Clear(); if ( !useBrushSavedPlaneSide ) { for ( b = head; b; b = b->next ) { b->Split( plane, planeNum, &front, &back ); if ( front ) { frontList.AddToTail( front ); } if ( back ) { backList.AddToTail( back ); } } return; } for ( b = head; b; b = b->next ) { if ( b->savedPlaneSide & BRUSH_PLANESIDE_BOTH ) { b->Split( plane, planeNum, &front, &back ); if ( front ) { frontList.AddToTail( front ); } if ( back ) { backList.AddToTail( back ); } } else if ( b->savedPlaneSide & BRUSH_PLANESIDE_FRONT ) { frontList.AddToTail( b->Copy() ); } else { backList.AddToTail( b->Copy() ); } } } /* ============ idBrushList::Chop ============ */ void idBrushList::Chop( bool (*ChopAllowed)( idBrush *b1, idBrush *b2 ) ) { idBrush *b1, *b2, *next; idBrushList sub1, sub2, keep; int i, j, c1, c2; idPlaneSet planeList; #ifdef OUTPUT_CHOP_STATS common->Printf( "[Brush CSG]\n"); common->Printf( "%6d original brushes\n", this->Num() ); #endif CreatePlaneList( planeList ); for ( b1 = this->Head(); b1; b1 = this->Head() ) { for ( b2 = b1->next; b2; b2 = next ) { next = b2->next; for ( i = 0; i < 3; i++ ) { if ( b1->bounds[0][i] >= b2->bounds[1][i] ) { break; } if ( b1->bounds[1][i] <= b2->bounds[0][i] ) { break; } } if ( i < 3 ) { continue; } for ( i = 0; i < b1->GetNumSides(); i++ ) { for ( j = 0; j < b2->GetNumSides(); j++ ) { if ( b1->GetSide(i)->GetPlaneNum() == ( b2->GetSide(j)->GetPlaneNum() ^ 1 ) ) { // opposite planes, so not touching break; } } if ( j < b2->GetNumSides() ) { break; } } if ( i < b1->GetNumSides() ) { continue; } sub1.Clear(); sub2.Clear(); c1 = 999999; c2 = 999999; // if b2 may chop up b1 if ( !ChopAllowed || ChopAllowed( b2, b1 ) ) { if ( !b1->Subtract( b2, sub1 ) ) { // didn't really intersect continue; } if ( sub1.IsEmpty() ) { // b1 is swallowed by b2 this->Delete( b1 ); break; } c1 = sub1.Num(); } // if b1 may chop up b2 if ( !ChopAllowed || ChopAllowed( b1, b2 ) ) { if ( !b2->Subtract( b1, sub2 ) ) { // didn't really intersect continue; } if ( sub2.IsEmpty() ) { // b2 is swallowed by b1 sub1.Free(); this->Delete( b2 ); continue; } c2 = sub2.Num(); } if ( sub1.IsEmpty() && sub2.IsEmpty() ) { continue; } // don't allow too much fragmentation if ( c1 > 2 && c2 > 2 ) { sub1.Free(); sub2.Free(); continue; } if ( c1 < c2 ) { sub2.Free(); this->AddToTail( sub1 ); this->Delete( b1 ); break; } else { sub1.Free(); this->AddToTail( sub2 ); this->Delete( b2 ); continue; } } if ( !b2 ) { // b1 is no longer intersecting anything, so keep it this->Remove( b1 ); keep.AddToTail( b1 ); #ifdef OUTPUT_CHOP_STATS DisplayRealTimeString( "\r%6d", keep.numBrushes ); #endif } } *this = keep; #ifdef OUTPUT_CHOP_STATS common->Printf( "\r%6d output brushes\n", Num() ); #endif } /* ============ idBrushList::Merge ============ */ void idBrushList::Merge( bool (*MergeAllowed)( idBrush *b1, idBrush *b2 ) ) { idPlaneSet planeList; idBrush *b1, *b2, *nextb2; int numMerges; common->Printf( "[Brush Merge]\n"); common->Printf( "%6d original brushes\n", Num() ); CreatePlaneList( planeList ); numMerges = 0; for ( b1 = Head(); b1; b1 = b1->next ) { for ( b2 = Head(); b2; b2 = nextb2 ) { nextb2 = b2->Next(); if ( b2 == b1 ) { continue; } if ( MergeAllowed && !MergeAllowed( b1, b2 ) ) { continue; } if ( b1->TryMerge( b2, planeList ) ) { Delete( b2 ); DisplayRealTimeString( "\r%6d", ++numMerges ); nextb2 = Head(); } } } common->Printf( "\r%6d brushes merged\n", numMerges ); } /* ============ idBrushList::SetFlagOnFacingBrushSides ============ */ void idBrushList::SetFlagOnFacingBrushSides( const idPlane &plane, int flag ) { int i; idBrush *b; const idWinding *w; for ( b = head; b; b = b->next ) { if ( idMath::Fabs( b->GetBounds().PlaneDistance( plane ) ) > 0.1f ) { continue; } for ( i = 0; i < b->GetNumSides(); i++ ) { w = b->GetSide(i)->GetWinding(); if ( !w ) { if ( b->GetSide(i)->GetPlane().Compare( plane, BRUSH_PLANE_NORMAL_EPSILON, BRUSH_PLANE_DIST_EPSILON ) ) { b->GetSide(i)->SetFlag( flag ); } continue; } if ( w->PlaneSide( plane ) == SIDE_ON ) { b->GetSide(i)->SetFlag( flag ); } } } } /* ============ idBrushList::CreatePlaneList ============ */ void idBrushList::CreatePlaneList( idPlaneSet &planeList ) const { int i; idBrush *b; idBrushSide *side; planeList.Resize( 512, 128 ); for ( b = Head(); b; b = b->Next() ) { for ( i = 0; i < b->GetNumSides(); i++ ) { side = b->GetSide( i ); side->SetPlaneNum( planeList.FindPlane( side->GetPlane(), BRUSH_PLANE_NORMAL_EPSILON, BRUSH_PLANE_DIST_EPSILON ) ); } } } /* ============ idBrushList::CreatePlaneList ============ */ void idBrushList::WriteBrushMap( const idStr &fileName, const idStr &ext ) const { idBrushMap *map; map = new idBrushMap( fileName, ext ); map->WriteBrushList( *this ); delete map; } //=============================================================== // // idBrushMap // //=============================================================== /* ============ idBrushMap::idBrushMap ============ */ idBrushMap::idBrushMap( const idStr &fileName, const idStr &ext ) { idStr qpath; qpath = fileName; qpath.StripFileExtension(); qpath += ext; qpath.SetFileExtension( "map" ); common->Printf( "writing %s...\n", qpath.c_str() ); fp = fileSystem->OpenFileWrite( qpath, "fs_devpath" ); if ( !fp ) { common->Error( "Couldn't open %s\n", qpath.c_str() ); return; } texture = "textures/washroom/btile01"; fp->WriteFloatString( "Version %1.2f\n", (float) CURRENT_MAP_VERSION ); fp->WriteFloatString( "{\n" ); fp->WriteFloatString( "\"classname\" \"worldspawn\"\n" ); brushCount = 0; } /* ============ idBrushMap::~idBrushMap ============ */ idBrushMap::~idBrushMap( void ) { if ( !fp ) { return; } fp->WriteFloatString( "}\n" ); fileSystem->CloseFile( fp ); } /* ============ idBrushMap::WriteBrush ============ */ void idBrushMap::WriteBrush( const idBrush *brush ) { int i; idBrushSide *side; if ( !fp ) { return; } fp->WriteFloatString( "// primitive %d\n{\nbrushDef3\n{\n", brushCount++ ); for ( i = 0; i < brush->GetNumSides(); i++ ) { side = brush->GetSide( i ); fp->WriteFloatString( " ( %f %f %f %f ) ", side->GetPlane()[0], side->GetPlane()[1], side->GetPlane()[2], -side->GetPlane().Dist() ); fp->WriteFloatString( "( ( 0.031250 0 0 ) ( 0 0.031250 0 ) ) %s 0 0 0\n", texture.c_str() ); } fp->WriteFloatString( "}\n}\n" ); } /* ============ idBrushMap::WriteBrushList ============ */ void idBrushMap::WriteBrushList( const idBrushList &brushList ) { idBrush *b; if ( !fp ) { return; } for ( b = brushList.Head(); b; b = b->Next() ) { WriteBrush( b ); } }