ZoneBuilder/Source/Plugins/BuilderModes/ErrorChecks/ResultLineMissingSides.cs

206 lines
5.6 KiB
C#

#region ================== Copyright (c) 2007 Pascal vd Heiden
/*
* Copyright (c) 2007 Pascal vd Heiden, www.codeimp.com
* This program is released under GNU General Public License
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#endregion
#region ================== Namespaces
using System;
using System.Collections.Generic;
using CodeImp.DoomBuilder.Map;
using CodeImp.DoomBuilder.Rendering;
using CodeImp.DoomBuilder.Geometry;
#endregion
namespace CodeImp.DoomBuilder.BuilderModes
{
public class ResultLineMissingSides : ErrorResult
{
#region ================== Variables
private readonly Linedef line;
private readonly int buttons;
private readonly Sidedef copysidedeffront;
private readonly Sidedef copysidedefback;
#endregion
#region ================== Properties
public override int Buttons { get { return buttons; } }
public override string Button1Text { get { return "Create One Side"; } }
public override string Button2Text { get { return "Create Both Sides"; } }
#endregion
#region ================== Constructor / Destructor
// Constructor
public ResultLineMissingSides(Linedef l)
{
// Initialize
line = l;
viewobjects.Add(l);
hidden = l.IgnoredErrorChecks.Contains(this.GetType()); //mxd
description = "This linedef is missing front and back sidedefs." +
"A line must have at least a front side and optionally a back side!";
buttons = 0;
// Check if we can join a sector on the front side
bool fixable = false;
List<LinedefSide> sides = Tools.FindPotentialSectorAt(l, true);
if(sides != null)
{
foreach(LinedefSide sd in sides)
{
// If any of the sides lies along a sidedef, then we can copy
// that sidedef to fix the missing sidedef on this line.
if(sd.Front && (sd.Line.Front != null))
{
copysidedeffront = sd.Line.Front;
fixable = true;
break;
}
if(!sd.Front && (sd.Line.Back != null))
{
copysidedeffront = sd.Line.Back;
fixable = true;
break;
}
}
}
// Fixable?
if(fixable) buttons++;
// Check if we can join a sector on the back side
fixable = false;
sides = Tools.FindPotentialSectorAt(l, false);
if(sides != null)
{
foreach(LinedefSide sd in sides)
{
// If any of the sides lies along a sidedef, then we can copy
// that sidedef to fix the missing sidedef on this line.
if(sd.Front && (sd.Line.Front != null))
{
copysidedefback = sd.Line.Front;
fixable = true;
break;
}
if(!sd.Front && (sd.Line.Back != null))
{
copysidedefback = sd.Line.Back;
fixable = true;
break;
}
}
}
// Fixable?
if(fixable) buttons++;
// Now make a fine description
switch(buttons)
{
case 0: description += " Doom Builder could not find a solution to fix this line."; break;
case 1: description += " Click Create One Side to rebuild a single sidedef, making this line single-sided."; break;
case 2: description += " Click Create Both Side to rebuild both sides of the line, making this line double-sided."; break;
}
}
#endregion
#region ================== Methods
// This sets if this result is displayed in ErrorCheckForm (mxd)
internal override void Hide(bool hide)
{
hidden = hide;
Type t = this.GetType();
if(hide) line.IgnoredErrorChecks.Add(t);
else if(line.IgnoredErrorChecks.Contains(t)) line.IgnoredErrorChecks.Remove(t);
}
// This must return the string that is displayed in the listbox
public override string ToString()
{
return "Linedef " + line.Index + " is missing both sides";
}
// Rendering
public override void PlotSelection(IRenderer2D renderer)
{
renderer.PlotLinedef(line, General.Colors.Selection);
renderer.PlotVertex(line.Start, ColorCollection.VERTICES);
renderer.PlotVertex(line.End, ColorCollection.VERTICES);
}
// Fix a single side
public override bool Button1Click(bool batchMode)
{
// On which side can we fix?
if(copysidedeffront != null)
{
// Front
if(!batchMode) General.Map.UndoRedo.CreateUndo("Create front sidedef");
Sidedef newside = General.Map.Map.CreateSidedef(line, true, copysidedeffront.Sector);
if(newside == null) return false;
copysidedeffront.CopyPropertiesTo(newside);
}
else if(copysidedefback != null)
{
// Back
// Because the line is single-sided, we make the sidedef on the front.
// We will then flip it to make sure to ends up in the right position.
if(!batchMode) General.Map.UndoRedo.CreateUndo("Create front sidedef");
Sidedef newside = General.Map.Map.CreateSidedef(line, true, copysidedefback.Sector);
if(newside == null) return false;
copysidedefback.CopyPropertiesTo(newside);
line.FlipVertices();
}
line.ApplySidedFlags();
General.Map.Map.Update();
return true;
}
// Fix both sides
public override bool Button2Click(bool batchMode)
{
if(!batchMode) General.Map.UndoRedo.CreateUndo("Create sidedefs");
// Front
Sidedef newside = General.Map.Map.CreateSidedef(line, true, copysidedeffront.Sector);
if(newside == null) return false;
copysidedeffront.CopyPropertiesTo(newside);
// Back
newside = General.Map.Map.CreateSidedef(line, false, copysidedefback.Sector);
if(newside == null) return false;
copysidedefback.CopyPropertiesTo(newside);
line.ApplySidedFlags();
General.Map.Map.Update();
return true;
}
#endregion
}
}