mirror of
https://git.do.srb2.org/STJr/UltimateZoneBuilder.git
synced 2024-11-23 20:32:34 +00:00
0f7aa9f827
Added, DECORATE parser: damage types are now parsed. Added: the editor now reports duplicate textures/flats/patches/sprites/colormaps/voxels in the loaded wads. Added, all text parsers: added #region/#endregion support. Added TERRAIN parser. Added, Script Editor: added special handling for DECORATE special comments. Added, Sector Edit window, UDMF: Soundsequence value was setup incorrectly when showing the window for multiple sectors with mixed Soundsequence value. Fixed, Map Options window: "Strictly load patches between P_START and P_END" was not applied when applying the changes. Fixed, MAPINFO parser: MapInfo should be treated as defined when a map MAPINFO block corresponding to current map is encountered even if it doesn't define any properties recognized by the editor. Fixed, all text parsers: in some cases error line was calculated incorrectly when reporting an error detected by a text parser. Cosmetic: changed ' to " in the rest of Error and Warning messages. Internal: added text resource tracking. Updated ZDoom_DECORATE.cfg. Updated documentation ("Game Configuration - Basic Settings" page).
1652 lines
51 KiB
C#
1652 lines
51 KiB
C#
#region ================== Namespaces
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Drawing;
|
|
using System.Windows.Forms;
|
|
using CodeImp.DoomBuilder.Controls;
|
|
using CodeImp.DoomBuilder.Geometry;
|
|
using CodeImp.DoomBuilder.Map;
|
|
using CodeImp.DoomBuilder.Types;
|
|
|
|
#endregion
|
|
|
|
namespace CodeImp.DoomBuilder.Windows
|
|
{
|
|
internal partial class SectorEditFormUDMF : DelayedForm
|
|
{
|
|
#region ================== Events
|
|
|
|
public event EventHandler OnValuesChanged; //mxd
|
|
|
|
#endregion
|
|
|
|
#region ================== Constants
|
|
|
|
private const string NO_SOUND_SEQUENCE = "None"; //mxd
|
|
private const string NO_TERRAIN = "Default"; //mxd
|
|
private const string NO_DAMAGETYPE = "None"; //mxd
|
|
|
|
#endregion
|
|
|
|
#region ================== Variables
|
|
|
|
private ICollection<Sector> sectors;
|
|
private Dictionary<Sector, SectorProperties> sectorprops; //mxd
|
|
private bool preventchanges; //mxd
|
|
private bool undocreated; //mxd
|
|
private StepsList anglesteps; //mxd
|
|
private readonly string[] renderstyles; //mxd
|
|
|
|
//mxd. Persistent settings
|
|
private static bool linkCeilingScale;
|
|
private static bool linkFloorScale;
|
|
private static bool useFloorLineAngles;
|
|
private static bool useCeilLineAngles;
|
|
private static bool useFloorSlopeLineAngles;
|
|
private static bool useCeilSlopeLineAngles;
|
|
private static SlopePivotMode ceilpivotmode = SlopePivotMode.LOCAL;
|
|
private static SlopePivotMode floorpivotmode = SlopePivotMode.LOCAL;
|
|
|
|
//mxd. Window setup stuff
|
|
private static Point location = Point.Empty;
|
|
private static int activetab;
|
|
|
|
//mxd. Slope pivots
|
|
private Vector2D globalslopepivot;
|
|
private Dictionary<Sector, Vector2D> slopepivots;
|
|
|
|
#endregion
|
|
|
|
#region ================== Structs
|
|
|
|
private struct SectorProperties //mxd
|
|
{
|
|
public readonly int Brightness;
|
|
public readonly int FloorHeight;
|
|
public readonly int CeilHeight;
|
|
public readonly string FloorTexture;
|
|
public readonly string CeilTexture;
|
|
|
|
//UDMF stuff
|
|
public readonly int LightColor;
|
|
public readonly int FadeColor;
|
|
//public float Desaturation;
|
|
|
|
//UDMF Ceiling
|
|
public readonly float CeilOffsetX;
|
|
public readonly float CeilOffsetY;
|
|
public readonly float CeilScaleX;
|
|
public readonly float CeilScaleY;
|
|
//public float CeilAlpha;
|
|
public readonly float CeilRotation;
|
|
public readonly int CeilBrightness;
|
|
public readonly bool CeilLightAbsoulte;
|
|
|
|
//UDMF Floor
|
|
public readonly float FloorOffsetX;
|
|
public readonly float FloorOffsetY;
|
|
public readonly float FloorScaleX;
|
|
public readonly float FloorScaleY;
|
|
//public float FloorAlpha;
|
|
public readonly float FloorRotation;
|
|
public readonly int FloorBrightness;
|
|
public readonly bool FloorLightAbsoulte;
|
|
|
|
//UDMF slopes. Angles are in degrees
|
|
public readonly Vector3D FloorSlope;
|
|
public readonly Vector3D CeilSlope;
|
|
public readonly float FloorSlopeAngleXY;
|
|
public readonly float FloorSlopeAngleZ;
|
|
public readonly float FloorSlopeOffset;
|
|
public readonly float CeilSlopeAngleXY;
|
|
public readonly float CeilSlopeAngleZ;
|
|
public readonly float CeilSlopeOffset;
|
|
|
|
public SectorProperties(Sector s)
|
|
{
|
|
Brightness = s.Brightness;
|
|
FloorHeight = s.FloorHeight;
|
|
CeilHeight = s.CeilHeight;
|
|
FloorTexture = s.FloorTexture;
|
|
CeilTexture = s.CeilTexture;
|
|
|
|
//UDMF stuff
|
|
LightColor = UniFields.GetInteger(s.Fields, "lightcolor", 16777215);
|
|
FadeColor = UniFields.GetInteger(s.Fields, "fadecolor", 0);
|
|
|
|
//UDMF Ceiling
|
|
CeilOffsetX = UniFields.GetFloat(s.Fields, "xpanningceiling", 0f);
|
|
CeilOffsetY = UniFields.GetFloat(s.Fields, "ypanningceiling", 0f);
|
|
CeilScaleX = UniFields.GetFloat(s.Fields, "xscaleceiling", 1.0f);
|
|
CeilScaleY = UniFields.GetFloat(s.Fields, "yscaleceiling", 1.0f);
|
|
//CeilAlpha = UniFields.GetFloat(s.Fields, "alphaceiling", 1.0f);
|
|
CeilRotation = s.Fields.GetValue("rotationceiling", 0.0f);
|
|
CeilBrightness = s.Fields.GetValue("lightceiling", 0);
|
|
CeilLightAbsoulte = s.Fields.GetValue("lightceilingabsolute", false);
|
|
|
|
//UDMF Floor
|
|
FloorOffsetX = UniFields.GetFloat(s.Fields, "xpanningfloor", 0f);
|
|
FloorOffsetY = UniFields.GetFloat(s.Fields, "ypanningfloor", 0f);
|
|
FloorScaleX = UniFields.GetFloat(s.Fields, "xscalefloor", 1.0f);
|
|
FloorScaleY = UniFields.GetFloat(s.Fields, "yscalefloor", 1.0f);
|
|
//FloorAlpha = UniFields.GetFloat(s.Fields, "alphafloor", 1.0f);
|
|
FloorRotation = s.Fields.GetValue("rotationfloor", 0.0f);
|
|
FloorBrightness = s.Fields.GetValue("lightfloor", 0);
|
|
FloorLightAbsoulte = s.Fields.GetValue("lightfloorabsolute", false);
|
|
|
|
//UDMF slopes
|
|
if(s.FloorSlope.GetLengthSq() > 0)
|
|
{
|
|
FloorSlopeAngleXY = General.ClampAngle((float)Math.Round(Angle2D.RadToDeg(s.FloorSlope.GetAngleXY()) - 180, 1));
|
|
FloorSlopeAngleZ = -(float)Math.Round(Angle2D.RadToDeg(s.FloorSlope.GetAngleZ()) - 90, 1);
|
|
FloorSlopeOffset = (float.IsNaN(s.FloorSlopeOffset) ? s.FloorHeight : s.FloorSlopeOffset);
|
|
}
|
|
else
|
|
{
|
|
FloorSlopeAngleXY = 0;
|
|
FloorSlopeAngleZ = 0;
|
|
FloorSlopeOffset = -s.FloorHeight;
|
|
}
|
|
FloorSlope = s.FloorSlope;
|
|
|
|
if(s.CeilSlope.GetLengthSq() > 0)
|
|
{
|
|
CeilSlopeAngleXY = General.ClampAngle((float)Math.Round(Angle2D.RadToDeg(s.CeilSlope.GetAngleXY()) - 180, 1));
|
|
CeilSlopeAngleZ = -(float)Math.Round(270 - Angle2D.RadToDeg(s.CeilSlope.GetAngleZ()), 1);
|
|
CeilSlopeOffset = (float.IsNaN(s.CeilSlopeOffset) ? s.CeilHeight : s.CeilSlopeOffset);
|
|
}
|
|
else
|
|
{
|
|
CeilSlopeAngleXY = 0;
|
|
CeilSlopeAngleZ = 0;
|
|
CeilSlopeOffset = s.CeilHeight;
|
|
}
|
|
CeilSlope = s.CeilSlope;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Constructor
|
|
|
|
public SectorEditFormUDMF()
|
|
{
|
|
InitializeComponent();
|
|
|
|
//mxd. Widow setup
|
|
if(location != Point.Empty)
|
|
{
|
|
this.StartPosition = FormStartPosition.Manual;
|
|
this.Location = location;
|
|
if(General.Settings.StoreSelectedEditTab && activetab > 0) tabs.SelectTab(activetab);
|
|
}
|
|
|
|
// Fill flags list
|
|
foreach(KeyValuePair<string, string> lf in General.Map.Config.SectorFlags)
|
|
flags.Add(lf.Value, lf.Key);
|
|
flags.Enabled = General.Map.Config.SectorFlags.Count > 0;
|
|
|
|
// Setup renderstyles
|
|
renderstyles = new string[General.Map.Config.SectorRenderStyles.Count];
|
|
General.Map.Config.SectorRenderStyles.Keys.CopyTo(renderstyles, 0);
|
|
floorRenderStyle.Enabled = (General.Map.Config.SectorRenderStyles.Count > 0);
|
|
labelfloorrenderstyle.Enabled = (General.Map.Config.SectorRenderStyles.Count > 0);
|
|
ceilRenderStyle.Enabled = (General.Map.Config.SectorRenderStyles.Count > 0);
|
|
labelceilrenderstyle.Enabled = (General.Map.Config.SectorRenderStyles.Count > 0);
|
|
|
|
// Fill renderstyles
|
|
foreach(KeyValuePair<string, string> lf in General.Map.Config.SectorRenderStyles)
|
|
{
|
|
floorRenderStyle.Items.Add(lf.Value);
|
|
ceilRenderStyle.Items.Add(lf.Value);
|
|
}
|
|
|
|
// Fill effects list
|
|
effect.GeneralizedOptions = General.Map.Config.GenEffectOptions; //mxd
|
|
effect.AddInfo(General.Map.Config.SortedSectorEffects.ToArray());
|
|
|
|
// Fill sound sequences list
|
|
soundsequence.Items.Add(NO_SOUND_SEQUENCE);
|
|
soundsequence.Items.AddRange(General.Map.Data.SoundSequences);
|
|
|
|
// Fill damagetype list
|
|
damagetype.Items.Add(NO_DAMAGETYPE);
|
|
damagetype.Items.AddRange(General.Map.Data.DamageTypes);
|
|
|
|
// Fill terrain type lists
|
|
ceilterrain.Items.Add(NO_TERRAIN);
|
|
ceilterrain.Items.AddRange(General.Map.Data.TerrainNames);
|
|
floorterrain.Items.Add(NO_TERRAIN);
|
|
floorterrain.Items.AddRange(General.Map.Data.TerrainNames);
|
|
|
|
// Initialize custom fields editor
|
|
fieldslist.Setup("sector");
|
|
|
|
// Fill universal fields list
|
|
fieldslist.ListFixedFields(General.Map.Config.SectorFields);
|
|
|
|
// Initialize image selectors
|
|
floortex.Initialize();
|
|
ceilingtex.Initialize();
|
|
|
|
// Set steps for brightness field
|
|
brightness.StepValues = General.Map.Config.BrightnessLevels;
|
|
|
|
// Value linking
|
|
ceilScale.LinkValues = linkCeilingScale;
|
|
floorScale.LinkValues = linkFloorScale;
|
|
|
|
cbUseCeilLineAngles.Checked = useCeilLineAngles;
|
|
cbUseFloorLineAngles.Checked = useFloorLineAngles;
|
|
|
|
floorslopecontrol.UseLineAngles = useFloorSlopeLineAngles;
|
|
ceilingslopecontrol.UseLineAngles = useCeilSlopeLineAngles;
|
|
|
|
floorslopecontrol.PivotMode = floorpivotmode;
|
|
ceilingslopecontrol.PivotMode = ceilpivotmode;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Methods
|
|
|
|
// This sets up the form to edit the given sectors
|
|
public void Setup(ICollection<Sector> sectors)
|
|
{
|
|
preventchanges = true; //mxd
|
|
|
|
// Keep this list
|
|
this.sectors = sectors;
|
|
if(sectors.Count > 1) this.Text = "Edit Sectors (" + sectors.Count + ")";
|
|
sectorprops = new Dictionary<Sector, SectorProperties>(sectors.Count); //mxd
|
|
|
|
//mxd. Set default height offset
|
|
heightoffset.Text = "0";
|
|
|
|
CreateHelperProps(sectors); //mxd
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Set all options to the first sector properties
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Get first sector
|
|
Sector sc = General.GetByIndex(sectors, 0);
|
|
|
|
// Flags
|
|
foreach(CheckBox c in flags.Checkboxes)
|
|
if(sc.Flags.ContainsKey(c.Tag.ToString())) c.Checked = sc.Flags[c.Tag.ToString()];
|
|
|
|
// Effects
|
|
effect.Value = sc.Effect;
|
|
brightness.Text = sc.Brightness.ToString();
|
|
|
|
// Floor/ceiling
|
|
floorheight.Text = sc.FloorHeight.ToString();
|
|
ceilingheight.Text = sc.CeilHeight.ToString();
|
|
floortex.TextureName = sc.FloorTexture;
|
|
ceilingtex.TextureName = sc.CeilTexture;
|
|
|
|
//UDMF stuff
|
|
//Texture offsets
|
|
ceilOffsets.SetValuesFrom(sc.Fields, true);
|
|
floorOffsets.SetValuesFrom(sc.Fields, true);
|
|
|
|
//Texture scale
|
|
ceilScale.SetValuesFrom(sc.Fields, true);
|
|
floorScale.SetValuesFrom(sc.Fields, true);
|
|
|
|
//Texture rotation
|
|
float ceilAngle = sc.Fields.GetValue("rotationceiling", 0.0f);
|
|
float floorAngle = sc.Fields.GetValue("rotationfloor", 0.0f);
|
|
|
|
ceilRotation.Text = ceilAngle.ToString();
|
|
floorRotation.Text = floorAngle.ToString();
|
|
|
|
ceilAngleControl.Angle = General.ClampAngle(360 - (int)ceilAngle);
|
|
floorAngleControl.Angle = General.ClampAngle(360 - (int)floorAngle);
|
|
|
|
//Texture brightness
|
|
ceilBrightness.Text = sc.Fields.GetValue("lightceiling", 0).ToString();
|
|
floorBrightness.Text = sc.Fields.GetValue("lightfloor", 0).ToString();
|
|
ceilLightAbsolute.Checked = sc.Fields.GetValue("lightceilingabsolute", false);
|
|
floorLightAbsolute.Checked = sc.Fields.GetValue("lightfloorabsolute", false);
|
|
|
|
//Alpha
|
|
ceilAlpha.Text = General.Clamp(sc.Fields.GetValue("alphaceiling", 1.0f), 0f, 1f).ToString();
|
|
floorAlpha.Text = General.Clamp(sc.Fields.GetValue("alphafloor", 1.0f), 0f, 1f).ToString();
|
|
|
|
//Render style
|
|
ceilRenderStyle.SelectedIndex = Array.IndexOf(renderstyles, sc.Fields.GetValue("renderstyleceiling", "translucent"));
|
|
floorRenderStyle.SelectedIndex = Array.IndexOf(renderstyles, sc.Fields.GetValue("renderstylefloor", "translucent"));
|
|
|
|
//Damage
|
|
damagetype.Text = sc.Fields.GetValue("damagetype", NO_DAMAGETYPE);
|
|
damageamount.Text = sc.Fields.GetValue("damageamount", 0).ToString();
|
|
damageinterval.Text = sc.Fields.GetValue("damageinterval", 32).ToString();
|
|
leakiness.Text = General.Clamp(sc.Fields.GetValue("leakiness", 0), 0, 256).ToString();
|
|
|
|
//Terrain
|
|
ceilterrain.Text = sc.Fields.GetValue("ceilingterrain", NO_TERRAIN);
|
|
floorterrain.Text = sc.Fields.GetValue("floorterrain", NO_TERRAIN);
|
|
|
|
//Misc
|
|
soundsequence.Text = sc.Fields.GetValue("soundsequence", NO_SOUND_SEQUENCE);
|
|
gravity.Text = sc.Fields.GetValue("gravity", 1.0f).ToString();
|
|
desaturation.Text = General.Clamp(sc.Fields.GetValue("desaturation", 0.0f), 0f, 1f).ToString();
|
|
|
|
//Sector colors
|
|
fadeColor.SetValueFrom(sc.Fields);
|
|
lightColor.SetValueFrom(sc.Fields);
|
|
|
|
//Slopes
|
|
SetupFloorSlope(sc, true);
|
|
SetupCeilingSlope(sc, true);
|
|
|
|
// Action
|
|
tagsselector.Setup(UniversalType.SectorTag); //mxd
|
|
tagsselector.SetValue(sc.Tags, true);//mxd
|
|
|
|
// Custom fields
|
|
fieldslist.SetValues(sc.Fields, true);
|
|
|
|
//mxd. Comments
|
|
commenteditor.SetValues(sc.Fields, true);
|
|
|
|
anglesteps = new StepsList();
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Now go for all sectors and change the options when a setting is different
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Go for all sectors
|
|
foreach(Sector s in sectors)
|
|
{
|
|
// Flags
|
|
foreach(CheckBox c in flags.Checkboxes)
|
|
{
|
|
if(c.CheckState == CheckState.Indeterminate) continue; //mxd
|
|
if(s.IsFlagSet(c.Tag.ToString()) != c.Checked)
|
|
{
|
|
c.ThreeState = true;
|
|
c.CheckState = CheckState.Indeterminate;
|
|
}
|
|
}
|
|
|
|
// Effects
|
|
if(s.Effect != effect.Value) effect.Empty = true;
|
|
if(s.Brightness.ToString() != brightness.Text) brightness.Text = "";
|
|
|
|
// Floor/Ceiling
|
|
if(s.FloorHeight.ToString() != floorheight.Text) floorheight.Text = "";
|
|
if(s.CeilHeight.ToString() != ceilingheight.Text) ceilingheight.Text = "";
|
|
if(s.FloorTexture != floortex.TextureName)
|
|
{
|
|
floortex.MultipleTextures = true; //mxd
|
|
floortex.TextureName = "";
|
|
}
|
|
if(s.CeilTexture != ceilingtex.TextureName)
|
|
{
|
|
ceilingtex.MultipleTextures = true; //mxd
|
|
ceilingtex.TextureName = "";
|
|
}
|
|
|
|
//mxd. UDMF stuff
|
|
//Texture offsets
|
|
ceilOffsets.SetValuesFrom(s.Fields, false);
|
|
floorOffsets.SetValuesFrom(s.Fields, false);
|
|
|
|
//Texture scale
|
|
ceilScale.SetValuesFrom(s.Fields, false);
|
|
floorScale.SetValuesFrom(s.Fields, false);
|
|
|
|
//Texture rotation
|
|
if(s.Fields.GetValue("rotationceiling", 0.0f).ToString() != ceilRotation.Text)
|
|
{
|
|
ceilRotation.Text = "";
|
|
ceilAngleControl.Angle = GZBuilder.Controls.AngleControl.NO_ANGLE;
|
|
}
|
|
if(s.Fields.GetValue("rotationfloor", 0.0f).ToString() != floorRotation.Text)
|
|
{
|
|
floorRotation.Text = "";
|
|
floorAngleControl.Angle = GZBuilder.Controls.AngleControl.NO_ANGLE;
|
|
}
|
|
|
|
//Texture brightness
|
|
if(s.Fields.GetValue("lightceiling", 0).ToString() != ceilBrightness.Text) ceilBrightness.Text = "";
|
|
if(s.Fields.GetValue("lightfloor", 0).ToString() != floorBrightness.Text) floorBrightness.Text = "";
|
|
|
|
if(s.Fields.GetValue("lightceilingabsolute", false) != ceilLightAbsolute.Checked)
|
|
{
|
|
ceilLightAbsolute.ThreeState = true;
|
|
ceilLightAbsolute.CheckState = CheckState.Indeterminate;
|
|
}
|
|
if(s.Fields.GetValue("lightfloorabsolute", false) != floorLightAbsolute.Checked)
|
|
{
|
|
floorLightAbsolute.ThreeState = true;
|
|
floorLightAbsolute.CheckState = CheckState.Indeterminate;
|
|
}
|
|
|
|
//Alpha
|
|
if(s.Fields.GetValue("alphaceiling", 1.0f).ToString() != ceilAlpha.Text) ceilAlpha.Text = "";
|
|
if(s.Fields.GetValue("alphafloor", 1.0f).ToString() != floorAlpha.Text) floorAlpha.Text = "";
|
|
|
|
//Render style
|
|
if(ceilRenderStyle.SelectedIndex > -1 && ceilRenderStyle.SelectedIndex != Array.IndexOf(renderstyles, s.Fields.GetValue("renderstyleceiling", "translucent")))
|
|
ceilRenderStyle.SelectedIndex = -1;
|
|
if(floorRenderStyle.SelectedIndex > -1 && floorRenderStyle.SelectedIndex != Array.IndexOf(renderstyles, s.Fields.GetValue("renderstylefloor", "translucent")))
|
|
floorRenderStyle.SelectedIndex = -1;
|
|
|
|
//Damage
|
|
if(damagetype.SelectedIndex > -1 && s.Fields.GetValue("damagetype", NO_DAMAGETYPE) != damagetype.Text)
|
|
damagetype.SelectedIndex = -1;
|
|
if(s.Fields.GetValue("damageamount", 0).ToString() != damageamount.Text) damageamount.Text = "";
|
|
if(s.Fields.GetValue("damageinterval", 32).ToString() != damageinterval.Text) damageinterval.Text = "";
|
|
if(s.Fields.GetValue("leakiness", 0).ToString() != leakiness.Text) leakiness.Text = "";
|
|
|
|
//Terrain
|
|
if(ceilterrain.SelectedIndex > -1 && s.Fields.GetValue("ceilingterrain", NO_TERRAIN) != ceilterrain.Text)
|
|
ceilterrain.SelectedIndex = -1;
|
|
if(floorterrain.SelectedIndex > -1 && s.Fields.GetValue("floorterrain", NO_TERRAIN) != floorterrain.Text)
|
|
floorterrain.SelectedIndex = -1;
|
|
|
|
//Misc
|
|
if(soundsequence.SelectedIndex > -1 && s.Fields.GetValue("soundsequence", NO_SOUND_SEQUENCE) != soundsequence.Text)
|
|
soundsequence.SelectedIndex = -1;
|
|
if(s.Fields.GetValue("gravity", 1.0f).ToString() != gravity.Text) gravity.Text = "";
|
|
if(s.Fields.GetValue("desaturation", 0.0f).ToString() != desaturation.Text) desaturation.Text = "";
|
|
|
|
//Sector colors
|
|
fadeColor.SetValueFrom(s.Fields);
|
|
lightColor.SetValueFrom(s.Fields);
|
|
|
|
//Slopes
|
|
SetupFloorSlope(s, false);
|
|
SetupCeilingSlope(s, false);
|
|
|
|
// Action
|
|
tagsselector.SetValue(s.Tags, false); //mxd
|
|
|
|
// Custom fields
|
|
fieldslist.SetValues(s.Fields, false);
|
|
|
|
//mxd. Comments
|
|
commenteditor.SetValues(s.Fields, false);
|
|
|
|
//mxd. Angle steps
|
|
foreach(Sidedef side in s.Sidedefs)
|
|
{
|
|
int angle;
|
|
if(side.Line.Front != null && side.Index == side.Line.Front.Index)
|
|
angle = General.ClampAngle(270 - side.Line.AngleDeg);
|
|
else
|
|
angle = General.ClampAngle(90 - side.Line.AngleDeg);
|
|
|
|
if(!anglesteps.Contains(angle)) anglesteps.Add(angle);
|
|
}
|
|
}
|
|
|
|
//mxd. Update slope controls
|
|
ceilingslopecontrol.UpdateControls();
|
|
floorslopecontrol.UpdateControls();
|
|
|
|
// Show sector height
|
|
UpdateSectorHeight();
|
|
|
|
//mxd. Update some labels
|
|
labelCeilOffsets.Enabled = ceilOffsets.NonDefaultValue;
|
|
labelCeilScale.Enabled = ceilScale.NonDefaultValue;
|
|
labelFloorOffsets.Enabled = floorOffsets.NonDefaultValue;
|
|
labelFloorScale.Enabled = floorScale.NonDefaultValue;
|
|
|
|
//mxd. Update brightness reset buttons
|
|
resetceillight.Visible = (ceilLightAbsolute.CheckState != CheckState.Unchecked || ceilBrightness.GetResult(0) != 0);
|
|
resetfloorlight.Visible = (floorLightAbsolute.CheckState != CheckState.Unchecked || floorBrightness.GetResult(0) != 0);
|
|
|
|
//mxd. Angle steps
|
|
anglesteps.Sort();
|
|
if(useCeilLineAngles) ceilRotation.StepValues = anglesteps;
|
|
if(useFloorLineAngles) floorRotation.StepValues = anglesteps;
|
|
if(useCeilSlopeLineAngles) ceilingslopecontrol.StepValues = anglesteps;
|
|
if(useFloorSlopeLineAngles) floorslopecontrol.StepValues = anglesteps;
|
|
|
|
//mxd. Comments and Tags
|
|
commenteditor.FinishSetup();
|
|
tagsselector.FinishSetup();
|
|
|
|
preventchanges = false; //mxd
|
|
}
|
|
|
|
//mxd
|
|
private void MakeUndo()
|
|
{
|
|
if(undocreated) return;
|
|
undocreated = true;
|
|
|
|
//mxd. Make undo
|
|
General.Map.UndoRedo.CreateUndo("Edit " + (sectors.Count > 1 ? sectors.Count + " sectors" : "sector"));
|
|
foreach(Sector s in sectors) s.Fields.BeforeFieldsChange();
|
|
}
|
|
|
|
// mxd
|
|
private void CreateHelperProps(ICollection<Sector> sectors)
|
|
{
|
|
slopepivots = new Dictionary<Sector, Vector2D>(sectors.Count);
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(slopepivots.ContainsKey(s)) continue;
|
|
Vector2D pivot = new Vector2D(s.BBox.X + s.BBox.Width / 2, s.BBox.Y + s.BBox.Height / 2);
|
|
globalslopepivot += pivot;
|
|
slopepivots.Add(s, pivot);
|
|
|
|
// Store initial properties
|
|
sectorprops.Add(s, new SectorProperties(s));
|
|
}
|
|
|
|
globalslopepivot /= sectors.Count;
|
|
}
|
|
|
|
// This updates the sector height field
|
|
private void UpdateSectorHeight()
|
|
{
|
|
int delta = int.MinValue;
|
|
|
|
// Check all selected sectors
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(delta == int.MinValue)
|
|
{
|
|
// First sector in list
|
|
delta = s.CeilHeight - s.FloorHeight;
|
|
}
|
|
else if(delta != (s.CeilHeight - s.FloorHeight))
|
|
{
|
|
// We can't show heights because the delta
|
|
// heights for the sectors is different
|
|
delta = int.MinValue;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(delta != int.MinValue)
|
|
{
|
|
sectorheight.Text = delta.ToString();
|
|
sectorheight.Visible = true;
|
|
sectorheightlabel.Visible = true;
|
|
}
|
|
else
|
|
{
|
|
sectorheight.Visible = false;
|
|
sectorheightlabel.Visible = false;
|
|
}
|
|
}
|
|
|
|
//mxd
|
|
private void UpdateCeilingHeight()
|
|
{
|
|
int offset;
|
|
|
|
if(heightoffset.Text == "++" || heightoffset.Text == "--") // Raise or lower by sector height
|
|
{
|
|
int sign = (heightoffset.Text == "++" ? 1 : -1);
|
|
foreach(Sector s in sectors)
|
|
{
|
|
offset = sectorprops[s].CeilHeight - sectorprops[s].FloorHeight;
|
|
s.CeilHeight += offset * sign;
|
|
SynchCeilSlopeOffsetToHeight(s);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
offset = heightoffset.GetResult(0);
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(ceilingheight.Text))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.CeilHeight = sectorprops[s].CeilHeight + offset;
|
|
SynchCeilSlopeOffsetToHeight(s);
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.CeilHeight = ceilingheight.GetResult(sectorprops[s].CeilHeight) + offset;
|
|
SynchCeilSlopeOffsetToHeight(s);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//mxd
|
|
private void UpdateFloorHeight()
|
|
{
|
|
int offset;
|
|
|
|
if(heightoffset.Text == "++" || heightoffset.Text == "--")
|
|
{
|
|
// Raise or lower by sector height
|
|
int sign = (heightoffset.Text == "++" ? 1 : -1);
|
|
foreach(Sector s in sectors)
|
|
{
|
|
offset = sectorprops[s].CeilHeight - sectorprops[s].FloorHeight;
|
|
s.FloorHeight += offset * sign;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
offset = heightoffset.GetResult(0);
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(floorheight.Text))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.FloorHeight = sectorprops[s].FloorHeight + offset;
|
|
SynchFloorSlopeOffsetToHeight(s);
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.FloorHeight = floorheight.GetResult(sectorprops[s].FloorHeight) + offset;
|
|
SynchFloorSlopeOffsetToHeight(s);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//mxd
|
|
private void SynchCeilSlopeOffsetToHeight(Sector s)
|
|
{
|
|
Vector3D center = GetSectorCenter(s, s.CeilHeight, SlopePivotMode.LOCAL);
|
|
Plane p = new Plane(center, s.CeilSlope.GetAngleXY() - Angle2D.PIHALF, s.CeilSlope.GetAngleZ(), false);
|
|
s.CeilSlopeOffset = p.Offset;
|
|
}
|
|
|
|
//mxd
|
|
private void SynchFloorSlopeOffsetToHeight(Sector s)
|
|
{
|
|
Vector3D center = GetSectorCenter(s, s.FloorHeight, SlopePivotMode.LOCAL);
|
|
Plane p = new Plane(center, s.FloorSlope.GetAngleXY() + Angle2D.PIHALF, -s.FloorSlope.GetAngleZ(), true);
|
|
s.FloorSlopeOffset = p.Offset;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Events
|
|
|
|
private void apply_Click(object sender, EventArgs e)
|
|
{
|
|
// Verify the effect
|
|
if((effect.Value < General.Map.FormatInterface.MinEffect) || (effect.Value > General.Map.FormatInterface.MaxEffect))
|
|
{
|
|
General.ShowWarningMessage("Sector effect must be between " + General.Map.FormatInterface.MinEffect + " and " + General.Map.FormatInterface.MaxEffect + ".", MessageBoxButtons.OK);
|
|
return;
|
|
}
|
|
|
|
//mxd
|
|
string[] rskeys = null;
|
|
if(General.Map.Config.SectorRenderStyles.Count > 0)
|
|
{
|
|
rskeys = new string[General.Map.Config.SectorRenderStyles.Count];
|
|
General.Map.Config.SectorRenderStyles.Keys.CopyTo(rskeys, 0);
|
|
}
|
|
|
|
MakeUndo(); //mxd
|
|
|
|
// Go for all sectors
|
|
int tagoffset = 0; //mxd
|
|
foreach(Sector s in sectors)
|
|
{
|
|
// Apply all flags
|
|
foreach(CheckBox c in flags.Checkboxes)
|
|
{
|
|
switch(c.CheckState)
|
|
{
|
|
case CheckState.Checked: s.SetFlag(c.Tag.ToString(), true); break;
|
|
case CheckState.Unchecked: s.SetFlag(c.Tag.ToString(), false); break;
|
|
}
|
|
}
|
|
|
|
// Effects
|
|
if(!effect.Empty) s.Effect = effect.Value;
|
|
|
|
//mxd. Tag
|
|
tagsselector.ApplyTo(s, tagoffset++);
|
|
|
|
// Fields
|
|
fieldslist.Apply(s.Fields);
|
|
|
|
//mxd. Comments
|
|
commenteditor.Apply(s.Fields);
|
|
|
|
// Alpha
|
|
if(!string.IsNullOrEmpty(ceilAlpha.Text))
|
|
{
|
|
float ceilAlphaVal = General.Clamp(ceilAlpha.GetResultFloat(s.Fields.GetValue("alphaceiling", 1.0f)), 0f, 1f);
|
|
UniFields.SetFloat(s.Fields, "alphaceiling", ceilAlphaVal, 1.0f);
|
|
}
|
|
|
|
if(!string.IsNullOrEmpty(floorAlpha.Text))
|
|
{
|
|
float floorAlphaVal = General.Clamp(floorAlpha.GetResultFloat(s.Fields.GetValue("alphafloor", 1.0f)), 0f, 1f);
|
|
UniFields.SetFloat(s.Fields, "alphafloor", floorAlphaVal, 1.0f);
|
|
}
|
|
|
|
//renderstyle
|
|
if(rskeys != null)
|
|
{
|
|
if(ceilRenderStyle.SelectedIndex > -1)
|
|
UniFields.SetString(s.Fields, "renderstyleceiling", rskeys[ceilRenderStyle.SelectedIndex], "translucent");
|
|
if(floorRenderStyle.SelectedIndex > -1)
|
|
UniFields.SetString(s.Fields, "renderstylefloor", rskeys[floorRenderStyle.SelectedIndex], "translucent");
|
|
}
|
|
|
|
//Damage
|
|
if(!string.IsNullOrEmpty(damagetype.Text))
|
|
UniFields.SetString(s.Fields, "damagetype", damagetype.Text, NO_DAMAGETYPE);
|
|
if(!string.IsNullOrEmpty(damageamount.Text))
|
|
UniFields.SetInteger(s.Fields, "damageamount", damageamount.GetResult(s.Fields.GetValue("damageamount", 0)), 0);
|
|
if(!string.IsNullOrEmpty(damageinterval.Text))
|
|
UniFields.SetInteger(s.Fields, "damageinterval", damageinterval.GetResult(s.Fields.GetValue("damageinterval", 32)), 32);
|
|
if(!string.IsNullOrEmpty(leakiness.Text))
|
|
UniFields.SetInteger(s.Fields, "leakiness", General.Clamp(leakiness.GetResult(s.Fields.GetValue("leakiness", 0)), 0, 256), 0);
|
|
|
|
//Terrain
|
|
if(!string.IsNullOrEmpty(ceilterrain.Text))
|
|
UniFields.SetString(s.Fields, "ceilingterrain", ceilterrain.Text, NO_TERRAIN);
|
|
if(!string.IsNullOrEmpty(floorterrain.Text))
|
|
UniFields.SetString(s.Fields, "floorterrain", floorterrain.Text, NO_TERRAIN);
|
|
|
|
// Misc
|
|
if(!string.IsNullOrEmpty(soundsequence.Text))
|
|
UniFields.SetString(s.Fields, "soundsequence", soundsequence.Text, NO_SOUND_SEQUENCE);
|
|
if(!string.IsNullOrEmpty(gravity.Text))
|
|
UniFields.SetFloat(s.Fields, "gravity", gravity.GetResultFloat(s.Fields.GetValue("gravity", 1.0f)), 1.0f);
|
|
if(!string.IsNullOrEmpty(desaturation.Text))
|
|
{
|
|
float val = General.Clamp(desaturation.GetResultFloat(s.Fields.GetValue("desaturation", 0f)), 0f, 1f);
|
|
UniFields.SetFloat(s.Fields, "desaturation", val, 0f);
|
|
}
|
|
|
|
// Clear horizontal slopes
|
|
if((float)Math.Round(Angle2D.RadToDeg(s.FloorSlope.GetAngleZ()), 3) == 90f)
|
|
{
|
|
s.FloorSlope = new Vector3D();
|
|
s.FloorSlopeOffset = float.NaN;
|
|
}
|
|
if((float)Math.Round(Angle2D.RadToDeg(s.CeilSlope.GetAngleZ()), 3) == 270f)
|
|
{
|
|
s.CeilSlope = new Vector3D();
|
|
s.CeilSlopeOffset = float.NaN;
|
|
}
|
|
}
|
|
|
|
// Update the used textures
|
|
General.Map.Data.UpdateUsedTextures();
|
|
|
|
// Store value linking
|
|
linkCeilingScale = ceilScale.LinkValues;
|
|
linkFloorScale = floorScale.LinkValues;
|
|
useCeilLineAngles = cbUseCeilLineAngles.Checked;
|
|
useFloorLineAngles = cbUseFloorLineAngles.Checked;
|
|
useCeilSlopeLineAngles = ceilingslopecontrol.UseLineAngles;
|
|
useFloorSlopeLineAngles = floorslopecontrol.UseLineAngles;
|
|
floorpivotmode = floorslopecontrol.PivotMode;
|
|
ceilpivotmode = ceilingslopecontrol.PivotMode;
|
|
|
|
// Done
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); //mxd
|
|
this.DialogResult = DialogResult.OK;
|
|
this.Close();
|
|
}
|
|
|
|
private void cancel_Click(object sender, EventArgs e)
|
|
{
|
|
//mxd. Let's pretend nothing of this really happened...
|
|
if(undocreated) General.Map.UndoRedo.WithdrawUndo();
|
|
|
|
// Be gone
|
|
this.DialogResult = DialogResult.Cancel;
|
|
this.Close();
|
|
}
|
|
|
|
private void browseeffect_Click(object sender, EventArgs e)
|
|
{
|
|
effect.Value = EffectBrowserForm.BrowseEffect(this, effect.Value);
|
|
}
|
|
|
|
//mxd
|
|
private void SectorEditFormUDMF_FormClosing(object sender, FormClosingEventArgs e)
|
|
{
|
|
location = this.Location;
|
|
activetab = tabs.SelectedIndex;
|
|
}
|
|
|
|
private void SectorEditFormUDMF_HelpRequested(object sender, HelpEventArgs hlpevent)
|
|
{
|
|
General.ShowHelp("w_sectoredit.html");
|
|
hlpevent.Handled = true;
|
|
}
|
|
|
|
private void tabcustom_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
fieldslist.Focus();
|
|
}
|
|
|
|
private void ceilAngleControl_AngleChanged(object sender, EventArgs e)
|
|
{
|
|
ceilRotation.Text = (General.ClampAngle(360 - ceilAngleControl.Angle)).ToString();
|
|
}
|
|
|
|
private void floorAngleControl_AngleChanged(object sender, EventArgs e)
|
|
{
|
|
floorRotation.Text = (General.ClampAngle(360 - floorAngleControl.Angle)).ToString();
|
|
}
|
|
|
|
private void cbUseCeilLineAngles_CheckedChanged(object sender, EventArgs e)
|
|
{
|
|
ceilRotation.ButtonStepsWrapAround = cbUseCeilLineAngles.Checked;
|
|
ceilRotation.StepValues = (cbUseCeilLineAngles.Checked ? anglesteps : null);
|
|
}
|
|
|
|
private void cbUseFloorLineAngles_CheckedChanged(object sender, EventArgs e)
|
|
{
|
|
floorRotation.ButtonStepsWrapAround = cbUseFloorLineAngles.Checked;
|
|
floorRotation.StepValues = (cbUseFloorLineAngles.Checked ? anglesteps : null);
|
|
}
|
|
|
|
private void resetfloorterrain_Click(object sender, EventArgs e)
|
|
{
|
|
floorterrain.Text = NO_TERRAIN;
|
|
}
|
|
|
|
private void floorterrain_TextChanged(object sender, EventArgs e)
|
|
{
|
|
resetfloorterrain.Visible = (floorterrain.Text != NO_TERRAIN);
|
|
}
|
|
|
|
private void floorterrain_MouseDown(object sender, MouseEventArgs e)
|
|
{
|
|
if(floorterrain.Text == NO_TERRAIN) floorterrain.SelectAll();
|
|
}
|
|
|
|
private void resetceilterrain_Click(object sender, EventArgs e)
|
|
{
|
|
ceilterrain.Text = NO_TERRAIN;
|
|
}
|
|
|
|
private void ceilterrain_TextChanged(object sender, EventArgs e)
|
|
{
|
|
resetceilterrain.Visible = (ceilterrain.Text != NO_TERRAIN);
|
|
}
|
|
|
|
private void ceilterrain_MouseDown(object sender, MouseEventArgs e)
|
|
{
|
|
if(ceilterrain.Text == NO_TERRAIN) ceilterrain.SelectAll();
|
|
}
|
|
|
|
private void resetdamagetype_Click(object sender, EventArgs e)
|
|
{
|
|
damagetype.Focus();
|
|
damagetype.Text = NO_DAMAGETYPE;
|
|
}
|
|
|
|
private void damagetype_TextChanged(object sender, EventArgs e)
|
|
{
|
|
resetdamagetype.Visible = (damagetype.Text != NO_DAMAGETYPE);
|
|
}
|
|
|
|
private void damagetype_MouseDown(object sender, MouseEventArgs e)
|
|
{
|
|
if(damagetype.Text == NO_DAMAGETYPE) damagetype.SelectAll();
|
|
}
|
|
|
|
private void resetsoundsequence_Click(object sender, EventArgs e)
|
|
{
|
|
soundsequence.Focus();
|
|
soundsequence.Text = NO_SOUND_SEQUENCE;
|
|
}
|
|
|
|
private void soundsequence_TextChanged(object sender, EventArgs e)
|
|
{
|
|
resetsoundsequence.Visible = (soundsequence.Text != NO_SOUND_SEQUENCE);
|
|
}
|
|
|
|
private void soundsequence_MouseDown(object sender, MouseEventArgs e)
|
|
{
|
|
if(soundsequence.Text == NO_SOUND_SEQUENCE) soundsequence.SelectAll();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Sector Realtime events (mxd)
|
|
|
|
private void ceilingheight_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
UpdateCeilingHeight();
|
|
UpdateSectorHeight();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorheight_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
UpdateFloorHeight();
|
|
UpdateSectorHeight();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void heightoffset_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
UpdateFloorHeight();
|
|
UpdateCeilingHeight();
|
|
UpdateSectorHeight();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void brightness_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(brightness.Text))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.Brightness = sectorprops[s].Brightness;
|
|
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.Brightness = General.Clamp(brightness.GetResult(sectorprops[s].Brightness), General.Map.FormatInterface.MinBrightness, General.Map.FormatInterface.MaxBrightness);
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilingtex_OnValueChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(ceilingtex.TextureName))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.SetCeilTexture(sectorprops[s].CeilTexture);
|
|
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.SetCeilTexture(ceilingtex.GetResult(s.CeilTexture));
|
|
}
|
|
|
|
// Update the used textures
|
|
General.Map.Data.UpdateUsedTextures();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floortex_OnValueChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(floortex.TextureName))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.SetFloorTexture(sectorprops[s].FloorTexture);
|
|
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
s.SetFloorTexture(floortex.GetResult(s.FloorTexture));
|
|
}
|
|
|
|
// Update the used textures
|
|
General.Map.Data.UpdateUsedTextures();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorRotation_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(floorRotation.Text))
|
|
{
|
|
floorAngleControl.Angle = GZBuilder.Controls.AngleControl.NO_ANGLE;
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetFloat(s.Fields, "rotationfloor", sectorprops[s].FloorRotation, 0f);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
floorAngleControl.Angle = (int)General.ClampAngle(360 - floorRotation.GetResultFloat(0));
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetFloat(s.Fields, "rotationfloor", floorRotation.GetResultFloat(sectorprops[s].FloorRotation), 0f);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilRotation_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(ceilRotation.Text))
|
|
{
|
|
ceilAngleControl.Angle = GZBuilder.Controls.AngleControl.NO_ANGLE;
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetFloat(s.Fields, "rotationceiling", sectorprops[s].CeilRotation, 0f);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
ceilAngleControl.Angle = (int)General.ClampAngle(360 - ceilRotation.GetResultFloat(0));
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetFloat(s.Fields, "rotationceiling", ceilRotation.GetResultFloat(sectorprops[s].CeilRotation), 0f);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void lightColor_OnValueChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
lightColor.ApplyTo(s.Fields, sectorprops[s].LightColor);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void fadeColor_OnValueChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
fadeColor.ApplyTo(s.Fields, sectorprops[s].FadeColor);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Ceiling/Floor realtime events (mxd)
|
|
|
|
private void ceilOffsets_OnValuesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
ceilOffsets.ApplyTo(s.Fields, General.Map.FormatInterface.MinTextureOffset, General.Map.FormatInterface.MaxTextureOffset, sectorprops[s].CeilOffsetX, sectorprops[s].CeilOffsetY);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
labelCeilOffsets.Enabled = ceilOffsets.NonDefaultValue;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorOffsets_OnValuesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
floorOffsets.ApplyTo(s.Fields, General.Map.FormatInterface.MinTextureOffset, General.Map.FormatInterface.MaxTextureOffset, sectorprops[s].FloorOffsetX, sectorprops[s].FloorOffsetY);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
labelFloorOffsets.Enabled = floorOffsets.NonDefaultValue;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilScale_OnValuesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
ceilScale.ApplyTo(s.Fields, General.Map.FormatInterface.MinTextureOffset, General.Map.FormatInterface.MaxTextureOffset, sectorprops[s].CeilScaleX, sectorprops[s].CeilScaleY);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
labelCeilScale.Enabled = ceilScale.NonDefaultValue;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorScale_OnValuesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
floorScale.ApplyTo(s.Fields, General.Map.FormatInterface.MinTextureOffset, General.Map.FormatInterface.MaxTextureOffset, sectorprops[s].FloorScaleX, sectorprops[s].FloorScaleY);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
labelFloorScale.Enabled = floorScale.NonDefaultValue;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilBrightness_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(ceilBrightness.Text))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetInteger(s.Fields, "lightceiling", sectorprops[s].CeilBrightness, 0);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
bool absolute = false;
|
|
switch(ceilLightAbsolute.CheckState)
|
|
{
|
|
case CheckState.Indeterminate:
|
|
absolute = s.Fields.GetValue("lightceilingabsolute", false);
|
|
break;
|
|
case CheckState.Checked:
|
|
absolute = true;
|
|
break;
|
|
}
|
|
|
|
int value = General.Clamp(ceilBrightness.GetResult(sectorprops[s].CeilBrightness), (absolute ? 0 : -255), 255);
|
|
UniFields.SetInteger(s.Fields, "lightceiling", value, 0);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
|
|
resetceillight.Visible = (ceilLightAbsolute.CheckState != CheckState.Unchecked || ceilBrightness.Text != "0");
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorBrightness_WhenTextChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//restore values
|
|
if(string.IsNullOrEmpty(floorBrightness.Text))
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
UniFields.SetInteger(s.Fields, "lightfloor", sectorprops[s].FloorBrightness, 0);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else //update values
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
bool absolute = false;
|
|
switch(floorLightAbsolute.CheckState)
|
|
{
|
|
case CheckState.Indeterminate:
|
|
absolute = s.Fields.GetValue("lightfloorabsolute", false);
|
|
break;
|
|
case CheckState.Checked:
|
|
absolute = true;
|
|
break;
|
|
}
|
|
|
|
int value = General.Clamp(floorBrightness.GetResult(sectorprops[s].FloorBrightness), (absolute ? 0 : -255), 255);
|
|
UniFields.SetInteger(s.Fields, "lightfloor", value, 0);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
|
|
resetfloorlight.Visible = (floorLightAbsolute.CheckState != CheckState.Unchecked || floorBrightness.Text != "0");
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilLightAbsolute_CheckedChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
if(ceilLightAbsolute.Checked)
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.Fields["lightceilingabsolute"] = new UniValue(UniversalType.Boolean, true);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else if(ceilLightAbsolute.CheckState == CheckState.Indeterminate)
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(sectorprops[s].CeilLightAbsoulte)
|
|
{
|
|
s.Fields["lightceilingabsolute"] = new UniValue(UniversalType.Boolean, true);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
else if(s.Fields.ContainsKey("lightceilingabsolute"))
|
|
{
|
|
s.Fields.Remove("lightceilingabsolute");
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(s.Fields.ContainsKey("lightceilingabsolute"))
|
|
{
|
|
s.Fields.Remove("lightceilingabsolute");
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
resetceillight.Visible = (ceilLightAbsolute.CheckState != CheckState.Unchecked || ceilBrightness.Text != "0");
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorLightAbsolute_CheckedChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
if(floorLightAbsolute.Checked)
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.Fields["lightfloorabsolute"] = new UniValue(UniversalType.Boolean, true);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
else if(floorLightAbsolute.CheckState == CheckState.Indeterminate)
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(sectorprops[s].FloorLightAbsoulte)
|
|
{
|
|
s.Fields["lightfloorabsolute"] = new UniValue(UniversalType.Boolean, true);
|
|
s.UpdateNeeded = true;
|
|
}
|
|
else if(s.Fields.ContainsKey("lightfloorabsolute"))
|
|
{
|
|
s.Fields.Remove("lightfloorabsolute");
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(s.Fields.ContainsKey("lightfloorabsolute"))
|
|
{
|
|
s.Fields.Remove("lightfloorabsolute");
|
|
s.UpdateNeeded = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
resetfloorlight.Visible = (floorLightAbsolute.CheckState != CheckState.Unchecked || floorBrightness.Text != "0");
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void resetceillight_Click(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
|
|
preventchanges = true;
|
|
|
|
ceilLightAbsolute.Checked = false;
|
|
ceilBrightness.Text = "0";
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(s.Fields.ContainsKey("lightceilingabsolute")) s.Fields.Remove("lightceilingabsolute");
|
|
if(s.Fields.ContainsKey("lightceiling")) s.Fields.Remove("lightceiling");
|
|
}
|
|
|
|
preventchanges = false;
|
|
|
|
resetceillight.Visible = false;
|
|
ceilBrightness.Focus();
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void resetfloorlight_Click(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
|
|
preventchanges = true;
|
|
|
|
floorLightAbsolute.Checked = false;
|
|
floorBrightness.Text = "0";
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
if(s.Fields.ContainsKey("lightfloorabsolute")) s.Fields.Remove("lightfloorabsolute");
|
|
if(s.Fields.ContainsKey("lightfloor")) s.Fields.Remove("lightfloor");
|
|
}
|
|
|
|
preventchanges = false;
|
|
|
|
resetfloorlight.Visible = false;
|
|
floorBrightness.Focus();
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Slope Utility (mxd)
|
|
|
|
private void SetupFloorSlope(Sector s, bool first)
|
|
{
|
|
if(s.FloorSlope.GetLengthSq() > 0)
|
|
{
|
|
float anglexy = General.ClampAngle((float)Math.Round(Angle2D.RadToDeg(s.FloorSlope.GetAngleXY()) - 180, 1));
|
|
float anglez = -(float)Math.Round(Angle2D.RadToDeg(s.FloorSlope.GetAngleZ()) - 90, 1);
|
|
float offset = (float)Math.Round(GetVirtualSlopeOffset(s, floorslopecontrol.PivotMode, true), 1);
|
|
|
|
if(anglexy >= 180 && anglez < 0)
|
|
{
|
|
anglexy -= 180;
|
|
anglez = -anglez;
|
|
}
|
|
|
|
floorslopecontrol.SetValues(anglexy, anglez, offset, first);
|
|
}
|
|
else
|
|
{
|
|
floorslopecontrol.SetValues(0f, 0f, s.FloorHeight, first);
|
|
}
|
|
}
|
|
|
|
private void SetupCeilingSlope(Sector s, bool first)
|
|
{
|
|
if(s.CeilSlope.GetLengthSq() > 0)
|
|
{
|
|
float anglexy = General.ClampAngle((float)Math.Round(Angle2D.RadToDeg(s.CeilSlope.GetAngleXY()) - 180, 1));
|
|
float anglez = -(float)(270 - Math.Round(Angle2D.RadToDeg(s.CeilSlope.GetAngleZ()), 1));
|
|
float offset = (float)Math.Round(GetVirtualSlopeOffset(s, ceilingslopecontrol.PivotMode, false), 1);
|
|
|
|
if(anglexy >= 180 && anglez < 0)
|
|
{
|
|
anglexy -= 180;
|
|
anglez = -anglez;
|
|
}
|
|
|
|
ceilingslopecontrol.SetValues(anglexy, anglez, offset, first);
|
|
}
|
|
else
|
|
{
|
|
ceilingslopecontrol.SetValues(0f, 0f, s.CeilHeight, first);
|
|
}
|
|
}
|
|
|
|
// Gets the offset to be displayed in a SectorSlopeControl
|
|
private float GetVirtualSlopeOffset(Sector s, SlopePivotMode mode, bool floor)
|
|
{
|
|
float offset = (floor ? s.FloorSlopeOffset : s.CeilSlopeOffset);
|
|
if(float.IsNaN(offset))
|
|
{
|
|
offset = (floor ? s.FloorHeight : s.CeilHeight);
|
|
}
|
|
|
|
Vector3D normal = (floor ? s.FloorSlope : s.CeilSlope);
|
|
|
|
if(normal.GetLengthSq() > 0)
|
|
{
|
|
Vector3D center = GetSectorCenter(s, 0, mode);
|
|
Plane p = new Plane(normal, offset);
|
|
return p.GetZ(center);
|
|
}
|
|
|
|
return offset;
|
|
}
|
|
|
|
// Gets the offset to be displayed in a SectorSlopeControl
|
|
private float GetInitialVirtualSlopeOffset(Sector s, SlopePivotMode mode, bool floor)
|
|
{
|
|
float offset = (floor ? sectorprops[s].FloorSlopeOffset : sectorprops[s].CeilSlopeOffset);
|
|
Vector3D normal = (floor ? sectorprops[s].FloorSlope : sectorprops[s].CeilSlope);
|
|
|
|
if(normal.GetLengthSq() > 0)
|
|
{
|
|
Vector3D center = GetSectorCenter(s, 0, mode);
|
|
Plane p = new Plane(normal, offset);
|
|
return p.GetZ(center);
|
|
}
|
|
|
|
return offset;
|
|
}
|
|
|
|
private Vector3D GetSectorCenter(Sector s, float offset, SlopePivotMode mode)
|
|
{
|
|
switch(mode)
|
|
{
|
|
case SlopePivotMode.GLOBAL: //translate from the center of selection
|
|
return new Vector3D(globalslopepivot, offset);
|
|
|
|
case SlopePivotMode.LOCAL: //translate from sector's bounding box center
|
|
return new Vector3D(slopepivots[s], offset);
|
|
|
|
case SlopePivotMode.ORIGIN: //don't translate
|
|
return new Vector3D(0, 0, offset);
|
|
|
|
default:
|
|
throw new NotImplementedException("Unknown SlopePivotMode: " + (int)mode);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Slopes realtime events (mxd)
|
|
|
|
private void ceilingslopecontrol_OnAnglesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//Set or restore values
|
|
foreach(Sector s in sectors)
|
|
{
|
|
float anglexy = General.ClampAngle(ceilingslopecontrol.GetAngleXY(sectorprops[s].CeilSlopeAngleXY) + 270);
|
|
float anglez = -(ceilingslopecontrol.GetAngleZ(sectorprops[s].CeilSlopeAngleZ) + 90);
|
|
|
|
float virtualoffset = GetInitialVirtualSlopeOffset(s, ceilingslopecontrol.PivotMode, false);
|
|
Vector3D center = GetSectorCenter(s, ceilingslopecontrol.GetOffset(virtualoffset), ceilingslopecontrol.PivotMode);
|
|
Plane p = new Plane(center, Angle2D.DegToRad(anglexy), Angle2D.DegToRad(anglez), false);
|
|
s.CeilSlope = p.Normal;
|
|
s.CeilSlopeOffset = p.Offset;
|
|
s.CeilHeight = (int)Math.Round(p.GetZ(center.x, center.y));
|
|
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorslopecontrol_OnAnglesChanged(object sender, EventArgs e)
|
|
{
|
|
if(preventchanges) return;
|
|
MakeUndo(); //mxd
|
|
|
|
//Set or restore values
|
|
foreach(Sector s in sectors)
|
|
{
|
|
float anglexy = General.ClampAngle(floorslopecontrol.GetAngleXY(sectorprops[s].FloorSlopeAngleXY) + 90);
|
|
float anglez = -(floorslopecontrol.GetAngleZ(sectorprops[s].FloorSlopeAngleZ) + 90);
|
|
|
|
float virtualoffset = GetInitialVirtualSlopeOffset(s, floorslopecontrol.PivotMode, true);
|
|
Vector3D center = GetSectorCenter(s, floorslopecontrol.GetOffset(virtualoffset), floorslopecontrol.PivotMode);
|
|
Plane p = new Plane(center, Angle2D.DegToRad(anglexy), Angle2D.DegToRad(anglez), true);
|
|
s.FloorSlope = p.Normal;
|
|
s.FloorSlopeOffset = p.Offset;
|
|
s.FloorHeight = (int)Math.Round(p.GetZ(center.x, center.y));
|
|
|
|
s.UpdateNeeded = true;
|
|
}
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
// Update displayed ceiling offset value
|
|
private void ceilingslopecontrol_OnPivotModeChanged(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
bool first = true;
|
|
foreach(Sector s in sectors)
|
|
{
|
|
SetupCeilingSlope(s, first);
|
|
first = false;
|
|
}
|
|
}
|
|
|
|
// Update displayed floor offset value
|
|
private void floorslopecontrol_OnPivotModeChanged(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
bool first = true;
|
|
foreach(Sector s in sectors)
|
|
{
|
|
SetupFloorSlope(s, first);
|
|
first = false;
|
|
}
|
|
}
|
|
|
|
private void ceilingslopecontrol_OnResetClicked(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
ceilingslopecontrol.SetOffset(General.GetByIndex(sectors, 0).CeilHeight, true);
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.CeilSlope = new Vector3D();
|
|
s.CeilSlopeOffset = float.NaN;
|
|
s.CeilHeight = ceilingheight.GetResult(sectorprops[s].CeilHeight);
|
|
s.UpdateNeeded = true;
|
|
ceilingslopecontrol.SetOffset(s.CeilHeight, false);
|
|
}
|
|
|
|
ceilingslopecontrol.UpdateOffset();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void floorslopecontrol_OnResetClicked(object sender, EventArgs e)
|
|
{
|
|
MakeUndo(); //mxd
|
|
floorslopecontrol.SetOffset(General.GetByIndex(sectors, 0).FloorHeight, true);
|
|
|
|
foreach(Sector s in sectors)
|
|
{
|
|
s.FloorSlope = new Vector3D();
|
|
s.FloorSlopeOffset = float.NaN;
|
|
s.FloorHeight = floorheight.GetResult(sectorprops[s].FloorHeight);
|
|
s.UpdateNeeded = true;
|
|
floorslopecontrol.SetOffset(s.FloorHeight, false);
|
|
}
|
|
|
|
floorslopecontrol.UpdateOffset();
|
|
|
|
General.Map.IsChanged = true;
|
|
if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty);
|
|
}
|
|
|
|
private void ceilingslopecontrol_OnUseLineAnglesChanged(object sender, EventArgs e)
|
|
{
|
|
ceilingslopecontrol.StepValues = (ceilingslopecontrol.UseLineAngles ? anglesteps : null);
|
|
}
|
|
|
|
private void floorslopecontrol_OnUseLineAnglesChanged(object sender, EventArgs e)
|
|
{
|
|
floorslopecontrol.StepValues = (floorslopecontrol.UseLineAngles ? anglesteps : null);
|
|
}
|
|
|
|
#endregion
|
|
|
|
}
|
|
}
|