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 ;
2014-12-22 21:36:49 +00:00
using System.Drawing ;
Added, Visual mode, GLDEFS, GLOOME: subtractive glow is now supported.
Changed, Visual mode: changed thing fog calculation logic. Should be closer to GZDoom now.
Fixed, GLDEFS parser: "height" texture parameter was not treated as optional.
Fixed, text lump parsers: in some cases incorrect line number was displayed in error and warning messages.
Fixed, Visual mode: glow effect was not applied to sectors with 3 sidedefs.
Fixed, Visual mode: in some cases glow effect was not updated when replacing textures.
Fixed, general interface: "Full Brightness" button state was not updated during map loading.
Fixed, Drag Linedefs/Vertices/Sectors/Things modes: positions of line length labels were not updated while panning the view.
Cosmetic: added a bunch of new icons.
Cosmetic: changed Visual mode crosshair.
2015-08-27 20:46:49 +00:00
using System.Globalization ;
2016-03-23 14:52:33 +00:00
using System.IO ;
2009-04-19 18:07:22 +00:00
using System.Windows.Forms ;
using CodeImp.DoomBuilder.Data ;
2014-12-22 21:36:49 +00:00
using CodeImp.DoomBuilder.Geometry ;
Added, Visual mode, GLDEFS, GLOOME: subtractive glow is now supported.
Changed, Visual mode: changed thing fog calculation logic. Should be closer to GZDoom now.
Fixed, GLDEFS parser: "height" texture parameter was not treated as optional.
Fixed, text lump parsers: in some cases incorrect line number was displayed in error and warning messages.
Fixed, Visual mode: glow effect was not applied to sectors with 3 sidedefs.
Fixed, Visual mode: in some cases glow effect was not updated when replacing textures.
Fixed, general interface: "Full Brightness" button state was not updated during map loading.
Fixed, Drag Linedefs/Vertices/Sectors/Things modes: positions of line length labels were not updated while panning the view.
Cosmetic: added a bunch of new icons.
Cosmetic: changed Visual mode crosshair.
2015-08-27 20:46:49 +00:00
using CodeImp.DoomBuilder.GZBuilder.Data ;
2009-04-19 18:07:22 +00:00
using CodeImp.DoomBuilder.Map ;
using CodeImp.DoomBuilder.Rendering ;
using CodeImp.DoomBuilder.VisualModes ;
#endregion
namespace CodeImp.DoomBuilder.BuilderModes
{
internal abstract class BaseVisualGeometrySidedef : VisualGeometry , IVisualEventReceiver
{
#region = = = = = = = = = = = = = = = = = = Constants
private const float DRAG_ANGLE_TOLERANCE = 0.06f ;
#endregion
#region = = = = = = = = = = = = = = = = = = Variables
2014-12-22 21:36:49 +00:00
protected readonly BaseVisualMode mode ;
2009-04-19 18:07:22 +00:00
2012-11-27 21:12:20 +00:00
protected Plane top ;
protected Plane bottom ;
2009-04-19 18:07:22 +00:00
protected long setuponloadedtexture ;
2019-12-31 02:44:36 +00:00
private long lastsetuponloadedtexture ;
// UV dragging
2020-05-21 12:20:02 +00:00
private double dragstartanglexy ;
private double dragstartanglez ;
2009-04-19 18:07:22 +00:00
private Vector3D dragorigin ;
private Vector3D deltaxy ;
private Vector3D deltaz ;
private int startoffsetx ;
private int startoffsety ;
protected bool uvdragging ;
2009-05-03 19:22:32 +00:00
private int prevoffsetx ; // We have to provide delta offsets, but I don't
private int prevoffsety ; // want to calculate with delta offsets to prevent
// inaccuracy in the dragging.
2016-01-16 12:46:44 +00:00
private static List < BaseVisualSector > updatelist ; //mxd
private bool performautoselection ; //mxd
2013-07-19 15:30:58 +00:00
2009-04-19 18:07:22 +00:00
// Undo/redo
2016-04-19 20:40:42 +00:00
protected int undoticket ;
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Properties
public bool IsDraggingUV { get { return uvdragging ; } }
new public BaseVisualSector Sector { get { return ( BaseVisualSector ) base . Sector ; } }
#endregion
#region = = = = = = = = = = = = = = = = = = Constructor / Destructor
// Constructor for sidedefs
2015-05-27 12:38:03 +00:00
protected BaseVisualGeometrySidedef ( BaseVisualMode mode , VisualSector vs , Sidedef sd ) : base ( vs , sd )
2009-04-19 18:07:22 +00:00
{
this . mode = mode ;
this . deltaz = new Vector3D ( 0.0f , 0.0f , 1.0f ) ;
this . deltaxy = ( sd . Line . End . Position - sd . Line . Start . Position ) * sd . Line . LengthInv ;
if ( ! sd . IsFront ) this . deltaxy = - this . deltaxy ;
2016-01-16 12:46:44 +00:00
this . performautoselection = ( mode . UseSelectionFromClassicMode & & sd . Line . Selected ) ; //mxd
}
#endregion
2013-03-18 13:52:27 +00:00
2016-01-16 12:46:44 +00:00
#region = = = = = = = = = = = = = = = = = = Methods
//mxd
override protected void PerformAutoSelection ( )
{
if ( ! performautoselection ) return ;
if ( Triangles > 0 )
2014-12-03 23:15:26 +00:00
{
2016-01-16 12:46:44 +00:00
dragstartanglexy = General . Map . VisualCamera . AngleXY ;
dragstartanglez = General . Map . VisualCamera . AngleZ ;
dragorigin = pickintersect ;
Point texoffset = GetTextureOffset ( ) ;
startoffsetx = texoffset . X ;
startoffsety = texoffset . Y ;
prevoffsetx = texoffset . X ;
prevoffsety = texoffset . Y ;
2013-03-18 13:52:27 +00:00
this . selected = true ;
mode . AddSelectedObject ( this ) ;
}
2009-04-19 18:07:22 +00:00
2016-01-16 12:46:44 +00:00
performautoselection = false ;
}
2009-04-19 18:07:22 +00:00
2012-11-27 21:12:20 +00:00
// This sets the renderstyle from linedef information and returns the alpha value or the vertices
protected byte SetLinedefRenderstyle ( bool solidasmask )
{
2014-02-21 14:42:12 +00:00
byte alpha ;
2015-09-27 21:09:14 +00:00
bool canhavealpha = ( this is VisualMiddleDouble | | this is VisualMiddle3D | | this is VisualMiddleBack ) ; //mxd
2012-11-27 21:12:20 +00:00
// From TranslucentLine action
if ( Sidedef . Line . Action = = 208 )
{
alpha = ( byte ) General . Clamp ( Sidedef . Line . Args [ 1 ] , 0 , 255 ) ;
2015-09-27 21:09:14 +00:00
if ( canhavealpha & & Sidedef . Line . Args [ 2 ] = = 1 )
2012-11-27 21:12:20 +00:00
this . RenderPass = RenderPass . Additive ;
2015-09-27 21:09:14 +00:00
else if ( canhavealpha & & ( alpha < 255 | | Texture . IsTranslucent ) )
2012-11-27 21:12:20 +00:00
this . RenderPass = RenderPass . Alpha ;
else if ( solidasmask )
this . RenderPass = RenderPass . Mask ;
else
this . RenderPass = RenderPass . Solid ;
}
2015-09-27 21:09:14 +00:00
// From UDMF field
2012-11-27 21:12:20 +00:00
else
{
2020-05-22 19:39:18 +00:00
alpha = ( byte ) ( Sidedef . Line . Fields . GetValue ( "alpha" , 1.0 ) * 255.0 ) ;
2015-02-06 09:01:33 +00:00
if ( alpha = = 255 & & Sidedef . Line . IsFlagSet ( "transparent" ) ) alpha = 64 ; //mxd
2014-05-05 14:24:57 +00:00
2015-09-27 21:09:14 +00:00
if ( canhavealpha & & Sidedef . Line . Fields . GetValue ( "renderstyle" , "translucent" ) = = "add" )
2012-11-27 21:12:20 +00:00
this . RenderPass = RenderPass . Additive ;
2015-09-27 21:09:14 +00:00
else if ( canhavealpha & & ( alpha < 255 | | Texture . IsTranslucent ) )
2012-11-27 21:12:20 +00:00
this . RenderPass = RenderPass . Alpha ;
else if ( solidasmask )
this . RenderPass = RenderPass . Mask ;
else
this . RenderPass = RenderPass . Solid ;
}
return alpha ;
}
2009-04-19 18:07:22 +00:00
// This performs a fast test in object picking
public override bool PickFastReject ( Vector3D from , Vector3D to , Vector3D dir )
{
// Check if intersection point is between top and bottom
2012-11-27 21:12:20 +00:00
return ( pickintersect . z > = bottom . GetZ ( pickintersect ) ) & & ( pickintersect . z < = top . GetZ ( pickintersect ) ) ;
2009-04-19 18:07:22 +00:00
}
2012-11-27 21:12:20 +00:00
2009-04-19 18:07:22 +00:00
// This performs an accurate test for object picking
2020-05-21 12:20:02 +00:00
public override bool PickAccurate ( Vector3D from , Vector3D to , Vector3D dir , ref double u_ray )
2009-04-19 18:07:22 +00:00
{
// The fast reject pass is already as accurate as it gets,
// so we just return the intersection distance here
u_ray = pickrayu ;
return true ;
}
2012-11-27 21:12:20 +00:00
// This creates vertices from a wall polygon and applies lighting
2015-02-06 09:01:33 +00:00
/ * protected List < WorldVertex > CreatePolygonVertices ( WallPolygon poly , TexturePlane tp , SectorData sd , int lightvalue , bool lightabsolute )
2012-11-27 21:12:20 +00:00
{
List < WallPolygon > polylist = new List < WallPolygon > ( 1 ) ;
polylist . Add ( poly ) ;
return CreatePolygonVertices ( polylist , tp , sd , lightvalue , lightabsolute ) ;
2015-02-06 09:01:33 +00:00
} * /
2012-11-27 21:12:20 +00:00
// This creates vertices from a wall polygon and applies lighting
protected List < WorldVertex > CreatePolygonVertices ( List < WallPolygon > poly , TexturePlane tp , SectorData sd , int lightvalue , bool lightabsolute )
{
List < WallPolygon > polygons = new List < WallPolygon > ( poly ) ;
List < WorldVertex > verts = new List < WorldVertex > ( ) ;
2015-11-20 14:31:54 +00:00
//mxd. Do complicated light level shenanigans only when there are extrafloors
if ( sd . LightLevels . Count > 2 )
2012-11-27 21:12:20 +00:00
{
2015-11-20 14:31:54 +00:00
SectorLevel prevlight = null ; //mxd
2012-11-27 21:12:20 +00:00
2015-11-20 14:31:54 +00:00
// Go for all levels to build geometry
for ( int i = sd . LightLevels . Count - 1 ; i > = 0 ; i - - )
2012-11-27 21:12:20 +00:00
{
2015-11-20 14:31:54 +00:00
SectorLevel l = sd . LightLevels [ i ] ;
2015-02-25 19:59:17 +00:00
2015-11-20 14:31:54 +00:00
//mxd. Skip current light level when it's between TYPE1 and TYPE1_BOTTOM
if ( prevlight ! = null
& & prevlight . type = = SectorLevelType . Light & & l . type = = SectorLevelType . Light
& & ( prevlight . lighttype = = LightLevelType . TYPE1 & & l . lighttype ! = LightLevelType . TYPE1_BOTTOM ) ) continue ;
2015-02-25 19:59:17 +00:00
2015-11-20 14:31:54 +00:00
if ( ( l ! = sd . Floor ) & & ( l ! = sd . Ceiling ) & & ( l . type ! = SectorLevelType . Floor | | l . splitsides /*(l.alpha < 255)*/ ) )
{
// Go for all polygons
int num = polygons . Count ;
Plane plane = ( l . type = = SectorLevelType . Floor ? l . plane . GetInverted ( ) : l . plane ) ; //mxd
for ( int pi = 0 ; pi < num ; pi + + )
{
// Split by plane
WallPolygon p = polygons [ pi ] ;
WallPolygon np = SplitPoly ( ref p , plane , false ) ;
if ( np . Count > 0 )
2012-11-27 21:12:20 +00:00
{
Fixed, Visual mode: in some cases ceiling glow effect was interfering with Transfer Brightness effect resulting in incorrectly lit sidedef geometry.
Fixed, Visual mode: UDMF sidedef brightness should be ignored when a wall section is affected by Transfer Brightness effect.
Fixed, Visual mode: any custom fog should be rendered regardless of sector brightness.
Fixed, Visual mode: "fogdensity" and "outsidefogdensity" MAPINFO values were processed incorrectly.
Fixed, Visual mode: in some cases Things were rendered twice during a render pass.
Fixed, Visual mode: floor glow effect should affect thing brightness only when applied to floor of the sector thing is in.
Fixed, TEXTURES parser: TEXTURES group was named incorrectly in the Textures Browser window when parsed from a WAD file.
Fixed, MAPINFO, GLDEFS, DECORATE parsers: "//$GZDB_SKIP" special comment was processed incorrectly.
Fixed, MAPINFO parser: "fogdensity" and "outsidefogdensity" properties are now initialized using GZDoom default value (255) instead of 0.
2016-01-25 13:42:53 +00:00
if ( l . type = = SectorLevelType . Glow )
{
//mxd. Glow levels should not affect light level
np . color = p . color ;
}
2015-11-20 14:31:54 +00:00
else
Fixed, Visual mode: in some cases ceiling glow effect was interfering with Transfer Brightness effect resulting in incorrectly lit sidedef geometry.
Fixed, Visual mode: UDMF sidedef brightness should be ignored when a wall section is affected by Transfer Brightness effect.
Fixed, Visual mode: any custom fog should be rendered regardless of sector brightness.
Fixed, Visual mode: "fogdensity" and "outsidefogdensity" MAPINFO values were processed incorrectly.
Fixed, Visual mode: in some cases Things were rendered twice during a render pass.
Fixed, Visual mode: floor glow effect should affect thing brightness only when applied to floor of the sector thing is in.
Fixed, TEXTURES parser: TEXTURES group was named incorrectly in the Textures Browser window when parsed from a WAD file.
Fixed, MAPINFO, GLDEFS, DECORATE parsers: "//$GZDB_SKIP" special comment was processed incorrectly.
Fixed, MAPINFO parser: "fogdensity" and "outsidefogdensity" properties are now initialized using GZDoom default value (255) instead of 0.
2016-01-25 13:42:53 +00:00
{
//mxd. Determine color
int lightlevel ;
// Sidedef part is not affected by 3d floor brightness
if ( l . type ! = SectorLevelType . Light & & ( l . disablelighting | | ! l . extrafloor ) )
lightlevel = ( lightabsolute ? lightvalue : l . brightnessbelow + lightvalue ) ;
// 3d floors and light transfer effects transfers brightness below them ignoring sidedef's brightness
else
lightlevel = l . brightnessbelow ;
PixelColor wallbrightness = PixelColor . FromInt ( mode . CalculateBrightness ( lightlevel , Sidedef ) ) ; //mxd
np . color = PixelColor . Modulate ( l . colorbelow , wallbrightness ) . WithAlpha ( 255 ) . ToInt ( ) ;
}
2015-11-20 14:31:54 +00:00
if ( p . Count = = 0 )
{
polygons [ pi ] = np ;
}
else
{
polygons [ pi ] = p ;
polygons . Add ( np ) ;
}
2012-11-27 21:12:20 +00:00
}
else
{
polygons [ pi ] = p ;
}
}
}
2015-04-14 11:33:57 +00:00
2015-11-20 14:31:54 +00:00
//mxd
if ( l . type = = SectorLevelType . Light ) prevlight = l ;
}
2012-11-27 21:12:20 +00:00
}
2015-11-20 14:31:54 +00:00
2012-11-27 21:12:20 +00:00
// Go for all polygons to make geometry
foreach ( WallPolygon p in polygons )
{
// Find texture coordinates for each vertex in the polygon
List < Vector2D > texc = new List < Vector2D > ( p . Count ) ;
2015-04-14 11:33:57 +00:00
foreach ( Vector3D v in p ) texc . Add ( tp . GetTextureCoordsAt ( v ) ) ;
2012-11-27 21:12:20 +00:00
// Now we create triangles from the polygon.
// The polygon is convex and clockwise, so this is a piece of cake.
2015-04-14 11:33:57 +00:00
if ( p . Count > 2 )
2012-11-27 21:12:20 +00:00
{
for ( int k = 1 ; k < ( p . Count - 1 ) ; k + + )
{
verts . Add ( new WorldVertex ( p [ 0 ] , p . color , texc [ 0 ] ) ) ;
verts . Add ( new WorldVertex ( p [ k ] , p . color , texc [ k ] ) ) ;
verts . Add ( new WorldVertex ( p [ k + 1 ] , p . color , texc [ k + 1 ] ) ) ;
}
}
}
2015-04-14 11:33:57 +00:00
//mxd. Interpolate vertex colors?
2017-02-01 07:48:13 +00:00
for ( int i = 0 ; i < verts . Count ; i + + )
2015-04-14 11:33:57 +00:00
{
2017-02-01 07:48:13 +00:00
//if(verts[i].c == PixelColor.INT_WHITE) continue; // Fullbright verts are not affected by glows.
verts [ i ] = InterpolateVertexColor ( verts [ i ] , sd ) ;
2015-04-14 11:33:57 +00:00
}
2012-11-27 21:12:20 +00:00
return verts ;
}
2015-04-14 11:33:57 +00:00
//mxd
private static WorldVertex InterpolateVertexColor ( WorldVertex v , SectorData data )
{
2017-02-01 07:48:13 +00:00
// don't process glows if fullbright.
2017-02-01 07:57:00 +00:00
//if (v.c == PixelColor.INT_WHITE)
// return v;
2017-02-01 07:48:13 +00:00
2015-04-14 11:33:57 +00:00
// Apply ceiling glow?
if ( data . CeilingGlow ! = null )
{
2020-05-21 12:20:02 +00:00
double cgz = data . CeilingGlowPlane . GetZ ( v . x , v . y ) ;
2015-04-14 11:33:57 +00:00
// Vertex is above ceiling glow plane?
if ( v . z > cgz )
{
2020-05-21 12:20:02 +00:00
double cz = data . Ceiling . plane . GetZ ( v . x , v . y ) ;
double delta = 1.0f - ( ( ( v . z - cgz ) / ( cz - cgz ) ) * 0.9f ) ;
2016-11-09 10:07:21 +00:00
PixelColor vertexcolor = PixelColor . FromInt ( v . c ) ;
PixelColor glowcolor = PixelColor . Add ( vertexcolor , data . CeilingGlow . Color ) ;
v . c = InterpolationTools . InterpolateColor ( glowcolor , vertexcolor , delta ) . WithAlpha ( 255 ) . ToInt ( ) ;
2015-04-14 11:33:57 +00:00
}
}
// Apply floor glow?
if ( data . FloorGlow ! = null )
{
2020-05-21 12:20:02 +00:00
double fgz = data . FloorGlowPlane . GetZ ( v . x , v . y ) ;
2015-04-14 11:33:57 +00:00
// Vertex is below floor glow plane?
if ( v . z < fgz )
{
2020-05-21 12:20:02 +00:00
double fz = data . Floor . plane . GetZ ( v . x , v . y ) ;
double delta = 1.0f - ( ( ( v . z - fz ) / ( fgz - fz ) ) * 0.9f ) ;
2016-11-09 10:07:21 +00:00
PixelColor vertexcolor = PixelColor . FromInt ( v . c ) ;
PixelColor glowcolor = PixelColor . Add ( vertexcolor , data . FloorGlow . Color ) ;
v . c = InterpolationTools . InterpolateColor ( vertexcolor , glowcolor , delta ) . WithAlpha ( 255 ) . ToInt ( ) ;
2015-04-14 11:33:57 +00:00
}
}
2017-02-01 07:57:00 +00:00
// [ZZ] process sector top and bottom colors.
// block
{
2020-05-21 12:20:02 +00:00
double cz = data . Ceiling . plane . GetZ ( v . x , v . y ) ;
double cgz = data . Floor . plane . GetZ ( v . x , v . y ) ;
double delta = 1.0f - ( ( ( v . z - cgz ) / ( cz - cgz ) ) * 0.9f ) ;
2017-02-01 07:57:00 +00:00
PixelColor vertexcolor = PixelColor . FromInt ( v . c ) ;
PixelColor topcolor = PixelColor . Modulate ( vertexcolor , data . ColorWallTop ) ;
PixelColor bottomcolor = PixelColor . Modulate ( vertexcolor , data . ColorWallBottom ) ;
v . c = InterpolationTools . InterpolateColor ( topcolor , bottomcolor , delta ) . WithAlpha ( 255 ) . ToInt ( ) ;
}
return v ;
2015-04-14 11:33:57 +00:00
}
2012-11-27 21:12:20 +00:00
// This splits a polygon with a plane and returns the other part as a new polygon
// The polygon is expected to be convex and clockwise
2014-12-22 21:36:49 +00:00
protected static WallPolygon SplitPoly ( ref WallPolygon poly , Plane p , bool keepfront )
2012-11-27 21:12:20 +00:00
{
2020-05-21 12:20:02 +00:00
const double NEAR_ZERO = 0.01f ;
2012-11-27 21:12:20 +00:00
WallPolygon front = new WallPolygon ( poly . Count ) ;
WallPolygon back = new WallPolygon ( poly . Count ) ;
poly . CopyProperties ( front ) ;
poly . CopyProperties ( back ) ;
if ( poly . Count > 0 )
{
// Go for all vertices to see which side they have to be on
Vector3D v1 = poly [ poly . Count - 1 ] ;
2020-05-21 12:20:02 +00:00
double side1 = p . Distance ( v1 ) ;
2012-11-27 21:12:20 +00:00
for ( int i = 0 ; i < poly . Count ; i + + )
{
// Fetch vertex and determine side
Vector3D v2 = poly [ i ] ;
2020-05-21 12:20:02 +00:00
double side2 = p . Distance ( v2 ) ;
2012-11-27 21:12:20 +00:00
// Front?
if ( side2 > NEAR_ZERO )
{
if ( side1 < - NEAR_ZERO )
{
// Split line with plane and insert the vertex
2020-05-21 12:20:02 +00:00
double u = 0.0f ;
2012-11-27 21:12:20 +00:00
p . GetIntersection ( v1 , v2 , ref u ) ;
Vector3D v3 = v1 + ( v2 - v1 ) * u ;
front . Add ( v3 ) ;
back . Add ( v3 ) ;
}
front . Add ( v2 ) ;
}
// Back?
else if ( side2 < - NEAR_ZERO )
{
if ( side1 > NEAR_ZERO )
{
// Split line with plane and insert the vertex
2020-05-21 12:20:02 +00:00
double u = 0.0f ;
2012-11-27 21:12:20 +00:00
p . GetIntersection ( v1 , v2 , ref u ) ;
Vector3D v3 = v1 + ( v2 - v1 ) * u ;
front . Add ( v3 ) ;
back . Add ( v3 ) ;
}
back . Add ( v2 ) ;
}
else
{
// On the plane, add to both polygons
front . Add ( v2 ) ;
back . Add ( v2 ) ;
}
// Next
v1 = v2 ;
side1 = side2 ;
}
}
if ( keepfront )
{
poly = front ;
return back ;
}
else
{
poly = back ;
return front ;
}
}
// This crops a polygon with a plane and keeps only a certain part of the polygon
2014-12-22 21:36:49 +00:00
protected static void CropPoly ( ref WallPolygon poly , Plane p , bool keepfront )
2012-11-27 21:12:20 +00:00
{
2020-05-21 12:20:02 +00:00
const double NEAR_ZERO = 0.01f ;
double sideswitch = keepfront ? 1 : - 1 ;
2012-11-27 21:12:20 +00:00
WallPolygon newp = new WallPolygon ( poly . Count ) ;
poly . CopyProperties ( newp ) ;
if ( poly . Count > 0 )
{
// First split lines that cross the plane so that we have vertices on the plane where the lines cross
Vector3D v1 = poly [ poly . Count - 1 ] ;
2020-05-21 12:20:02 +00:00
double side1 = p . Distance ( v1 ) * sideswitch ;
2012-11-27 21:12:20 +00:00
for ( int i = 0 ; i < poly . Count ; i + + )
{
// Fetch vertex and determine side
Vector3D v2 = poly [ i ] ;
2020-05-21 12:20:02 +00:00
double side2 = p . Distance ( v2 ) * sideswitch ;
2012-11-27 21:12:20 +00:00
// Front?
if ( side2 > NEAR_ZERO )
{
if ( side1 < - NEAR_ZERO )
{
// Split line with plane and insert the vertex
2020-05-21 12:20:02 +00:00
double u = 0.0f ;
2012-11-27 21:12:20 +00:00
p . GetIntersection ( v1 , v2 , ref u ) ;
Vector3D v3 = v1 + ( v2 - v1 ) * u ;
newp . Add ( v3 ) ;
}
newp . Add ( v2 ) ;
}
// Back?
else if ( side2 < - NEAR_ZERO )
{
if ( side1 > NEAR_ZERO )
{
// Split line with plane and insert the vertex
2020-05-21 12:20:02 +00:00
double u = 0.0f ;
2012-11-27 21:12:20 +00:00
p . GetIntersection ( v1 , v2 , ref u ) ;
Vector3D v3 = v1 + ( v2 - v1 ) * u ;
newp . Add ( v3 ) ;
}
}
else
{
// On the plane
newp . Add ( v2 ) ;
}
// Next
v1 = v2 ;
side1 = side2 ;
}
}
poly = newp ;
}
2015-11-20 14:31:54 +00:00
//mxd. This clips given polys by extrafloors
protected void ClipExtraFloors ( List < WallPolygon > polygons , List < Effect3DFloor > extrafloors , bool clipalways )
{
foreach ( Effect3DFloor ef in extrafloors )
{
//mxd. Walls should be clipped by 3D floors
if ( ef . ClipSidedefs | | clipalways )
{
int num = polygons . Count ;
for ( int pi = 0 ; pi < num ; pi + + )
{
// Split by floor plane of 3D floor
WallPolygon p = polygons [ pi ] ;
WallPolygon np = SplitPoly ( ref p , ef . Ceiling . plane , true ) ;
if ( np . Count > 0 )
{
// Split part below floor by the ceiling plane of 3D floor
// and keep only the part below the ceiling (front)
SplitPoly ( ref np , ef . Floor . plane , true ) ;
if ( p . Count = = 0 )
{
polygons [ pi ] = np ;
}
else
{
polygons [ pi ] = p ;
polygons . Add ( np ) ;
}
}
else
{
polygons [ pi ] = p ;
}
}
}
}
}
2015-02-06 09:01:33 +00:00
//mxd
protected void GetLightValue ( out int lightvalue , out bool lightabsolute )
{
2022-01-29 13:48:41 +00:00
string partstr = string . Empty , partstrabs = string . Empty ;
2022-01-29 17:59:24 +00:00
if ( General . Map . Config . DistinctSidedefPartBrightness )
2022-01-29 13:48:41 +00:00
{
switch ( geometrytype )
{
case VisualGeometryType . WALL_UPPER :
partstr = "light_top" ;
partstrabs = "lightabsolute_top" ;
break ;
case VisualGeometryType . WALL_LOWER :
partstr = "light_bottom" ;
partstrabs = "lightabsolute_bottom" ;
break ;
case VisualGeometryType . WALL_MIDDLE :
case VisualGeometryType . WALL_MIDDLE_3D :
partstr = "light_mid" ;
partstrabs = "lightabsolute_mid" ;
break ;
}
}
bool lightglobalabsolute = Sidedef . Fields . GetValue ( "lightabsolute" , false ) ;
2022-01-29 17:59:24 +00:00
bool lightpartabsolute = General . Map . Config . DistinctSidedefPartBrightness ? Sidedef . Fields . GetValue ( partstrabs , false ) : false ;
2022-01-29 13:48:41 +00:00
lightabsolute = lightglobalabsolute | | lightpartabsolute ;
2021-08-14 09:33:52 +00:00
bool affectedbyfog = General . Map . Data . MapInfo . HasFadeColor | | ( Sector . Sector . HasSkyCeiling & & General . Map . Data . MapInfo . HasOutsideFogColor ) | | Sector . Sector . Fields . ContainsKey ( "fadecolor" ) ;
2015-02-06 09:01:33 +00:00
bool ignorelight = affectedbyfog & & ! Sidedef . IsFlagSet ( "lightfog" ) & & ! lightabsolute ;
lightvalue = ignorelight ? 0 : Sidedef . Fields . GetValue ( "light" , 0 ) ; //mxd
2022-01-29 13:48:41 +00:00
if ( ignorelight )
{
lightvalue = 0 ;
}
else
{
// Absolute value of upper/middle/lower always has precedence
if ( lightpartabsolute )
lightvalue = Sidedef . Fields . GetValue ( partstr , 0 ) ;
else
2022-01-29 17:59:24 +00:00
lightvalue = Sidedef . Fields . GetValue ( "light" , 0 ) + ( General . Map . Config . DistinctSidedefPartBrightness ? Sidedef . Fields . GetValue ( partstr , 0 ) : 0 ) ;
2022-01-29 13:48:41 +00:00
}
2015-02-06 09:01:33 +00:00
if ( ignorelight ) lightabsolute = false ;
}
2019-04-14 16:24:37 +00:00
// biwa
2020-05-22 19:39:18 +00:00
protected static double GetNewTexutreOffset ( double oldValue , double offset , double textureSize )
2019-04-14 16:24:37 +00:00
{
2022-02-02 18:37:43 +00:00
return GetRoundedTextureOffset ( oldValue , offset , 1.0 , textureSize ) ;
2019-04-14 16:24:37 +00:00
}
2013-09-11 09:47:53 +00:00
//mxd
2020-05-22 19:39:18 +00:00
protected static double GetRoundedTextureOffset ( double oldValue , double offset , double scale , double textureSize )
2014-12-03 23:15:26 +00:00
{
2022-02-02 18:37:43 +00:00
if ( offset = = 0 | | offset % textureSize = = 0 ) return oldValue ;
2020-05-22 19:39:18 +00:00
double scaledOffset = offset * Math . Abs ( scale ) ;
double result = Math . Round ( oldValue + scaledOffset ) ;
2015-12-28 15:01:53 +00:00
if ( textureSize > 0 ) result % = textureSize ;
2022-02-02 18:37:43 +00:00
if ( result = = oldValue ) result + = ( scaledOffset < 0 ? - 1 : 1 ) ; // biwa. Why?
2013-09-11 09:47:53 +00:00
return result ;
}
2013-03-18 13:52:27 +00:00
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
//mxd
2017-01-26 13:33:07 +00:00
public void SelectNeighbours ( bool select , bool matchtexture , bool matchheight )
{
SelectNeighbours ( select , matchtexture , matchheight , true , true ) ;
}
private void SelectNeighbours ( bool select , bool matchtexture , bool matchheight , bool clearlinedefs , bool forward )
2014-09-18 22:06:35 +00:00
{
2016-01-15 18:20:47 +00:00
if ( Sidedef . Sector = = null | | Triangles < 1 | | ( ! matchtexture & & ! matchheight ) ) return ;
2013-04-01 11:06:01 +00:00
2014-12-22 21:36:49 +00:00
Rectangle rect = BuilderModesTools . GetSidedefPartSize ( this ) ;
2014-09-18 22:06:35 +00:00
if ( rect . Height = = 0 ) return ;
if ( select & & ! selected )
{
2013-04-01 11:06:01 +00:00
selected = true ;
mode . AddSelectedObject ( this ) ;
2014-09-18 22:06:35 +00:00
}
else if ( ! select & & selected )
{
2013-04-01 11:06:01 +00:00
selected = false ;
mode . RemoveSelectedObject ( this ) ;
}
2017-01-26 13:33:07 +00:00
// [ZZ] use the marking system.
if ( clearlinedefs ) General . Map . Map . ClearMarkedLinedefs ( false ) ;
Sidedef . Line . Marked = true ;
// Select
Vertex v ;
if ( forward )
{
v = Sidedef . IsFront ? Sidedef . Line . End : Sidedef . Line . Start ;
SelectNeighbourLines ( v . Linedefs , v , rect , select , matchtexture , matchheight , true ) ;
v = Sidedef . IsFront ? Sidedef . Line . Start : Sidedef . Line . End ;
SelectNeighbourLines ( v . Linedefs , v , rect , select , matchtexture , matchheight , false ) ;
}
else
{
v = Sidedef . IsFront ? Sidedef . Line . Start : Sidedef . Line . End ;
SelectNeighbourLines ( v . Linedefs , v , rect , select , matchtexture , matchheight , false ) ;
v = Sidedef . IsFront ? Sidedef . Line . End : Sidedef . Line . Start ;
SelectNeighbourLines ( v . Linedefs , v , rect , select , matchtexture , matchheight , true ) ;
}
2016-01-15 18:20:47 +00:00
}
2013-04-01 11:06:01 +00:00
2016-01-15 18:20:47 +00:00
//mxd
2017-01-26 13:33:07 +00:00
private void SelectNeighbourLines ( IEnumerable < Linedef > lines , Vertex v , Rectangle sourcerect , bool select , bool matchtexture , bool matchheight , bool forward )
2016-01-15 18:20:47 +00:00
{
foreach ( Linedef line in lines )
2014-09-18 22:06:35 +00:00
{
2017-01-26 13:33:07 +00:00
// [ZZ] decide which side of the next linedef to iterate.
// do NOT do both
Sidedef next = null ;
Sidedef side1 = forward ? line . Front : line . Back ;
Sidedef side2 = forward ? line . Back : line . Front ;
if ( line . Start = = v )
next = side1 ;
else if ( line . End = = v )
next = side2 ;
2013-10-21 10:19:31 +00:00
2017-01-26 13:33:07 +00:00
if ( next = = null | | next . Sector = = null )
continue ;
2013-10-21 10:19:31 +00:00
2017-01-26 13:33:07 +00:00
SelectNeighbourSideParts ( next , sourcerect , select , matchtexture , matchheight , forward ) ;
2016-01-15 18:20:47 +00:00
}
}
2013-10-21 10:19:31 +00:00
2016-01-15 18:20:47 +00:00
//mxd
2017-01-26 13:33:07 +00:00
private void SelectNeighbourSideParts ( Sidedef side , Rectangle sourcerect , bool select , bool matchtexture , bool matchheight , bool forward )
2016-01-15 18:20:47 +00:00
{
2017-01-26 13:33:07 +00:00
if ( side . Line . Marked )
return ;
2016-04-19 20:40:42 +00:00
BaseVisualSector s = ( BaseVisualSector ) mode . GetVisualSector ( side . Sector ) ;
2016-01-15 18:20:47 +00:00
if ( s ! = null )
{
VisualSidedefParts parts = s . GetSidedefParts ( side ) ;
2017-01-26 13:33:07 +00:00
SelectNeighbourSidePart ( parts . lower , sourcerect , select , matchtexture , matchheight , forward ) ;
SelectNeighbourSidePart ( parts . middlesingle , sourcerect , select , matchtexture , matchheight , forward ) ;
SelectNeighbourSidePart ( parts . middledouble , sourcerect , select , matchtexture , matchheight , forward ) ;
SelectNeighbourSidePart ( parts . upper , sourcerect , select , matchtexture , matchheight , forward ) ;
2014-09-18 22:06:35 +00:00
2016-01-15 18:20:47 +00:00
if ( parts . middle3d ! = null )
2014-09-18 22:06:35 +00:00
{
2016-01-15 18:20:47 +00:00
foreach ( VisualMiddle3D middle3D in parts . middle3d )
2017-01-26 13:33:07 +00:00
SelectNeighbourSidePart ( middle3D , sourcerect , select , matchtexture , matchheight , forward ) ;
2013-04-01 11:06:01 +00:00
}
2016-01-15 18:20:47 +00:00
}
}
2013-04-01 11:06:01 +00:00
2016-01-15 18:20:47 +00:00
//mxd
2017-01-26 13:33:07 +00:00
private void SelectNeighbourSidePart ( BaseVisualGeometrySidedef visualside , Rectangle sourcerect , bool select , bool matchtexture , bool matchheight , bool forward )
2016-01-15 18:20:47 +00:00
{
if ( visualside ! = null & & visualside . Triangles > 0 & & visualside . Selected ! = select )
{
Rectangle r = BuilderModesTools . GetSidedefPartSize ( visualside ) ;
if ( r . Width = = 0 | | r . Height = = 0 ) return ;
2016-05-12 11:32:10 +00:00
if ( ( ! matchtexture | | ( visualside . Texture = = Texture & & r . IntersectsWith ( sourcerect ) ) ) & &
( ! matchheight | | ( sourcerect . Height = = r . Height & & sourcerect . Y = = r . Y ) ) )
2014-09-18 22:06:35 +00:00
{
2017-01-26 13:33:07 +00:00
visualside . SelectNeighbours ( select , matchtexture , matchheight , false , forward ) ;
2014-09-18 22:06:35 +00:00
}
2013-04-01 11:06:01 +00:00
}
}
2014-12-22 21:36:49 +00:00
//mxd
protected void FitTexture ( FitTextureOptions options )
{
// Create undo name
string s ;
if ( options . FitWidth & & options . FitHeight ) s = "width and height" ;
else if ( options . FitWidth ) s = "width" ;
else s = "height" ;
2015-03-10 18:49:29 +00:00
// Create undo
2014-12-22 21:36:49 +00:00
mode . CreateUndo ( "Fit texture (" + s + ")" , UndoGroup . TextureOffsetChange , Sector . Sector . FixedIndex ) ;
Sidedef . Fields . BeforeFieldsChange ( ) ;
2015-03-10 18:49:29 +00:00
// Get proper control side...
Linedef controlline = GetControlLinedef ( ) ;
Sidedef controlside ;
if ( controlline ! = Sidedef . Line )
{
controlside = controlline . Front ;
controlside . Fields . BeforeFieldsChange ( ) ;
}
else
{
controlside = Sidedef ;
}
2014-12-22 21:36:49 +00:00
// Fit width
if ( options . FitWidth )
{
2020-05-21 12:20:02 +00:00
double scalex , offsetx ;
double linelength = Math . Round ( Sidedef . Line . Length ) ; // Let's use ZDoom-compatible line length here
double patternwidth = ( options . AutoWidth & & options . PatternWidth > 0 ) ? options . PatternWidth : Texture . Width ;
double horizontalrepeat = options . HorizontalRepeat ;
2014-12-22 21:36:49 +00:00
2017-07-18 11:56:27 +00:00
if ( options . FitAcrossSurfaces )
2014-12-22 21:36:49 +00:00
{
2017-07-18 11:56:27 +00:00
if ( options . AutoWidth )
{
2020-05-21 12:20:02 +00:00
horizontalrepeat = Math . Round ( ( double ) options . GlobalBounds . Width / patternwidth ) ;
2017-07-18 11:56:27 +00:00
if ( horizontalrepeat = = 0 )
horizontalrepeat = 1.0f ;
if ( options . PatternWidth > 0 )
horizontalrepeat / = Texture . Width / patternwidth ;
}
scalex = Texture . ScaledWidth / ( linelength * ( options . GlobalBounds . Width / linelength ) ) * horizontalrepeat ;
2020-05-21 12:20:02 +00:00
offsetx = Math . Round ( ( options . Bounds . X * scalex - Sidedef . OffsetX - options . ControlSideOffsetX ) , General . Map . FormatInterface . VertexDecimals ) ;
2016-09-02 19:18:37 +00:00
if ( Texture . IsImageLoaded ) offsetx % = Texture . Width ;
2014-12-22 21:36:49 +00:00
}
else
{
2017-07-18 11:56:27 +00:00
if ( options . AutoWidth )
{
2020-05-22 20:30:32 +00:00
horizontalrepeat = Math . Round ( linelength / patternwidth ) ;
2017-07-18 11:56:27 +00:00
if ( horizontalrepeat = = 0 )
horizontalrepeat = 1.0f ;
if ( options . PatternWidth > 0 )
horizontalrepeat / = Texture . Width / patternwidth ;
}
scalex = Texture . ScaledWidth / linelength * horizontalrepeat ;
2015-03-10 18:49:29 +00:00
offsetx = - Sidedef . OffsetX - options . ControlSideOffsetX ;
2014-12-22 21:36:49 +00:00
}
2020-05-22 20:30:32 +00:00
UniFields . SetFloat ( controlside . Fields , "scalex_" + partname , Math . Round ( scalex , General . Map . FormatInterface . VertexDecimals ) , 1.0 ) ;
UniFields . SetFloat ( Sidedef . Fields , "offsetx_" + partname , offsetx , 0.0 ) ;
2014-12-22 21:36:49 +00:00
}
else
{
// Restore initial offsets
2020-05-22 20:30:32 +00:00
UniFields . SetFloat ( controlside . Fields , "scalex_" + partname , options . InitialScaleX , 1.0 ) ;
UniFields . SetFloat ( Sidedef . Fields , "offsetx_" + partname , options . InitialOffsetX , 0.0 ) ;
2014-12-22 21:36:49 +00:00
}
// Fit height
if ( options . FitHeight )
{
if ( Sidedef . Sector ! = null )
{
2020-05-21 12:20:02 +00:00
double scaley , offsety ;
double patternheight = ( options . AutoHeight & & options . PatternHeight > 0 ) ? options . PatternHeight : Texture . Height ;
double verticalrepeat = options . VerticalRepeat ;
2014-12-22 21:36:49 +00:00
2017-07-18 11:56:27 +00:00
if ( options . FitAcrossSurfaces )
2014-12-22 21:36:49 +00:00
{
2017-07-18 11:56:27 +00:00
if ( options . AutoHeight )
{
2020-05-22 20:30:32 +00:00
verticalrepeat = Math . Round ( ( float ) options . GlobalBounds . Height / patternheight ) ;
2017-07-18 11:56:27 +00:00
if ( verticalrepeat = = 0 )
verticalrepeat = 1.0f ;
if ( options . PatternHeight > 0 )
verticalrepeat / = Texture . Height / patternheight ;
}
2020-05-22 20:30:32 +00:00
scaley = Texture . ScaledHeight / ( options . Bounds . Height * ( ( double ) options . GlobalBounds . Height / options . Bounds . Height ) ) * verticalrepeat ;
2014-12-22 21:36:49 +00:00
if ( this is VisualLower ) // Special cases, special cases...
{
2015-01-29 21:41:16 +00:00
offsety = GetLowerOffsetY ( scaley ) ;
2014-12-22 21:36:49 +00:00
}
else if ( this is VisualMiddleDouble )
{
2022-01-30 07:39:16 +00:00
if ( Sidedef . Line . IsFlagSet ( General . Map . Config . PegMidtextureFlag ) )
2014-12-23 12:32:08 +00:00
offsety = ( options . Bounds . Y - Sidedef . GetHighHeight ( ) - Sidedef . GetLowHeight ( ) ) * scaley - Sidedef . OffsetY ;
2014-12-22 21:36:49 +00:00
else
offsety = options . Bounds . Y * scaley - Sidedef . OffsetY ;
}
else
{
2016-09-02 19:18:37 +00:00
offsety = Tools . GetSidedefOffsetY ( Sidedef , geometrytype , options . Bounds . Y * scaley - Sidedef . OffsetY - options . ControlSideOffsetY , scaley , true ) ;
if ( Texture . IsImageLoaded ) offsety % = Texture . Height ;
2014-12-22 21:36:49 +00:00
}
}
else
{
2017-07-18 11:56:27 +00:00
if ( options . AutoHeight )
{
2020-05-22 20:30:32 +00:00
verticalrepeat = Math . Round ( ( double ) options . Bounds . Height / patternheight ) ;
2017-07-18 11:56:27 +00:00
if ( verticalrepeat = = 0 )
2020-05-22 20:30:32 +00:00
verticalrepeat = 1.0 ;
2017-07-18 11:56:27 +00:00
if ( options . PatternHeight > 0 )
verticalrepeat / = Texture . Height / patternheight ;
}
scaley = Texture . ScaledHeight / options . Bounds . Height * verticalrepeat ;
2015-01-29 21:41:16 +00:00
2016-09-02 19:18:37 +00:00
// Special cases, special cases...
if ( this is VisualLower )
{
2015-01-29 21:41:16 +00:00
offsety = GetLowerOffsetY ( scaley ) ;
2016-09-02 19:18:37 +00:00
}
2015-01-29 21:41:16 +00:00
else
2016-09-02 19:18:37 +00:00
{
offsety = Tools . GetSidedefOffsetY ( Sidedef , geometrytype , - Sidedef . OffsetY - options . ControlSideOffsetY , scaley , true ) ;
if ( Texture . IsImageLoaded ) offsety % = Texture . Height ;
}
2014-12-22 21:36:49 +00:00
}
2020-05-22 20:30:32 +00:00
UniFields . SetFloat ( controlside . Fields , "scaley_" + partname , ( float ) Math . Round ( scaley , General . Map . FormatInterface . VertexDecimals ) , 1.0 ) ;
UniFields . SetFloat ( Sidedef . Fields , "offsety_" + partname , ( float ) Math . Round ( offsety , General . Map . FormatInterface . VertexDecimals ) , 0.0 ) ;
2014-12-22 21:36:49 +00:00
}
}
else
{
// Restore initial offsets
2020-05-22 20:30:32 +00:00
UniFields . SetFloat ( controlside . Fields , "scaley_" + partname , options . InitialScaleY , 1.0 ) ;
UniFields . SetFloat ( Sidedef . Fields , "offsety_" + partname , options . InitialOffsetY , 0.0 ) ;
2014-12-22 21:36:49 +00:00
}
}
2015-01-29 21:41:16 +00:00
//mxd. Oh so special cases...
2020-05-21 12:20:02 +00:00
private double GetLowerOffsetY ( double scaley )
2015-01-29 21:41:16 +00:00
{
2020-05-21 12:20:02 +00:00
double offsety ;
2015-01-29 21:41:16 +00:00
if ( Sidedef . Line . IsFlagSet ( General . Map . Config . LowerUnpeggedFlag ) )
2016-09-02 19:18:37 +00:00
offsety = ( - Sidedef . OffsetY - Sidedef . GetMiddleHeight ( ) - Sidedef . GetHighHeight ( ) ) * scaley ;
else
offsety = - Sidedef . OffsetY * scaley ;
if ( Texture . IsImageLoaded ) offsety % = Texture . Height ;
return offsety ;
2015-01-29 21:41:16 +00:00
}
2012-11-27 21:12:20 +00:00
2009-04-19 18:07:22 +00:00
#endregion
#region = = = = = = = = = = = = = = = = = = Events
// Unused
public virtual void OnEditBegin ( ) { }
protected virtual void SetTexture ( string texturename ) { }
public abstract bool Setup ( ) ;
2012-11-27 21:12:20 +00:00
protected abstract void SetTextureOffsetX ( int x ) ;
protected abstract void SetTextureOffsetY ( int y ) ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
protected virtual void ResetTextureScale ( ) { } //mxd
2016-09-06 12:05:47 +00:00
protected abstract void MoveTextureOffset ( int offsetx , int offsety ) ;
2012-11-27 21:12:20 +00:00
protected abstract Point GetTextureOffset ( ) ;
2014-12-22 21:36:49 +00:00
public virtual void OnTextureFit ( FitTextureOptions options ) { } //mxd
2019-01-19 07:56:13 +00:00
public virtual void OnPaintSelectEnd ( ) { } // biwa
2009-04-19 18:07:22 +00:00
// Insert middle texture
public virtual void OnInsert ( )
{
// No middle texture yet?
2013-07-31 12:38:47 +00:00
if ( ! Sidedef . MiddleRequired ( ) & & ( string . IsNullOrEmpty ( Sidedef . MiddleTexture ) | | ( Sidedef . MiddleTexture = = "-" ) ) )
2009-04-19 18:07:22 +00:00
{
// Make it now
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Create middle texture" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Created middle texture." ) ;
2009-04-19 18:07:22 +00:00
General . Settings . FindDefaultDrawSettings ( ) ;
2013-11-21 10:53:11 +00:00
Sidedef . SetTextureMid ( General . Map . Options . DefaultWallTexture ) ;
2009-04-19 18:07:22 +00:00
// Update
2009-05-01 20:31:17 +00:00
Sector . Changed = true ;
2009-04-19 18:07:22 +00:00
// Other side as well
2013-07-31 12:38:47 +00:00
if ( string . IsNullOrEmpty ( Sidedef . Other . MiddleTexture ) | | ( Sidedef . Other . MiddleTexture = = "-" ) )
2009-04-19 18:07:22 +00:00
{
2013-11-21 10:53:11 +00:00
Sidedef . Other . SetTextureMid ( General . Map . Options . DefaultWallTexture ) ;
2009-04-19 18:07:22 +00:00
// Update
VisualSector othersector = mode . GetVisualSector ( Sidedef . Other . Sector ) ;
2016-04-19 20:40:42 +00:00
if ( othersector is BaseVisualSector ) ( ( BaseVisualSector ) othersector ) . Changed = true ;
2009-04-19 18:07:22 +00:00
}
}
}
// Delete texture
public virtual void OnDelete ( )
{
// Remove texture
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Delete texture" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Deleted a texture." ) ;
2009-04-19 18:07:22 +00:00
SetTexture ( "-" ) ;
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2009-04-19 18:07:22 +00:00
}
// Processing
2016-03-14 10:25:27 +00:00
public virtual void OnProcess ( long deltatime )
2009-04-19 18:07:22 +00:00
{
2019-12-31 02:44:36 +00:00
// If the texture was not loaded, but is loaded now, then re-setup geometry
if ( setuponloadedtexture ! = lastsetuponloadedtexture )
{
if ( setuponloadedtexture ! = 0 )
{
ImageData t = General . Map . Data . GetTextureImage ( setuponloadedtexture ) ;
if ( t ! = null & & t . IsImageLoaded )
{
lastsetuponloadedtexture = setuponloadedtexture ;
Setup ( ) ;
}
}
else
{
lastsetuponloadedtexture = setuponloadedtexture ;
}
}
}
2009-04-19 18:07:22 +00:00
// Change target height
public virtual void OnChangeTargetHeight ( int amount )
{
switch ( BuilderPlug . Me . ChangeHeightBySidedef )
{
// Change ceiling
case 1 :
2015-11-20 14:31:54 +00:00
if ( ! this . Sector . Ceiling . Changed ) this . Sector . Ceiling . OnChangeTargetHeight ( amount ) ;
2009-04-19 18:07:22 +00:00
break ;
// Change floor
case 2 :
2015-11-20 14:31:54 +00:00
if ( ! this . Sector . Floor . Changed ) this . Sector . Floor . OnChangeTargetHeight ( amount ) ;
2009-04-19 18:07:22 +00:00
break ;
2009-05-05 11:26:50 +00:00
// Change both
case 3 :
2015-11-20 14:31:54 +00:00
if ( ! this . Sector . Floor . Changed ) this . Sector . Floor . OnChangeTargetHeight ( amount ) ;
if ( ! this . Sector . Ceiling . Changed ) this . Sector . Ceiling . OnChangeTargetHeight ( amount ) ;
2009-05-05 11:26:50 +00:00
break ;
2009-04-19 18:07:22 +00:00
}
}
// Reset texture offsets
public virtual void OnResetTextureOffset ( )
{
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Reset texture offsets" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Texture offsets reset." ) ;
2009-04-19 18:07:22 +00:00
// Apply offsets
2013-09-19 09:17:49 +00:00
Sidedef . OffsetX = 0 ;
Sidedef . OffsetY = 0 ;
2012-11-27 21:12:20 +00:00
2009-04-19 18:07:22 +00:00
// Update sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-19 18:07:22 +00:00
}
2013-09-19 09:17:49 +00:00
//mxd
2014-12-03 23:15:26 +00:00
public virtual void OnResetLocalTextureOffset ( )
{
2015-12-28 15:01:53 +00:00
if ( ! General . Map . UDMF )
2014-12-03 23:15:26 +00:00
{
2013-09-19 09:17:49 +00:00
OnResetTextureOffset ( ) ;
return ;
}
Added, Visual mode, UDMF: added "Change Pitch Clockwise", "Change Pitch Counterclockwise", "Change Roll Clockwise" and "Change Roll Counterclockwise" actions.
Added, Visual mode, UDMF: added "Increase Scale" and "Decrease Scale" actions.
Added, Visual mode, UDMF: "Reset Texture Offsets" action now resets scale when used on things.
Added, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets scale, pitch and roll when used on things.
Changed, Visual mode, UDMF: "Reset Texture Offsets" action now only resets texture offsets (previously it also reset texture scale).
Changed, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets texture offsets, scale and brightness of sidedefs and also rotation of floors/ceilings.
Changed, Visual mode, UDMF: thing box arrow now displays pitch and roll for things, which have attached model and appropriate MODELDEF flags.
Changed, Thing Edit Form, UDMF: negative pitch and roll can now be entered.
Updated documentation.
2015-03-30 21:44:04 +00:00
mode . CreateUndo ( "Reset local texture offsets, scale and brightness" ) ;
mode . SetActionResult ( "Local texture offsets, scale and brightness reset." ) ;
2013-09-19 09:17:49 +00:00
Added, Visual mode, UDMF: added "Change Pitch Clockwise", "Change Pitch Counterclockwise", "Change Roll Clockwise" and "Change Roll Counterclockwise" actions.
Added, Visual mode, UDMF: added "Increase Scale" and "Decrease Scale" actions.
Added, Visual mode, UDMF: "Reset Texture Offsets" action now resets scale when used on things.
Added, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets scale, pitch and roll when used on things.
Changed, Visual mode, UDMF: "Reset Texture Offsets" action now only resets texture offsets (previously it also reset texture scale).
Changed, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets texture offsets, scale and brightness of sidedefs and also rotation of floors/ceilings.
Changed, Visual mode, UDMF: thing box arrow now displays pitch and roll for things, which have attached model and appropriate MODELDEF flags.
Changed, Thing Edit Form, UDMF: negative pitch and roll can now be entered.
Updated documentation.
2015-03-30 21:44:04 +00:00
// Reset texture offsets
2013-09-19 09:17:49 +00:00
SetTextureOffsetX ( 0 ) ;
SetTextureOffsetY ( 0 ) ;
Added, Visual mode, UDMF: added "Change Pitch Clockwise", "Change Pitch Counterclockwise", "Change Roll Clockwise" and "Change Roll Counterclockwise" actions.
Added, Visual mode, UDMF: added "Increase Scale" and "Decrease Scale" actions.
Added, Visual mode, UDMF: "Reset Texture Offsets" action now resets scale when used on things.
Added, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets scale, pitch and roll when used on things.
Changed, Visual mode, UDMF: "Reset Texture Offsets" action now only resets texture offsets (previously it also reset texture scale).
Changed, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets texture offsets, scale and brightness of sidedefs and also rotation of floors/ceilings.
Changed, Visual mode, UDMF: thing box arrow now displays pitch and roll for things, which have attached model and appropriate MODELDEF flags.
Changed, Thing Edit Form, UDMF: negative pitch and roll can now be entered.
Updated documentation.
2015-03-30 21:44:04 +00:00
// Scale
2013-09-19 09:17:49 +00:00
ResetTextureScale ( ) ;
Added, Visual mode, UDMF: added "Change Pitch Clockwise", "Change Pitch Counterclockwise", "Change Roll Clockwise" and "Change Roll Counterclockwise" actions.
Added, Visual mode, UDMF: added "Increase Scale" and "Decrease Scale" actions.
Added, Visual mode, UDMF: "Reset Texture Offsets" action now resets scale when used on things.
Added, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets scale, pitch and roll when used on things.
Changed, Visual mode, UDMF: "Reset Texture Offsets" action now only resets texture offsets (previously it also reset texture scale).
Changed, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets texture offsets, scale and brightness of sidedefs and also rotation of floors/ceilings.
Changed, Visual mode, UDMF: thing box arrow now displays pitch and roll for things, which have attached model and appropriate MODELDEF flags.
Changed, Thing Edit Form, UDMF: negative pitch and roll can now be entered.
Updated documentation.
2015-03-30 21:44:04 +00:00
// And brightness
2016-04-19 20:40:42 +00:00
bool setupallparts = false ;
if ( Sidedef . Fields . ContainsKey ( "light" ) )
{
Sidedef . Fields . Remove ( "light" ) ;
setupallparts = true ;
}
if ( Sidedef . Fields . ContainsKey ( "lightabsolute" ) )
{
Sidedef . Fields . Remove ( "lightabsolute" ) ;
setupallparts = true ;
}
Added, Visual mode, UDMF: added "Change Pitch Clockwise", "Change Pitch Counterclockwise", "Change Roll Clockwise" and "Change Roll Counterclockwise" actions.
Added, Visual mode, UDMF: added "Increase Scale" and "Decrease Scale" actions.
Added, Visual mode, UDMF: "Reset Texture Offsets" action now resets scale when used on things.
Added, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets scale, pitch and roll when used on things.
Changed, Visual mode, UDMF: "Reset Texture Offsets" action now only resets texture offsets (previously it also reset texture scale).
Changed, Visual mode, UDMF: "Reset Local Texture Offsets" action now resets texture offsets, scale and brightness of sidedefs and also rotation of floors/ceilings.
Changed, Visual mode, UDMF: thing box arrow now displays pitch and roll for things, which have attached model and appropriate MODELDEF flags.
Changed, Thing Edit Form, UDMF: negative pitch and roll can now be entered.
Updated documentation.
2015-03-30 21:44:04 +00:00
2016-04-19 20:40:42 +00:00
if ( setupallparts )
{
// Update all sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
parts . SetupAllParts ( ) ;
}
else
{
// Update this part only
this . Setup ( ) ;
}
2013-09-19 09:17:49 +00:00
}
2009-04-19 18:07:22 +00:00
// Toggle upper-unpegged
public virtual void OnToggleUpperUnpegged ( )
{
2009-06-11 21:21:20 +00:00
if ( this . Sidedef . Line . IsFlagSet ( General . Map . Config . UpperUnpeggedFlag ) )
2009-04-19 18:07:22 +00:00
{
// Remove flag
2009-05-03 19:22:32 +00:00
mode . ApplyUpperUnpegged ( false ) ;
}
else
{
// Add flag
mode . ApplyUpperUnpegged ( true ) ;
}
}
// Toggle lower-unpegged
public virtual void OnToggleLowerUnpegged ( )
{
2009-06-11 21:21:20 +00:00
if ( this . Sidedef . Line . IsFlagSet ( General . Map . Config . LowerUnpeggedFlag ) )
2009-05-03 19:22:32 +00:00
{
// Remove flag
mode . ApplyLowerUnpegged ( false ) ;
}
else
{
// Add flag
mode . ApplyLowerUnpegged ( true ) ;
}
}
// This sets the Upper Unpegged flag
public virtual void ApplyUpperUnpegged ( bool set )
{
if ( ! set )
{
// Remove flag
mode . CreateUndo ( "Remove upper-unpegged setting" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Removed upper-unpegged setting." ) ;
2009-06-11 21:21:20 +00:00
this . Sidedef . Line . SetFlag ( General . Map . Config . UpperUnpeggedFlag , false ) ;
2009-04-19 18:07:22 +00:00
}
else
{
// Add flag
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Set upper-unpegged setting" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Set upper-unpegged setting." ) ;
2009-06-11 21:21:20 +00:00
this . Sidedef . Line . SetFlag ( General . Map . Config . UpperUnpeggedFlag , true ) ;
2009-04-19 18:07:22 +00:00
}
2009-05-03 19:22:32 +00:00
2009-04-19 18:07:22 +00:00
// Update sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-27 17:32:23 +00:00
// Update other sidedef geometry
if ( Sidedef . Other ! = null )
{
BaseVisualSector othersector = ( BaseVisualSector ) mode . GetVisualSector ( Sidedef . Other . Sector ) ;
parts = othersector . GetSidedefParts ( Sidedef . Other ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-27 17:32:23 +00:00
}
2009-04-19 18:07:22 +00:00
}
2009-05-03 19:22:32 +00:00
// This sets the Lower Unpegged flag
public virtual void ApplyLowerUnpegged ( bool set )
2009-04-19 18:07:22 +00:00
{
2009-05-03 19:22:32 +00:00
if ( ! set )
2009-04-19 18:07:22 +00:00
{
// Remove flag
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Remove lower-unpegged setting" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Removed lower-unpegged setting." ) ;
2009-06-11 21:21:20 +00:00
this . Sidedef . Line . SetFlag ( General . Map . Config . LowerUnpeggedFlag , false ) ;
2009-04-19 18:07:22 +00:00
}
else
{
// Add flag
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Set lower-unpegged setting" ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Set lower-unpegged setting." ) ;
2009-06-11 21:21:20 +00:00
this . Sidedef . Line . SetFlag ( General . Map . Config . LowerUnpeggedFlag , true ) ;
2009-04-19 18:07:22 +00:00
}
2009-05-03 19:22:32 +00:00
2009-04-19 18:07:22 +00:00
// Update sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-27 17:32:23 +00:00
// Update other sidedef geometry
if ( Sidedef . Other ! = null )
{
BaseVisualSector othersector = ( BaseVisualSector ) mode . GetVisualSector ( Sidedef . Other . Sector ) ;
parts = othersector . GetSidedefParts ( Sidedef . Other ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-27 17:32:23 +00:00
}
2009-04-19 18:07:22 +00:00
}
2009-05-03 19:22:32 +00:00
2009-04-19 18:07:22 +00:00
// Flood-fill textures
public virtual void OnTextureFloodfill ( )
{
if ( BuilderPlug . Me . CopiedTexture ! = null )
{
string oldtexture = GetTextureName ( ) ;
string newtexture = BuilderPlug . Me . CopiedTexture ;
if ( newtexture ! = oldtexture )
{
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Flood-fill textures with " + newtexture ) ;
2009-05-02 14:59:05 +00:00
mode . SetActionResult ( "Flood-filled textures with " + newtexture + "." ) ;
2009-04-19 18:07:22 +00:00
mode . Renderer . SetCrosshairBusy ( true ) ;
General . Interface . RedrawDisplay ( ) ;
// Get the texture
ImageData newtextureimage = General . Map . Data . GetTextureImage ( newtexture ) ;
if ( newtextureimage ! = null )
{
2009-06-12 09:44:38 +00:00
if ( mode . IsSingleSelection )
{
// Clear all marks, this will align everything it can
General . Map . Map . ClearMarkedSidedefs ( false ) ;
}
else
{
// Limit the alignment to selection only
General . Map . Map . ClearMarkedSidedefs ( true ) ;
List < Sidedef > sides = mode . GetSelectedSidedefs ( ) ;
2017-01-29 02:56:00 +00:00
foreach ( Sidedef sd in sides )
{
sd . Marked = false ;
if ( sd . Other ! = null )
sd . Other . Marked = false ;
}
2009-06-12 09:44:38 +00:00
}
2016-04-07 20:01:59 +00:00
//mxd. We potentially need to deal with 2 textures (because of long and short texture names)...
HashSet < long > texturehashes = new HashSet < long > { Texture . LongName } ;
switch ( this . GeometryType )
{
case VisualGeometryType . WALL_LOWER :
texturehashes . Add ( Sidedef . LongLowTexture ) ;
break ;
case VisualGeometryType . WALL_MIDDLE :
case VisualGeometryType . WALL_MIDDLE_3D :
texturehashes . Add ( Sidedef . LongMiddleTexture ) ;
break ;
case VisualGeometryType . WALL_UPPER :
texturehashes . Add ( Sidedef . LongHighTexture ) ;
break ;
}
2009-06-12 09:44:38 +00:00
2009-04-19 18:07:22 +00:00
// Do the alignment
2016-05-12 11:32:10 +00:00
BuilderModesTools . FloodfillTextures ( mode , Sidedef , texturehashes , newtexture , false ) ;
2009-04-19 18:07:22 +00:00
// Get the changed sidedefs
List < Sidedef > changes = General . Map . Map . GetMarkedSidedefs ( true ) ;
foreach ( Sidedef sd in changes )
{
// Update the parts for this sidedef!
if ( mode . VisualSectorExists ( sd . Sector ) )
{
2016-04-19 20:40:42 +00:00
BaseVisualSector vs = ( BaseVisualSector ) mode . GetVisualSector ( sd . Sector ) ;
2009-04-19 18:07:22 +00:00
VisualSidedefParts parts = vs . GetSidedefParts ( sd ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-19 18:07:22 +00:00
}
}
General . Map . Data . UpdateUsedTextures ( ) ;
mode . Renderer . SetCrosshairBusy ( false ) ;
mode . ShowTargetInfo ( ) ;
}
}
}
}
2013-04-26 12:32:51 +00:00
// Auto-align texture offsets
2009-04-19 18:07:22 +00:00
public virtual void OnTextureAlign ( bool alignx , bool aligny )
{
2013-04-26 12:32:51 +00:00
//mxd
2014-02-21 14:42:12 +00:00
string rest ;
2013-04-26 12:32:51 +00:00
if ( alignx & & aligny ) rest = "(X and Y)" ;
else if ( alignx ) rest = "(X)" ;
else rest = "(Y)" ;
mode . CreateUndo ( "Auto-align textures " + rest ) ;
mode . SetActionResult ( "Auto-aligned textures " + rest + "." ) ;
2009-04-19 18:07:22 +00:00
// Make sure the texture is loaded (we need the texture size)
2020-01-12 22:10:57 +00:00
if ( ! base . Texture . IsImageLoaded ) base . Texture . LoadImageNow ( ) ;
2009-04-19 18:07:22 +00:00
2009-06-12 09:44:38 +00:00
if ( mode . IsSingleSelection )
{
// Clear all marks, this will align everything it can
General . Map . Map . ClearMarkedSidedefs ( false ) ;
}
else
{
// Limit the alignment to selection only
General . Map . Map . ClearMarkedSidedefs ( true ) ;
List < Sidedef > sides = mode . GetSelectedSidedefs ( ) ;
2017-01-29 02:56:00 +00:00
foreach ( Sidedef sd in sides )
{
sd . Marked = false ;
if ( sd . Other ! = null )
sd . Other . Marked = false ;
}
2009-06-12 09:44:38 +00:00
}
2009-04-19 18:07:22 +00:00
// Do the alignment
2013-09-03 09:34:28 +00:00
mode . AutoAlignTextures ( this , base . Texture , alignx , aligny , false , true ) ;
2009-04-19 18:07:22 +00:00
// Get the changed sidedefs
List < Sidedef > changes = General . Map . Map . GetMarkedSidedefs ( true ) ;
foreach ( Sidedef sd in changes )
{
// Update the parts for this sidedef!
if ( mode . VisualSectorExists ( sd . Sector ) )
{
2015-11-20 14:31:54 +00:00
BaseVisualSector vs = ( BaseVisualSector ) mode . GetVisualSector ( sd . Sector ) ;
2009-04-19 18:07:22 +00:00
VisualSidedefParts parts = vs . GetSidedefParts ( sd ) ;
2009-05-03 19:22:32 +00:00
parts . SetupAllParts ( ) ;
2009-04-19 18:07:22 +00:00
}
}
}
// Select texture
public virtual void OnSelectTexture ( )
{
if ( General . Interface . IsActiveWindow )
{
string oldtexture = GetTextureName ( ) ;
string newtexture = General . Interface . BrowseTexture ( General . Interface , oldtexture ) ;
if ( newtexture ! = oldtexture )
{
2009-05-03 19:22:32 +00:00
mode . ApplySelectTexture ( newtexture , false ) ;
2009-04-19 18:07:22 +00:00
}
}
}
2009-05-03 19:22:32 +00:00
// Apply Texture
public virtual void ApplyTexture ( string texture )
{
mode . CreateUndo ( "Change texture " + texture ) ;
SetTexture ( texture ) ;
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2009-05-03 19:22:32 +00:00
}
2009-04-19 18:07:22 +00:00
// Paste texture
public virtual void OnPasteTexture ( )
{
if ( BuilderPlug . Me . CopiedTexture ! = null )
{
2016-03-08 20:41:06 +00:00
mode . CreateUndo ( "Paste texture \"" + BuilderPlug . Me . CopiedTexture + "\"" ) ;
mode . SetActionResult ( "Pasted texture \"" + BuilderPlug . Me . CopiedTexture + "\"." ) ;
2009-04-19 18:07:22 +00:00
SetTexture ( BuilderPlug . Me . CopiedTexture ) ;
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2009-04-19 18:07:22 +00:00
}
}
// Paste texture offsets
public virtual void OnPasteTextureOffsets ( )
{
2009-05-03 19:22:32 +00:00
mode . CreateUndo ( "Paste texture offsets" ) ;
2021-03-16 20:44:12 +00:00
if ( General . Map . UDMF & & General . Map . Config . UseLocalSidedefTextureOffsets )
2014-12-03 23:15:26 +00:00
{
2013-09-06 07:55:02 +00:00
SetTextureOffsetX ( BuilderPlug . Me . CopiedOffsets . X ) ;
SetTextureOffsetY ( BuilderPlug . Me . CopiedOffsets . Y ) ;
2016-04-19 20:40:42 +00:00
// Update sidedef part geometry
this . Setup ( ) ;
2014-12-03 23:15:26 +00:00
}
else
{
2013-09-06 07:55:02 +00:00
Sidedef . OffsetX = BuilderPlug . Me . CopiedOffsets . X ;
Sidedef . OffsetY = BuilderPlug . Me . CopiedOffsets . Y ;
2016-04-19 20:40:42 +00:00
// Update sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
parts . SetupAllParts ( ) ;
2013-09-06 07:55:02 +00:00
}
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2012-11-27 21:12:20 +00:00
mode . SetActionResult ( "Pasted texture offsets " + BuilderPlug . Me . CopiedOffsets . X + ", " + BuilderPlug . Me . CopiedOffsets . Y + "." ) ;
2009-04-19 18:07:22 +00:00
}
// Copy texture
public virtual void OnCopyTexture ( )
{
2016-03-23 14:52:33 +00:00
//mxd. When UseLongTextureNames is enabled and the image filename is longer than 8 chars, use full name,
// otherwise use texture name as stored in Sidedef
string texturename = ( ( General . Map . Options . UseLongTextureNames & & Texture ! = null & & Texture . UsedInMap
& & Path . GetFileNameWithoutExtension ( Texture . Name ) . Length > DataManager . CLASIC_IMAGE_NAME_LENGTH )
? Texture . Name : GetTextureName ( ) ) ;
2015-02-19 13:00:19 +00:00
BuilderPlug . Me . CopiedTexture = texturename ;
if ( General . Map . Config . MixTexturesFlats ) BuilderPlug . Me . CopiedFlat = texturename ;
2016-03-08 20:41:06 +00:00
mode . SetActionResult ( "Copied texture \"" + texturename + "\"." ) ;
2009-04-19 18:07:22 +00:00
}
// Copy texture offsets
public virtual void OnCopyTextureOffsets ( )
{
2013-09-06 07:55:02 +00:00
//mxd
2021-03-16 20:44:12 +00:00
BuilderPlug . Me . CopiedOffsets = ( General . Map . UDMF & & General . Map . Config . UseLocalSidedefTextureOffsets ) ? GetTextureOffset ( ) : new Point ( Sidedef . OffsetX , Sidedef . OffsetY ) ;
2012-11-27 21:12:20 +00:00
mode . SetActionResult ( "Copied texture offsets " + BuilderPlug . Me . CopiedOffsets . X + ", " + BuilderPlug . Me . CopiedOffsets . Y + "." ) ;
2009-04-19 18:07:22 +00:00
}
// Copy properties
public virtual void OnCopyProperties ( )
{
Removed "Paste Properties Options" action.
Added "Paste Properties Special" actions in "Classic" and "Visual" categories. They work the same way as "Paste Special" action.
Added: "Copy Properties", "Paste Properties" and "Paste Properties Special" options are now shown in the Edit menu if current classic mode supports them.
Changed, Paste Properties Special window: only options relevant to current map format are now displayed.
Changed, Paste Properties Special window, UDMF: all UI-managed options are now available.
Fixed: MAPINFO parser was unable to process "include" directives.
Fixed, General interface: selection info was reset to "Nothing selected" after few seconds regardless of current selection.
Fixed, Visual mode: thing bounding boxes were not updated when changing things positions using Randomize mode.
Fixed, Visual mode: event lines were displayed at incorrect height when entering Visual mode for the first time.
Fixed, Texture Browser window: when MixTexturesFlats Game Configuration option is disabled, textures/flats are no longer shown in the Used group when flats/textures with the same names are used in the map.
Fixed(?): probably fixed an exception some users reported when trying to initialize a Classic mode after switching from Visual mode with "Sync cameras" option enabled.
Changed, Game configurations, Thing Categories: a block must have at least one thing category property to be recognized as a thing category.
Changed, Visplane Explorer: the plugin now outputs more info when it fails to initialize vpo.dll.
Cosmetic, Thing Edit window, Doom/Hexen map format: adjusted UI layout so thing flags control no longer displays scrollbars in Hexen map format.
Internal: merged methods from UDMFTools into UniFields, removed UDMFTools.
Updated Inno Setup script (added VC++ 2008 SP1 distributive).
Updated ZDoom_DECORATE.cfg (A_CheckBlock).
Updated documentation (added "System Requirements" page).
2015-10-09 12:38:12 +00:00
BuilderPlug . Me . CopiedLinedefProps = new LinedefProperties ( Sidedef . Line ) ; //mxd
2009-04-19 18:07:22 +00:00
BuilderPlug . Me . CopiedSidedefProps = new SidedefProperties ( Sidedef ) ;
Removed "Paste Properties Options" action.
Added "Paste Properties Special" actions in "Classic" and "Visual" categories. They work the same way as "Paste Special" action.
Added: "Copy Properties", "Paste Properties" and "Paste Properties Special" options are now shown in the Edit menu if current classic mode supports them.
Changed, Paste Properties Special window: only options relevant to current map format are now displayed.
Changed, Paste Properties Special window, UDMF: all UI-managed options are now available.
Fixed: MAPINFO parser was unable to process "include" directives.
Fixed, General interface: selection info was reset to "Nothing selected" after few seconds regardless of current selection.
Fixed, Visual mode: thing bounding boxes were not updated when changing things positions using Randomize mode.
Fixed, Visual mode: event lines were displayed at incorrect height when entering Visual mode for the first time.
Fixed, Texture Browser window: when MixTexturesFlats Game Configuration option is disabled, textures/flats are no longer shown in the Used group when flats/textures with the same names are used in the map.
Fixed(?): probably fixed an exception some users reported when trying to initialize a Classic mode after switching from Visual mode with "Sync cameras" option enabled.
Changed, Game configurations, Thing Categories: a block must have at least one thing category property to be recognized as a thing category.
Changed, Visplane Explorer: the plugin now outputs more info when it fails to initialize vpo.dll.
Cosmetic, Thing Edit window, Doom/Hexen map format: adjusted UI layout so thing flags control no longer displays scrollbars in Hexen map format.
Internal: merged methods from UDMFTools into UniFields, removed UDMFTools.
Updated Inno Setup script (added VC++ 2008 SP1 distributive).
Updated ZDoom_DECORATE.cfg (A_CheckBlock).
Updated documentation (added "System Requirements" page).
2015-10-09 12:38:12 +00:00
mode . SetActionResult ( "Copied linedef and sidedef properties." ) ;
2009-04-19 18:07:22 +00:00
}
// Paste properties
Removed "Paste Properties Options" action.
Added "Paste Properties Special" actions in "Classic" and "Visual" categories. They work the same way as "Paste Special" action.
Added: "Copy Properties", "Paste Properties" and "Paste Properties Special" options are now shown in the Edit menu if current classic mode supports them.
Changed, Paste Properties Special window: only options relevant to current map format are now displayed.
Changed, Paste Properties Special window, UDMF: all UI-managed options are now available.
Fixed: MAPINFO parser was unable to process "include" directives.
Fixed, General interface: selection info was reset to "Nothing selected" after few seconds regardless of current selection.
Fixed, Visual mode: thing bounding boxes were not updated when changing things positions using Randomize mode.
Fixed, Visual mode: event lines were displayed at incorrect height when entering Visual mode for the first time.
Fixed, Texture Browser window: when MixTexturesFlats Game Configuration option is disabled, textures/flats are no longer shown in the Used group when flats/textures with the same names are used in the map.
Fixed(?): probably fixed an exception some users reported when trying to initialize a Classic mode after switching from Visual mode with "Sync cameras" option enabled.
Changed, Game configurations, Thing Categories: a block must have at least one thing category property to be recognized as a thing category.
Changed, Visplane Explorer: the plugin now outputs more info when it fails to initialize vpo.dll.
Cosmetic, Thing Edit window, Doom/Hexen map format: adjusted UI layout so thing flags control no longer displays scrollbars in Hexen map format.
Internal: merged methods from UDMFTools into UniFields, removed UDMFTools.
Updated Inno Setup script (added VC++ 2008 SP1 distributive).
Updated ZDoom_DECORATE.cfg (A_CheckBlock).
Updated documentation (added "System Requirements" page).
2015-10-09 12:38:12 +00:00
public virtual void OnPasteProperties ( bool usecopysettings )
2009-04-19 18:07:22 +00:00
{
Removed "Paste Properties Options" action.
Added "Paste Properties Special" actions in "Classic" and "Visual" categories. They work the same way as "Paste Special" action.
Added: "Copy Properties", "Paste Properties" and "Paste Properties Special" options are now shown in the Edit menu if current classic mode supports them.
Changed, Paste Properties Special window: only options relevant to current map format are now displayed.
Changed, Paste Properties Special window, UDMF: all UI-managed options are now available.
Fixed: MAPINFO parser was unable to process "include" directives.
Fixed, General interface: selection info was reset to "Nothing selected" after few seconds regardless of current selection.
Fixed, Visual mode: thing bounding boxes were not updated when changing things positions using Randomize mode.
Fixed, Visual mode: event lines were displayed at incorrect height when entering Visual mode for the first time.
Fixed, Texture Browser window: when MixTexturesFlats Game Configuration option is disabled, textures/flats are no longer shown in the Used group when flats/textures with the same names are used in the map.
Fixed(?): probably fixed an exception some users reported when trying to initialize a Classic mode after switching from Visual mode with "Sync cameras" option enabled.
Changed, Game configurations, Thing Categories: a block must have at least one thing category property to be recognized as a thing category.
Changed, Visplane Explorer: the plugin now outputs more info when it fails to initialize vpo.dll.
Cosmetic, Thing Edit window, Doom/Hexen map format: adjusted UI layout so thing flags control no longer displays scrollbars in Hexen map format.
Internal: merged methods from UDMFTools into UniFields, removed UDMFTools.
Updated Inno Setup script (added VC++ 2008 SP1 distributive).
Updated ZDoom_DECORATE.cfg (A_CheckBlock).
Updated documentation (added "System Requirements" page).
2015-10-09 12:38:12 +00:00
if ( BuilderPlug . Me . CopiedLinedefProps ! = null )
2009-04-19 18:07:22 +00:00
{
2015-12-13 19:18:09 +00:00
bool pastesideprops = ( BuilderPlug . Me . CopiedSidedefProps ! = null ) ; //mxd
Removed "Paste Properties Options" action.
Added "Paste Properties Special" actions in "Classic" and "Visual" categories. They work the same way as "Paste Special" action.
Added: "Copy Properties", "Paste Properties" and "Paste Properties Special" options are now shown in the Edit menu if current classic mode supports them.
Changed, Paste Properties Special window: only options relevant to current map format are now displayed.
Changed, Paste Properties Special window, UDMF: all UI-managed options are now available.
Fixed: MAPINFO parser was unable to process "include" directives.
Fixed, General interface: selection info was reset to "Nothing selected" after few seconds regardless of current selection.
Fixed, Visual mode: thing bounding boxes were not updated when changing things positions using Randomize mode.
Fixed, Visual mode: event lines were displayed at incorrect height when entering Visual mode for the first time.
Fixed, Texture Browser window: when MixTexturesFlats Game Configuration option is disabled, textures/flats are no longer shown in the Used group when flats/textures with the same names are used in the map.
Fixed(?): probably fixed an exception some users reported when trying to initialize a Classic mode after switching from Visual mode with "Sync cameras" option enabled.
Changed, Game configurations, Thing Categories: a block must have at least one thing category property to be recognized as a thing category.
Changed, Visplane Explorer: the plugin now outputs more info when it fails to initialize vpo.dll.
Cosmetic, Thing Edit window, Doom/Hexen map format: adjusted UI layout so thing flags control no longer displays scrollbars in Hexen map format.
Internal: merged methods from UDMFTools into UniFields, removed UDMFTools.
Updated Inno Setup script (added VC++ 2008 SP1 distributive).
Updated ZDoom_DECORATE.cfg (A_CheckBlock).
Updated documentation (added "System Requirements" page).
2015-10-09 12:38:12 +00:00
string pastetarget = ( pastesideprops ? "linedef and sidedef" : "linedef" ) ; //mxd
mode . CreateUndo ( "Paste " + pastetarget + " properties" ) ;
mode . SetActionResult ( "Pasted " + pastetarget + " properties." ) ;
2016-10-11 12:58:35 +00:00
BuilderPlug . Me . CopiedLinedefProps . Apply ( new List < Linedef > { Sidedef . Line } , usecopysettings , false ) ; //mxd
if ( pastesideprops ) BuilderPlug . Me . CopiedSidedefProps . Apply ( new List < Sidedef > { Sidedef } , usecopysettings ) ; //mxd. Added "usecopysettings"
2009-04-19 18:07:22 +00:00
// Update sectors on both sides
BaseVisualSector front = ( BaseVisualSector ) mode . GetVisualSector ( Sidedef . Sector ) ;
2009-05-01 20:31:17 +00:00
if ( front ! = null ) front . Changed = true ;
2009-04-19 18:07:22 +00:00
if ( Sidedef . Other ! = null )
{
BaseVisualSector back = ( BaseVisualSector ) mode . GetVisualSector ( Sidedef . Other . Sector ) ;
2009-05-01 20:31:17 +00:00
if ( back ! = null ) back . Changed = true ;
2009-04-19 18:07:22 +00:00
}
mode . ShowTargetInfo ( ) ;
}
}
// Return texture name
public virtual string GetTextureName ( ) { return "" ; }
// Select button pressed
public virtual void OnSelectBegin ( )
{
2009-05-05 09:50:23 +00:00
mode . LockTarget ( ) ;
2009-04-19 18:07:22 +00:00
dragstartanglexy = General . Map . VisualCamera . AngleXY ;
dragstartanglez = General . Map . VisualCamera . AngleZ ;
dragorigin = pickintersect ;
2016-01-16 12:46:44 +00:00
Point texoffset = GetTextureOffset ( ) ; //mxd
startoffsetx = texoffset . X ;
startoffsety = texoffset . Y ;
prevoffsetx = texoffset . X ;
prevoffsety = texoffset . Y ;
2009-04-19 18:07:22 +00:00
}
// Select button released
public virtual void OnSelectEnd ( )
{
2009-05-05 09:50:23 +00:00
mode . UnlockTarget ( ) ;
2009-04-19 18:07:22 +00:00
// Was dragging?
if ( uvdragging )
{
// Dragging stops now
uvdragging = false ;
}
else
{
// Add/remove selection
2009-07-07 11:29:56 +00:00
if ( this . selected )
{
this . selected = false ;
mode . RemoveSelectedObject ( this ) ;
}
else
{
this . selected = true ;
mode . AddSelectedObject ( this ) ;
}
2009-04-19 18:07:22 +00:00
}
}
// Edit button released
public virtual void OnEditEnd ( )
{
2010-08-14 18:07:38 +00:00
if ( General . Interface . IsActiveWindow )
2009-04-19 18:07:22 +00:00
{
2010-08-14 18:07:38 +00:00
List < Linedef > linedefs = mode . GetSelectedLinedefs ( ) ;
2016-01-16 12:46:44 +00:00
updatelist = new List < BaseVisualSector > ( ) ; //mxd
2014-12-03 23:15:26 +00:00
foreach ( Linedef l in linedefs )
{
2013-07-19 15:30:58 +00:00
if ( l . Front ! = null & & mode . VisualSectorExists ( l . Front . Sector ) )
2016-01-16 12:46:44 +00:00
updatelist . Add ( ( BaseVisualSector ) mode . GetVisualSector ( l . Front . Sector ) ) ;
2013-07-19 15:30:58 +00:00
if ( l . Back ! = null & & mode . VisualSectorExists ( l . Back . Sector ) )
2016-01-16 12:46:44 +00:00
updatelist . Add ( ( BaseVisualSector ) mode . GetVisualSector ( l . Back . Sector ) ) ;
2013-07-19 15:30:58 +00:00
}
2016-04-19 20:40:42 +00:00
//mxd. Always select front side for extrafloors
Linedef sourceline = GetControlLinedef ( ) ;
Sidedef target = ( sourceline ! = Sidedef . Line & & sourceline . Front ! = null ? sourceline . Front : Sidedef ) ;
2014-02-21 14:42:12 +00:00
General . Interface . OnEditFormValuesChanged + = Interface_OnEditFormValuesChanged ;
2013-07-19 15:30:58 +00:00
mode . StartRealtimeInterfaceUpdate ( SelectionType . Linedefs ) ;
2016-04-19 20:40:42 +00:00
DialogResult result = General . Interface . ShowEditLinedefs ( linedefs , target . IsFront , ! target . IsFront ) ;
2013-07-19 15:30:58 +00:00
mode . StopRealtimeInterfaceUpdate ( SelectionType . Linedefs ) ;
General . Interface . OnEditFormValuesChanged - = Interface_OnEditFormValuesChanged ;
2016-01-16 12:46:44 +00:00
updatelist . Clear ( ) ;
updatelist = null ;
2013-07-19 15:30:58 +00:00
2015-04-14 11:33:57 +00:00
//mxd. Effects may need updating...
if ( result = = DialogResult . OK ) mode . RebuildElementData ( ) ;
2009-04-19 18:07:22 +00:00
}
}
2013-07-19 15:30:58 +00:00
//mxd
2014-12-03 23:15:26 +00:00
private void Interface_OnEditFormValuesChanged ( object sender , EventArgs e )
{
2016-01-16 12:46:44 +00:00
foreach ( BaseVisualSector vs in updatelist ) vs . UpdateSectorGeometry ( false ) ;
2013-07-19 15:30:58 +00:00
}
2009-04-19 18:07:22 +00:00
// Mouse moves
public virtual void OnMouseMove ( MouseEventArgs e )
{
// Dragging UV?
if ( uvdragging )
{
UpdateDragUV ( ) ;
}
2019-01-19 07:56:13 +00:00
else if ( mode . PaintSelectPressed ) // biwa. Paint selection going on?
{
if ( mode . PaintSelectType = = this . GetType ( ) . BaseType & & mode . Highlighted ! = this ) // using BaseType so that middle, upper, lower, etc can be selecting in one go
{
// toggle selected state
2019-06-20 13:38:41 +00:00
if ( General . Interface . ShiftState ^ BuilderPlug . Me . AdditivePaintSelect )
2019-01-19 07:56:13 +00:00
{
2021-02-13 21:59:06 +00:00
if ( ! selected )
{
selected = true ;
mode . AddSelectedObject ( this ) ;
}
2019-01-19 07:56:13 +00:00
}
else if ( General . Interface . CtrlState )
{
2021-02-13 21:59:06 +00:00
if ( selected )
{
selected = false ;
mode . RemoveSelectedObject ( this ) ;
}
2019-01-19 07:56:13 +00:00
}
else
{
2021-02-13 21:59:06 +00:00
if ( selected )
2019-01-19 07:56:13 +00:00
mode . RemoveSelectedObject ( this ) ;
else
mode . AddSelectedObject ( this ) ;
2021-02-13 21:59:06 +00:00
selected = ! selected ;
2019-01-19 07:56:13 +00:00
}
}
return ;
}
2009-04-19 18:07:22 +00:00
else
{
// Select button pressed?
if ( General . Actions . CheckActionActive ( General . ThisAssembly , "visualselect" ) )
{
// Check if tolerance is exceeded to start UV dragging
2020-05-21 12:20:02 +00:00
double deltaxy = General . Map . VisualCamera . AngleXY - dragstartanglexy ;
double deltaz = General . Map . VisualCamera . AngleZ - dragstartanglez ;
2009-04-19 18:07:22 +00:00
if ( ( Math . Abs ( deltaxy ) + Math . Abs ( deltaz ) ) > DRAG_ANGLE_TOLERANCE )
{
2013-09-11 09:47:53 +00:00
mode . PreAction ( UndoGroup . TextureOffsetChange ) ;
mode . CreateUndo ( "Change texture offsets" ) ;
// Start drag now
uvdragging = true ;
mode . Renderer . ShowSelection = false ;
mode . Renderer . ShowHighlight = false ;
UpdateDragUV ( ) ;
2009-04-19 18:07:22 +00:00
}
}
}
}
// This is called to update UV dragging
protected virtual void UpdateDragUV ( )
{
2020-05-21 12:20:02 +00:00
double u_ray ;
2009-04-19 18:07:22 +00:00
// Calculate intersection position
Line2D ray = new Line2D ( General . Map . VisualCamera . Position , General . Map . VisualCamera . Target ) ;
Sidedef . Line . Line . GetIntersection ( ray , out u_ray ) ;
Vector3D intersect = General . Map . VisualCamera . Position + ( General . Map . VisualCamera . Target - General . Map . VisualCamera . Position ) * u_ray ;
// Calculate offsets
Vector3D dragdelta = intersect - dragorigin ;
Vector3D dragdeltaxy = dragdelta * deltaxy ;
Vector3D dragdeltaz = dragdelta * deltaz ;
2020-05-21 12:20:02 +00:00
double offsetx = dragdeltaxy . GetLength ( ) ;
double offsety = dragdeltaz . GetLength ( ) ;
2009-04-19 18:07:22 +00:00
if ( ( Math . Sign ( dragdeltaxy . x ) < 0 ) | | ( Math . Sign ( dragdeltaxy . y ) < 0 ) | | ( Math . Sign ( dragdeltaxy . z ) < 0 ) ) offsetx = - offsetx ;
if ( ( Math . Sign ( dragdeltaz . x ) < 0 ) | | ( Math . Sign ( dragdeltaz . y ) < 0 ) | | ( Math . Sign ( dragdeltaz . z ) < 0 ) ) offsety = - offsety ;
// Apply offsets
2014-12-03 23:15:26 +00:00
if ( General . Interface . CtrlState & & General . Interface . ShiftState )
{
//mxd. Clamp to grid size?
2013-12-05 09:24:55 +00:00
int newoffsetx = startoffsetx - ( int ) Math . Round ( offsetx ) ;
int newoffsety = startoffsety + ( int ) Math . Round ( offsety ) ;
int dx = prevoffsetx - newoffsetx ;
int dy = prevoffsety - newoffsety ;
2017-01-06 10:01:59 +00:00
if ( Math . Abs ( dx ) > = General . Map . Grid . GridSize )
2014-12-03 23:15:26 +00:00
{
2017-01-06 10:01:59 +00:00
dx = General . Map . Grid . GridSize * Math . Sign ( dx ) ;
2013-12-05 09:24:55 +00:00
prevoffsetx = newoffsetx ;
2014-12-03 23:15:26 +00:00
}
else
{
2013-12-05 09:24:55 +00:00
dx = 0 ;
}
2017-01-06 10:01:59 +00:00
if ( Math . Abs ( dy ) > = General . Map . Grid . GridSize )
2014-12-03 23:15:26 +00:00
{
2017-01-06 10:01:59 +00:00
dy = General . Map . Grid . GridSize * Math . Sign ( dy ) ;
2013-12-05 09:24:55 +00:00
prevoffsety = newoffsety ;
2014-12-03 23:15:26 +00:00
}
else
{
2013-12-05 09:24:55 +00:00
dy = 0 ;
}
if ( dx ! = 0 | | dy ! = 0 ) mode . ApplyTextureOffsetChange ( dx , dy ) ;
2014-12-03 23:15:26 +00:00
}
else
{
//mxd. Constraint to axis?
2013-12-05 09:24:55 +00:00
int newoffsetx = ( General . Interface . CtrlState ? startoffsetx : startoffsetx - ( int ) Math . Round ( offsetx ) ) ; //mxd
int newoffsety = ( General . Interface . ShiftState ? startoffsety : startoffsety + ( int ) Math . Round ( offsety ) ) ; //mxd
mode . ApplyTextureOffsetChange ( prevoffsetx - newoffsetx , prevoffsety - newoffsety ) ;
prevoffsetx = newoffsetx ;
prevoffsety = newoffsety ;
}
2009-04-19 18:07:22 +00:00
mode . ShowTargetInfo ( ) ;
}
// Sector brightness change
public virtual void OnChangeTargetBrightness ( bool up )
{
2014-12-22 21:36:49 +00:00
//mxd. Change UDMF wall light?
2022-01-29 17:59:24 +00:00
if ( General . Map . UDMF & & ( General . Map . Config . DistinctWallBrightness | | General . Map . Config . DistinctSidedefPartBrightness ) )
2014-12-22 21:36:49 +00:00
{
2022-01-29 13:48:41 +00:00
string fieldname = "light" ;
string fieldabsolutename = "lightabsolute" ;
2022-01-29 17:59:24 +00:00
if ( General . Map . Config . DistinctSidedefPartBrightness )
2022-01-29 13:48:41 +00:00
{
fieldname + = "_" + partname ;
fieldabsolutename + = "_" + partname ;
}
int light = Sidedef . Fields . GetValue ( fieldname , 0 ) ;
bool absolute = Sidedef . Fields . GetValue ( fieldabsolutename , false ) ;
2015-02-12 22:04:49 +00:00
int newlight ;
2014-12-22 21:36:49 +00:00
if ( up )
2015-02-12 22:04:49 +00:00
newlight = General . Map . Config . BrightnessLevels . GetNextHigher ( light , absolute ) ;
2014-12-22 21:36:49 +00:00
else
2015-02-12 22:04:49 +00:00
newlight = General . Map . Config . BrightnessLevels . GetNextLower ( light , absolute ) ;
2014-12-22 21:36:49 +00:00
2015-02-12 22:04:49 +00:00
if ( newlight = = light ) return ;
2014-12-22 21:36:49 +00:00
2016-04-19 20:40:42 +00:00
// Create undo
2014-12-22 21:36:49 +00:00
mode . CreateUndo ( "Change wall brightness" , UndoGroup . SurfaceBrightnessChange , Sector . Sector . FixedIndex ) ;
Sidedef . Fields . BeforeFieldsChange ( ) ;
2016-04-19 20:40:42 +00:00
// Apply changes
2022-01-29 13:48:41 +00:00
UniFields . SetInteger ( Sidedef . Fields , fieldname , newlight , ( absolute ? int . MinValue : 0 ) ) ;
2015-02-12 22:04:49 +00:00
Tools . UpdateLightFogFlag ( Sidedef ) ;
mode . SetActionResult ( "Changed wall brightness to " + newlight + "." ) ;
2014-12-22 21:36:49 +00:00
2017-02-09 13:20:39 +00:00
// Update this part only
//this.Setup();
// [ZZ] why the hell was maxed updating only this part? sidedef change is global per sidedef, not only upper/lower/middle part.
// find this sidedef in sector, update all parts.
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
parts . SetupAllParts ( ) ;
2014-12-22 21:36:49 +00:00
}
else if ( ! Sector . Changed )
2009-04-19 18:07:22 +00:00
{
2009-05-03 19:22:32 +00:00
// Change brightness
mode . CreateUndo ( "Change sector brightness" , UndoGroup . SectorBrightnessChange , Sector . Sector . FixedIndex ) ;
if ( up )
Sector . Sector . Brightness = General . Map . Config . BrightnessLevels . GetNextHigher ( Sector . Sector . Brightness ) ;
else
Sector . Sector . Brightness = General . Map . Config . BrightnessLevels . GetNextLower ( Sector . Sector . Brightness ) ;
mode . SetActionResult ( "Changed sector brightness to " + Sector . Sector . Brightness + "." ) ;
Sector . Sector . UpdateCache ( ) ;
// Rebuild sector
2012-11-27 21:12:20 +00:00
Sector . UpdateSectorGeometry ( false ) ;
2009-05-03 19:22:32 +00:00
// Go for all things in this sector
foreach ( Thing t in General . Map . Map . Things )
2009-04-19 18:07:22 +00:00
{
2009-05-03 19:22:32 +00:00
if ( t . Sector = = Sector . Sector )
2009-04-19 18:07:22 +00:00
{
2009-05-03 19:22:32 +00:00
if ( mode . VisualThingExists ( t ) )
{
// Update thing
2016-04-19 20:40:42 +00:00
BaseVisualThing vt = ( BaseVisualThing ) mode . GetVisualThing ( t ) ;
2009-05-03 19:22:32 +00:00
vt . Changed = true ;
}
2009-04-19 18:07:22 +00:00
}
}
}
}
// Texture offset change
2013-04-26 12:32:51 +00:00
public virtual void OnChangeTextureOffset ( int horizontal , int vertical , bool doSurfaceAngleCorrection )
2009-04-19 18:07:22 +00:00
{
if ( ( General . Map . UndoRedo . NextUndo = = null ) | | ( General . Map . UndoRedo . NextUndo . TicketID ! = undoticket ) )
2009-05-03 19:22:32 +00:00
undoticket = mode . CreateUndo ( "Change texture offsets" ) ;
2009-04-19 18:07:22 +00:00
2013-09-11 09:47:53 +00:00
//mxd
2016-09-05 18:36:22 +00:00
if ( General . Map . UDMF & & General . Map . Config . UseLocalSidedefTextureOffsets )
2014-12-03 23:15:26 +00:00
{
2016-09-05 18:36:22 +00:00
// Apply per-texture offsets
2016-09-06 12:05:47 +00:00
MoveTextureOffset ( - horizontal , - vertical ) ;
2013-09-11 09:47:53 +00:00
Point p = GetTextureOffset ( ) ;
mode . SetActionResult ( "Changed texture offsets to " + p . X + ", " + p . Y + "." ) ;
2016-04-19 20:40:42 +00:00
// Update this part only
this . Setup ( ) ;
2014-12-03 23:15:26 +00:00
}
else
{
2013-08-02 12:50:53 +00:00
//mxd. Apply classic offsets
2016-09-02 19:18:37 +00:00
bool textureloaded = ( Texture ! = null & & Texture . IsImageLoaded ) ;
2013-08-28 14:53:21 +00:00
Sidedef . OffsetX = ( Sidedef . OffsetX - horizontal ) ;
2016-09-02 19:18:37 +00:00
if ( textureloaded ) Sidedef . OffsetX % = Texture . Width ;
2013-08-28 14:53:21 +00:00
Sidedef . OffsetY = ( Sidedef . OffsetY - vertical ) ;
2016-09-02 19:18:37 +00:00
if ( geometrytype ! = VisualGeometryType . WALL_MIDDLE & & textureloaded ) Sidedef . OffsetY % = Texture . Height ;
2013-08-02 12:50:53 +00:00
2013-09-11 09:47:53 +00:00
mode . SetActionResult ( "Changed texture offsets to " + Sidedef . OffsetX + ", " + Sidedef . OffsetY + "." ) ;
2016-04-19 20:40:42 +00:00
// Update all sidedef geometry
VisualSidedefParts parts = Sector . GetSidedefParts ( Sidedef ) ;
parts . SetupAllParts ( ) ;
2013-09-11 09:47:53 +00:00
}
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2009-04-19 18:07:22 +00:00
}
2013-04-01 11:06:01 +00:00
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
//mxd
2015-03-03 09:42:54 +00:00
public virtual void OnChangeScale ( int incrementX , int incrementY )
2014-12-03 23:15:26 +00:00
{
2016-09-02 19:18:37 +00:00
if ( ! General . Map . UDMF | | Texture = = null | | ! Texture . IsImageLoaded ) return ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
if ( ( General . Map . UndoRedo . NextUndo = = null ) | | ( General . Map . UndoRedo . NextUndo . TicketID ! = undoticket ) )
undoticket = mode . CreateUndo ( "Change wall scale" ) ;
string keyX ;
string keyY ;
2014-12-03 23:15:26 +00:00
switch ( GeometryType )
{
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
case VisualGeometryType . WALL_UPPER :
keyX = "scalex_top" ;
keyY = "scaley_top" ;
break ;
case VisualGeometryType . WALL_MIDDLE :
keyX = "scalex_mid" ;
keyY = "scaley_mid" ;
break ;
case VisualGeometryType . WALL_LOWER :
keyX = "scalex_bottom" ;
keyY = "scaley_bottom" ;
break ;
default :
throw new Exception ( "OnChangeTextureScale(): Got unknown GeometryType: " + GeometryType ) ;
}
2020-05-22 19:39:18 +00:00
double scaleX = Sidedef . Fields . GetValue ( keyX , 1.0 ) ;
double scaleY = Sidedef . Fields . GetValue ( keyY , 1.0 ) ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
Sidedef . Fields . BeforeFieldsChange ( ) ;
2015-03-03 09:42:54 +00:00
if ( incrementX ! = 0 )
2014-12-03 23:15:26 +00:00
{
2020-05-22 19:39:18 +00:00
double pix = ( int ) Math . Round ( Texture . Width * scaleX ) - incrementX ;
double newscaleX = Math . Round ( pix / Texture . Width , 3 ) ;
2015-03-03 09:42:54 +00:00
scaleX = ( newscaleX = = 0 ? scaleX * - 1 : newscaleX ) ;
2020-05-22 19:39:18 +00:00
UniFields . SetFloat ( Sidedef . Fields , keyX , scaleX , 1.0 ) ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
}
2014-12-03 23:15:26 +00:00
if ( incrementY ! = 0 )
{
2020-05-22 19:39:18 +00:00
double pix = ( int ) Math . Round ( Texture . Height * scaleY ) - incrementY ;
double newscaleY = Math . Round ( pix / Texture . Height , 3 ) ;
2015-03-03 09:42:54 +00:00
scaleY = ( newscaleY = = 0 ? scaleY * - 1 : newscaleY ) ;
2020-05-22 19:39:18 +00:00
UniFields . SetFloat ( Sidedef . Fields , keyY , scaleY , 1.0 ) ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
}
2016-04-19 20:40:42 +00:00
// Update geometry
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
Setup ( ) ;
2016-04-19 20:40:42 +00:00
//mxd. Update linked effects
SectorData sd = mode . GetSectorDataEx ( Sector . Sector ) ;
if ( sd ! = null ) sd . Reset ( true ) ;
2015-03-03 09:42:54 +00:00
mode . SetActionResult ( "Wall scale changed to " + scaleX . ToString ( "F03" , CultureInfo . InvariantCulture ) + ", " + scaleY . ToString ( "F03" , CultureInfo . InvariantCulture ) + " (" + ( int ) Math . Round ( Texture . Width / scaleX ) + " x " + ( int ) Math . Round ( Texture . Height / scaleY ) + ")." ) ;
Visual mode, UDMF: added "Scale Texture Up (X)", "Scale Texture Down (X)", "Scale Texture Up (Y)", "Scale Texture Down (Y)" actions. Default keys are Num6, Num4, Num8, Num5.
Visual mode, UDMF: renamed "Rotate Thing Clockwise" and "Rotate Thing Counterclockwise" actions to "Rotate Clockwise" and "Rotate Counterclockwise". These actions can now be used to change rotation of floor/ceiling textures.
Visual mode, UDMF: "Reset Texture Offsets" action now also resets sidedef's scale and floor/ceiling's scale and rotation.
Visual mode, UDMF: control line's OffsetX and OffsetY were not taken into account when calculating texture offsets of 3d floors' sides.
Visual mode, UDMF: fixed a ton of bugs in Auto align functions.
Visual mode, UDMF: when using "Move Texture Left/Right/Up/Down by 1" actions texture offsets were not updated properly when texture's scale was < 1.0.
Visual mode, UDMF: OffsetX and OffsetY were not taken into account in "Fit Texture Width/Height" actions.
Dockers Panel: added Pin/Unpin button, which acts the same as "Preferences -> Interface -> Side panels -> Auto hide" checkbox.
Texture size labels can now be disabled by unchecking "Preferences -> Interface -> Show texture and flat sizes in browsers" checkbox.
Texture size labels now are not shown for unknown textures.
Most of texture size labels had incorrect bg color.
ZDoom_linedefs.cfg: action specials 223 and 224 had incorrect Arg0.
2013-06-24 14:21:13 +00:00
}
2019-01-19 07:56:13 +00:00
// biwa
public virtual void OnPaintSelectBegin ( )
{
mode . PaintSelectType = this . GetType ( ) . BaseType ; // using BaseType so that middle, upper, lower, etc can be selecting in one go
// toggle selected state
2019-06-20 13:38:41 +00:00
if ( General . Interface . ShiftState ^ BuilderPlug . Me . AdditivePaintSelect )
2019-01-19 07:56:13 +00:00
{
2021-01-30 23:12:07 +00:00
if ( ! selected )
{
selected = true ;
mode . AddSelectedObject ( this ) ;
}
2019-01-19 07:56:13 +00:00
}
else if ( General . Interface . CtrlState )
{
2021-01-30 23:12:07 +00:00
if ( selected )
{
selected = false ;
mode . RemoveSelectedObject ( this ) ;
}
2019-01-19 07:56:13 +00:00
}
else
{
2021-01-30 23:12:07 +00:00
if ( selected )
2019-01-19 07:56:13 +00:00
mode . RemoveSelectedObject ( this ) ;
else
mode . AddSelectedObject ( this ) ;
2021-01-30 23:12:07 +00:00
selected = ! selected ;
2019-01-19 07:56:13 +00:00
}
}
2009-04-19 18:07:22 +00:00
#endregion
}
}