我创建了一个派生自System.Windows.Forms.ContextMenuStrip类的类,而不是用户控件,只是一个带有构造函数和一个事件处理程序的普通.cs类。
当我将这个类从工具箱拖到设计器上时,它会为它创建一个私有成员和几个属性,但不会实例化一个对象。
因此,在运行时我得到“对象引用未设置为对象的实例。”,因为设计者从不生成该行:
this.searchGridContextMenu1 = new SearchGridContextMenu();
在InitializeComponent中。
它曾用于生成这一行,事实上,我一直把它从我的Vault存储库中重新插入,但设计师只是“再次”使用它。
更新:我现在尝试使用同一个类创建一个用户控件,它只是这样做有同样的问题。
答案 0 :(得分:6)
我在另一个问题上交叉发表了这条评论,但由于这与此有关,因此又是。
当用户控件无法加载到Visual Studio设计器中时,您需要执行此操作。这些指令适用于vb.net项目,但c#应该类似。此外,在此之前关闭所有打开的窗口(或至少您正在处理的控件的源和设计器文件。)
最后一件事。您应该做的第一件事是确保重新启动visual studio不能解决问题。如果没有,您可以尝试以下步骤。这些说明假定错误的用户控件位于visual studio中的控件库项目中。如果没有,你应该能够稍微调整方向以使其工作,但当控件在自己的项目中时,它会容易得多。
执行以下操作:
注意:这意味着当您运行解决方案时,它将启动另一个Visual Studio实例,而不是实际运行您的解决方案。 Visual Studion的第一个实例(INSTANCE_1)将在您运行时“托管”Visual Studio的第二个实例(INSTANCE_2)。
注意:这将确保INSTANCE_1在任何运行时错误都会发生故障,即使它在try块中命中也是如此。
您应该会发现Visual Studio的INSTANCE_1应该停在导致设计者不加载控件的代码行上。修复代码(这通常意味着在引用对象属性之前测试IsNot Nothing ......但可能意味着其他事情。)
此外,有时我发现控件将在INSTANCE_2中加载,而不是在INSTANCE_1中打破错误。在这种情况下,只需停止调试...关闭INSTANCE_2。保存/重新启动INSTANCE_1,您的问题通常会消失。
教训是这样的。用户控件必须能够加载/引用所有对象及其成员,以便将其加载到设计器中。因此,对于将放置在其他容器上的用户控件,我通常会设计事件来通知父级,而不是尝试将对象推送到子控件中。
希望这有助于将来参考这个老问题。
赛斯
答案 1 :(得分:3)
我读过这个吗? Visual Studio是否删除构造控件实例的行?
你的构造函数是公开的吗?在某些时候,我将构造函数的可访问性从公共更改为内部 - 试图变得更好,阻止讨厌的用户访问他们不应该访问的东西 - 然后我遭受了与您描述的相同的影响,我不得不添加我的构造函数回到InitializeComponent。
我花了几个月才意识到自己的错误......只是一直认为这是Visual Studio中的一个错误。将其改回公众并没有问题。
答案 2 :(得分:1)
继承控件是不够的。这是我必须实现的最低要求:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WinForms
{
class Foo : System.Windows.Forms.ContextMenuStrip
{
public Foo()
{
InitializeComponent();
}
protected override void OnPaint(PaintEventArgs pe)
{
base.OnPaint(pe);
}
/// <summary>
/// Required designer variable.
/// </summary>
private System.ComponentModel.IContainer components = null;
/// <summary>
/// Clean up any resources being used.
/// </summary>
/// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
#region Component Designer generated code
/// <summary>
/// Required method for Designer support - do not modify
/// the contents of this method with the code editor.
/// </summary>
private void InitializeComponent()
{
components = new System.ComponentModel.Container();
}
#endregion
}
}
答案 3 :(得分:1)
当InitializeComponent()与基本和继承的组件交叉时,会发生此问题。解决方案是调用组件的设计者方法。否则将调用基本方法。
public Form() { this.InitializeComponent(); // base.InitializeComponent <-- default one is thisone }
答案 4 :(得分:0)
将我的无参数构造函数从内部更改为公共解决了问题