繁体   English   中英

选择选项卡时,TabControl 重新排序 ControlCollection

[英]TabControl reordering ControlCollection when selecting a tab

好吧,我又来了一个奇怪的人。

我有一个正在运行的项目,我正在构建一个给定形式的控件/组件树,并使用它来生成到其中任何一个的最短唯一路径。 就像实际问题的上下文一样。

现在我在处理 TabControl 时遇到了很糟糕的时间,它在很多层面上都做得很糟糕,但大部分都可以通过 WinAPI 调用来处理。 美好的。 现在我遇到了一个问题,即使在查看 Microsoft 参考代码库时也无法弄清楚。

出于某种未知原因,TabControl 会在选择选项卡时根据 ControlCollection 重新排序。 从一个很小的测试应用程序来看,它似乎比我最初预期的要随机得多。 不幸的是,这破坏了我处理未命名控件的后备方法,稳定的索引对我来说非常重要。

using System;
using System.Windows.Forms;

namespace TabControlOrderTest
{
    using System.Threading;
    using System.Threading.Tasks;

    public partial class Form1 : Form
    {
        private TabControl tabControl1;
        private ListBox listBox1;

        public Form1()
        {
            this.tabControl1 = new TabControl();
            this.listBox1 = new ListBox();
            this.InitializeComponent();
            this.SuspendLayout();
            // 
            // tabControl1
            // 
            this.tabControl1.Location = new System.Drawing.Point(12, 12);
            this.tabControl1.Name = "tabControl1";
            this.tabControl1.SelectedIndex = 0;
            this.tabControl1.Size = new System.Drawing.Size(566, 244);
            this.tabControl1.TabIndex = 0;
            this.tabControl1.SelectedIndexChanged += new EventHandler(this.tabControl1_SelectedIndexChanged);
            // 
            // listBox1
            // 
            this.listBox1.FormattingEnabled = true;
            this.listBox1.Location = new System.Drawing.Point(658, 34);
            this.listBox1.Name = "listBox1";
            this.listBox1.Size = new System.Drawing.Size(130, 251);
            this.listBox1.TabIndex = 2;

            this.Controls.Add(this.listBox1);
            this.Controls.Add(this.tabControl1);

            this.ResumeLayout(false);


            for (var i = 0; i < 10; i++)
            {
                this.tabControl1.TabPages.Add(
                    new TabPage($"tabPage{i}")
                    {
                        Name = $"tabPage{i}"
                    });
            }
        }

        protected override void OnShown(EventArgs e)
        {
            base.OnShown(e);

            Task.Run(
                () =>
                {
                    for (var i = 9; i >= 0; i--)
                    {
                        Thread.Sleep(500);
                        this.tabControl1.SelectedIndex = i;
                    }
                });
        }

        private void tabControl1_SelectedIndexChanged(object sender, EventArgs e)
        {
            var pages = this.tabControl1.TabPages;
            var controls = this.tabControl1.Controls;
            this.Text = $"{pages.Count};{controls.Count}";
            this.listBox1.Items.Clear();
            for (int i = 0, count = this.tabControl1.TabCount; i < count; i++)
            {
                if (ReferenceEquals(pages[i], controls[i]))
                {
                    continue;

                }

                this.listBox1.Items.Add($"{pages[i].Name} != {controls[i].Name}");
            }
        }     
    }
}

有时它非常稳定,有时每次点击都会变得更加混乱。 有人可以给我一个提示,为什么会发生这种情况? 如果有任何线索,我将不胜感激。

编辑:我特别询问 Controls 属性,因为这是 Controls 存储任何基础项目的通用位置。 我可以将我的生产代码切换到 TabPageCollection 专门用于 TabControl,但它可能会破坏比修复更多的东西,所以这就是我试图理解这种行为的原因。

好的,我已经找到了这个原因。

它是由 Control.UpdateChildControlIndex(Control ctl) 完成的,它在处理 WmWindowPosChanged 期间被调用

https://referencesource.microsoft.com/System.Windows.Forms/R/13dbf65f74593e7c.html

从第一行可以明显看出,它可以被选择性地禁用。

在 .NET 4.6 之前有点痛苦(因为它需要最丑陋的反射代码才能做到)。 从 4.6+ 开始,您只需调用

AppContext.SetSwitch("Switch.System.Windows.Forms.AllowUpdateChildControlIndexForTabControls", false);

和利润。 希望。

就我而言,我不喜欢为我的编码 UI 测试切换 .NET FW 的某些内部状态的想法,因此我将切换到 TabControl 的不同控件集合,正如我在我的问题中所讨论的那样。 但很高兴在我做之前知道为什么。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM