WinForms按钮:Autosize Maximumsize

时间:2015-07-26 07:59:36

标签: c# winforms button autosize

我想将Buttons添加到FlowLayoutPanel。按钮可能包含较长的文本,文字之间有空格。按钮是Autosize = true和AutoSizeMode = AutoSizeMode.GrowAndShrink。我还将MaximumSize属性设置为(maxwidth,0)。 maxwidth是面板的宽度。因此按钮不会变得太宽。

我看到的是,Button的宽度受MaximumSize属性的限制,但是当文本换行发生时,Button的高度不会自动调整到包装文本的高度。这个问题有解决方案吗?

我也尝试过这样手动调整按钮大小:

using (Graphics cg = this.CreateGraphics()) {
SizeF size = cg.MeasureString(button.Text, button.Font, 200);
  button.Width = (int)size.Width+20;
  button.Height = (int)size.Height+20;
  button.Text = someLongTextWithSpaces;
}

但请注意,我在计算出的尺寸上加了20。它有效,但有没有一种正确的方法来确定这个额外的大小?也许是2x Padding + ?????

几个小时后......

我来到这个版本似乎工作得很好。

using (Graphics cg = this.CreateGraphics()) {
  var fmt = TextFormatFlags.HorizontalCenter | TextFormatFlags.VerticalCenter | TextFormatFlags.WordBreak;
  var prop = new Size(tableLayoutPanel1.Width - 20, 0);
  var size = TextRenderer.MeasureText(button.Text, button.Font, prop, fmt);

  int border = button.Height - button.Font.Height;
  button.Width = (int)size.Width + border;
  button.Height = (int)size.Height + border;
  button.Text = someLongTextWithSpaces;
}

似乎初始按钮高度是边框+字体的高度。所以我计算了边框减去按钮.Height-button.font.Height。

根据Hans的说法,我现在使用TextRenderer.MeasureText。我在没有启用VisualStyles的情况下测试了它并且工作正常。对此有何评论?

3 个答案:

答案 0 :(得分:3)

有一种正确的方法,但它并不是非常微妙。从ButtonRenderer类源代码(绘制按钮文本的Winforms类)反向设计它,您必须使用TextRenderer类来测量文本。并且必须使用VisualStyleRenderer.GetBackgroundContentRectangle()方法来获取有效的绘图边界。请注意,它小于按钮的大小,因为边框和边距取决于所选的视觉样式。

非常重要的问题是将计算出的内容矩形映射回外部按钮大小,并处理未启用视觉样式的旧机器。示例代码似乎达到了正确的大小:

    private static void SetButtonSize(Graphics gr, Button button) {
        VisualStyleElement ButtonElement = VisualStyleElement.Button.PushButton.Normal;
        var visualStyleRenderer = new VisualStyleRenderer(ButtonElement.ClassName, ButtonElement.Part, 0);
        var bounds = visualStyleRenderer.GetBackgroundContentRectangle(gr, button.Bounds);
        var margin =  button.Height - bounds.Height;
        var fmt = TextFormatFlags.HorizontalCenter | TextFormatFlags.VerticalCenter | TextFormatFlags.WordBreak;
        var prop = new Size(bounds.Width, 0);
        var size = TextRenderer.MeasureText(button.Text, button.Font, prop, fmt);
        button.ClientSize = new Size(button.ClientSize.Width, size.Height - margin);
    }

    protected override void OnLoad(EventArgs e) {
        using (var gr = this.CreateGraphics()) {
            SetButtonSize(gr, this.button1);
        }
        base.OnLoad(e);
    }

没有广泛测试角落情况,不能说我推荐这个。

答案 1 :(得分:2)

似乎初始按钮高度是边框+字体的高度。所以我计算了边框减去button.Height-button.font.Height。 (见我原帖的最后一块)

这也适用于启用/禁用VisualStyles。

答案 2 :(得分:0)

您应该通过在文本中添加换行符来控制换行符。自动文本包装不仅仅适用于空格:

button1.Text = "123232131232\r\nfgfdgfdgdfgdfgdf\r\nASDSADSDASD";

或者:

button1.Text = "123232131232" + Environment.NewLine + 
           "fgfdgfdgdfgdfgdf" + Environment.NewLine + "ASDSADSDASD";

如果您更喜欢自动换行,可以尝试使用TextMeasure来确定文字所需的高度,然后相应地设置按钮的高度,但可能需要额外注意..

但我建议考虑使用标签。对于标签来说,包装是开箱即用的。不同大小的巨大按钮是非标准的UI元素。