可滚动区域如何检索正在显示的内容的大小?

时间:2014-12-30 21:27:23

标签: c# winforms scroll

在我的申请中,我遇到以下情况:

我有一个带有多个标签的Tab控件的Windows窗体。每个选项卡都包含在启动时或运行时由其他类添加的任意内容。

我希望设置标签的方式是,只要表单太小而标签面板无法显示所有内容,滚动条就会自动显示。

我到目前为止所做的是设置标签页AutoScroll = true并将AutoScrollMinSize属性设置为面板的大小。

由于专家组Size似乎总是(200,100)独立于其内容,因此无法正常工作。

我已经创建了一个小示例应用程序(下面的代码)来演示该问题。如果您调整表单大小,您将看到仅当表单小于面板(默认大小为(200,100))而不是面板中的文本框(大小为300,150)时才会显示滚动条。如果手动设置AutoScrollMinSize(取消注释第34行),则其行为与预期一致。

问题是:标签页如何检索其中显示内容的实际大小?

我可以通过所有控件递归并尝试自己计算尺寸 - 但这感觉非常糟糕。

PS:请不要建议将面板的大小设置为标签的大小,因为实际的面板要复杂得多。 ; - )


代码:

只需在Visual Studio中创建一个应用程序,并使用以下代码覆盖Program.cs:

using System;
using System.Windows.Forms;

namespace ScrollbarTest
{
    static class Program
    {
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);

            var sampleForm = CreateSampleForm();

            Application.Run(sampleForm);
        }

        private static Form CreateSampleForm()
        {
            var sampleForm = new Form() { };
            var tabControl = new TabControl() { Dock = DockStyle.Fill };
            var tabPage = new TabPage("Test") { AutoScroll = true };
            sampleForm.Controls.Add(tabControl);
            tabControl.TabPages.Add(tabPage);

            var samplePanel = CreateSamplePanel();
            tabPage.Controls.Add(samplePanel);

            // this does not provide the right size
            tabPage.AutoScrollMinSize = samplePanel.Size;

            // uncomment this to make it work
            //tabPage.AutoScrollMinSize = new System.Drawing.Size(300, 150);

            return sampleForm;
        }

        private static Control CreateSamplePanel()
        {
            // As an example, create a panel with a text box with a fixed size.
            var samplePanel = new Panel() { Dock = DockStyle.Fill };
            var sampleSize = new System.Drawing.Size(300, 150);
            var textBox = new TextBox() 
            { 
                Dock = DockStyle.Fill,
                MinimumSize = sampleSize,
                MaximumSize = sampleSize,
                Size = sampleSize
            };
            samplePanel.Controls.Add(textBox);

            return samplePanel;
        }
    }
}

1 个答案:

答案 0 :(得分:1)

samplePanel.Size返回(200,100)。在CreateSamplePanel方法中,如果您设置samplePanel.MinimumSize = sampleSize;,那么您的代码就可以使用。

小组不会根据其子控件计算其大小属性(例如SizeMinimumSizePreferredSize)。您必须继承Panel并提供该行为。即使TableLayoutPanelFlowLayoutPanel也无法正确计算PreferredSize属性,这是令人惊讶的。通常,您通常会覆盖GetPreferredSize(Size proposedSize)方法,并可选择让MinimumSize属性返回PreferredSize属性。

值得注意的是DockStyle.FillMinimumSize相互矛盾。 TabPage控件本身就是DockStyle.Fill模式,这就是您必须设置AutoScrollMinSize属性的原因。

编辑:还没有现有的功能可以检索控件列表所需的总大小(递归),例如:通过他们的X / Y和尺寸?

由主机容器本身(例如TableLayoutPanel)正确计算其PreferredSize,因为只有它知道其布局执行方式的具体细节。

您可以将AutoSize属性设置为true,然后希望GetPreferredSize(...)/PreferredSize计算正确的大小。对于TableLayoutPanel,我记得有一种情况是它没有正确计算,我必须将其子类化并覆盖GetPreferredSize(...)方法。除非GetPreferredSize(...)成立,否则AutoSize不会被调用。

如果您正在谈论普通PanelUserControl,默认情况下这些会使用WYSIWYG LayoutEngine,而不会计算PreferredSize。您可以子类化,然后计算最大control.X + control.Width和高度相同的事物,并将其用作首选大小。

首先尝试将AutoSize设置为true,看看它是否适合您。如果没有,您可能必须覆盖GetPreferredSize(...)方法。这是一个粗略的例子:

   [STAThread]
    static void Main()
    {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);

        var sampleForm = new Form() { AutoScroll = true };

        var panel = new MyPanel() { AutoSize = true, AutoSizeMode = AutoSizeMode.GrowAndShrink, BackColor = Color.LightYellow };

        for (int i = 0; i < 6; i++) {
            for (int j = 0; j < 3; j++) {
                Button b = new Button { Text = "Button" + panel.Controls.Count, AutoSize = true };
                b.Click += delegate {
                    MessageBox.Show("Preferred Size: " + panel.PreferredSize);
                };
                panel.Controls.Add(b, j, i);
            }
        }

        sampleForm.Controls.Add(panel);
        Application.Run(sampleForm);
    }

    private class MyPanel : TableLayoutPanel {
        public override Size MinimumSize {
            get {
                return PreferredSize;
            }
            set {

            }
        }

        public override Size GetPreferredSize(Size proposedSize) {
            Size s = new Size();
            int[] harr = new int[100];//this.RowCount];
            int[] warr = new int[100];//this.ColumnCount];
            foreach (Control c in this.Controls) {
                var cell = this.GetPositionFromControl(c);
                var ps = c.PreferredSize;
                Padding m = c.Margin;
                int w = ps.Width + m.Horizontal;
                int h = ps.Height + m.Vertical;
                if (w > warr[cell.Column])
                    warr[cell.Column] = w;
                if (h > harr[cell.Row])
                    harr[cell.Row] = h;
            }

            foreach (int w in warr)
                s.Width += w;
            foreach (int h in harr)
                s.Height += h;

            return s;
        }
    }