2

我有一个自定义控件,将来可能会有用户自定义的字体(缩放已经实现)。我必须在两位数下填充一个矩形,形成一个以 10 为基数的数字。我对零、一个或两个数字有不同的颜色。

使用字体 {Name = Microsoft Sans SerifSize= 16} 和以下Graphics.MeasureString方法调用:

g.MeasureString("00", Font);
g.MeasureString("0", Font);

我得到:

  • “00”的大小是{Width = 31.5486088Height = 26.8124962}
  • “0”的大小是{Width = 19.3298588Height = 26.8124962}

“0”的宽度比“00”的一半大得多。

我知道方法Graphics.MeasureString,它有很多重载,我也知道 StringFormat 类。如何正确计算“0”字符的宽度?

因为字体将是用户可自定义的,所以我不想使用等宽字体来解决问题。

如果我使用以下调用:

g.MeasureString("00", Font, 999, StringFormat.GenericTypographic);
g.MeasureString("0", Font, 999, StringFormat.GenericTypographic);

“0”的宽度似乎是“00”宽度的一半,但是用较小的字体大小绘制时数字重叠:

截屏

更新:在 UserControl 的 OnPaint 方法中,我有以下代码:

Graphics g = e.Graphics;

int[] indices = { 0, 1 };
CharacterRange[] charRanges = new CharacterRange[indices.Length];
for (int chx = 0; chx < indices.Length; ++chx)
{
    charRanges[chx] = new CharacterRange(indices[chx], 1);
}

StringFormat sf = new StringFormat(StringFormat.GenericDefault);
sf.SetMeasurableCharacterRanges(charRanges);

Region[] regions = e.Graphics.MeasureCharacterRanges("01", Font, e.ClipRectangle, sf);

RectangleF[] r = new RectangleF[regions.Length];
int i = 0;
foreach (Region rr in regions)
{
    r[i] = rr.GetBounds(g);
    g.DrawRectangle(Pens.Blue, r[i].X, r[i].Y, r[i].Width, r[i].Height);
    ++i;
}

g.DrawString("0", Font, Brushes.Black, r[0], sf);
g.DrawString("1", Font, Brushes.Black, r[1], sf);

字体是{Name = "Microsoft Sans Serif" Size=25}。运行程序时,这是可见的:

截图 2

我想让数字以蓝色矩形为中心。矩形在 UserControl 中必须尽可能大,但也要​​为 UserControl 的高度留出一定百分比的空间。字体应该适应矩形。

4

1 回答 1

1

需要进行一些小的调整才能按预期进行这项工作:

  1. TextRenderingHint.ClearTypeGridFit在渲染文本时给出更好的结果。
    它更精确,并且与Graphics.DrawString.
    有关此问题的更多信息,请参阅您可以在下面链接的答案中找到的注释。
  2. StringFormat在水平和垂直维度上对齐。
  3. 一种允许绘制任意长度字符串的修改方法。
    如果字符串大于容器,它将使用当前设置进行包装。
  4. 无关:BrushPen在 Paint 事件之外声明,以便在需要时允许重新定义。

这里的不同实现MeasureCharacterRanges
How to highlight Wrapped text in a control

关于Graphics.DrawStringTextRenderingHint.ClearTypeGridFit
在位图上绘制长字符串会导致绘制问题


字体 48em:
测量字符范围 1

字体 16em:
测量字符范围 2

字体 9em:
测量字符范围 3

Pen pen = new Pen(Color.LightGreen, 1);
Brush brush = new SolidBrush(Color.White);
string sourceDigits = "010011001";

private void panel1_Paint(object sender, PaintEventArgs e)
{
    e.Graphics.TextRenderingHint = TextRenderingHint.ClearTypeGridFit;

    CharacterRange[] charRanges = new CharacterRange[sourceDigits.Length];
    for (int chx = 0; chx < sourceDigits.Length; ++chx) {
        charRanges[chx] = new CharacterRange(chx, 1);
    }

    using (StringFormat sf = new StringFormat())
    {
        sf.Alignment = StringAlignment.Center;
        sf.LineAlignment = StringAlignment.Center;
        sf.SetMeasurableCharacterRanges(charRanges);

        Region[] regions = e.Graphics.MeasureCharacterRanges(sourceDigits, Font, e.ClipRectangle, sf);
        for (int i = 0; i < regions.Length; i++) {
            RectangleF rect = regions[i].GetBounds(e.Graphics);
            e.Graphics.DrawRectangle(pen, rect.X, rect.Y, rect.Width, rect.Height);
            e.Graphics.DrawString(char.ToString(sourceDigits[i]), Font, brush, rect, sf);
        }
    }
}
于 2019-02-19T17:47:42.620 回答