记录一次:Winform的控件的Visible属性异常问题

发布时间 2023-09-21 16:27:46作者: Light_Yang

一:背景

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差异造成。