是列表<>变量可能吗?

时间:2011-11-11 06:44:55

标签: c# list variables object foreach

这可能是一个很长的镜头,但我正在努力减少我正在进行的程序中的重复,并且遇到了麻烦。从下面的ClearTextBoxes()方法中可以看出,我有一个非常重复的代码,我宁愿放在foreach循环中以简洁。 (原来foreach (object box in customBoxes)循环不存在)。我尝试使用以下列表执行此操作,但无济于事。我不确定这是不可能的,或者我只是做错了。我很感激你能给予的任何帮助,如果不能这样做,那么我该如何缩小这个代码块?

谢谢!

List<object> customBoxes = new List<object>();

customBoxes.AddRange(new[] { "TextBox", "DateBox", "DigitBox", "PhoneBox", "WaterTextBox" });



public void ClearTextBoxes()
    {
        ChildControls ccChildren = new ChildControls();

        foreach (object o in ccChildren.GetChildren(rvraDockPanel, 2))
        {
            foreach (object box in customBoxes)
            {
                if (o.GetType() == typeof(TextBox))
                {
                    TextBox txt = (TextBox)o;
                    txt.Text = "";
                }

                if (o.GetType() == typeof(DigitBox))
                {
                    DigitBox digit = (DigitBox)o;
                    digit.Text = "";
                }

                if (o.GetType() == typeof(PhoneBox))
                {
                    PhoneBox phone = (PhoneBox)o;
                    phone.Text = "";
                }

                if (o.GetType() == typeof(DateBox))
                {
                    DateBox date = (DateBox)o;
                    date.Text = "";
                }

                if (o.GetType() == typeof(WatermarkTextBox))
                {
                    WatermarkTextBox water = (WatermarkTextBox)o;
                    water.Text = "";
                }
            }
        }
    }

5 个答案:

答案 0 :(得分:1)

List<Type> customBoxes = new List<Type>();

customBoxes.AddRange(new[] { typeof(PhoneBox), typeof(DigitBox), ....." });

foreach (Control c in this.Controls)
{
  if (customBoxes.Contains(c.GetType()))
  {
    c.Text = string.Empty;
  }
}

答案 1 :(得分:0)

您可以通过某种方式使用反射来模仿某些管道类型行为,但我不会去寻找解决方案,因为它不具备高效性和丑陋性。

        foreach (object box in customBoxes) 
        { 
            var boxType = box.GetType();
            var textProperty = boxType.GetProperty("Text");
            if (textProperty != null && textProperty.CanWrite)
            {
                textProperty.SetValue(box, "", null);
            }
        }

或者您可以使用动态来实现相同的结果:

      foreach (dynamic box in customBoxes)
      {
           box.Text = "";
      }

要做的就是让你的自定义控件实现一个接口IWithTextProperty,其中哪个接口暴露了text属性。

答案 2 :(得分:0)

我会使用ClearText()方法创建interface

interface IClearable
{
  public void ClearText();
}

然后,您可以从每个控件继承并应用该接口:

class ClearableDigitBox : DigitBox, IClearable
{
  public void ClearText() {
    Text = String.Empty;
  }
}
// etc...

所以它只是:

var list = new List<IClearable>;
// ...
foreach (IClearable control in list) control.ClearText();

答案 3 :(得分:0)

Aren将所有输入框都作为Control对象的一部分吗?

如果是这样,并且您想要清除控件中的所有文本 那么我可能会有一个方法:

public void ClearText(List<Control> items)
    {
        foreach (Control control in items)
        {
            control.Text = string.Empty;
        }
    }

如果您只想找到特定类型的控件

public void ClearText(List<Control> items)
        {
            foreach (Control control in items)
            {
                if (control is TextBox)
                    ((TextBox)control).Text = string.Empty;
                else if (control is DigitBox)
                    ((DigitBox)control).Text = string.Empty;
                else
                { // Handle anything else.}
            }
        }

答案 4 :(得分:0)

对于到目前为止的几个回复,这是我对自定义框的类文件。 NumberTextBox类是VS添加的默认代码段。我没用过它,也没有删除它。除了DateBox(折叠以节省空间)类之外,还有一个继承自DigitBox的PhoneBox类。 DigitBox继承的WatermarkTextBox类位于WpfToolkit.Extended.dll中。这些类中唯一真正的区别是每个类都添加了一个允许/禁止按下格式键的方法(括号,句点,连字符等)。

这个类基本上是由于尝试合并我在网络上发现的几个不同的片段而产生的,但这些盒子的目的是启用水印并限制可以输入到这些盒子中的字符。

public class NumberTextBox : Control
{
    static NumberTextBox()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(NumberTextBox), new FrameworkPropertyMetadata(typeof(NumberTextBox)));
    }

}    

public class DigitBox : WatermarkTextBox, IClearable
{
    #region Constructors
    ///<summary>
    ///The default constructor
    /// </summary>
    public DigitBox()
    {
        TextChanged += new TextChangedEventHandler(OnTextChanged);
        KeyDown += new KeyEventHandler(OnKeyDown);
        PreviewKeyDown += new KeyEventHandler(OnPreviewDown);
    }
    #endregion

    #region Properties
    new public String Text
    {
        get { return base.Text; }
        set
        {
            base.Text = LeaveOnlyNumbers(value);
        }
    }
    #endregion

    #region Functions
    public bool IsNumberKey(Key inKey)
    {
        if (inKey < Key.D0 || inKey > Key.D9)
        {
            if (inKey < Key.NumPad0 || inKey > Key.NumPad9)
            {
                return false;
            }
        }
        return true;
    }

    public bool IsActionKey(Key inKey)
    {
        return inKey == Key.Delete || inKey == Key.Back || inKey == Key.Tab || inKey == Key.Return;
    }

    public string LeaveOnlyNumbers(String inString)
    {
        String tmp = inString;
        foreach (char c in inString.ToCharArray())
        {
            if (!IsDigit(c))
            {
                tmp = tmp.Replace(c.ToString(), "");
            }
        }
        return tmp;
    }

    public bool IsSpaceKey(Key inKey)
    {
        if (inKey == Key.Space)
        {
            return true;
        }
        return false;
    }

    public bool IsDigit(char c)
    {
        return (c >= '0' || c <='9');
    }
    #endregion

    #region Event Functions
    protected virtual void OnKeyDown(object sender, KeyEventArgs e)
    {
        e.Handled = !IsNumberKey(e.Key) && !IsActionKey(e.Key) && !IsSpaceKey(e.Key);
    }

    protected virtual void OnTextChanged(object sender, TextChangedEventArgs e)
    {
        base.Text = LeaveOnlyNumbers(Text);
    }

    protected virtual void OnPreviewDown(object sender, KeyEventArgs e)
    {
        if (e.Key == Key.Space)
        {
            e.Handled = true;
        }
    }
    #endregion
}


public class DateBox : DigitBox