在我的申请中,我遇到以下情况:
我有一个带有多个标签的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;
}
}
}
答案 0 :(得分:1)
samplePanel.Size
返回(200,100)。在CreateSamplePanel
方法中,如果您设置samplePanel.MinimumSize = sampleSize;
,那么您的代码就可以使用。
小组不会根据其子控件计算其大小属性(例如Size
,MinimumSize
,PreferredSize
)。您必须继承Panel并提供该行为。即使TableLayoutPanel
和FlowLayoutPanel
也无法正确计算PreferredSize
属性,这是令人惊讶的。通常,您通常会覆盖GetPreferredSize(Size proposedSize)
方法,并可选择让MinimumSize
属性返回PreferredSize
属性。
值得注意的是DockStyle.Fill
和MinimumSize
相互矛盾。 TabPage
控件本身就是DockStyle.Fill
模式,这就是您必须设置AutoScrollMinSize
属性的原因。
编辑:还没有现有的功能可以检索控件列表所需的总大小(递归),例如:通过他们的X / Y和尺寸?
由主机容器本身(例如TableLayoutPanel
)正确计算其PreferredSize
,因为只有它知道其布局执行方式的具体细节。
您可以将AutoSize
属性设置为true,然后希望GetPreferredSize(...)/PreferredSize
计算正确的大小。对于TableLayoutPanel
,我记得有一种情况是它没有正确计算,我必须将其子类化并覆盖GetPreferredSize(...)
方法。除非GetPreferredSize(...)
成立,否则AutoSize
不会被调用。
如果您正在谈论普通Panel
或UserControl
,默认情况下这些会使用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;
}
}