记录一次:Winform的控件的Visible属性异常问题
一:背景
1. 讲故事
有一次同事找到我,说以下代码中:btnPlanAppend控件:客户电脑显示正常、开发者电脑调试时无法显示
- btnAppend可以在界面中显示出来
- btnPlanAppend控件在界面上就是不显示
private void Check_Privilege()
{
string sPrivilege = ClientUtils.GetPrivilege(g_sUserID, g_sFunction, g_sProgram).ToString();
btnAppend.Visible = SajetCommon.CheckEnabled("INSERT", sPrivilege);
btnPlanAppend.Visible = btnAppend.Visible;
}
二、分析
1. 使用VS调试
- SajetCommon.CheckEnabled("INSERT", sPrivilege)方法的返回结果:true;
- btnAppend控件也能正常显示
- 调试到这一行(btnPlanAppend.Visible = btnAppend.Visible;)btnAppend.Visible属性始终返回false.
2. 分析ToolStripItem控件源码
- 通过此网站https://referencesource.microsoft.com/可以查询.Net Framework源码。
- btnAppend与btnPlanAppend控件,都是ToolStripItem控件,找到Visible属性,代码如下:
public bool Visible {
get {
return (ParentInternal!=null) && (ParentInternal.Visible) && Available;
}
set {
SetVisibleCore(value);
}
}
- 从代码中可以看出,出现问题应该是由于ParentInternal不满足条件。通过Parent英语单词可以看出,ParentInternal属性应该是与此控件的父控件有关系。
- 再看一下ParentInternal属性的代码
internal ToolStrip ParentInternal {
get {
return parent;
}
set {
if (parent != value) {
ToolStrip oldParent = parent;
parent = value;
OnParentChanged(oldParent, value);
}
}
}
- 控件parent就应该是保存父控件引用。
- 第一段代码中Check_Privilege方法,是放在Form_Loaded事件中调用,可以推断,调用Form_Loaded事件时,ToolStripItem还没有真正的现在Form中显示出来,parent应该为null.
3. 分析:现场电脑为什么可以正常显示
- 通过VS的查看,编译环境中:目标框架:.NET Framework 2.0。看到这个好古老。
- 开发者电脑没有.NET Framework 2.0的环境,调试时使用是新的.NET Framework库。
三、总结
1. 修改代码
- 通过分析,既然Form_Loaded事件中,获得到btnAppend.Visible一直False。就不要使用btnAppend.Visible作为中转变量。
- 创建一个临时变量,btnAppend与btnPlanAppend的Visible都有临时变量赋值就可以了。
2. 总结
- .NET Framework 版本很多,版本之间一定有差异,此类问题就是NET Framework差异造成。