I've built this custom control that has Items inside of it class HeaderItem : Component, INotifyPropertyChanged
. The list where these items is a BindingList<HeaderItem>
I've been having this problem with my custom control lately, where everytime I build the project with the form designer, the control looses all its items but they stay in the form.designer.cs file.
Here's the most important code:
[DefaultEvent("SelectedIndexChanged"), DefaultProperty("Items")]
class Header : Control
{
[field: NonSerialized]
private BindingList<HeaderItem> _items = new BindingList<HeaderItem>();
public Header()
{
_items.ListChanged += (sender, args) =>
{
if (SelectedIndex > Items.Count) SelectedIndex = Items.Count - 1;
Invalidate();
};
}
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
public BindingList<HeaderItem> Items
{
get { return _items; }
}
[field: NonSerialized]
public event EventHandler SelectedIndexChanged;
protected virtual void OnSelectedIndexChanged()
{
EventHandler handler = SelectedIndexChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
}
[DesignTimeVisible(false)]
class HeaderItem : Component, INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected virtual void OnPropertyChanged(string propertyName)
{
PropertyChangedEventHandler handler = PropertyChanged;
if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
}
}
The first problem is that you are telling VS not to serialize it when you use the
.Hidden
setting.Content
indicates the property is an object and so the contents need to be serialized:Your original code was correct in omitting the setter1. You probably do not want something changing or resetting the contents - you already have some liability for that by exposing a
List
object as a property rather than as a collection class1. The VS serializer will not use or need the setter (nor will any Collection Editor), items are added using the Add/AddRange methods:Other Problems and Issues
It is hard to say for certain because the classes posted are clearly simplified for here. For instance,
HeaderItem
probably includes apropertyName
property to save what it is watching. But tweaking it a bit to get it to compile, these issues emerge:A. As noted, a
List
property type can be dangerous.B. You probably need to may want to implement
ShouldSerializexxx
andResetxxx
. These are generally a good idea for this sort of thing.C. Why does
HeaderItem
inherit from Component?It doesnt seem like it needs to be a Component2. I suspect you read somewhere that this was an easy way to let VS/Net create a unique name and serialize it for you without having to write a TypeConverter. That's largely true, but if you delete your
Header
control/object, the attendantHeaderItems
will likely remain orphaned in the form designer:Normally, the user deletes Components from the form tray but yours are hidden. Unlike Components, collection class objects get removed from the designer when the control is removed. So, unless you really need some Component capability, consider changing it.
With simple properties and a parameterless ctor, you probably wont need a TypeConverter, maybe some Attributes. This near-clone has three string properties and does not rely on

Component
:(Click for larger image)
Beforehand, it displays the simple TypeName (or it could be something static like
NewHeaderItem
), then once the property is set, the Name displays.This CodeProject article might seem unrelated, but has some good starting stuff relating to Collections, TypeConverters, Designer Serialization and the like. The Code is in VB, but the concepts should be clear enough. Disclaimer: I wrote the article.
1 See MSDN Guidelines for Collections.
2 Based on whats there.