#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 System.Drawing; using System.Windows.Forms; using CodeImp.DoomBuilder.Map; using CodeImp.DoomBuilder.Data; using CodeImp.DoomBuilder.Config; using CodeImp.DoomBuilder.Geometry; using CodeImp.DoomBuilder.GZBuilder.Data; using CodeImp.DoomBuilder.Types; #endregion namespace CodeImp.DoomBuilder.Windows { /// /// Dialog window that allows viewing and editing of Thing properties. /// internal partial class ThingEditForm : DelayedForm { #region ================== Events public event EventHandler OnValuesChanged; //mxd #endregion #region ================== Variables private ICollection things; private ThingTypeInfo thinginfo; private bool preventchanges; //mxd private static bool useAbsoluteHeight; private List thingProps; //mxd //mxd. Window setup stuff private static Point location = Point.Empty; private static int activeTab; private struct ThingProperties //mxd { //public readonly int Type; public readonly int AngleDoom; public readonly float X; public readonly float Y; public readonly float Z; public ThingProperties(Thing t) { X = t.Position.x; Y = t.Position.y; Z = t.Position.z; //Type = t.Type; AngleDoom = t.AngleDoom; } } #endregion #region ================== Constructor // Constructor public ThingEditForm() { // Initialize InitializeComponent(); //mxd. Widow setup if(location != Point.Empty) { this.StartPosition = FormStartPosition.Manual; this.Location = location; if(activeTab > 0 && activeTab < tabs.TabCount) { tabs.SelectTab(activeTab); } else { activeTab = 0; } } // Fill flags list foreach(KeyValuePair tf in General.Map.Config.ThingFlags) flags.Add(tf.Value, tf.Key); // Fill actions list action.GeneralizedCategories = General.Map.Config.GenActionCategories; action.AddInfo(General.Map.Config.SortedLinedefActions.ToArray()); // Tag/Effects? if(!General.Map.FormatInterface.HasThingAction && !General.Map.FormatInterface.HasThingTag){ tabs.TabPages.Remove(tabeffects); } else { //mxd. Setup script numbers scriptNumbers.Location = arg0.Location; foreach(ScriptItem si in General.Map.NumberedScripts) scriptNumbers.Items.Add(si); scriptNumbers.DropDownWidth = Tools.GetDropDownWidth(scriptNumbers); } // Thing height? posZ.Visible = General.Map.FormatInterface.HasThingHeight; zlabel.Visible = General.Map.FormatInterface.HasThingHeight; cbAbsoluteHeight.Visible = General.Map.FormatInterface.HasThingHeight; //mxd //mxd. Decimals allowed? if(General.Map.FormatInterface.VertexDecimals > 0) { posX.AllowDecimal = true; posY.AllowDecimal = true; posZ.AllowDecimal = true; } // Setup types list thingtype.Setup(); } // This sets up the form to edit the given things public void Setup(ICollection things) { preventchanges = true; // Keep this list this.things = things; if (things.Count > 1) this.Text = "Edit Things (" + things.Count + ")"; hint.Visible = things.Count > 1; //mxd hintlabel.Visible = things.Count > 1; //mxd //mxd. Make undo string undodesc = "thing"; if(things.Count > 1) undodesc = things.Count + " things"; General.Map.UndoRedo.CreateUndo("Edit " + undodesc); //////////////////////////////////////////////////////////////////////// // Set all options to the first thing properties //////////////////////////////////////////////////////////////////////// Thing ft = General.GetByIndex(things, 0); // Set type thingtype.SelectType(ft.Type); // Flags foreach(CheckBox c in flags.Checkboxes) if(ft.Flags.ContainsKey(c.Tag.ToString())) c.Checked = ft.Flags[c.Tag.ToString()]; // Coordination angle.Text = ft.AngleDoom.ToString(); zlabel.Text = useAbsoluteHeight ? "Abs. Z:" : "Z:"; //mxd cbAbsoluteHeight.Checked = useAbsoluteHeight; //mxd //mxd ft.DetermineSector(); int floorheight = (ft.Sector != null ? ft.Sector.FloorHeight :0); posX.Text = ((int)ft.Position.x).ToString(); posY.Text = ((int)ft.Position.y).ToString(); posZ.Text = useAbsoluteHeight ? ((int)ft.Position.z + floorheight).ToString() : ((int)ft.Position.z).ToString(); posX.ButtonStep = General.Map.Grid.GridSize; posY.ButtonStep = General.Map.Grid.GridSize; posZ.ButtonStep = General.Map.Grid.GridSize; // Action/tags action.Value = ft.Action; if(General.Map.FormatInterface.HasThingTag) {//mxd tagSelector.Setup(UniversalType.ThingTag); tagSelector.SetTag(ft.Tag); } arg0.SetValue(ft.Args[0]); arg1.SetValue(ft.Args[1]); arg2.SetValue(ft.Args[2]); arg3.SetValue(ft.Args[3]); arg4.SetValue(ft.Args[4]); //////////////////////////////////////////////////////////////////////// // Now go for all lines and change the options when a setting is different //////////////////////////////////////////////////////////////////////// thingProps = new List(); // Go for all things foreach(Thing t in things) { //mxd. Update sector info t.DetermineSector(); // Type does not match? ThingTypeInfo info = thingtype.GetSelectedInfo(); //mxd if(info != null && info.Index != t.Type) thingtype.ClearSelectedType(); // Flags foreach(CheckBox c in flags.Checkboxes) { if(c.CheckState == CheckState.Indeterminate) continue; //mxd if(t.IsFlagSet(c.Tag.ToString()) != c.Checked) { c.ThreeState = true; c.CheckState = CheckState.Indeterminate; } } // Coordination if(t.AngleDoom.ToString() != angle.Text) angle.Text = ""; //mxd. Position if(((int)t.Position.x).ToString() != posX.Text) posX.Text = ""; if(((int)t.Position.y).ToString() != posY.Text) posY.Text = ""; if (useAbsoluteHeight && t.Sector != null) { if(((int)t.Position.z + t.Sector.FloorHeight).ToString() != posZ.Text) posZ.Text = ""; } else if(((int)t.Position.z).ToString() != posZ.Text){ posZ.Text = ""; } // Action/tags if(t.Action != action.Value) action.Empty = true; if(General.Map.FormatInterface.HasThingTag && t.Tag != ft.Tag) tagSelector.ClearTag(); //mxd if(t.Args[0] != arg0.GetResult(-1)) arg0.ClearValue(); if(t.Args[1] != arg1.GetResult(-1)) arg1.ClearValue(); if(t.Args[2] != arg2.GetResult(-1)) arg2.ClearValue(); if(t.Args[3] != arg3.GetResult(-1)) arg3.ClearValue(); if(t.Args[4] != arg4.GetResult(-1)) arg4.ClearValue(); //mxd. Store initial properties thingProps.Add(new ThingProperties(t)); } preventchanges = false; //mxd. Trigger updates manually... angle_WhenTextChanged(angle, EventArgs.Empty); flags_OnValueChanged(flags, EventArgs.Empty); updateScriptControls(); //mxd //mxd. Set intial script-related values, if required if(Array.IndexOf(GZBuilder.GZGeneral.ACS_SPECIALS, action.Value) != -1) { int a0 = arg0.GetResult(0); if(a0 > 0) { for(int i = 0; i < General.Map.NumberedScripts.Count; i++) { if(General.Map.NumberedScripts[i].Index == a0) { scriptNumbers.SelectedIndex = i; break; } } if(scriptNumbers.SelectedIndex == -1) { scriptNumbers.Items.Add(new ScriptItem(a0, "Script " + a0)); scriptNumbers.SelectedIndex = scriptNumbers.Items.Count - 1; } } else { scriptNumbers.Text = arg0.Text; } } else { scriptNumbers.Text = "0"; } } //mxd private void updateScriptControls() { scriptNumbers.Visible = (Array.IndexOf(GZBuilder.GZGeneral.ACS_SPECIALS, action.Value) != -1); } #endregion #region ================== Events //mxd private void thingtype_OnTypeDoubleClicked() { apply_Click(this, EventArgs.Empty); } // Action changes private void action_ValueChanges(object sender, EventArgs e) { int showaction = 0; ArgumentInfo[] arginfo; // Only when line type is known, otherwise use the thing arguments if(General.Map.Config.LinedefActions.ContainsKey(action.Value)) showaction = action.Value; if((showaction == 0) && (thinginfo != null)) arginfo = thinginfo.Args; else arginfo = General.Map.Config.LinedefActions[showaction].Args; // Change the argument descriptions arg0label.Text = arginfo[0].Title + ":"; arg1label.Text = arginfo[1].Title + ":"; arg2label.Text = arginfo[2].Title + ":"; arg3label.Text = arginfo[3].Title + ":"; arg4label.Text = arginfo[4].Title + ":"; arg0label.Enabled = arginfo[0].Used; arg1label.Enabled = arginfo[1].Used; arg2label.Enabled = arginfo[2].Used; arg3label.Enabled = arginfo[3].Used; arg4label.Enabled = arginfo[4].Used; arg0.ForeColor = (arg0label.Enabled ? SystemColors.WindowText : SystemColors.GrayText); arg1.ForeColor = (arg1label.Enabled ? SystemColors.WindowText : SystemColors.GrayText); arg2.ForeColor = (arg2label.Enabled ? SystemColors.WindowText : SystemColors.GrayText); arg3.ForeColor = (arg3label.Enabled ? SystemColors.WindowText : SystemColors.GrayText); arg4.ForeColor = (arg4label.Enabled ? SystemColors.WindowText : SystemColors.GrayText); arg0.Setup(arginfo[0]); arg1.Setup(arginfo[1]); arg2.Setup(arginfo[2]); arg3.Setup(arginfo[3]); arg4.Setup(arginfo[4]); // Zero all arguments when linedef action 0 (normal) is chosen if(!preventchanges && (showaction == 0)) { //mxd arg0.SetDefaultValue(); arg1.SetDefaultValue(); arg2.SetDefaultValue(); arg3.SetDefaultValue(); arg4.SetDefaultValue(); } if(!preventchanges) updateScriptControls(); //mxd } // Browse Action clicked private void browseaction_Click(object sender, EventArgs e) { action.Value = ActionBrowserForm.BrowseAction(this, action.Value); } // Angle text changes private void angle_WhenTextChanged(object sender, EventArgs e) { if(preventchanges) return; preventchanges = true; anglecontrol.Angle = angle.GetResult(int.MinValue); preventchanges = false; updateAngle(); //mxd } //mxd. Angle control clicked private void anglecontrol_AngleChanged() { if(preventchanges) return; angle.Text = anglecontrol.Angle.ToString(); updateAngle(); } // Apply clicked private void apply_Click(object sender, EventArgs e) { List defaultflags = new List(); // Verify the tag if(General.Map.FormatInterface.HasThingTag) //mxd { tagSelector.ValidateTag();//mxd if(((tagSelector.GetTag(0) < General.Map.FormatInterface.MinTag) || (tagSelector.GetTag(0) > General.Map.FormatInterface.MaxTag))) { General.ShowWarningMessage("Thing tag must be between " + General.Map.FormatInterface.MinTag + " and " + General.Map.FormatInterface.MaxTag + ".", MessageBoxButtons.OK); return; } } // Verify the type if(((thingtype.GetResult(0) < General.Map.FormatInterface.MinThingType) || (thingtype.GetResult(0) > General.Map.FormatInterface.MaxThingType))) { General.ShowWarningMessage("Thing type must be between " + General.Map.FormatInterface.MinThingType + " and " + General.Map.FormatInterface.MaxThingType + ".", MessageBoxButtons.OK); return; } // Verify the action if(General.Map.FormatInterface.HasThingAction && ((action.Value < General.Map.FormatInterface.MinAction) || (action.Value > General.Map.FormatInterface.MaxAction))) { General.ShowWarningMessage("Thing action must be between " + General.Map.FormatInterface.MinAction + " and " + General.Map.FormatInterface.MaxAction + ".", MessageBoxButtons.OK); return; } bool hasAcs = !action.Empty && Array.IndexOf(GZBuilder.GZGeneral.ACS_SPECIALS, action.Value) != -1; //mxd // Go for all the things int tagoffset = 0; //mxd foreach(Thing t in things) { // Coordination if(cbRandomAngle.Checked) t.Rotate(General.Random(0, 359)); //mxd //mxd. Check position float px = General.Clamp(t.Position.x, General.Map.Config.LeftBoundary, General.Map.Config.RightBoundary); float py = General.Clamp(t.Position.y, General.Map.Config.BottomBoundary, General.Map.Config.TopBoundary); if(t.Position.x != px || t.Position.y != py) t.Move(new Vector2D(px, py)); // Apply all flags foreach(CheckBox c in flags.Checkboxes) { if(c.CheckState == CheckState.Checked) t.SetFlag(c.Tag.ToString(), true); else if(c.CheckState == CheckState.Unchecked) t.SetFlag(c.Tag.ToString(), false); } // Action/tags t.Tag = General.Clamp(tagSelector.GetSmartTag(t.Tag, tagoffset++), General.Map.FormatInterface.MinTag, General.Map.FormatInterface.MaxTag); //mxd if (!action.Empty) { t.Action = action.Value; //mxd. Script number handling if(hasAcs) { if(!string.IsNullOrEmpty(scriptNumbers.Text)) { if(scriptNumbers.SelectedItem != null) t.Args[0] = ((ScriptItem)scriptNumbers.SelectedItem).Index; else if(!int.TryParse(scriptNumbers.Text.Trim(), out t.Args[0])) t.Args[0] = 0; if(t.Fields.ContainsKey("arg0str")) t.Fields.Remove("arg0str"); } } else { t.Args[0] = arg0.GetResult(t.Args[0]); if(t.Fields.ContainsKey("arg0str")) t.Fields.Remove("arg0str"); } }else{ t.Args[0] = arg0.GetResult(t.Args[0]); } t.Args[1] = arg1.GetResult(t.Args[1]); t.Args[2] = arg2.GetResult(t.Args[2]); t.Args[3] = arg3.GetResult(t.Args[3]); t.Args[4] = arg4.GetResult(t.Args[4]); // Update settings t.UpdateConfiguration(); } // Set as defaults foreach(CheckBox c in flags.Checkboxes) if(c.CheckState == CheckState.Checked) defaultflags.Add(c.Tag.ToString()); General.Settings.DefaultThingType = thingtype.GetResult(General.Settings.DefaultThingType); General.Settings.DefaultThingAngle = Angle2D.DegToRad((float)angle.GetResult((int)Angle2D.RadToDeg(General.Settings.DefaultThingAngle) - 90) + 90); General.Settings.SetDefaultThingFlags(defaultflags); // Done General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); //mxd this.DialogResult = DialogResult.OK; this.Close(); } // Cancel clicked private void cancel_Click(object sender, EventArgs e) { //mxd. perform undo General.Map.UndoRedo.WithdrawUndo(); // Be gone this.DialogResult = DialogResult.Cancel; this.Close(); } //mxd private void cbAbsoluteHeight_CheckedChanged(object sender, EventArgs e) { if(preventchanges) return; useAbsoluteHeight = cbAbsoluteHeight.Checked; zlabel.Text = (useAbsoluteHeight ? "Abs. Z:" : "Z:"); preventchanges = true; //update label text Thing ft = General.GetByIndex(things, 0); float z = ft.Position.z; if(useAbsoluteHeight && ft.Sector != null) z += ft.Sector.FloorHeight; posZ.Text = z.ToString(); foreach(Thing t in things) { z = t.Position.z; if(useAbsoluteHeight && t.Sector != null) z += t.Sector.FloorHeight; if (posZ.Text != z.ToString()) { posZ.Text = ""; break; } } preventchanges = false; } //mxd private void cbRandomAngle_CheckedChanged(object sender, EventArgs e) { angle.Enabled = !cbRandomAngle.Checked; anglecontrol.Enabled = !cbRandomAngle.Checked; labelAngle.Enabled = !cbRandomAngle.Checked; } //mxd private void ThingEditForm_FormClosing(object sender, FormClosingEventArgs e) { location = this.Location; activeTab = tabs.SelectedIndex; } // Help private void ThingEditForm_HelpRequested(object sender, HelpEventArgs hlpevent) { General.ShowHelp("w_thingeditor.html"); hlpevent.Handled = true; } #endregion #region ================== mxd. Realtime events private void posX_WhenTextChanged(object sender, EventArgs e) { if(preventchanges) return; int i = 0; // Update values foreach(Thing t in things) t.Move(new Vector2D(posX.GetResultFloat(thingProps[i++].X), t.Position.y)); General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); } private void posY_WhenTextChanged(object sender, EventArgs e) { if(preventchanges) return; int i = 0; // Update values foreach(Thing t in things) t.Move(new Vector2D(t.Position.x, posY.GetResultFloat(thingProps[i++].Y))); General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); } private void posZ_WhenTextChanged(object sender, EventArgs e) { if(preventchanges) return; int i = 0; if(string.IsNullOrEmpty(posZ.Text)) { // Restore values foreach(Thing t in things) t.Move(new Vector3D(t.Position.x, t.Position.y, thingProps[i++].Z)); } else { // Update values foreach(Thing t in things) { float z = posZ.GetResultFloat(thingProps[i++].Z); if(useAbsoluteHeight && !posZ.CheckIsRelative() && t.Sector != null) z -= t.Sector.FloorHeight; t.Move(new Vector3D(t.Position.x, t.Position.y, z)); } } General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); } // Selected type changes private void thingtype_OnTypeChanged(ThingTypeInfo value) { thinginfo = value; // Update preview image if(thinginfo != null) { if(thinginfo.Sprite.ToLowerInvariant().StartsWith(DataManager.INTERNAL_PREFIX) && (thinginfo.Sprite.Length > DataManager.INTERNAL_PREFIX.Length)) { General.DisplayZoomedImage(spritetex, General.Map.Data.GetSpriteImage(thinginfo.Sprite).GetBitmap()); } else if((thinginfo.Sprite.Length <= 8) && (thinginfo.Sprite.Length > 0)) { General.DisplayZoomedImage(spritetex, General.Map.Data.GetSpriteImage(thinginfo.Sprite).GetPreview()); } else { spritetex.BackgroundImage = null; } } else { spritetex.BackgroundImage = null; } // Update arguments action_ValueChanges(this, EventArgs.Empty); //mxd. Update things if(preventchanges) return; if(((thingtype.GetResult(0) < General.Map.FormatInterface.MinThingType) || (thingtype.GetResult(0) > General.Map.FormatInterface.MaxThingType))) return; foreach(Thing t in things) { //Set type t.Type = thingtype.GetResult(t.Type); // Update settings t.UpdateConfiguration(); } General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); } //mxd private void updateAngle() { if(preventchanges) return; int i = 0; //restore values if(string.IsNullOrEmpty(angle.Text)) { // Apply rotation foreach(Thing t in things) t.Rotate(thingProps[i++].AngleDoom); } else { //update values // Apply rotation foreach(Thing t in things) t.Rotate(angle.GetResult(thingProps[i++].AngleDoom)); } General.Map.IsChanged = true; if(OnValuesChanged != null) OnValuesChanged(this, EventArgs.Empty); } //mxd private void flags_OnValueChanged(object sender, EventArgs e) { if(preventchanges) return; foreach(KeyValuePair> group in General.Map.Config.ThingFlagsCompare) { if(group.Value.Count < 2) continue; int enabledcount = 0; foreach(CheckBox cb in flags.Checkboxes) { if (group.Value.ContainsKey(cb.Tag.ToString()) && cb.CheckState != CheckState.Unchecked) { enabledcount++; } } if (enabledcount == 0) { switch(group.Key) { case "skills": tooltip.SetToolTip(missingflags, "Thing is not used in any skill level."); break; case "gamemodes": tooltip.SetToolTip(missingflags, "Thing is not used in any game mode."); break; case "classes": tooltip.SetToolTip(missingflags, "Thing is not used by any class."); break; default: tooltip.SetToolTip(missingflags, "At least one '" + group.Key + "' flag should be set."); break; } missingflags.Visible = true; settingsgroup.ForeColor = Color.DarkRed; return; } } missingflags.Visible = false; settingsgroup.ForeColor = SystemColors.ControlText; } #endregion } }