mirror of
https://git.do.srb2.org/STJr/UltimateZoneBuilder.git
synced 2024-11-30 15:41:30 +00:00
3a35b7603a
Fixed, Map Analysis mode: fixed a crash when trying to dissolve an invalid sector when one of it's linedefs referenced it on the both sides. Fixed, Sectors mode: fixed incorrect undo description when deleting sectors. Internal: joined declaration and assignment of some more variables.
351 lines
9.1 KiB
C#
351 lines
9.1 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 System.Globalization;
|
|
using System.Windows.Forms;
|
|
using CodeImp.DoomBuilder.Config;
|
|
|
|
#endregion
|
|
|
|
namespace CodeImp.DoomBuilder.Windows
|
|
{
|
|
internal partial class ActionBrowserForm : DelayedForm
|
|
{
|
|
// Constants
|
|
private const int MAX_OPTIONS = 8;
|
|
|
|
// Variables
|
|
private int selectedaction;
|
|
private readonly ComboBox[] options;
|
|
private readonly Label[] optionlbls;
|
|
private readonly TreeNode[] allNodes; //mxd
|
|
|
|
// Properties
|
|
public int SelectedAction { get { return selectedaction; } }
|
|
|
|
// Constructor
|
|
public ActionBrowserForm(int action)
|
|
{
|
|
// Initialize
|
|
InitializeComponent();
|
|
|
|
// Make array references for controls
|
|
options = new[] { option0, option1, option2, option3, option4, option5, option6, option7 };
|
|
optionlbls = new[] { option0label, option1label, option2label, option3label, option4label,
|
|
option5label, option6label, option7label };
|
|
|
|
// Show prefixes panel only for doom type maps
|
|
if(!General.Map.FormatInterface.HasBuiltInActivations)
|
|
{
|
|
prefixespanel.Visible = false;
|
|
actions.Height += prefixespanel.Height + 3; //mxd
|
|
}
|
|
|
|
// Go for all predefined categories
|
|
CreateActionCategories(action);
|
|
allNodes = new TreeNode[actions.Nodes.Count];
|
|
actions.Nodes.CopyTo(allNodes, 0);
|
|
|
|
// Using generalized actions?
|
|
if(General.Map.Config.GeneralizedActions)
|
|
{
|
|
// Add for all generalized categories to the combobox
|
|
category.Items.AddRange(General.Map.Config.GenActionCategories.ToArray());
|
|
|
|
// Given action is generalized?
|
|
if(GameConfiguration.IsGeneralized(action, General.Map.Config.GenActionCategories))
|
|
{
|
|
// Open the generalized tab
|
|
tabs.SelectedTab = tabgeneralized;
|
|
|
|
// Select category
|
|
foreach(GeneralizedCategory ac in category.Items)
|
|
if((action >= ac.Offset) && (action < (ac.Offset + ac.Length))) category.SelectedItem = ac;
|
|
|
|
// Anything selected?
|
|
if(category.SelectedIndex > -1)
|
|
{
|
|
// Go for all options in selected category
|
|
GeneralizedCategory sc = category.SelectedItem as GeneralizedCategory;
|
|
int actionbits = action - sc.Offset;
|
|
for(int i = 0; i < MAX_OPTIONS; i++)
|
|
{
|
|
// Option used?
|
|
if(i < sc.Options.Count)
|
|
{
|
|
// Go for all bits
|
|
foreach(GeneralizedBit ab in sc.Options[i].Bits)
|
|
{
|
|
// Select this setting if matches
|
|
if((actionbits & ab.Index) == ab.Index) options[i].SelectedItem = ab;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Remove generalized tab
|
|
tabs.TabPages.Remove(tabgeneralized);
|
|
}
|
|
}
|
|
|
|
// This browses for an action
|
|
// Returns the new action or the same action when cancelled
|
|
public static int BrowseAction(IWin32Window owner, int action)
|
|
{
|
|
ActionBrowserForm f = new ActionBrowserForm(action);
|
|
if(f.ShowDialog(owner) == DialogResult.OK) action = f.SelectedAction;
|
|
f.Dispose();
|
|
return action;
|
|
}
|
|
|
|
//mxd
|
|
private void CreateActionCategories(int action)
|
|
{
|
|
actions.BeginUpdate();
|
|
actions.ShowLines = true;
|
|
foreach(LinedefActionCategory ac in General.Map.Config.ActionCategories)
|
|
{
|
|
// Empty category names will not be created
|
|
// (those actions will go in the root of the tree)
|
|
if(ac.Title.Length > 0)
|
|
{
|
|
// Create category
|
|
TreeNode cn = actions.Nodes.Add(ac.Title);
|
|
foreach(LinedefActionInfo ai in ac.Actions)
|
|
{
|
|
// Create action
|
|
TreeNode n = cn.Nodes.Add(ai.Title);
|
|
n.Tag = ai;
|
|
|
|
// This is the given action?
|
|
if(ai.Index == action)
|
|
{
|
|
// Select this and expand the category
|
|
cn.Expand();
|
|
actions.SelectedNode = n;
|
|
n.EnsureVisible();
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Put actions in the tree root
|
|
foreach(LinedefActionInfo ai in ac.Actions)
|
|
{
|
|
// Create action
|
|
TreeNode n = actions.Nodes.Add(ai.Title);
|
|
n.Tag = ai;
|
|
}
|
|
}
|
|
}
|
|
actions.EndUpdate();
|
|
}
|
|
|
|
//mxd
|
|
private void FilterActions(string p)
|
|
{
|
|
List<TreeNode> filteredNodes = new List<TreeNode>();
|
|
|
|
foreach(TreeNode n in allNodes)
|
|
{
|
|
foreach(TreeNode cn in n.Nodes)
|
|
{
|
|
LinedefActionInfo ai = cn.Tag as LinedefActionInfo;
|
|
if(ai.Title.ToLowerInvariant().IndexOf(p) != -1)
|
|
filteredNodes.Add(cn);
|
|
}
|
|
}
|
|
|
|
actions.BeginUpdate();
|
|
actions.Nodes.Clear();
|
|
actions.ShowLines = false;
|
|
actions.Nodes.AddRange(filteredNodes.ToArray());
|
|
actions.EndUpdate();
|
|
}
|
|
|
|
// OK clicked
|
|
private void apply_Click(object sender, EventArgs e)
|
|
{
|
|
// Presume no result
|
|
selectedaction = 0;
|
|
|
|
// Predefined action?
|
|
if(tabs.SelectedTab == tabactions)
|
|
{
|
|
// Action node selected?
|
|
if((actions.SelectedNode != null) && (actions.SelectedNode.Tag is LinedefActionInfo))
|
|
{
|
|
// Our result
|
|
selectedaction = (actions.SelectedNode.Tag as LinedefActionInfo).Index;
|
|
}
|
|
}
|
|
// Generalized action
|
|
else
|
|
{
|
|
// Category selected?
|
|
if(category.SelectedIndex > -1)
|
|
{
|
|
// Add category bits and go for all options
|
|
GeneralizedCategory sc = category.SelectedItem as GeneralizedCategory;
|
|
selectedaction = sc.Offset;
|
|
for(int i = 0; i < MAX_OPTIONS; i++)
|
|
{
|
|
// Option used?
|
|
if(i < sc.Options.Count)
|
|
{
|
|
// Add selected bits
|
|
if(options[i].SelectedIndex > -1)
|
|
selectedaction += (options[i].SelectedItem as GeneralizedBit).Index;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Done
|
|
this.DialogResult = DialogResult.OK;
|
|
this.Close();
|
|
}
|
|
|
|
// Cancel clicked
|
|
private void cancel_Click(object sender, EventArgs e)
|
|
{
|
|
// Leave
|
|
this.DialogResult = DialogResult.Cancel;
|
|
this.Close();
|
|
}
|
|
|
|
// Generalized category selected
|
|
private void category_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
// Category selected?
|
|
if(category.SelectedIndex > -1)
|
|
{
|
|
// Get the category
|
|
GeneralizedCategory ac = category.SelectedItem as GeneralizedCategory;
|
|
|
|
// Go for all options
|
|
for(int i = 0; i < MAX_OPTIONS; i++)
|
|
{
|
|
// Option used in selected category?
|
|
if(i < ac.Options.Count)
|
|
{
|
|
// Setup controls
|
|
optionlbls[i].Text = ac.Options[i].Name + ":";
|
|
options[i].Items.Clear();
|
|
options[i].Items.AddRange(ac.Options[i].Bits.ToArray());
|
|
|
|
// Show option
|
|
options[i].Visible = true;
|
|
optionlbls[i].Visible = true;
|
|
}
|
|
else
|
|
{
|
|
// Hide option
|
|
options[i].Visible = false;
|
|
optionlbls[i].Visible = false;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Hide all options
|
|
for(int i = 0; i < MAX_OPTIONS; i++)
|
|
{
|
|
options[i].Visible = false;
|
|
optionlbls[i].Visible = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Double clicking on item
|
|
private void actions_DoubleClick(object sender, EventArgs e)
|
|
{
|
|
// Action node selected?
|
|
if((actions.SelectedNode != null) && (actions.SelectedNode.Tag is LinedefActionInfo))
|
|
{
|
|
if(apply.Enabled) apply_Click(this, EventArgs.Empty);
|
|
}
|
|
}
|
|
|
|
//mxd
|
|
private void tbFilter_TextChanged(object sender, EventArgs e)
|
|
{
|
|
if(!string.IsNullOrEmpty(tbFilter.Text.Trim()))
|
|
{
|
|
FilterActions(tbFilter.Text);
|
|
}
|
|
else
|
|
{
|
|
actions.Nodes.Clear();
|
|
CreateActionCategories(actions.SelectedNode != null ? ((LinedefActionInfo)actions.SelectedNode.Tag).Index : 0);
|
|
}
|
|
}
|
|
|
|
//mxd. Switch focus to actions list?
|
|
private void tbFilter_KeyUp(object sender, KeyEventArgs e)
|
|
{
|
|
if(e.KeyCode == Keys.Down && actions.Nodes.Count > 0)
|
|
{
|
|
actions.SelectedNode = actions.Nodes[0];
|
|
actions.Focus();
|
|
}
|
|
}
|
|
|
|
//mxd
|
|
private void btnClearFilter_Click(object sender, EventArgs e)
|
|
{
|
|
tbFilter.Clear();
|
|
}
|
|
|
|
//mxd
|
|
private void ActionBrowserForm_Shown(object sender, EventArgs e)
|
|
{
|
|
if(tabs.SelectedTab == tabactions) tbFilter.Focus();
|
|
}
|
|
|
|
//mxd
|
|
private void actions_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
actions.Focus();
|
|
}
|
|
|
|
//mxd. Transfer focus to Filter textbox
|
|
private void actions_KeyPress(object sender, KeyPressEventArgs e)
|
|
{
|
|
tbFilter.Focus();
|
|
if(e.KeyChar == '\b') // Any better way to check for Backspace?..
|
|
{
|
|
if(!string.IsNullOrEmpty(tbFilter.Text) && tbFilter.SelectionStart > 0 && tbFilter.SelectionLength == 0)
|
|
{
|
|
int s = tbFilter.SelectionStart - 1;
|
|
tbFilter.Text = tbFilter.Text.Remove(s, 1);
|
|
tbFilter.SelectionStart = s;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
tbFilter.AppendText(e.KeyChar.ToString(CultureInfo.InvariantCulture));
|
|
}
|
|
}
|
|
}
|
|
}
|