首页 > 解决方案 > 让设计人员为 UserControl 的简单 List 属性生成 AddRange

问题描述

我创建了一个新的 UserControl,如下所示:

    public partial class MyControl : UserControl {
    List<Fruit> _fruits = new List<Fruit>();

    public List<Fruit> Fruits {
        get {
            return _fruits;
        }
        set {
            _fruits = value;
        }
    }

    public UserControl1() {
        InitializeComponent();
    }
}

该类Fruit仅包含两个 get/set 属性,仅此而已:

public class Fruit {
    public bool Edible {
        get;
        set;
    }

    public string Name {
        get;
        set;
    }
}

当我从窗体上的 Visual Studio 工具箱中拖动 MyControl 的实例,然后继续使用 Visual Studio 设计器将Fruit对象添加到FruitsMyControl 的集合中时,我希望设计器生成新Fruit实例并自动将它们添加到Fruits设计器中的集合中- 通过生成对集合AddRangeAdd方法的调用来生成代码。

但是,它不会生成任何AddRange代码将它们添加到MyControlFruits集合中,因此我最终Fruit在代码隐藏中得到了“挥之不去的”实例。我已经尝试将[DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]属性添加到属性中Fruits,但什么也没做。我错过了什么?

标签: c#visual-studiowinformsuser-controlswindows-forms-designer

解决方案


[DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]需要Fruits让设计者知道序列化其内容。此外,该Fruits属性不需要公共设置器:

using System.Collections.Generic;
using System.ComponentModel;
using System.Windows.Forms;
public partial class MyControl : UserControl
{
    public MyControl()
    {
        Fruits = new List<Fruit>();
        InitializeComponent();
    }
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
    public List<Fruit> Fruits { get; private set; }
}

结果将生成以下代码:

Sample.Fruit fruit1 = new Sample.Fruit();
Sample.Fruit fruit2 = new Sample.Fruit();
Sample.Fruit fruit3 = new Sample.Fruit();
this.myControl1 = new Sample.MyControl();
// 
// myControl1
// 
fruit1.Edible = true;
fruit1.Name = "Apple";
fruit2.Edible = true;
fruit2.Name = "Orange";
fruit3.Edible = true;
fruit3.Name = "Banana";
this.myControl1.Fruits.Add(fruit1);
this.myControl1.Fruits.Add(fruit2);
this.myControl1.Fruits.Add(fruit3);

更清洁的设计器生成的代码

如果您想生成更清晰的代码,如下所示:

this.myControl1 = new Sample.MyControl();
// 
// myControl1
// 
this.myControl1.Fruits.Add(new Sample.Fruit(true, "Apple"));
this.myControl1.Fruits.Add(new Sample.Fruit(true, "Orange"));
this.myControl1.Fruits.Add(new Sample.Fruit(true, "Banana"));

您需要TypeConverter为您的Fruit类创建一个用于InstanceDescriptor创建该类的实例的类:

using System;
using System.ComponentModel;
using System.ComponentModel.Design.Serialization;
using System.Globalization;
[TypeConverter(typeof(FruitConverter))]
public class Fruit
{
    public Fruit() { }
    public Fruit(bool edible, string name) : this()
    {
        Edible = edible;
        Name = name;
    }
    public bool Edible { get; set; }
    public string Name { get; set; }
}
public class FruitConverter : TypeConverter
{
    public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
    {
        if (destinationType == typeof(InstanceDescriptor)) return true;
        return base.CanConvertTo(context, destinationType);
    }
    public override object ConvertTo(ITypeDescriptorContext context, 
        CultureInfo culture, object value, Type destinationType)
    {
        if (destinationType == typeof(InstanceDescriptor)) {
            var ci = typeof(Fruit).GetConstructor(new Type[] { 
                typeof(bool), typeof(string) });
            var t = (Fruit)value;
            return new InstanceDescriptor(ci, new object[] { t.Edible, t.Name });
        }
        return base.ConvertTo(context, culture, value, destinationType);
    }
}

推荐阅读