2009-04-19 18:07:22 +00:00
#region = = = = = = = = = = = = = = = = = = Copyright ( c ) 2007 Pascal vd Heiden
/ *
* Copyright ( c ) 2007 Pascal vd Heiden , www . codeimp . com
* This program is released under GNU General Public License
*
* This program 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 .
*
* /
#endregion
#region = = = = = = = = = = = = = = = = = = Namespaces
using System ;
using System.Collections.Generic ;
using CodeImp.DoomBuilder.Geometry ;
2010-08-15 19:43:00 +00:00
using CodeImp.DoomBuilder.Windows ;
2009-04-19 18:07:22 +00:00
using System.Drawing ;
using CodeImp.DoomBuilder.IO ;
using CodeImp.DoomBuilder.Types ;
using System.IO ;
2009-07-09 14:03:47 +00:00
using CodeImp.DoomBuilder.Config ;
2009-04-19 18:07:22 +00:00
#endregion
namespace CodeImp.DoomBuilder.Map
{
2009-05-19 09:59:19 +00:00
/// <summary>
/// Manages all geometry structures and things in a map. Also provides
/// methods to works with selections and marking elements for any purpose.
/// Note that most methods are of O(n) complexity.
/// </summary>
2009-04-19 18:07:22 +00:00
public sealed class MapSet
{
#region = = = = = = = = = = = = = = = = = = Constants
2009-05-19 09:59:19 +00:00
/// <summary>Stiching distance. This is only to get around inaccuracies. Basically,
/// geometry only stitches when exactly on top of each other.</summary>
2009-04-19 18:07:22 +00:00
public const float STITCH_DISTANCE = 0.001f ;
// Virtual sector identification
// This contains a character that is invalid in the UDMF standard, but valid
// in our parser, so that it can only be used by Doom Builder and will never
// conflict with any other valid UDMF field.
internal const string VIRTUAL_SECTOR_FIELD = "!virtual_sector" ;
2009-07-09 14:03:47 +00:00
// Handler for tag fields
public delegate void TagHandler < T > ( MapElement element , bool actionargument , UniversalType type , ref int value , T obj ) ;
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Variables
// Sector indexing
private List < int > indexholes ;
private int lastsectorindex ;
// Sidedef indexing for (de)serialization
private Sidedef [ ] sidedefindices ;
// Map structures
2009-06-05 19:03:56 +00:00
private Vertex [ ] vertices ;
private Linedef [ ] linedefs ;
private Sidedef [ ] sidedefs ;
private Sector [ ] sectors ;
private Thing [ ] things ;
private int numvertices ;
private int numlinedefs ;
private int numsidedefs ;
private int numsectors ;
private int numthings ;
2013-03-18 13:52:27 +00:00
//mxd
2013-04-01 11:06:01 +00:00
private GroupInfo [ ] groupInfos ;
2009-06-05 19:03:56 +00:00
2009-06-07 10:26:06 +00:00
// Behavior
private int freezearrays ;
private bool autoremove ;
2009-04-19 18:07:22 +00:00
// Selected elements
private LinkedList < Vertex > sel_vertices ;
private LinkedList < Linedef > sel_linedefs ;
private LinkedList < Sector > sel_sectors ;
private LinkedList < Thing > sel_things ;
private SelectionType sel_type ;
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Statics
private static long emptylongname ;
private static UniValue virtualsectorvalue ;
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Disposing
2014-01-10 15:08:39 +00:00
private bool isdisposed ;
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Properties
2009-07-09 14:03:47 +00:00
/// <summary>Returns the number of selected sectors.</summary>
public int SelectedSectorsCount { get { return sel_sectors . Count ; } }
/// <summary>Returns the number of selected linedefs.</summary>
public int SelectedLinedefsCount { get { return sel_linedefs . Count ; } }
2009-04-19 18:07:22 +00:00
2009-07-09 14:03:47 +00:00
/// <summary>Returns the number of selected vertices.</summary>
public int SelectedVerticessCount { get { return sel_vertices . Count ; } }
/// <summary>Returns the number of selected things.</summary>
public int SelectedThingsCount { get { return sel_things . Count ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of all vertices.</summary>
2009-06-05 19:03:56 +00:00
public ICollection < Vertex > Vertices { get { if ( freezearrays = = 0 ) return vertices ; else return new MapElementCollection < Vertex > ( ref vertices , numvertices ) ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of all linedefs.</summary>
2009-06-05 19:03:56 +00:00
public ICollection < Linedef > Linedefs { get { if ( freezearrays = = 0 ) return linedefs ; else return new MapElementCollection < Linedef > ( ref linedefs , numlinedefs ) ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of all sidedefs.</summary>
2009-06-05 19:03:56 +00:00
public ICollection < Sidedef > Sidedefs { get { if ( freezearrays = = 0 ) return sidedefs ; else return new MapElementCollection < Sidedef > ( ref sidedefs , numsidedefs ) ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of all sectors.</summary>
2009-06-05 19:03:56 +00:00
public ICollection < Sector > Sectors { get { if ( freezearrays = = 0 ) return sectors ; else return new MapElementCollection < Sector > ( ref sectors , numsectors ) ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of all things.</summary>
2009-06-05 19:03:56 +00:00
public ICollection < Thing > Things { get { if ( freezearrays = = 0 ) return things ; else return new MapElementCollection < Thing > ( ref things , numthings ) ; } }
2009-04-19 18:07:22 +00:00
2009-05-19 09:59:19 +00:00
/// <summary>Indicates if the map is disposed.</summary>
public bool IsDisposed { get { return isdisposed ; } }
/// <summary>Returns a reference to the list of selected vertices.</summary>
2009-04-19 18:07:22 +00:00
internal LinkedList < Vertex > SelectedVertices { get { return sel_vertices ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of selected linedefs.</summary>
2009-04-19 18:07:22 +00:00
internal LinkedList < Linedef > SelectedLinedefs { get { return sel_linedefs ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of selected sectors.</summary>
2009-04-19 18:07:22 +00:00
internal LinkedList < Sector > SelectedSectors { get { return sel_sectors ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns a reference to the list of selected things.</summary>
2009-04-19 18:07:22 +00:00
internal LinkedList < Thing > SelectedThings { get { return sel_things ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns the current type of selection.</summary>
public SelectionType SelectionType { get { return sel_type ; } set { sel_type = value ; } }
/// <summary>Long name to indicate "no texture". This is the long name for a single dash.</summary>
2009-04-19 18:07:22 +00:00
public static long EmptyLongName { get { return emptylongname ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns the name of the custom field that marks virtual sectors in pasted geometry.</summary>
2009-04-19 18:07:22 +00:00
public static string VirtualSectorField { get { return VIRTUAL_SECTOR_FIELD ; } }
2009-05-19 09:59:19 +00:00
/// <summary>Returns the value of the custom field that marks virtual sectors in pasted geometry.</summary>
2009-04-19 18:07:22 +00:00
public static UniValue VirtualSectorValue { get { return virtualsectorvalue ; } }
internal Sidedef [ ] SidedefIndices { get { return sidedefindices ; } }
2009-06-07 10:26:06 +00:00
internal bool AutoRemove { get { return autoremove ; } set { autoremove = value ; } }
2013-04-01 11:06:01 +00:00
public GroupInfo [ ] GroupInfos { get { return groupInfos ; } } //mxd
2013-03-18 13:52:27 +00:00
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Constructor / Disposer
// Constructor for new empty map
internal MapSet ( )
{
// Initialize
2009-06-05 19:03:56 +00:00
vertices = new Vertex [ 0 ] ;
linedefs = new Linedef [ 0 ] ;
sidedefs = new Sidedef [ 0 ] ;
sectors = new Sector [ 0 ] ;
things = new Thing [ 0 ] ;
2009-04-19 18:07:22 +00:00
sel_vertices = new LinkedList < Vertex > ( ) ;
sel_linedefs = new LinkedList < Linedef > ( ) ;
sel_sectors = new LinkedList < Sector > ( ) ;
sel_things = new LinkedList < Thing > ( ) ;
indexholes = new List < int > ( ) ;
lastsectorindex = 0 ;
2009-06-07 10:26:06 +00:00
autoremove = true ;
2013-04-01 11:06:01 +00:00
groupInfos = new GroupInfo [ 10 ] ; //mxd
2009-04-19 18:07:22 +00:00
// We have no destructor
GC . SuppressFinalize ( this ) ;
}
// Constructor for map to deserialize
internal MapSet ( MemoryStream stream )
{
// Initialize
2009-06-05 19:03:56 +00:00
vertices = new Vertex [ 0 ] ;
linedefs = new Linedef [ 0 ] ;
sidedefs = new Sidedef [ 0 ] ;
sectors = new Sector [ 0 ] ;
things = new Thing [ 0 ] ;
2009-04-19 18:07:22 +00:00
sel_vertices = new LinkedList < Vertex > ( ) ;
sel_linedefs = new LinkedList < Linedef > ( ) ;
sel_sectors = new LinkedList < Sector > ( ) ;
sel_things = new LinkedList < Thing > ( ) ;
indexholes = new List < int > ( ) ;
lastsectorindex = 0 ;
2009-06-07 10:26:06 +00:00
autoremove = true ;
2013-04-01 11:06:01 +00:00
groupInfos = new GroupInfo [ 10 ] ; //mxd
2009-04-19 18:07:22 +00:00
// Deserialize
Deserialize ( stream ) ;
// We have no destructor
GC . SuppressFinalize ( this ) ;
}
// Disposer
internal void Dispose ( )
{
// Not already disposed?
if ( ! isdisposed )
{
// Already set isdisposed so that changes can be prohibited
isdisposed = true ;
2009-06-05 19:03:56 +00:00
BeginAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
// Dispose all things
2009-06-05 19:03:56 +00:00
while ( ( things . Length > 0 ) & & ( things [ 0 ] ! = null ) )
things [ 0 ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
// Dispose all sectors
2009-06-05 19:03:56 +00:00
while ( ( sectors . Length > 0 ) & & ( sectors [ 0 ] ! = null ) )
sectors [ 0 ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
// Dispose all sidedefs
2009-06-05 19:03:56 +00:00
while ( ( sidedefs . Length > 0 ) & & ( sidedefs [ 0 ] ! = null ) )
sidedefs [ 0 ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
// Dispose all linedefs
2009-06-05 19:03:56 +00:00
while ( ( linedefs . Length > 0 ) & & ( linedefs [ 0 ] ! = null ) )
linedefs [ 0 ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
// Dispose all vertices
2009-06-05 19:03:56 +00:00
while ( ( vertices . Length > 0 ) & & ( vertices [ 0 ] ! = null ) )
vertices [ 0 ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
// Clean up
vertices = null ;
linedefs = null ;
sidedefs = null ;
sectors = null ;
things = null ;
sel_vertices = null ;
sel_linedefs = null ;
sel_sectors = null ;
sel_things = null ;
indexholes = null ;
2013-04-01 11:06:01 +00:00
groupInfos = null ; //mxd
2009-04-19 18:07:22 +00:00
// Done
isdisposed = true ;
}
}
// Static initializer
internal static void Initialize ( )
{
emptylongname = Lump . MakeLongName ( "-" ) ;
2013-08-14 09:02:37 +00:00
virtualsectorvalue = new UniValue ( ( int ) UniversalType . Integer , 0 ) ;
2009-04-19 18:07:22 +00:00
}
#endregion
#region = = = = = = = = = = = = = = = = = = Management
2009-05-19 09:59:19 +00:00
2009-06-05 19:03:56 +00:00
// This begins large add/remove operations
public void BeginAddRemove ( )
{
freezearrays + + ;
}
// This allocates the arrays to a minimum size so that
// a lot of items can be created faster. This function will never
// allocate less than the current number of items.
public void SetCapacity ( int nvertices , int nlinedefs , int nsidedefs , int nsectors , int nthings )
{
if ( freezearrays = = 0 )
throw new Exception ( "You must call BeginAddRemove before setting the reserved capacity." ) ;
if ( numvertices < nvertices )
Array . Resize ( ref vertices , nvertices ) ;
if ( numlinedefs < nlinedefs )
Array . Resize ( ref linedefs , nlinedefs ) ;
if ( numsidedefs < nsidedefs )
Array . Resize ( ref sidedefs , nsidedefs ) ;
if ( numsectors < nsectors )
Array . Resize ( ref sectors , nsectors ) ;
if ( numthings < nthings )
Array . Resize ( ref things , nthings ) ;
}
// This ends add/remove operations and crops the arrays
public void EndAddRemove ( )
{
if ( freezearrays > 0 )
freezearrays - - ;
if ( freezearrays = = 0 )
{
if ( numvertices < vertices . Length )
Array . Resize ( ref vertices , numvertices ) ;
if ( numlinedefs < linedefs . Length )
Array . Resize ( ref linedefs , numlinedefs ) ;
if ( numsidedefs < sidedefs . Length )
Array . Resize ( ref sidedefs , numsidedefs ) ;
if ( numsectors < sectors . Length )
Array . Resize ( ref sectors , numsectors ) ;
if ( numthings < things . Length )
Array . Resize ( ref things , numthings ) ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>
/// This makes a deep copy and returns the new MapSet.
/// </summary>
2009-04-19 18:07:22 +00:00
public MapSet Clone ( )
{
Linedef nl ;
Sidedef nd ;
// Create the map set
MapSet newset = new MapSet ( ) ;
2009-06-05 19:03:56 +00:00
newset . BeginAddRemove ( ) ;
2009-06-12 08:02:47 +00:00
newset . SetCapacity ( numvertices , numlinedefs , numsidedefs , numsectors , numthings ) ;
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Go for all vertices
foreach ( Vertex v in vertices )
{
// Make new vertex
v . Clone = newset . CreateVertex ( v . Position ) ;
v . CopyPropertiesTo ( v . Clone ) ;
}
// Go for all sectors
foreach ( Sector s in sectors )
{
// Make new sector
s . Clone = newset . CreateSector ( ) ;
s . CopyPropertiesTo ( s . Clone ) ;
}
// Go for all linedefs
foreach ( Linedef l in linedefs )
{
// Make new linedef
nl = newset . CreateLinedef ( l . Start . Clone , l . End . Clone ) ;
l . CopyPropertiesTo ( nl ) ;
// Linedef has a front side?
if ( l . Front ! = null )
{
// Make new sidedef
nd = newset . CreateSidedef ( nl , true , l . Front . Sector . Clone ) ;
l . Front . CopyPropertiesTo ( nd ) ;
}
// Linedef has a back side?
if ( l . Back ! = null )
{
// Make new sidedef
nd = newset . CreateSidedef ( nl , false , l . Back . Sector . Clone ) ;
l . Back . CopyPropertiesTo ( nd ) ;
}
}
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Go for all things
foreach ( Thing t in things )
{
// Make new thing
Thing nt = newset . CreateThing ( ) ;
t . CopyPropertiesTo ( nt ) ;
}
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Remove clone references
foreach ( Vertex v in vertices ) v . Clone = null ;
foreach ( Sector s in sectors ) s . Clone = null ;
// Return the new set
2009-06-05 19:03:56 +00:00
newset . EndAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
return newset ;
}
// This makes a deep copy of the marked geometry and binds missing sectors to a virtual sector
internal MapSet CloneMarked ( )
{
Sector virtualsector = null ;
// Create the map set
MapSet newset = new MapSet ( ) ;
2009-06-05 19:03:56 +00:00
newset . BeginAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
// Get marked geometry
ICollection < Vertex > mvertices = GetMarkedVertices ( true ) ;
ICollection < Linedef > mlinedefs = GetMarkedLinedefs ( true ) ;
ICollection < Sector > msectors = GetMarkedSectors ( true ) ;
ICollection < Thing > mthings = GetMarkedThings ( true ) ;
2009-06-12 08:02:47 +00:00
newset . SetCapacity ( mvertices . Count , mlinedefs . Count , numsidedefs , msectors . Count , mthings . Count ) ;
2009-04-19 18:07:22 +00:00
// Go for all vertices
foreach ( Vertex v in mvertices )
{
// Make new vertex
v . Clone = newset . CreateVertex ( v . Position ) ;
v . CopyPropertiesTo ( v . Clone ) ;
}
// Go for all sectors
foreach ( Sector s in msectors )
{
// Make new sector
s . Clone = newset . CreateSector ( ) ;
s . CopyPropertiesTo ( s . Clone ) ;
}
// Go for all linedefs
foreach ( Linedef l in mlinedefs )
{
// Make new linedef
Linedef nl = newset . CreateLinedef ( l . Start . Clone , l . End . Clone ) ;
l . CopyPropertiesTo ( nl ) ;
// Linedef has a front side?
if ( l . Front ! = null )
{
Sidedef nd ;
// Sector on front side marked?
if ( l . Front . Sector . Marked )
{
// Make new sidedef
nd = newset . CreateSidedef ( nl , true , l . Front . Sector . Clone ) ;
}
else
{
// Make virtual sector if needed
if ( virtualsector = = null )
{
virtualsector = newset . CreateSector ( ) ;
l . Front . Sector . CopyPropertiesTo ( virtualsector ) ;
2009-06-11 21:21:20 +00:00
virtualsector . Fields . BeforeFieldsChange ( ) ;
2009-04-19 18:07:22 +00:00
virtualsector . Fields [ VIRTUAL_SECTOR_FIELD ] = new UniValue ( virtualsectorvalue ) ;
}
// Make new sidedef that links to the virtual sector
nd = newset . CreateSidedef ( nl , true , virtualsector ) ;
}
l . Front . CopyPropertiesTo ( nd ) ;
}
// Linedef has a back side?
if ( l . Back ! = null )
{
Sidedef nd ;
// Sector on front side marked?
if ( l . Back . Sector . Marked )
{
// Make new sidedef
nd = newset . CreateSidedef ( nl , false , l . Back . Sector . Clone ) ;
}
else
{
// Make virtual sector if needed
if ( virtualsector = = null )
{
virtualsector = newset . CreateSector ( ) ;
l . Back . Sector . CopyPropertiesTo ( virtualsector ) ;
2009-06-11 21:21:20 +00:00
virtualsector . Fields . BeforeFieldsChange ( ) ;
2009-04-19 18:07:22 +00:00
virtualsector . Fields [ VIRTUAL_SECTOR_FIELD ] = new UniValue ( virtualsectorvalue ) ;
}
// Make new sidedef that links to the virtual sector
nd = newset . CreateSidedef ( nl , false , virtualsector ) ;
}
l . Back . CopyPropertiesTo ( nd ) ;
}
}
// Go for all things
foreach ( Thing t in mthings )
{
// Make new thing
Thing nt = newset . CreateThing ( ) ;
t . CopyPropertiesTo ( nt ) ;
}
// Remove clone references
foreach ( Vertex v in vertices ) v . Clone = null ;
foreach ( Sector s in sectors ) s . Clone = null ;
// Return the new set
2009-06-05 19:03:56 +00:00
newset . EndAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
return newset ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates a new vertex and returns it.</summary>
2009-04-19 18:07:22 +00:00
public Vertex CreateVertex ( Vector2D pos )
{
2010-08-15 19:43:00 +00:00
if ( numvertices = = General . Map . FormatInterface . MaxVertices )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of vertices reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Make the vertex
2009-06-05 19:03:56 +00:00
Vertex v = new Vertex ( this , numvertices , pos ) ;
2009-06-07 09:00:14 +00:00
AddItem ( v , ref vertices , numvertices , ref numvertices ) ;
return v ;
}
/// <summary>This creates a new vertex and returns it.</summary>
public Vertex CreateVertex ( int index , Vector2D pos )
{
2010-08-15 19:43:00 +00:00
if ( numvertices = = General . Map . FormatInterface . MaxVertices )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of vertices reached." ) ;
return null ;
}
2009-06-07 09:00:14 +00:00
// Make the vertex
Vertex v = new Vertex ( this , index , pos ) ;
AddItem ( v , ref vertices , index , ref numvertices ) ;
2009-04-19 18:07:22 +00:00
return v ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates a new linedef and returns it.</summary>
2009-04-19 18:07:22 +00:00
public Linedef CreateLinedef ( Vertex start , Vertex end )
{
2010-08-15 19:43:00 +00:00
if ( numlinedefs = = General . Map . FormatInterface . MaxLinedefs )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of linedefs reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Make the linedef
2009-06-05 19:03:56 +00:00
Linedef l = new Linedef ( this , numlinedefs , start , end ) ;
2009-06-07 09:00:14 +00:00
AddItem ( l , ref linedefs , numlinedefs , ref numlinedefs ) ;
return l ;
}
2009-04-19 18:07:22 +00:00
2009-06-07 09:00:14 +00:00
/// <summary>This creates a new linedef and returns it.</summary>
public Linedef CreateLinedef ( int index , Vertex start , Vertex end )
{
2010-08-15 19:43:00 +00:00
if ( numlinedefs = = General . Map . FormatInterface . MaxLinedefs )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of linedefs reached." ) ;
return null ;
}
2009-06-07 09:00:14 +00:00
// Make the linedef
Linedef l = new Linedef ( this , index , start , end ) ;
AddItem ( l , ref linedefs , index , ref numlinedefs ) ;
2009-04-19 18:07:22 +00:00
return l ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates a new sidedef and returns it.</summary>
2009-04-19 18:07:22 +00:00
public Sidedef CreateSidedef ( Linedef l , bool front , Sector s )
{
2010-08-15 19:43:00 +00:00
if ( numsidedefs = = int . MaxValue )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of sidedefs reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Make the sidedef
2009-06-05 19:03:56 +00:00
Sidedef sd = new Sidedef ( this , numsidedefs , l , front , s ) ;
2009-06-07 09:00:14 +00:00
AddItem ( sd , ref sidedefs , numsidedefs , ref numsidedefs ) ;
return sd ;
}
2009-04-19 18:07:22 +00:00
2009-06-07 09:00:14 +00:00
/// <summary>This creates a new sidedef and returns it.</summary>
public Sidedef CreateSidedef ( int index , Linedef l , bool front , Sector s )
{
2010-08-15 19:43:00 +00:00
if ( numsidedefs = = int . MaxValue )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of sidedefs reached." ) ;
return null ;
}
2009-06-07 09:00:14 +00:00
// Make the sidedef
Sidedef sd = new Sidedef ( this , index , l , front , s ) ;
AddItem ( sd , ref sidedefs , index , ref numsidedefs ) ;
2009-04-19 18:07:22 +00:00
return sd ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates a new sector and returns it.</summary>
2009-04-19 18:07:22 +00:00
public Sector CreateSector ( )
{
2009-06-07 09:00:14 +00:00
// Make the sector
return CreateSector ( numsectors ) ;
}
/// <summary>This creates a new sector and returns it.</summary>
public Sector CreateSector ( int index )
{
int fixedindex ;
2010-08-15 19:43:00 +00:00
if ( numsectors = = General . Map . FormatInterface . MaxSectors )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of sectors reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Do we have any index holes we can use?
if ( indexholes . Count > 0 )
{
// Take one of the index holes
2009-06-07 09:00:14 +00:00
fixedindex = indexholes [ indexholes . Count - 1 ] ;
2009-04-19 18:07:22 +00:00
indexholes . RemoveAt ( indexholes . Count - 1 ) ;
}
else
{
// Make a new index
2009-06-07 09:00:14 +00:00
fixedindex = lastsectorindex + + ;
2009-04-19 18:07:22 +00:00
}
// Make the sector
2009-06-07 09:00:14 +00:00
return CreateSectorEx ( fixedindex , index ) ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
// This creates a new sector with a specific fixed index
2009-06-07 09:00:14 +00:00
private Sector CreateSectorEx ( int fixedindex , int index )
2009-04-19 18:07:22 +00:00
{
2010-08-15 19:43:00 +00:00
if ( numsectors = = General . Map . FormatInterface . MaxSectors )
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of sectors reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Make the sector
2009-06-07 09:00:14 +00:00
Sector s = new Sector ( this , index , fixedindex ) ;
AddItem ( s , ref sectors , index , ref numsectors ) ;
2009-04-19 18:07:22 +00:00
return s ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates a new thing and returns it.</summary>
2009-04-19 18:07:22 +00:00
public Thing CreateThing ( )
{
2013-09-11 09:47:53 +00:00
if ( numthings = = General . Map . FormatInterface . MaxThings )
2010-08-15 19:43:00 +00:00
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of things reached." ) ;
return null ;
}
2009-04-19 18:07:22 +00:00
// Make the thing
2009-06-05 19:03:56 +00:00
Thing t = new Thing ( this , numthings ) ;
2009-06-07 09:00:14 +00:00
AddItem ( t , ref things , numthings , ref numthings ) ;
return t ;
}
2009-04-19 18:07:22 +00:00
2009-06-07 09:00:14 +00:00
/// <summary>This creates a new thing and returns it.</summary>
public Thing CreateThing ( int index )
{
2013-09-11 09:47:53 +00:00
if ( numthings = = General . Map . FormatInterface . MaxThings )
2010-08-15 19:43:00 +00:00
{
General . Interface . DisplayStatus ( StatusType . Warning , "Failed to complete operation: maximum number of things reached." ) ;
return null ;
}
2009-06-07 09:00:14 +00:00
// Make the thing
Thing t = new Thing ( this , index ) ;
AddItem ( t , ref things , index , ref numthings ) ;
2009-04-19 18:07:22 +00:00
return t ;
}
2009-06-07 09:00:14 +00:00
// This increases the size of the array to add an item
private void AddItem < T > ( T item , ref T [ ] array , int index , ref int counter ) where T : MapElement
{
// Only resize when there are no more free entries
if ( counter = = array . Length )
{
if ( freezearrays = = 0 )
Array . Resize ( ref array , counter + 1 ) ;
else
Array . Resize ( ref array , counter + 10 ) ;
}
// Move item at the given index if the new item is not added at the end
if ( index ! = counter )
{
array [ counter ] = array [ index ] ;
array [ counter ] . Index = counter ;
}
// Add item
array [ index ] = item ;
counter + + ;
}
2009-05-19 09:59:19 +00:00
2009-04-19 18:07:22 +00:00
// This adds a sector index hole
2009-05-19 09:59:19 +00:00
internal void AddSectorIndexHole ( int index )
2009-04-19 18:07:22 +00:00
{
indexholes . Add ( index ) ;
}
2009-06-05 19:03:56 +00:00
private void RemoveItem < T > ( ref T [ ] array , int index , ref int counter ) where T : MapElement
{
if ( index = = ( counter - 1 ) )
{
array [ index ] = null ;
}
else
{
array [ index ] = array [ counter - 1 ] ;
array [ index ] . Index = index ;
array [ counter - 1 ] = null ;
}
counter - - ;
if ( freezearrays = = 0 )
Array . Resize ( ref array , counter ) ;
}
internal void RemoveVertex ( int index )
{
RemoveItem ( ref vertices , index , ref numvertices ) ;
}
internal void RemoveLinedef ( int index )
{
RemoveItem ( ref linedefs , index , ref numlinedefs ) ;
}
internal void RemoveSidedef ( int index )
{
RemoveItem ( ref sidedefs , index , ref numsidedefs ) ;
}
internal void RemoveSector ( int index )
{
RemoveItem ( ref sectors , index , ref numsectors ) ;
}
2009-04-19 18:07:22 +00:00
2009-06-05 19:03:56 +00:00
internal void RemoveThing ( int index )
{
RemoveItem ( ref things , index , ref numthings ) ;
}
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Serialization
// This serializes the MapSet
internal MemoryStream Serialize ( )
{
MemoryStream stream = new MemoryStream ( 20000000 ) ; // Yes that is about 20 MB.
SerializerStream serializer = new SerializerStream ( stream ) ;
// Index the sidedefs
int sidedefindex = 0 ;
foreach ( Sidedef sd in sidedefs )
sd . SerializedIndex = sidedefindex + + ;
serializer . Begin ( ) ;
// Write private data
serializer . wInt ( lastsectorindex ) ;
serializer . wInt ( indexholes . Count ) ;
foreach ( int i in indexholes ) serializer . wInt ( i ) ;
// Write map data
WriteVertices ( serializer ) ;
WriteSectors ( serializer ) ;
WriteLinedefs ( serializer ) ;
WriteSidedefs ( serializer ) ;
WriteThings ( serializer ) ;
serializer . End ( ) ;
// Reallocate to keep only the used memory
stream . Capacity = ( int ) stream . Length ;
return stream ;
}
// This serializes things
private void WriteThings ( SerializerStream stream )
{
2009-06-05 19:03:56 +00:00
stream . wInt ( numthings ) ;
2009-04-19 18:07:22 +00:00
// Go for all things
foreach ( Thing t in things )
{
t . ReadWrite ( stream ) ;
}
}
// This serializes vertices
private void WriteVertices ( SerializerStream stream )
{
2009-06-05 19:03:56 +00:00
stream . wInt ( numvertices ) ;
2009-04-19 18:07:22 +00:00
// Go for all vertices
int index = 0 ;
foreach ( Vertex v in vertices )
{
v . SerializedIndex = index + + ;
v . ReadWrite ( stream ) ;
}
}
// This serializes linedefs
private void WriteLinedefs ( SerializerStream stream )
{
2009-06-05 19:03:56 +00:00
stream . wInt ( numlinedefs ) ;
2009-04-19 18:07:22 +00:00
// Go for all lines
int index = 0 ;
foreach ( Linedef l in linedefs )
{
l . SerializedIndex = index + + ;
stream . wInt ( l . Start . SerializedIndex ) ;
stream . wInt ( l . End . SerializedIndex ) ;
l . ReadWrite ( stream ) ;
}
}
// This serializes sidedefs
private void WriteSidedefs ( SerializerStream stream )
{
2009-06-05 19:03:56 +00:00
stream . wInt ( numsidedefs ) ;
2009-04-19 18:07:22 +00:00
// Go for all sidedefs
foreach ( Sidedef sd in sidedefs )
{
stream . wInt ( sd . Line . SerializedIndex ) ;
stream . wInt ( sd . Sector . SerializedIndex ) ;
stream . wBool ( sd . IsFront ) ;
sd . ReadWrite ( stream ) ;
}
}
// This serializes sectors
private void WriteSectors ( SerializerStream stream )
{
2009-06-05 19:03:56 +00:00
stream . wInt ( numsectors ) ;
2009-04-19 18:07:22 +00:00
// Go for all sectors
int index = 0 ;
foreach ( Sector s in sectors )
{
s . SerializedIndex = index + + ;
s . ReadWrite ( stream ) ;
}
}
#endregion
#region = = = = = = = = = = = = = = = = = = Deserialization
// This serializes the MapSet
private void Deserialize ( MemoryStream stream )
{
stream . Seek ( 0 , SeekOrigin . Begin ) ;
DeserializerStream deserializer = new DeserializerStream ( stream ) ;
deserializer . Begin ( ) ;
// Read private data
int c ;
deserializer . rInt ( out lastsectorindex ) ;
deserializer . rInt ( out c ) ;
indexholes = new List < int > ( c ) ;
for ( int i = 0 ; i < c ; i + + )
{
int index ; deserializer . rInt ( out index ) ;
indexholes . Add ( index ) ;
}
// Read map data
Vertex [ ] verticesarray = ReadVertices ( deserializer ) ;
Sector [ ] sectorsarray = ReadSectors ( deserializer ) ;
Linedef [ ] linedefsarray = ReadLinedefs ( deserializer , verticesarray ) ;
ReadSidedefs ( deserializer , linedefsarray , sectorsarray ) ;
ReadThings ( deserializer ) ;
deserializer . End ( ) ;
// Make table of sidedef indices
2009-06-05 19:03:56 +00:00
sidedefindices = new Sidedef [ numsidedefs ] ;
2009-04-19 18:07:22 +00:00
foreach ( Sidedef sd in sidedefs )
sidedefindices [ sd . SerializedIndex ] = sd ;
// Call PostDeserialize
foreach ( Sector s in sectors )
s . PostDeserialize ( this ) ;
}
// This deserializes things
private void ReadThings ( DeserializerStream stream )
{
int c ; stream . rInt ( out c ) ;
// Go for all things
for ( int i = 0 ; i < c ; i + + )
{
Thing t = CreateThing ( ) ;
t . ReadWrite ( stream ) ;
}
}
// This deserializes vertices
private Vertex [ ] ReadVertices ( DeserializerStream stream )
{
int c ; stream . rInt ( out c ) ;
Vertex [ ] array = new Vertex [ c ] ;
// Go for all vertices
for ( int i = 0 ; i < c ; i + + )
{
2009-06-11 21:21:20 +00:00
array [ i ] = CreateVertex ( new Vector2D ( ) ) ;
array [ i ] . ReadWrite ( stream ) ;
2009-04-19 18:07:22 +00:00
}
return array ;
}
// This deserializes linedefs
private Linedef [ ] ReadLinedefs ( DeserializerStream stream , Vertex [ ] verticesarray )
{
int c ; stream . rInt ( out c ) ;
Linedef [ ] array = new Linedef [ c ] ;
// Go for all lines
for ( int i = 0 ; i < c ; i + + )
{
int start , end ;
stream . rInt ( out start ) ;
stream . rInt ( out end ) ;
2009-06-11 21:21:20 +00:00
array [ i ] = CreateLinedef ( verticesarray [ start ] , verticesarray [ end ] ) ;
array [ i ] . ReadWrite ( stream ) ;
2009-04-19 18:07:22 +00:00
}
return array ;
}
// This deserializes sidedefs
private void ReadSidedefs ( DeserializerStream stream , Linedef [ ] linedefsarray , Sector [ ] sectorsarray )
{
int c ; stream . rInt ( out c ) ;
// Go for all sidedefs
for ( int i = 0 ; i < c ; i + + )
{
int lineindex , sectorindex ;
bool front ;
stream . rInt ( out lineindex ) ;
stream . rInt ( out sectorindex ) ;
stream . rBool ( out front ) ;
2009-06-11 21:21:20 +00:00
Sidedef sd = CreateSidedef ( linedefsarray [ lineindex ] , front , sectorsarray [ sectorindex ] ) ;
sd . ReadWrite ( stream ) ;
2009-04-19 18:07:22 +00:00
}
}
// This deserializes sectors
private Sector [ ] ReadSectors ( DeserializerStream stream )
{
int c ; stream . rInt ( out c ) ;
Sector [ ] array = new Sector [ c ] ;
// Go for all sectors
for ( int i = 0 ; i < c ; i + + )
{
2009-06-11 21:21:20 +00:00
array [ i ] = CreateSector ( ) ;
array [ i ] . ReadWrite ( stream ) ;
2009-04-19 18:07:22 +00:00
}
return array ;
}
#endregion
#region = = = = = = = = = = = = = = = = = = Updating
2009-05-19 09:59:19 +00:00
/// <summary>
/// This updates the cache of all elements where needed. You must call this after making changes to the map.
/// </summary>
2009-04-19 18:07:22 +00:00
public void Update ( )
{
// Update all!
Update ( true , true ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>
/// This updates the cache of all elements where needed. It is not recommended to use this version, please use Update() instead.
/// </summary>
2009-04-19 18:07:22 +00:00
public void Update ( bool dolines , bool dosectors )
{
// Update all linedefs
if ( dolines ) foreach ( Linedef l in linedefs ) l . UpdateCache ( ) ;
2009-06-04 20:21:31 +00:00
2009-04-19 18:07:22 +00:00
// Update all sectors
2009-06-04 20:21:31 +00:00
if ( dosectors )
{
foreach ( Sector s in sectors ) s . Triangulate ( ) ;
General . Map . CRenderer2D . Surfaces . AllocateBuffers ( ) ;
foreach ( Sector s in sectors ) s . CreateSurfaces ( ) ;
General . Map . CRenderer2D . Surfaces . UnlockBuffers ( ) ;
}
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>
/// This updates the cache of all elements that is required after a configuration or settings change.
/// </summary>
2009-04-19 18:07:22 +00:00
public void UpdateConfiguration ( )
{
// Update all things
foreach ( Thing t in things ) t . UpdateConfiguration ( ) ;
}
#endregion
#region = = = = = = = = = = = = = = = = = = Selection
// This checks a flag in a selection type
private bool InSelectionType ( SelectionType value , SelectionType bits )
{
return ( value & bits ) = = bits ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This converts the current selection to a different type of selection as specified.
/// Note that this function uses the markings to convert the selection.</summary>
2009-04-19 18:07:22 +00:00
public void ConvertSelection ( SelectionType target )
{
ConvertSelection ( SelectionType . All , target ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This converts the current selection to a different type of selection as specified.
/// Note that this function uses the markings to convert the selection.</summary>
2009-04-19 18:07:22 +00:00
public void ConvertSelection ( SelectionType source , SelectionType target )
{
ICollection < Linedef > lines ;
ICollection < Vertex > verts ;
ClearAllMarks ( false ) ;
switch ( target )
{
// Convert geometry selection to vertices only
case SelectionType . Vertices :
if ( InSelectionType ( source , SelectionType . Linedefs ) ) MarkSelectedLinedefs ( true , true ) ;
if ( InSelectionType ( source , SelectionType . Sectors ) ) General . Map . Map . MarkSelectedSectors ( true , true ) ;
verts = General . Map . Map . GetVerticesFromLinesMarks ( true ) ;
foreach ( Vertex v in verts ) v . Selected = true ;
verts = General . Map . Map . GetVerticesFromSectorsMarks ( true ) ;
foreach ( Vertex v in verts ) v . Selected = true ;
General . Map . Map . ClearSelectedSectors ( ) ;
General . Map . Map . ClearSelectedLinedefs ( ) ;
break ;
// Convert geometry selection to linedefs only
case SelectionType . Linedefs :
if ( InSelectionType ( source , SelectionType . Vertices ) ) MarkSelectedVertices ( true , true ) ;
if ( ! InSelectionType ( source , SelectionType . Linedefs ) ) ClearSelectedLinedefs ( ) ;
lines = General . Map . Map . LinedefsFromMarkedVertices ( false , true , false ) ;
foreach ( Linedef l in lines ) l . Selected = true ;
if ( InSelectionType ( source , SelectionType . Sectors ) )
{
foreach ( Sector s in General . Map . Map . Sectors )
{
if ( s . Selected )
{
foreach ( Sidedef sd in s . Sidedefs )
sd . Line . Selected = true ;
}
}
}
General . Map . Map . ClearSelectedSectors ( ) ;
General . Map . Map . ClearSelectedVertices ( ) ;
break ;
// Convert geometry selection to sectors only
case SelectionType . Sectors :
if ( InSelectionType ( source , SelectionType . Vertices ) ) MarkSelectedVertices ( true , true ) ;
if ( ! InSelectionType ( source , SelectionType . Linedefs ) ) ClearSelectedLinedefs ( ) ;
lines = LinedefsFromMarkedVertices ( false , true , false ) ;
foreach ( Linedef l in lines ) l . Selected = true ;
ClearMarkedSectors ( true ) ;
foreach ( Linedef l in linedefs )
{
if ( ! l . Selected )
{
if ( l . Front ! = null ) l . Front . Sector . Marked = false ;
if ( l . Back ! = null ) l . Back . Sector . Marked = false ;
}
}
ClearSelectedLinedefs ( ) ;
ClearSelectedVertices ( ) ;
if ( InSelectionType ( source , SelectionType . Sectors ) )
{
foreach ( Sector s in General . Map . Map . Sectors )
{
if ( s . Marked | | s . Selected )
{
s . Selected = true ;
foreach ( Sidedef sd in s . Sidedefs )
sd . Line . Selected = true ;
}
}
}
else
{
foreach ( Sector s in General . Map . Map . Sectors )
{
if ( s . Marked )
{
s . Selected = true ;
foreach ( Sidedef sd in s . Sidedefs )
sd . Line . Selected = true ;
}
else
{
s . Selected = false ;
}
}
}
break ;
default :
throw new ArgumentException ( "Unsupported selection target conversion" ) ;
}
// New selection type
sel_type = target ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all selected items</summary>
2009-04-19 18:07:22 +00:00
public void ClearAllSelected ( )
{
ClearSelectedVertices ( ) ;
ClearSelectedThings ( ) ;
ClearSelectedLinedefs ( ) ;
ClearSelectedSectors ( ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears selected vertices.</summary>
2009-04-19 18:07:22 +00:00
public void ClearSelectedVertices ( )
{
sel_vertices . Clear ( ) ;
foreach ( Vertex v in vertices ) v . Selected = false ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears selected things.</summary>
2009-04-19 18:07:22 +00:00
public void ClearSelectedThings ( )
{
sel_things . Clear ( ) ;
foreach ( Thing t in things ) t . Selected = false ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears selected linedefs.</summary>
2009-04-19 18:07:22 +00:00
public void ClearSelectedLinedefs ( )
{
sel_linedefs . Clear ( ) ;
foreach ( Linedef l in linedefs ) l . Selected = false ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears selected sectors.</summary>
2009-04-19 18:07:22 +00:00
public void ClearSelectedSectors ( )
{
sel_sectors . Clear ( ) ;
foreach ( Sector s in sectors ) s . Selected = false ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of vertices that match a selected state.</summary>
2009-04-19 18:07:22 +00:00
public ICollection < Vertex > GetSelectedVertices ( bool selected )
{
if ( selected )
{
return new List < Vertex > ( sel_vertices ) ;
}
else
{
2009-06-05 19:03:56 +00:00
List < Vertex > list = new List < Vertex > ( numvertices - sel_vertices . Count ) ;
2009-04-19 18:07:22 +00:00
foreach ( Vertex v in vertices ) if ( ! v . Selected ) list . Add ( v ) ;
return list ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of things that match a selected state.</summary>
2009-04-19 18:07:22 +00:00
public ICollection < Thing > GetSelectedThings ( bool selected )
{
if ( selected )
{
return new List < Thing > ( sel_things ) ;
}
else
{
2009-06-05 19:03:56 +00:00
List < Thing > list = new List < Thing > ( numthings - sel_things . Count ) ;
2009-04-19 18:07:22 +00:00
foreach ( Thing t in things ) if ( ! t . Selected ) list . Add ( t ) ;
return list ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of linedefs that match a selected state.</summary>
2009-04-19 18:07:22 +00:00
public ICollection < Linedef > GetSelectedLinedefs ( bool selected )
{
if ( selected )
{
return new List < Linedef > ( sel_linedefs ) ;
}
else
{
2009-06-05 19:03:56 +00:00
List < Linedef > list = new List < Linedef > ( numlinedefs - sel_linedefs . Count ) ;
2009-04-19 18:07:22 +00:00
foreach ( Linedef l in linedefs ) if ( ! l . Selected ) list . Add ( l ) ;
return list ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of sidedefs that match a selected linedefs state.</summary>
2009-05-09 11:37:55 +00:00
public ICollection < Sidedef > GetSidedefsFromSelectedLinedefs ( bool selected )
{
if ( selected )
{
List < Sidedef > list = new List < Sidedef > ( sel_linedefs . Count ) ;
foreach ( Linedef ld in sel_linedefs )
{
if ( ld . Front ! = null ) list . Add ( ld . Front ) ;
if ( ld . Back ! = null ) list . Add ( ld . Back ) ;
}
return list ;
}
else
{
2009-06-05 19:03:56 +00:00
List < Sidedef > list = new List < Sidedef > ( numlinedefs - sel_linedefs . Count ) ;
2009-05-09 11:37:55 +00:00
foreach ( Linedef ld in linedefs )
{
if ( ! ld . Selected & & ( ld . Front ! = null ) ) list . Add ( ld . Front ) ;
if ( ! ld . Selected & & ( ld . Back ! = null ) ) list . Add ( ld . Back ) ;
}
return list ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of sectors that match a selected state.</summary>
2009-04-19 18:07:22 +00:00
public ICollection < Sector > GetSelectedSectors ( bool selected )
{
if ( selected )
{
return new List < Sector > ( sel_sectors ) ;
}
else
{
2009-06-05 19:03:56 +00:00
List < Sector > list = new List < Sector > ( numsectors - sel_sectors . Count ) ;
2009-04-19 18:07:22 +00:00
foreach ( Sector s in sectors ) if ( ! s . Selected ) list . Add ( s ) ;
return list ;
}
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects or deselectes geometry based on marked elements.</summary>
2009-04-19 18:07:22 +00:00
public void SelectMarkedGeometry ( bool mark , bool select )
{
SelectMarkedVertices ( mark , select ) ;
SelectMarkedLinedefs ( mark , select ) ;
SelectMarkedSectors ( mark , select ) ;
SelectMarkedThings ( mark , select ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects or deselectes geometry based on marked elements.</summary>
2009-04-19 18:07:22 +00:00
public void SelectMarkedVertices ( bool mark , bool select )
{
foreach ( Vertex v in vertices ) if ( v . Marked = = mark ) v . Selected = select ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects or deselectes geometry based on marked elements.</summary>
2009-04-19 18:07:22 +00:00
public void SelectMarkedLinedefs ( bool mark , bool select )
{
foreach ( Linedef l in linedefs ) if ( l . Marked = = mark ) l . Selected = select ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects or deselectes geometry based on marked elements.</summary>
2009-04-19 18:07:22 +00:00
public void SelectMarkedSectors ( bool mark , bool select )
{
foreach ( Sector s in sectors ) if ( s . Marked = = mark ) s . Selected = select ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects or deselectes geometry based on marked elements.</summary>
2009-04-19 18:07:22 +00:00
public void SelectMarkedThings ( bool mark , bool select )
{
foreach ( Thing t in things ) if ( t . Marked = = mark ) t . Selected = select ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects geometry by selection group index.</summary>
2009-04-19 18:07:22 +00:00
public void SelectVerticesByGroup ( int groupmask )
{
foreach ( SelectableElement e in vertices ) e . SelectByGroup ( groupmask ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects geometry by selection group index.</summary>
2009-04-19 18:07:22 +00:00
public void SelectLinedefsByGroup ( int groupmask )
{
foreach ( SelectableElement e in linedefs ) e . SelectByGroup ( groupmask ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects geometry by selection group index.</summary>
2009-04-19 18:07:22 +00:00
public void SelectSectorsByGroup ( int groupmask )
{
foreach ( SelectableElement e in sectors ) e . SelectByGroup ( groupmask ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This selects geometry by selection group index.</summary>
2009-04-19 18:07:22 +00:00
public void SelectThingsByGroup ( int groupmask )
{
foreach ( SelectableElement e in things ) e . SelectByGroup ( groupmask ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This adds the current selection to the specified selection group.</summary>
2013-04-01 11:06:01 +00:00
//mxd. switched groupmask to groupindex
public void AddSelectionToGroup ( int groupindex )
{
//mxd
int numSectors = 0 ;
int numLines = 0 ;
int numVerts = 0 ;
int numThings = 0 ;
int groupmask = 0x01 < < groupindex ;
foreach ( SelectableElement e in vertices ) {
if ( e . Selected ) {
numVerts + + ; //mxd
e . AddToGroup ( groupmask ) ;
}
}
foreach ( SelectableElement e in linedefs ) {
if ( e . Selected ) {
numLines + + ; //mxd
e . AddToGroup ( groupmask ) ;
}
}
foreach ( SelectableElement e in sectors ) {
if ( e . Selected ) {
numSectors + + ; //mxd
e . AddToGroup ( groupmask ) ;
}
}
foreach ( SelectableElement e in things ) {
if ( e . Selected ) {
numThings + + ; //mxd
e . AddToGroup ( groupmask ) ;
}
}
//mxd
if ( numSectors > 0 | | numLines > 0 | | numThings > 0 | | numVerts > 0 ) {
if ( groupInfos [ groupindex ] ! = null )
groupInfos [ groupindex ] . Append ( numSectors , numLines , numVerts , numThings ) ;
else
groupInfos [ groupindex ] = new GroupInfo ( numSectors , numLines , numVerts , numThings ) ;
}
}
//mxd
public void ClearGroup ( int groupmask , int groupindex ) {
2009-04-19 18:07:22 +00:00
foreach ( SelectableElement e in vertices )
2013-04-01 11:06:01 +00:00
e . RemoveFromGroup ( groupmask ) ;
2009-04-19 18:07:22 +00:00
foreach ( SelectableElement e in linedefs )
2013-04-01 11:06:01 +00:00
e . RemoveFromGroup ( groupmask ) ;
2009-04-19 18:07:22 +00:00
foreach ( SelectableElement e in sectors )
2013-04-01 11:06:01 +00:00
e . RemoveFromGroup ( groupmask ) ;
2009-04-19 18:07:22 +00:00
foreach ( SelectableElement e in things )
2013-04-01 11:06:01 +00:00
e . RemoveFromGroup ( groupmask ) ;
groupInfos [ groupindex ] = null ;
}
//mxd
public bool HaveSelectionGroups ( ) {
foreach ( GroupInfo info in groupInfos )
if ( info ! = null ) return true ;
return false ;
2009-04-19 18:07:22 +00:00
}
#endregion
#region = = = = = = = = = = = = = = = = = = Marking
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all elements.</summary>
2009-04-19 18:07:22 +00:00
public void ClearAllMarks ( bool mark )
{
ClearMarkedVertices ( mark ) ;
ClearMarkedThings ( mark ) ;
ClearMarkedLinedefs ( mark ) ;
ClearMarkedSectors ( mark ) ;
ClearMarkedSidedefs ( mark ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all vertices.</summary>
2009-04-19 18:07:22 +00:00
public void ClearMarkedVertices ( bool mark )
{
foreach ( Vertex v in vertices ) v . Marked = mark ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all things.</summary>
2009-04-19 18:07:22 +00:00
public void ClearMarkedThings ( bool mark )
{
foreach ( Thing t in things ) t . Marked = mark ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all linedefs.</summary>
2009-04-19 18:07:22 +00:00
public void ClearMarkedLinedefs ( bool mark )
{
foreach ( Linedef l in linedefs ) l . Marked = mark ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all sidedefs.</summary>
2009-04-19 18:07:22 +00:00
public void ClearMarkedSidedefs ( bool mark )
{
foreach ( Sidedef s in sidedefs ) s . Marked = mark ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This clears all marks on all sectors.</summary>
2009-04-19 18:07:22 +00:00
public void ClearMarkedSectors ( bool mark )
{
foreach ( Sector s in sectors ) s . Marked = mark ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all elements.</summary>
2009-04-19 18:07:22 +00:00
public void InvertAllMarks ( )
{
InvertMarkedVertices ( ) ;
InvertMarkedThings ( ) ;
InvertMarkedLinedefs ( ) ;
InvertMarkedSectors ( ) ;
InvertMarkedSidedefs ( ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all vertices.</summary>
2009-04-19 18:07:22 +00:00
public void InvertMarkedVertices ( )
{
foreach ( Vertex v in vertices ) v . Marked = ! v . Marked ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all things.</summary>
2009-04-19 18:07:22 +00:00
public void InvertMarkedThings ( )
{
foreach ( Thing t in things ) t . Marked = ! t . Marked ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all linedefs.</summary>
2009-04-19 18:07:22 +00:00
public void InvertMarkedLinedefs ( )
{
foreach ( Linedef l in linedefs ) l . Marked = ! l . Marked ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all sidedefs.</summary>
2009-04-19 18:07:22 +00:00
public void InvertMarkedSidedefs ( )
{
foreach ( Sidedef s in sidedefs ) s . Marked = ! s . Marked ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This inverts all marks on all sectors.</summary>
2009-04-19 18:07:22 +00:00
public void InvertMarkedSectors ( )
{
foreach ( Sector s in sectors ) s . Marked = ! s . Marked ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of vertices that match a marked state.</summary>
2009-04-19 18:07:22 +00:00
public List < Vertex > GetMarkedVertices ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Vertex > list = new List < Vertex > ( numvertices > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Vertex v in vertices ) if ( v . Marked = = mark ) list . Add ( v ) ;
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of things that match a marked state.</summary>
2009-04-19 18:07:22 +00:00
public List < Thing > GetMarkedThings ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Thing > list = new List < Thing > ( numthings > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Thing t in things ) if ( t . Marked = = mark ) list . Add ( t ) ;
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of linedefs that match a marked state.</summary>
2009-04-19 18:07:22 +00:00
public List < Linedef > GetMarkedLinedefs ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Linedef > list = new List < Linedef > ( numlinedefs > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Linedef l in linedefs ) if ( l . Marked = = mark ) list . Add ( l ) ;
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of sidedefs that match a marked state.</summary>
2009-04-19 18:07:22 +00:00
public List < Sidedef > GetMarkedSidedefs ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Sidedef > list = new List < Sidedef > ( numsidedefs > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Sidedef s in sidedefs ) if ( s . Marked = = mark ) list . Add ( s ) ;
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>Returns a collection of sectors that match a marked state.</summary>
2009-04-19 18:07:22 +00:00
public List < Sector > GetMarkedSectors ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Sector > list = new List < Sector > ( numsectors > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Sector s in sectors ) if ( s . Marked = = mark ) list . Add ( s ) ;
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This marks vertices based on selected vertices.</summary>
2009-04-19 18:07:22 +00:00
public void MarkSelectedVertices ( bool selected , bool mark )
{
2009-05-13 09:35:13 +00:00
foreach ( Vertex v in sel_vertices ) v . Marked = mark ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>This marks linedefs based on selected linedefs.</summary>
2009-04-19 18:07:22 +00:00
public void MarkSelectedLinedefs ( bool selected , bool mark )
{
2009-05-13 09:35:13 +00:00
foreach ( Linedef l in sel_linedefs ) l . Marked = mark ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>This marks sectors based on selected sectors.</summary>
2009-04-19 18:07:22 +00:00
public void MarkSelectedSectors ( bool selected , bool mark )
{
2009-05-13 09:35:13 +00:00
foreach ( Sector s in sel_sectors ) s . Marked = mark ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>This marks things based on selected things.</summary>
2009-04-19 18:07:22 +00:00
public void MarkSelectedThings ( bool selected , bool mark )
{
2009-05-13 09:35:13 +00:00
foreach ( Thing t in sel_things ) t . Marked = mark ;
2009-04-19 18:07:22 +00:00
}
/// <summary>
2009-05-19 09:59:19 +00:00
/// This marks the front and back sidedefs on linedefs with the matching mark.
2009-04-19 18:07:22 +00:00
/// </summary>
public void MarkSidedefsFromLinedefs ( bool matchmark , bool setmark )
{
foreach ( Linedef l in linedefs )
{
if ( l . Marked = = matchmark )
{
if ( l . Front ! = null ) l . Front . Marked = setmark ;
if ( l . Back ! = null ) l . Back . Marked = setmark ;
}
}
}
/// <summary>
2009-05-19 09:59:19 +00:00
/// This marks the sidedefs that make up the sectors with the matching mark.
2009-04-19 18:07:22 +00:00
/// </summary>
public void MarkSidedefsFromSectors ( bool matchmark , bool setmark )
{
foreach ( Sidedef sd in sidedefs )
{
if ( sd . Sector . Marked = = matchmark ) sd . Marked = setmark ;
}
}
/// <summary>
2009-05-19 09:59:19 +00:00
/// Returns a collection of vertices that match a marked state on the linedefs.
2009-04-19 18:07:22 +00:00
/// </summary>
public ICollection < Vertex > GetVerticesFromLinesMarks ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Vertex > list = new List < Vertex > ( numvertices > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Vertex v in vertices )
{
foreach ( Linedef l in v . Linedefs )
{
if ( l . Marked = = mark )
{
list . Add ( v ) ;
break ;
}
}
}
return list ;
}
/// <summary>
2009-05-19 09:59:19 +00:00
/// Returns a collection of vertices that match a marked state on the linedefs.
2009-04-19 18:07:22 +00:00
/// The difference with GetVerticesFromLinesMarks is that in this method
/// ALL linedefs of a vertex must match the specified marked state.
/// </summary>
public ICollection < Vertex > GetVerticesFromAllLinesMarks ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Vertex > list = new List < Vertex > ( numvertices > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Vertex v in vertices )
{
bool qualified = true ;
foreach ( Linedef l in v . Linedefs )
{
if ( l . Marked ! = mark )
{
qualified = false ;
break ;
}
}
if ( qualified ) list . Add ( v ) ;
}
return list ;
}
/// <summary>
2009-05-19 09:59:19 +00:00
/// Returns a collection of vertices that match a marked state on the linedefs.
2009-04-19 18:07:22 +00:00
/// </summary>
public ICollection < Vertex > GetVerticesFromSectorsMarks ( bool mark )
{
2009-06-05 19:03:56 +00:00
List < Vertex > list = new List < Vertex > ( numvertices > > 1 ) ;
2009-04-19 18:07:22 +00:00
foreach ( Vertex v in vertices )
{
foreach ( Linedef l in v . Linedefs )
{
2013-09-03 09:34:28 +00:00
if ( ( ( l . Front ! = null ) & & ( l . Front . Sector ! = null ) & & ( l . Front . Sector . Marked = = mark ) ) | |
( ( l . Back ! = null ) & & ( l . Back . Sector ! = null ) & & ( l . Back . Sector . Marked = = mark ) ) )
2009-04-19 18:07:22 +00:00
{
list . Add ( v ) ;
break ;
}
}
}
return list ;
}
/// <summary>
/// This marks all selected geometry, including sidedefs from sectors.
/// When sidedefsfromsectors is true, then the sidedefs are marked according to the
/// marked sectors. Otherwise the sidedefs are marked according to the marked linedefs.
/// </summary>
public void MarkAllSelectedGeometry ( bool mark , bool linedefsfromvertices , bool verticesfromlinedefs , bool sectorsfromlinedefs , bool sidedefsfromsectors )
{
General . Map . Map . ClearAllMarks ( ! mark ) ;
// Direct vertices
General . Map . Map . MarkSelectedVertices ( true , mark ) ;
// Direct linedefs
General . Map . Map . MarkSelectedLinedefs ( true , mark ) ;
// Linedefs from vertices
// We do this before "vertices from lines" because otherwise we get lines marked that we didn't select
if ( linedefsfromvertices )
{
ICollection < Linedef > lines = General . Map . Map . LinedefsFromMarkedVertices ( ! mark , mark , ! mark ) ;
foreach ( Linedef l in lines ) l . Marked = mark ;
}
// Vertices from linedefs
if ( verticesfromlinedefs )
{
ICollection < Vertex > verts = General . Map . Map . GetVerticesFromLinesMarks ( mark ) ;
foreach ( Vertex v in verts ) v . Marked = mark ;
}
// Mark sectors from linedefs (note: this must be the first to mark
// sectors, because this clears the sector marks!)
if ( sectorsfromlinedefs )
{
General . Map . Map . ClearMarkedSectors ( mark ) ;
foreach ( Linedef l in General . Map . Map . Linedefs )
{
if ( ! l . Selected )
{
if ( l . Front ! = null ) l . Front . Sector . Marked = ! mark ;
if ( l . Back ! = null ) l . Back . Sector . Marked = ! mark ;
}
}
}
// Direct sectors
General . Map . Map . MarkSelectedSectors ( true , mark ) ;
// Direct things
General . Map . Map . MarkSelectedThings ( true , mark ) ;
// Sidedefs from linedefs or sectors
if ( sidedefsfromsectors )
General . Map . Map . MarkSidedefsFromSectors ( true , mark ) ;
else
General . Map . Map . MarkSidedefsFromLinedefs ( true , mark ) ;
}
#endregion
#region = = = = = = = = = = = = = = = = = = Indexing
/// <summary>
2009-06-05 19:03:56 +00:00
/// Returns the vertex at the specified index. Returns null when index is out of range. This is an O(1) operation.
2009-04-19 18:07:22 +00:00
/// </summary>
public Vertex GetVertexByIndex ( int index )
{
2009-06-05 19:03:56 +00:00
return index < numvertices ? vertices [ index ] : null ;
2009-04-19 18:07:22 +00:00
}
/// <summary>
2009-06-05 19:03:56 +00:00
/// Returns the linedef at the specified index. Returns null when index is out of range. This is an O(1) operation.
2009-04-19 18:07:22 +00:00
/// </summary>
public Linedef GetLinedefByIndex ( int index )
{
2009-06-05 19:03:56 +00:00
return index < numlinedefs ? linedefs [ index ] : null ;
2009-04-19 18:07:22 +00:00
}
/// <summary>
2009-06-05 19:03:56 +00:00
/// Returns the sidedef at the specified index. Returns null when index is out of range. This is an O(1) operation.
2009-04-19 18:07:22 +00:00
/// </summary>
public Sidedef GetSidedefByIndex ( int index )
{
2009-06-05 19:03:56 +00:00
return index < numsidedefs ? sidedefs [ index ] : null ;
2009-04-19 18:07:22 +00:00
}
/// <summary>
2009-06-05 19:03:56 +00:00
/// Returns the sector at the specified index. Returns null when index is out of range. This is an O(1) operation.
2009-04-19 18:07:22 +00:00
/// </summary>
public Sector GetSectorByIndex ( int index )
{
2009-06-05 19:03:56 +00:00
return index < numsectors ? sectors [ index ] : null ;
2009-04-19 18:07:22 +00:00
}
/// <summary>
2009-06-05 19:03:56 +00:00
/// Returns the thing at the specified index. Returns null when index is out of range. This is an O(1) operation.
2009-04-19 18:07:22 +00:00
/// </summary>
public Thing GetThingByIndex ( int index )
{
2009-06-05 19:03:56 +00:00
return index < numthings ? things [ index ] : null ;
2009-04-19 18:07:22 +00:00
}
#endregion
#region = = = = = = = = = = = = = = = = = = Areas
2009-05-19 09:59:19 +00:00
/// <summary>This creates an initial, undefined area.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF CreateEmptyArea ( )
{
return new RectangleF ( float . MaxValue / 2 , float . MaxValue / 2 , - float . MaxValue , - float . MaxValue ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates an area from vertices.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF CreateArea ( ICollection < Vertex > verts )
{
float l = float . MaxValue ;
float t = float . MaxValue ;
float r = float . MinValue ;
float b = float . MinValue ;
// Go for all vertices
foreach ( Vertex v in verts )
{
// Adjust boundaries by vertices
if ( v . Position . x < l ) l = v . Position . x ;
if ( v . Position . x > r ) r = v . Position . x ;
if ( v . Position . y < t ) t = v . Position . y ;
if ( v . Position . y > b ) b = v . Position . y ;
}
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This increases and existing area with the given vertices.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF IncreaseArea ( RectangleF area , ICollection < Vertex > verts )
{
float l = area . Left ;
float t = area . Top ;
float r = area . Right ;
float b = area . Bottom ;
// Go for all vertices
foreach ( Vertex v in verts )
{
// Adjust boundaries by vertices
if ( v . Position . x < l ) l = v . Position . x ;
if ( v . Position . x > r ) r = v . Position . x ;
if ( v . Position . y < t ) t = v . Position . y ;
if ( v . Position . y > b ) b = v . Position . y ;
}
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This increases and existing area with the given things.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF IncreaseArea ( RectangleF area , ICollection < Thing > things )
{
float l = area . Left ;
float t = area . Top ;
float r = area . Right ;
float b = area . Bottom ;
// Go for all vertices
foreach ( Thing th in things )
{
// Adjust boundaries by vertices
if ( th . Position . x < l ) l = th . Position . x ;
if ( th . Position . x > r ) r = th . Position . x ;
if ( th . Position . y < t ) t = th . Position . y ;
if ( th . Position . y > b ) b = th . Position . y ;
}
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This increases and existing area with the given vertices.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF IncreaseArea ( RectangleF area , ICollection < Vector2D > verts )
{
float l = area . Left ;
float t = area . Top ;
float r = area . Right ;
float b = area . Bottom ;
// Go for all vertices
foreach ( Vector2D v in verts )
{
// Adjust boundaries by vertices
if ( v . x < l ) l = v . x ;
if ( v . x > r ) r = v . x ;
if ( v . y < t ) t = v . y ;
if ( v . y > b ) b = v . y ;
}
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This increases and existing area with the given vertex.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF IncreaseArea ( RectangleF area , Vector2D vert )
{
float l = area . Left ;
float t = area . Top ;
float r = area . Right ;
float b = area . Bottom ;
// Adjust boundaries by vertices
if ( vert . x < l ) l = vert . x ;
if ( vert . x > r ) r = vert . x ;
if ( vert . y < t ) t = vert . y ;
if ( vert . y > b ) b = vert . y ;
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This creates an area from linedefs.</summary>
2009-04-19 18:07:22 +00:00
public static RectangleF CreateArea ( ICollection < Linedef > lines )
{
float l = float . MaxValue ;
float t = float . MaxValue ;
float r = float . MinValue ;
float b = float . MinValue ;
// Go for all linedefs
foreach ( Linedef ld in lines )
{
// Adjust boundaries by vertices
if ( ld . Start . Position . x < l ) l = ld . Start . Position . x ;
if ( ld . Start . Position . x > r ) r = ld . Start . Position . x ;
if ( ld . Start . Position . y < t ) t = ld . Start . Position . y ;
if ( ld . Start . Position . y > b ) b = ld . Start . Position . y ;
if ( ld . End . Position . x < l ) l = ld . End . Position . x ;
if ( ld . End . Position . x > r ) r = ld . End . Position . x ;
if ( ld . End . Position . y < t ) t = ld . End . Position . y ;
if ( ld . End . Position . y > b ) b = ld . End . Position . y ;
}
// Return a rect
return new RectangleF ( l , t , r - l , b - t ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This filters lines by a rectangular area.</summary>
2009-04-19 18:07:22 +00:00
public static ICollection < Linedef > FilterByArea ( ICollection < Linedef > lines , ref RectangleF area )
{
ICollection < Linedef > newlines = new List < Linedef > ( lines . Count ) ;
// Go for all lines
foreach ( Linedef l in lines )
{
2013-12-11 09:47:35 +00:00
// Check the cs field bits
2013-12-17 13:18:44 +00:00
if ( ( GetCSFieldBits ( l . Start . Position , area ) & GetCSFieldBits ( l . End . Position , area ) ) = = 0 )
2013-12-11 09:47:35 +00:00
{
// The line could be in the area
newlines . Add ( l ) ;
}
2009-04-19 18:07:22 +00:00
}
// Return result
return newlines ;
}
2013-12-17 13:18:44 +00:00
/// <summary> This returns the cohen-sutherland field bits for a vector in a rectangle area</summary>
public static int GetCSFieldBits ( Vector2D v , RectangleF area )
2013-12-11 09:47:35 +00:00
{
int bits = 0 ;
2013-12-17 13:18:44 +00:00
if ( v . y < area . Top ) bits | = 0x01 ;
if ( v . y > area . Bottom ) bits | = 0x02 ;
if ( v . x < area . Left ) bits | = 0x04 ;
if ( v . x > area . Right ) bits | = 0x08 ;
2013-12-11 09:47:35 +00:00
return bits ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This filters vertices by a rectangular area.</summary>
2009-04-19 18:07:22 +00:00
public static ICollection < Vertex > FilterByArea ( ICollection < Vertex > verts , ref RectangleF area )
{
ICollection < Vertex > newverts = new List < Vertex > ( verts . Count ) ;
// Go for all verts
foreach ( Vertex v in verts )
{
// Within rect?
2013-12-11 09:47:35 +00:00
if ( ( v . Position . x < area . Left ) | | ( v . Position . x > area . Right ) | |
( v . Position . y < area . Top ) | | ( v . Position . y > area . Bottom ) ) continue ;
// The vertex is in the area
newverts . Add ( v ) ;
2009-04-19 18:07:22 +00:00
}
// Return result
return newverts ;
}
#endregion
#region = = = = = = = = = = = = = = = = = = Stitching
/// <summary>
2010-08-15 19:43:00 +00:00
/// Stitches marked geometry with non-marked geometry. Returns false when the operation failed.
2009-04-19 18:07:22 +00:00
/// </summary>
2010-08-15 19:43:00 +00:00
public bool StitchGeometry ( )
2009-04-19 18:07:22 +00:00
{
ICollection < Linedef > movinglines ;
ICollection < Linedef > fixedlines ;
ICollection < Vertex > nearbyfixedverts ;
ICollection < Vertex > movingverts ;
ICollection < Vertex > fixedverts ;
RectangleF editarea ;
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
// Find vertices
movingverts = General . Map . Map . GetMarkedVertices ( true ) ;
fixedverts = General . Map . Map . GetMarkedVertices ( false ) ;
// Find lines that moved during the drag
movinglines = LinedefsFromMarkedVertices ( false , true , true ) ;
// Find all non-moving lines
fixedlines = LinedefsFromMarkedVertices ( true , false , false ) ;
// Determine area in which we are editing
editarea = MapSet . CreateArea ( movinglines ) ;
editarea = MapSet . IncreaseArea ( editarea , movingverts ) ;
editarea . Inflate ( 1.0f , 1.0f ) ;
// Join nearby vertices
2009-06-05 19:03:56 +00:00
BeginAddRemove ( ) ;
2010-08-15 19:43:00 +00:00
MapSet . JoinVertices ( fixedverts , movingverts , true , MapSet . STITCH_DISTANCE ) ;
2009-06-05 19:03:56 +00:00
EndAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
// Update cached values of lines because we need their length/angle
Update ( true , false ) ;
2009-06-05 19:03:56 +00:00
BeginAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
// Split moving lines with unselected vertices
nearbyfixedverts = MapSet . FilterByArea ( fixedverts , ref editarea ) ;
2010-08-15 19:43:00 +00:00
if ( ! MapSet . SplitLinesByVertices ( movinglines , nearbyfixedverts , MapSet . STITCH_DISTANCE , movinglines ) )
return false ;
2009-04-19 18:07:22 +00:00
// Split non-moving lines with selected vertices
fixedlines = MapSet . FilterByArea ( fixedlines , ref editarea ) ;
2010-08-15 19:43:00 +00:00
if ( ! MapSet . SplitLinesByVertices ( fixedlines , movingverts , MapSet . STITCH_DISTANCE , movinglines ) )
return false ;
2009-04-19 18:07:22 +00:00
// Remove looped linedefs
2010-08-15 19:43:00 +00:00
MapSet . RemoveLoopedLinedefs ( movinglines ) ;
2009-04-19 18:07:22 +00:00
// Join overlapping lines
2010-08-15 19:43:00 +00:00
if ( ! MapSet . JoinOverlappingLines ( movinglines ) )
return false ;
2009-04-19 18:07:22 +00:00
2009-06-05 19:03:56 +00:00
EndAddRemove ( ) ;
2010-08-15 19:43:00 +00:00
return true ;
2009-04-19 18:07:22 +00:00
}
#endregion
#region = = = = = = = = = = = = = = = = = = Geometry Tools
2009-05-19 09:59:19 +00:00
/// <summary>This removes any virtual sectors in the map and returns the number of sectors removed.</summary>
2009-04-19 18:07:22 +00:00
public int RemoveVirtualSectors ( )
{
int count = 0 ;
2009-06-05 19:03:56 +00:00
int index = 0 ;
2009-04-19 18:07:22 +00:00
// Go for all sectors
2009-06-05 19:03:56 +00:00
while ( index < numsectors )
2009-04-19 18:07:22 +00:00
{
// Remove when virtual
2009-06-05 19:03:56 +00:00
if ( sectors [ index ] . Fields . ContainsKey ( VIRTUAL_SECTOR_FIELD ) )
2009-04-19 18:07:22 +00:00
{
2009-06-05 19:03:56 +00:00
sectors [ index ] . Dispose ( ) ;
2009-04-19 18:07:22 +00:00
count + + ;
}
2009-06-05 19:03:56 +00:00
else
{
index + + ;
}
2009-04-19 18:07:22 +00:00
}
2009-06-05 19:03:56 +00:00
2009-04-19 18:07:22 +00:00
return count ;
}
2009-05-17 14:37:34 +00:00
2009-05-19 09:59:19 +00:00
/// <summary>This removes unused sectors and returns the number of removed sectors.</summary>
2009-05-17 14:37:34 +00:00
public int RemoveUnusedSectors ( bool reportwarnings )
{
int count = 0 ;
2009-06-05 19:03:56 +00:00
int index = numsectors - 1 ;
2009-05-17 14:37:34 +00:00
// Go for all sectors
2009-06-05 19:03:56 +00:00
while ( index > = 0 )
2009-05-17 14:37:34 +00:00
{
// Remove when unused
2009-06-05 19:03:56 +00:00
if ( sectors [ index ] . Sidedefs . Count = = 0 )
2009-05-17 14:37:34 +00:00
{
if ( reportwarnings )
General . ErrorLogger . Add ( ErrorType . Warning , "Sector " + index + " was unused and has been removed." ) ;
2009-06-05 19:03:56 +00:00
sectors [ index ] . Dispose ( ) ;
2009-05-17 14:37:34 +00:00
count + + ;
}
2009-06-05 19:03:56 +00:00
index - - ;
2009-05-17 14:37:34 +00:00
}
2009-06-05 19:03:56 +00:00
2009-05-17 14:37:34 +00:00
return count ;
}
2009-05-19 09:59:19 +00:00
2010-08-15 19:43:00 +00:00
/// <summary>This joins overlapping lines together. Returns false when the operation failed.</summary>
public static bool JoinOverlappingLines ( ICollection < Linedef > lines )
2009-04-19 18:07:22 +00:00
{
bool joined ;
do
{
// No joins yet
joined = false ;
// Go for all the lines
foreach ( Linedef l1 in lines )
{
// Check if these vertices have lines that overlap
foreach ( Linedef l2 in l1 . Start . Linedefs )
{
2013-09-03 09:34:28 +00:00
//mxd. The same line?
if ( l1 . Index = = l2 . Index ) continue ;
2009-04-19 18:07:22 +00:00
// Sharing vertices?
if ( ( l1 . End = = l2 . End ) | |
( l1 . End = = l2 . Start ) )
{
2013-09-03 09:34:28 +00:00
bool oppositedirection = ( l1 . End = = l2 . Start ) ;
bool l2marked = l2 . Marked ;
// Merge these two linedefs
while ( lines . Remove ( l2 ) ) ;
if ( ! l2 . Join ( l1 ) ) return false ;
// If l2 was marked as new geometry, we have to make sure
// that l1's FrontInterior is correct for the drawing procedure
if ( l2marked )
2009-04-19 18:07:22 +00:00
{
2013-09-03 09:34:28 +00:00
l1 . FrontInterior = l2 . FrontInterior ^ oppositedirection ;
}
2009-05-12 11:36:52 +00:00
// If l1 is marked as new geometry, we may need to flip it to preserve
// orientation of the original geometry, and update its FrontInterior
2013-09-03 09:34:28 +00:00
else if ( l1 . Marked )
{
if ( oppositedirection )
2009-05-12 11:36:52 +00:00
{
2013-09-03 09:34:28 +00:00
l1 . FlipVertices ( ) ; // This also flips FrontInterior
l1 . FlipSidedefs ( ) ;
2009-05-06 10:06:45 +00:00
}
2009-04-19 18:07:22 +00:00
}
2013-09-03 09:34:28 +00:00
joined = true ;
break ;
2009-04-19 18:07:22 +00:00
}
}
// Will have to restart when joined
if ( joined ) break ;
// Check if these vertices have lines that overlap
foreach ( Linedef l2 in l1 . End . Linedefs )
{
2013-09-03 09:34:28 +00:00
//mxd. The same line?
if ( l1 . Index = = l2 . Index ) continue ;
2009-04-19 18:07:22 +00:00
// Sharing vertices?
if ( ( l1 . Start = = l2 . End ) | |
( l1 . Start = = l2 . Start ) )
{
2013-09-03 09:34:28 +00:00
bool oppositedirection = ( l1 . Start = = l2 . End ) ;
bool l2marked = l2 . Marked ;
// Merge these two linedefs
while ( lines . Remove ( l2 ) ) ;
if ( ! l2 . Join ( l1 ) ) return false ;
// If l2 was marked as new geometry, we have to make sure
// that l1's FrontInterior is correct for the drawing procedure
if ( l2marked )
2009-04-19 18:07:22 +00:00
{
2013-09-03 09:34:28 +00:00
l1 . FrontInterior = l2 . FrontInterior ^ oppositedirection ;
}
2009-05-12 11:36:52 +00:00
// If l1 is marked as new geometry, we may need to flip it to preserve
// orientation of the original geometry, and update its FrontInterior
2013-09-03 09:34:28 +00:00
else if ( l1 . Marked )
{
if ( oppositedirection )
2009-05-12 11:36:52 +00:00
{
2013-09-03 09:34:28 +00:00
l1 . FlipVertices ( ) ; // This also flips FrontInterior
l1 . FlipSidedefs ( ) ;
2009-05-06 10:06:45 +00:00
}
2009-04-19 18:07:22 +00:00
}
2013-09-03 09:34:28 +00:00
joined = true ;
break ;
2009-04-19 18:07:22 +00:00
}
}
// Will have to restart when joined
if ( joined ) break ;
}
}
while ( joined ) ;
// Return result
2010-08-15 19:43:00 +00:00
return true ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>This removes looped linedefs (linedefs which reference the same vertex for
/// start and end) and returns the number of linedefs removed.</summary>
2009-04-19 18:07:22 +00:00
public static int RemoveLoopedLinedefs ( ICollection < Linedef > lines )
{
int linesremoved = 0 ;
bool removedline ;
do
{
// Nothing removed yet
removedline = false ;
// Go for all the lines
foreach ( Linedef l in lines )
{
// Check if referencing the same vertex twice
if ( l . Start = = l . End )
{
// Remove this line
while ( lines . Remove ( l ) ) ;
l . Dispose ( ) ;
linesremoved + + ;
removedline = true ;
break ;
}
}
}
while ( removedline ) ;
// Return result
return linesremoved ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This joins nearby vertices from two collections. This does NOT join vertices
/// within the same collection, only if they exist in both collections.
/// The vertex from the second collection is moved to match the first vertex.
/// When keepsecond is true, the vertex in the second collection is kept,
/// otherwise the vertex in the first collection is kept.
/// Returns the number of joins made.</summary>
2009-04-19 18:07:22 +00:00
public static int JoinVertices ( ICollection < Vertex > set1 , ICollection < Vertex > set2 , bool keepsecond , float joindist )
{
float joindist2 = joindist * joindist ;
int joinsdone = 0 ;
bool joined ;
do
{
// No joins yet
joined = false ;
// Go for all vertices in the first set
foreach ( Vertex v1 in set1 )
{
// Go for all vertices in the second set
foreach ( Vertex v2 in set2 )
{
// Check if vertices are close enough
if ( v1 . DistanceToSq ( v2 . Position ) < = joindist2 )
{
// Check if not the same vertex
if ( v1 ! = v2 )
{
// Move the second vertex to match the first
v2 . Move ( v1 . Position ) ;
// Check which one to keep
if ( keepsecond )
{
// Join the first into the second
// Second is kept, first is removed
v1 . Join ( v2 ) ;
set1 . Remove ( v1 ) ;
set2 . Remove ( v1 ) ;
}
else
{
// Join the second into the first
// First is kept, second is removed
v2 . Join ( v1 ) ;
set1 . Remove ( v2 ) ;
set2 . Remove ( v2 ) ;
}
// Count the join
joinsdone + + ;
joined = true ;
break ;
}
}
}
// Will have to restart when joined
if ( joined ) break ;
}
}
while ( joined ) ;
// Return result
return joinsdone ;
}
2013-12-17 13:18:44 +00:00
/// <summary>This joins nearby vertices in the same collection </summary>
public static int JoinVertices ( List < Vertex > set , float joindist ) {
float joindist2 = joindist * joindist ;
int joinsdone = 0 ;
bool joined ;
Vertex v1 , v2 ;
do {
// No joins yet
joined = false ;
// Go for all vertices in the first set
for ( int i = 0 ; i < set . Count - 1 ; i + + ) {
for ( int c = i + 1 ; c < set . Count ; c + + ) {
v1 = set [ i ] ;
v2 = set [ c ] ;
// Check if vertices are close enough
if ( v1 . DistanceToSq ( v2 . Position ) < = joindist2 ) {
// Check if not the same vertex
if ( v1 . Index ! = v2 . Index ) {
// Move the second vertex to match the first
v2 . Move ( v1 . Position ) ;
// Join the second into the first
v2 . Join ( v1 ) ;
set . Remove ( v2 ) ;
// Count the join
joinsdone + + ;
joined = true ;
break ;
}
}
}
}
} while ( joined ) ;
// Return result
return joinsdone ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This corrects lines that have a back sidedef but no front sidedef by flipping them. Returns the number of flips made.</summary>
2009-04-19 18:07:22 +00:00
public static int FlipBackwardLinedefs ( ICollection < Linedef > lines )
{
int flipsdone = 0 ;
// Examine all lines
foreach ( Linedef l in lines )
{
// Back side but no front side?
if ( ( l . Back ! = null ) & & ( l . Front = = null ) )
{
// Flip that linedef!
l . FlipVertices ( ) ;
l . FlipSidedefs ( ) ;
flipsdone + + ;
}
}
// Return result
return flipsdone ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This splits the given lines with the given vertices. All affected lines
2010-08-15 19:43:00 +00:00
/// will be added to changedlines. Returns false when the operation failed.</summary>
public static bool SplitLinesByVertices ( ICollection < Linedef > lines , ICollection < Vertex > verts , float splitdist , ICollection < Linedef > changedlines )
2009-04-19 18:07:22 +00:00
{
2013-12-24 09:21:18 +00:00
if ( verts . Count = = 0 | | lines . Count = = 0 ) return true ; //mxd
2009-04-19 18:07:22 +00:00
float splitdist2 = splitdist * splitdist ;
bool splitted ;
2013-12-17 13:18:44 +00:00
//mxd. Create blockmap
RectangleF area = CreateArea ( verts ) ;
BlockMap < BlockEntry > blockmap = new BlockMap < BlockEntry > ( area ) ;
blockmap . AddVerticesSet ( verts ) ;
blockmap . AddLinedefsSet ( lines ) ;
int bmWidth = blockmap . Size . Width ;
int bmHeight = blockmap . Size . Height ;
BlockEntry [ , ] bmap = blockmap . Map ;
BlockEntry block ;
int w , h ;
2009-04-19 18:07:22 +00:00
do
{
// No split yet
splitted = false ;
2013-12-17 13:18:44 +00:00
for ( w = 0 ; w < bmWidth ; w + + ) {
for ( h = 0 ; h < bmHeight ; h + + ) {
block = bmap [ w , h ] ;
if ( block . Vertices . Count = = 0 | | block . Lines . Count = = 0 ) continue ;
// Go for all the lines
foreach ( Linedef l in block . Lines ) {
// Go for all the vertices
foreach ( Vertex v in block . Vertices ) {
// Check if v is close enough to l for splitting
if ( l . DistanceToSq ( v . Position , true ) < = splitdist2 ) {
// Line is not already referencing v?
Vector2D deltastart = l . Start . Position - v . Position ;
Vector2D deltaend = l . End . Position - v . Position ;
if ( ( ( Math . Abs ( deltastart . x ) > 0.001f ) | | ( Math . Abs ( deltastart . y ) > 0.001f ) ) & &
( ( Math . Abs ( deltaend . x ) > 0.001f ) | | ( Math . Abs ( deltaend . y ) > 0.001f ) ) ) {
// Split line l with vertex v
Linedef nl = l . Split ( v ) ;
if ( nl = = null ) return false ;
// Add the new line to the list
lines . Add ( nl ) ;
2013-12-24 09:21:18 +00:00
blockmap . AddLinedef ( nl ) ;
2013-12-17 13:18:44 +00:00
// Both lines must be updated because their new length
// is relevant for next iterations!
l . UpdateCache ( ) ;
nl . UpdateCache ( ) ;
// Add both lines to changedlines
if ( changedlines ! = null ) {
changedlines . Add ( l ) ;
changedlines . Add ( nl ) ;
}
// Count the split
splitted = true ;
break ;
}
2009-04-19 18:07:22 +00:00
}
}
2013-12-17 13:18:44 +00:00
// Will have to restart when splitted
// TODO: If we make (linked) lists from the collections first,
// we don't have to restart when splitted?
if ( splitted ) break ;
2009-04-19 18:07:22 +00:00
}
}
}
}
while ( splitted ) ;
2010-08-15 19:43:00 +00:00
return true ;
2009-04-19 18:07:22 +00:00
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the side closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Sidedef NearestSidedef ( ICollection < Sidedef > selection , Vector2D pos )
{
Sidedef closest = null ;
float distance = float . MaxValue ;
// Go for all sidedefs in selection
foreach ( Sidedef sd in selection )
{
// Calculate distance and check if closer than previous find
float d = sd . Line . SafeDistanceToSq ( pos , true ) ;
2009-05-06 07:26:12 +00:00
if ( d = = distance )
{
// Same distance, so only pick the one that is on the right side of the line
float side = sd . Line . SideOfLine ( pos ) ;
if ( ( ( side < = 0.0f ) & & sd . IsFront ) | | ( ( side > 0.0f ) & & ! sd . IsFront ) )
{
closest = sd ;
distance = d ;
}
}
else if ( d < distance )
2009-04-19 18:07:22 +00:00
{
// This one is closer
closest = sd ;
distance = d ;
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the line closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Linedef NearestLinedef ( ICollection < Linedef > selection , Vector2D pos )
{
Linedef closest = null ;
float distance = float . MaxValue ;
// Go for all linedefs in selection
foreach ( Linedef l in selection )
{
// Calculate distance and check if closer than previous find
float d = l . SafeDistanceToSq ( pos , true ) ;
if ( d < distance )
{
// This one is closer
closest = l ;
distance = d ;
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the line closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Linedef NearestLinedefRange ( ICollection < Linedef > selection , Vector2D pos , float maxrange )
{
Linedef closest = null ;
float distance = float . MaxValue ;
float maxrangesq = maxrange * maxrange ;
float d ;
// Go for all linedefs in selection
foreach ( Linedef l in selection )
{
// Calculate distance and check if closer than previous find
d = l . SafeDistanceToSq ( pos , true ) ;
2013-12-10 12:19:27 +00:00
if ( d < distance & & d < = maxrangesq )
2009-04-19 18:07:22 +00:00
{
// This one is closer
closest = l ;
distance = d ;
}
}
// Return result
return closest ;
}
2013-03-18 13:52:27 +00:00
/// <summary>mxd. This finds the line closest to the specified position excluding given list of linedefs.</summary>
public Linedef NearestLinedef ( Vector2D pos , List < Linedef > linesToExclude ) {
Linedef closest = null ;
float distance = float . MaxValue ;
// Go for all linedefs in selection
foreach ( Linedef l in linedefs ) {
if ( linesToExclude . Contains ( l ) ) continue ;
// Calculate distance and check if closer than previous find
float d = l . SafeDistanceToSq ( pos , true ) ;
if ( d < distance ) {
// This one is closer
closest = l ;
distance = d ;
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the vertex closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Vertex NearestVertex ( ICollection < Vertex > selection , Vector2D pos )
{
Vertex closest = null ;
float distance = float . MaxValue ;
float d ;
// Go for all vertices in selection
foreach ( Vertex v in selection )
{
// Calculate distance and check if closer than previous find
d = v . DistanceToSq ( pos ) ;
if ( d < distance )
{
// This one is closer
closest = v ;
distance = d ;
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the thing closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Thing NearestThing ( ICollection < Thing > selection , Vector2D pos )
{
Thing closest = null ;
float distance = float . MaxValue ;
float d ;
// Go for all things in selection
foreach ( Thing t in selection )
{
// Calculate distance and check if closer than previous find
d = t . DistanceToSq ( pos ) ;
if ( d < distance )
{
// This one is closer
closest = t ;
distance = d ;
}
}
// Return result
return closest ;
}
2013-03-18 13:52:27 +00:00
/// <summary>mxd. This finds the thing closest to the specified thing.</summary>
public static Thing NearestThing ( ICollection < Thing > selection , Thing thing ) {
Thing closest = null ;
float distance = float . MaxValue ;
float d ;
// Go for all things in selection
foreach ( Thing t in selection ) {
if ( t = = thing ) continue ;
// Calculate distance and check if closer than previous find
d = t . DistanceToSq ( thing . Position ) ;
if ( d < distance ) {
// This one is closer
closest = t ;
distance = d ;
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the vertex closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Vertex NearestVertexSquareRange ( ICollection < Vertex > selection , Vector2D pos , float maxrange )
{
RectangleF range = RectangleF . FromLTRB ( pos . x - maxrange , pos . y - maxrange , pos . x + maxrange , pos . y + maxrange ) ;
Vertex closest = null ;
float distance = float . MaxValue ;
2013-12-10 12:19:27 +00:00
float d , px , py ;
2009-04-19 18:07:22 +00:00
// Go for all vertices in selection
foreach ( Vertex v in selection )
{
2013-12-10 12:19:27 +00:00
px = v . Position . x ;
py = v . Position . y ;
2013-12-11 09:47:35 +00:00
//mxd. Within range?
if ( ( v . Position . x < range . Left ) | | ( v . Position . x > range . Right )
| | ( v . Position . y < range . Top ) | | ( v . Position . y > range . Bottom ) )
continue ;
2013-12-10 12:19:27 +00:00
// Close than previous find?
d = Math . Abs ( px - pos . x ) + Math . Abs ( py - pos . y ) ;
if ( d < distance )
2009-04-19 18:07:22 +00:00
{
2013-12-10 12:19:27 +00:00
// This one is closer
closest = v ;
distance = d ;
2009-04-19 18:07:22 +00:00
}
}
// Return result
return closest ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This finds the thing closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public static Thing NearestThingSquareRange ( ICollection < Thing > selection , Vector2D pos , float maxrange )
{
RectangleF range = RectangleF . FromLTRB ( pos . x - maxrange , pos . y - maxrange , pos . x + maxrange , pos . y + maxrange ) ;
Thing closest = null ;
float distance = float . MaxValue ;
2013-12-10 12:19:27 +00:00
float d , px , py ;
2009-04-19 18:07:22 +00:00
2013-12-10 12:19:27 +00:00
// Go for all things in selection
2009-04-19 18:07:22 +00:00
foreach ( Thing t in selection )
{
2013-12-10 12:19:27 +00:00
px = t . Position . x ;
py = t . Position . y ;
//mxd. Within range?
if ( px < range . Left - t . Size | | px > range . Right + t . Size | | py < range . Top - t . Size | | py > range . Bottom + t . Size ) continue ;
// Close than previous find?
d = Math . Abs ( px - pos . x ) + Math . Abs ( py - pos . y ) ;
if ( d < distance )
2009-04-19 18:07:22 +00:00
{
2013-12-10 12:19:27 +00:00
// This one is closer
closest = t ;
distance = d ;
2009-04-19 18:07:22 +00:00
}
}
// Return result
return closest ;
}
#endregion
#region = = = = = = = = = = = = = = = = = = Tools
2009-05-19 09:59:19 +00:00
/// <summary>This snaps all vertices to the map format accuracy. Call this to ensure the vertices are at valid coordinates.</summary>
2009-04-19 18:07:22 +00:00
public void SnapAllToAccuracy ( )
{
foreach ( Vertex v in vertices ) v . SnapToAccuracy ( ) ;
foreach ( Thing t in things ) t . SnapToAccuracy ( ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This returns the next unused tag number.</summary>
2009-04-19 18:07:22 +00:00
public int GetNewTag ( )
{
Dictionary < int , bool > usedtags = new Dictionary < int , bool > ( ) ;
2009-07-09 14:03:47 +00:00
ForAllTags ( NewTagHandler , false , usedtags ) ;
ForAllTags ( NewTagHandler , true , usedtags ) ;
2009-04-19 18:07:22 +00:00
2009-07-09 14:03:47 +00:00
// Now find the first unused index
for ( int i = 1 ; i < = General . Map . FormatInterface . MaxTag ; i + + )
if ( ! usedtags . ContainsKey ( i ) ) return i ;
2009-04-19 18:07:22 +00:00
2009-07-09 14:03:47 +00:00
// All tags used!
return 0 ;
}
2013-08-28 08:29:06 +00:00
//mxd
/// <summary>This returns the tag number, which is not used by any map element of given type. This method doesn't check action arguments!</summary>
public int GetNewTag ( UniversalType elementType )
{
Dictionary < int , bool > usedtags = new Dictionary < int , bool > ( ) ;
switch ( elementType ) {
case UniversalType . ThingTag :
for ( int i = 0 ; i < things . Length ; i + + ) {
if ( things [ i ] . Tag > 0 & & ! usedtags . ContainsKey ( things [ i ] . Tag ) )
usedtags . Add ( things [ i ] . Tag , false ) ;
}
break ;
case UniversalType . LinedefTag :
for ( int i = 0 ; i < linedefs . Length ; i + + ) {
if ( linedefs [ i ] . Tag > 0 & & ! usedtags . ContainsKey ( linedefs [ i ] . Tag ) )
usedtags . Add ( linedefs [ i ] . Tag , false ) ;
}
break ;
case UniversalType . SectorTag :
for ( int i = 0 ; i < sectors . Length ; i + + ) {
if ( sectors [ i ] . Tag > 0 & & ! usedtags . ContainsKey ( sectors [ i ] . Tag ) )
usedtags . Add ( sectors [ i ] . Tag , false ) ;
}
break ;
}
// Now find the first unused index
for ( int i = 1 ; i < = General . Map . FormatInterface . MaxTag ; i + + )
if ( ! usedtags . ContainsKey ( i ) ) return i ;
// All tags used!
return 0 ;
}
2009-07-09 14:03:47 +00:00
/// <summary>This returns the next unused tag number within the marked geometry.</summary>
public int GetNewTag ( bool marked )
{
Dictionary < int , bool > usedtags = new Dictionary < int , bool > ( ) ;
ForAllTags ( NewTagHandler , marked , usedtags ) ;
2009-04-19 18:07:22 +00:00
// Now find the first unused index
for ( int i = 1 ; i < = General . Map . FormatInterface . MaxTag ; i + + )
if ( ! usedtags . ContainsKey ( i ) ) return i ;
2009-07-09 14:03:47 +00:00
// All tags used!
2009-04-19 18:07:22 +00:00
return 0 ;
}
2009-05-19 09:59:19 +00:00
2009-07-09 14:03:47 +00:00
/// <summary>This returns the next unused tag number.</summary>
public List < int > GetMultipleNewTags ( int count )
{
List < int > newtags = new List < int > ( count ) ;
2009-07-10 08:25:04 +00:00
if ( count > 0 )
2009-07-09 14:03:47 +00:00
{
2009-07-10 08:25:04 +00:00
Dictionary < int , bool > usedtags = new Dictionary < int , bool > ( ) ;
ForAllTags ( NewTagHandler , false , usedtags ) ;
ForAllTags ( NewTagHandler , true , usedtags ) ;
// Find unused tags and add them
for ( int i = 1 ; i < = General . Map . FormatInterface . MaxTag ; i + + )
2009-07-09 14:03:47 +00:00
{
2009-07-10 08:25:04 +00:00
if ( ! usedtags . ContainsKey ( i ) )
{
newtags . Add ( i ) ;
if ( newtags . Count = = count ) break ;
}
2009-07-09 14:03:47 +00:00
}
}
return newtags ;
}
/// <summary>This returns the next unused tag number within the marked geometry.</summary>
public List < int > GetMultipleNewTags ( int count , bool marked )
{
List < int > newtags = new List < int > ( count ) ;
2009-07-10 08:25:04 +00:00
if ( count > 0 )
2009-07-09 14:03:47 +00:00
{
2009-07-10 08:25:04 +00:00
Dictionary < int , bool > usedtags = new Dictionary < int , bool > ( ) ;
ForAllTags ( NewTagHandler , marked , usedtags ) ;
// Find unused tags and add them
for ( int i = 1 ; i < = General . Map . FormatInterface . MaxTag ; i + + )
2009-07-09 14:03:47 +00:00
{
2009-07-10 08:25:04 +00:00
if ( ! usedtags . ContainsKey ( i ) )
{
newtags . Add ( i ) ;
if ( newtags . Count = = count ) break ;
}
2009-07-09 14:03:47 +00:00
}
}
2009-07-10 08:25:04 +00:00
2009-07-09 14:03:47 +00:00
return newtags ;
}
// Handler for finding a new tag
private void NewTagHandler ( MapElement element , bool actionargument , UniversalType type , ref int value , Dictionary < int , bool > usedtags )
{
usedtags [ value ] = true ;
}
/// <summary>This calls a function for all tag fields in the marked or unmarked geometry. The obj parameter can be anything you wish to pass on to your TagHandler function.</summary>
public void ForAllTags < T > ( TagHandler < T > handler , bool marked , T obj )
{
// Remove tags from sectors
foreach ( Sector s in sectors )
if ( s . Marked = = marked )
{
int tag = s . Tag ;
handler ( s , false , UniversalType . SectorTag , ref tag , obj ) ;
if ( tag ! = s . Tag ) s . Tag = tag ;
}
// Remove tags from things
if ( General . Map . FormatInterface . HasThingTag )
{
foreach ( Thing t in things )
if ( t . Marked = = marked )
{
int tag = t . Tag ;
handler ( t , false , UniversalType . ThingTag , ref tag , obj ) ;
if ( tag ! = t . Tag ) t . Tag = tag ;
}
}
// Remove tags from thing actions
if ( General . Map . FormatInterface . HasThingAction & &
General . Map . FormatInterface . HasActionArgs )
{
foreach ( Thing t in things )
{
if ( t . Marked = = marked )
{
LinedefActionInfo info = General . Map . Config . GetLinedefActionInfo ( t . Action ) ;
for ( int i = 0 ; i < Thing . NUM_ARGS ; i + + )
if ( info . Args [ i ] . Used & & CheckIsTagType ( info . Args [ i ] . Type ) )
{
int tag = t . Args [ i ] ;
handler ( t , true , ( UniversalType ) ( info . Args [ i ] . Type ) , ref tag , obj ) ;
if ( tag ! = t . Args [ i ] ) t . Args [ i ] = tag ;
}
}
}
}
// Remove tags from linedefs
if ( General . Map . FormatInterface . HasLinedefTag )
{
foreach ( Linedef l in linedefs )
if ( l . Marked = = marked )
{
int tag = l . Tag ;
handler ( l , false , UniversalType . LinedefTag , ref tag , obj ) ;
if ( tag ! = l . Tag ) l . Tag = tag ;
}
}
// Remove tags from linedef actions
if ( General . Map . FormatInterface . HasActionArgs )
{
foreach ( Linedef l in linedefs )
{
if ( l . Marked = = marked )
{
LinedefActionInfo info = General . Map . Config . GetLinedefActionInfo ( l . Action ) ;
for ( int i = 0 ; i < Linedef . NUM_ARGS ; i + + )
if ( info . Args [ i ] . Used & & CheckIsTagType ( info . Args [ i ] . Type ) )
{
int tag = l . Args [ i ] ;
handler ( l , true , ( UniversalType ) ( info . Args [ i ] . Type ) , ref tag , obj ) ;
if ( tag ! = l . Args [ i ] ) l . Args [ i ] = tag ;
}
}
}
}
}
// This checks if the given action argument type is a tag type
private bool CheckIsTagType ( int argtype )
{
return ( argtype = = ( int ) UniversalType . LinedefTag ) | |
( argtype = = ( int ) UniversalType . SectorTag ) | |
( argtype = = ( int ) UniversalType . ThingTag ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This makes a list of lines related to marked vertices.
/// A line is unstable when one vertex is marked and the other isn't.</summary>
2009-04-19 18:07:22 +00:00
public ICollection < Linedef > LinedefsFromMarkedVertices ( bool includeunselected , bool includestable , bool includeunstable )
{
2009-06-05 19:03:56 +00:00
List < Linedef > list = new List < Linedef > ( ( numlinedefs / 2 ) + 1 ) ;
2009-04-19 18:07:22 +00:00
// Go for all lines
foreach ( Linedef l in linedefs )
{
// Check if this is to be included
if ( ( includestable & & ( l . Start . Marked & & l . End . Marked ) ) | |
( includeunstable & & ( l . Start . Marked ^ l . End . Marked ) ) | |
( includeunselected & & ( ! l . Start . Marked & & ! l . End . Marked ) ) )
{
// Add to list
list . Add ( l ) ;
}
}
// Return result
return list ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This makes a list of unstable lines from the given vertices.
/// A line is unstable when one vertex is selected and the other isn't.</summary>
2009-04-19 18:07:22 +00:00
public static ICollection < Linedef > UnstableLinedefsFromVertices ( ICollection < Vertex > verts )
{
Dictionary < Linedef , Linedef > lines = new Dictionary < Linedef , Linedef > ( ) ;
// Go for all vertices
foreach ( Vertex v in verts )
{
// Go for all lines
foreach ( Linedef l in v . Linedefs )
{
// If the line exists in the list
if ( lines . ContainsKey ( l ) )
{
// Remove it
lines . Remove ( l ) ;
}
// Otherwise add it
else
{
// Add the line
lines . Add ( l , l ) ;
}
}
}
// Return result
return new List < Linedef > ( lines . Values ) ;
}
2009-05-19 09:59:19 +00:00
2013-09-11 09:47:53 +00:00
//mxd
/// <summary>This returns a sector if given coordinates lie inide one.</summary>
public Sector GetSectorByCoordinates ( Vector2D pos ) {
Linedef nl ;
Sector sector = null ;
nl = NearestLinedef ( pos ) ;
if ( nl ! = null ) {
// Check what side of line we are at
if ( nl . SideOfLine ( pos ) < 0f ) {
// Front side
if ( nl . Front ! = null ) sector = nl . Front . Sector ; else sector = null ;
} else {
// Back side
if ( nl . Back ! = null ) sector = nl . Back . Sector ; else sector = null ;
}
}
return sector ;
}
2012-06-07 01:06:37 +00:00
2009-05-19 09:59:19 +00:00
/// <summary>This finds the line closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public Linedef NearestLinedef ( Vector2D pos ) { return MapSet . NearestLinedef ( linedefs , pos ) ; }
2009-05-19 09:59:19 +00:00
/// <summary>This finds the line closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public Linedef NearestLinedefRange ( Vector2D pos , float maxrange ) { return MapSet . NearestLinedefRange ( linedefs , pos , maxrange ) ; }
2009-05-19 09:59:19 +00:00
/// <summary>This finds the vertex closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public Vertex NearestVertex ( Vector2D pos ) { return MapSet . NearestVertex ( vertices , pos ) ; }
2009-05-19 09:59:19 +00:00
/// <summary>This finds the vertex closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public Vertex NearestVertexSquareRange ( Vector2D pos , float maxrange ) { return MapSet . NearestVertexSquareRange ( vertices , pos , maxrange ) ; }
2009-05-19 09:59:19 +00:00
/// <summary>This finds the thing closest to the specified position.</summary>
2009-04-19 18:07:22 +00:00
public Thing NearestThingSquareRange ( Vector2D pos , float maxrange ) { return MapSet . NearestThingSquareRange ( things , pos , maxrange ) ; }
2009-05-19 09:59:19 +00:00
/// <summary>This finds the closest unselected linedef that is not connected to the given vertex.</summary>
2009-04-19 18:07:22 +00:00
public Linedef NearestUnselectedUnreferencedLinedef ( Vector2D pos , float maxrange , Vertex v , out float distance )
{
Linedef closest = null ;
distance = float . MaxValue ;
float maxrangesq = maxrange * maxrange ;
float d ;
// Go for all linedefs in selection
foreach ( Linedef l in linedefs )
{
// Calculate distance and check if closer than previous find
d = l . SafeDistanceToSq ( pos , true ) ;
if ( ( d < = maxrangesq ) & & ( d < distance ) )
{
// Check if not selected
// Check if linedef is not connected to v
if ( ( l . Start ! = v ) & & ( l . End ! = v ) )
{
// This one is closer
closest = l ;
distance = d ;
}
}
}
// Return result
return closest ;
}
// This performs sidedefs compression
// Note: Only use this for saving, because this messes up the expected data structure horribly.
internal void CompressSidedefs ( )
{
2009-06-05 19:03:56 +00:00
Dictionary < uint , List < Sidedef > > storedsides = new Dictionary < uint , List < Sidedef > > ( numsidedefs ) ;
int originalsidescount = numsidedefs ;
2012-07-12 22:34:12 +00:00
float starttime = General . Clock . CurrentTime ;
2010-01-09 19:23:54 +00:00
BeginAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
2009-06-05 19:03:56 +00:00
int sn = 0 ;
while ( sn < numsidedefs )
2009-04-19 18:07:22 +00:00
{
Sidedef stored = null ;
2009-06-05 19:03:56 +00:00
Sidedef snsd = sidedefs [ sn ] ;
2010-01-09 19:07:56 +00:00
2009-04-19 18:07:22 +00:00
// Check if checksum is stored
bool samesidedef = false ;
2009-06-05 19:03:56 +00:00
uint checksum = snsd . GetChecksum ( ) ;
2009-04-19 18:07:22 +00:00
bool checksumstored = storedsides . ContainsKey ( checksum ) ;
if ( checksumstored )
{
List < Sidedef > othersides = storedsides [ checksum ] ;
foreach ( Sidedef os in othersides )
{
// They must be in the same sector
2010-01-09 19:07:56 +00:00
if ( snsd . Sector = = os . Sector )
2009-04-19 18:07:22 +00:00
{
// Check if sidedefs are really the same
stored = os ;
MemoryStream sidemem = new MemoryStream ( 1024 ) ;
SerializerStream sidedata = new SerializerStream ( sidemem ) ;
MemoryStream othermem = new MemoryStream ( 1024 ) ;
SerializerStream otherdata = new SerializerStream ( othermem ) ;
2009-06-05 19:03:56 +00:00
snsd . ReadWrite ( sidedata ) ;
2009-04-19 18:07:22 +00:00
os . ReadWrite ( otherdata ) ;
2010-01-09 19:07:56 +00:00
if ( sidemem . Length = = othermem . Length )
2009-04-19 18:07:22 +00:00
{
samesidedef = true ;
sidemem . Seek ( 0 , SeekOrigin . Begin ) ;
othermem . Seek ( 0 , SeekOrigin . Begin ) ;
2010-01-09 19:07:56 +00:00
for ( int i = 0 ; i < sidemem . Length ; i + + )
2009-04-19 18:07:22 +00:00
{
2010-01-09 19:07:56 +00:00
if ( sidemem . ReadByte ( ) ! = othermem . ReadByte ( ) )
2009-04-19 18:07:22 +00:00
{
samesidedef = false ;
break ;
}
}
}
2010-01-09 19:07:56 +00:00
if ( samesidedef ) break ;
2009-04-19 18:07:22 +00:00
}
}
}
// Same sidedef?
if ( samesidedef )
{
// Replace with stored sidedef
2009-06-05 19:03:56 +00:00
bool isfront = snsd . IsFront ;
2010-01-09 19:07:56 +00:00
Linedef ld = snsd . Line ;
2009-06-11 21:21:20 +00:00
snsd . Line . DetachSidedefP ( snsd ) ;
2009-04-19 18:07:22 +00:00
if ( isfront )
2010-01-09 19:07:56 +00:00
ld . AttachFront ( stored ) ;
2009-04-19 18:07:22 +00:00
else
2010-01-09 19:07:56 +00:00
ld . AttachBack ( stored ) ;
2009-04-19 18:07:22 +00:00
// Remove the sidedef
2009-06-11 21:21:20 +00:00
snsd . SetSector ( null ) ;
2009-06-05 19:03:56 +00:00
RemoveSidedef ( sn ) ;
2009-04-19 18:07:22 +00:00
}
else
{
// Store this new one
if ( checksumstored )
{
2009-06-05 19:03:56 +00:00
storedsides [ checksum ] . Add ( snsd ) ;
2009-04-19 18:07:22 +00:00
}
else
{
2013-08-14 09:02:37 +00:00
List < Sidedef > newlist = new List < Sidedef > ( 4 ) { snsd } ;
2009-04-19 18:07:22 +00:00
storedsides . Add ( checksum , newlist ) ;
}
2009-06-05 19:03:56 +00:00
// Next
sn + + ;
2009-04-19 18:07:22 +00:00
}
}
2010-01-09 19:23:54 +00:00
EndAddRemove ( ) ;
2009-04-19 18:07:22 +00:00
// Output info
2013-09-11 09:47:53 +00:00
float endtime = General . Clock . CurrentTime ;
2012-07-12 22:34:12 +00:00
float deltatimesec = ( endtime - starttime ) / 1000.0f ;
2013-08-14 09:02:37 +00:00
float ratio = 100.0f - ( ( numsidedefs / ( float ) originalsidescount ) * 100.0f ) ;
2009-06-05 19:03:56 +00:00
General . WriteLogLine ( "Sidedefs compressed: " + numsidedefs + " remaining out of " + originalsidescount + " (" + ratio . ToString ( "########0.00" ) + "%) in " + deltatimesec . ToString ( "########0.00" ) + " seconds" ) ;
2009-04-19 18:07:22 +00:00
}
// This converts flags and activations to UDMF fields
internal void TranslateToUDMF ( )
{
foreach ( Linedef l in linedefs ) l . TranslateToUDMF ( ) ;
foreach ( Thing t in things ) t . TranslateToUDMF ( ) ;
}
// This converts UDMF fields back into flags and activations
// NOTE: Only converts the marked items
internal void TranslateFromUDMF ( )
{
foreach ( Linedef l in linedefs ) if ( l . Marked ) l . TranslateFromUDMF ( ) ;
foreach ( Thing t in things ) if ( t . Marked ) t . TranslateFromUDMF ( ) ;
}
2009-05-19 09:59:19 +00:00
/// <summary>This removes unused vertices.</summary>
2009-04-19 18:07:22 +00:00
public void RemoveUnusedVertices ( )
{
// Go for all vertices
2009-06-05 19:03:56 +00:00
int index = numvertices - 1 ;
while ( index > = 0 )
2009-04-19 18:07:22 +00:00
{
2009-06-05 19:03:56 +00:00
if ( ( vertices [ index ] ! = null ) & & ( vertices [ index ] . Linedefs . Count = = 0 ) )
vertices [ index ] . Dispose ( ) ;
else
index - - ;
2009-04-19 18:07:22 +00:00
}
}
2013-03-18 13:52:27 +00:00
2013-09-11 09:47:53 +00:00
//mxd
public void UpdateCustomLinedefColors ( ) {
foreach ( Linedef l in linedefs )
l . UpdateColorPreset ( ) ;
}
2009-04-19 18:07:22 +00:00
#endregion
}
}