winforms 如何调整标签和组合框,使两个文本在同一基线?

3zwjbxry  于 2023-10-23  发布在  其他
关注(0)|答案(1)|浏览(119)

你好
我想在运行时创建一个标签和一个组合框。

如何相对于ComboBox.Top计算Label.Top,以使两个控件的文本位于一行上?

它应该适用于不同的字体和边框样式。两个控件的相同的Top和Height值不会执行此技巧。
Thanks in advance

ijnw1ujt

ijnw1ujt1#

文本基线计算需要使用GetTextData API:
gettextmetrics (gdi32)
GetTextMetrics function (wingdi.h)

我更喜欢VB.net:

Imports System.Runtime.InteropServices

Public Module Font_Metrics
    Public Structure FontMetrics
        Dim Ascent As Integer
        Dim Descent As Integer
        Dim Height As Integer
    End Structure

    Public Function GetMetrics(gfx As Graphics, fnt As Font) As FontMetrics
        Dim Hdc As IntPtr = gfx.GetHdc()
        Dim Hfont As IntPtr = fnt.ToHfont()
        Dim Metric As TEXTMETRIC
        Dim HfontDefault As IntPtr = SelectObject(Hdc, Hfont)
        GetTextMetrics(Hdc, Metric)
        SelectObject(Hdc, HfontDefault)
        DeleteObject(Hfont)
        gfx.ReleaseHdc(Hdc)

        Dim fm As FontMetrics
        fm.Ascent = Metric.tmAscent
        fm.Descent = Metric.tmDescent
        fm.Height = Metric.tmHeight
        Return fm
    End Function

    Private Structure TEXTMETRIC
        Dim tmHeight As Integer
        Dim tmAscent As Integer
        Dim tmDescent As Integer
        Dim tmInternalLeading As Integer
        Dim tmExternalLeading As Integer
        Dim tmAveCharWidth As Integer
        Dim tmMaxCharWidth As Integer
        Dim tmWeight As Integer
        Dim tmOverhang As Integer
        Dim tmDigitizedAspectX As Integer
        Dim tmDigitizedAspectY As Integer
        Dim tmFirstChar As Char
        Dim tmLastChar As Char
        Dim tmDefaultChar As Char
        Dim tmBreakChar As Char
        Dim tmItalic As Byte
        Dim tmUnderlined As Byte
        Dim tmStruckOut As Byte
        Dim tmPitchAndFamily As Byte
        Dim tmCharSet As Byte
    End Structure

    <DllImport("gdi32.dll", CharSet:=CharSet.Unicode)>
    Private Function SelectObject(hdc As IntPtr, hgdiobj As IntPtr) As IntPtr
    End Function

    <DllImport("gdi32.dll", CharSet:=CharSet.Unicode)>
    Private Function GetTextMetrics(hdc As IntPtr, <Out> ByRef lptm As TEXTMETRIC) As Boolean
    End Function

    <DllImport("gdi32.dll", CharSet:=CharSet.Unicode)>
    Private Function DeleteObject(hdc As IntPtr) As Boolean
    End Function

End Module

现在你可以计算combobox的文本基线,并根据它对齐标签的顶部:

Dim combo_fm As FontMetrics = GetMetrics(CreateGraphics, ComboBox.Font)
Dim label_fm As FontMetrics = GetMetrics(CreateGraphics, Label.Font)
SuspendLayout()
Label.Left = ComboBox.Left - Label.Width 
Dim BaseLine_y As Integer = ComboBox.Top + (ComboBox.Height - ComboBox.ItemHeight) / 2 + combo_fm.Ascent
Label.Top = BaseLine_y - Label.Padding.Top - label_fm.Ascent
ResumeLayout()

相关问题