1

Compact Framework.Net 2 SP 2を使用して、現在発生している問題について誰かが助けてくれることを願っています。

現在、一連のテキストボックスを備えたUIがあり、各テキストボックスにはデータベースフィールドの内容が表示されています。これらは、フォームの右側にスクロールバーが表示され、上下に表示されます。各テキストボックスには、幅が設定されています。

各テキストボックスでスクロールバーを使用しないように、保持している行数、フォントサイズ、およびフォントに基づいて各テキストボックスの高さを調整したいと思います。

現時点では、テストアプリケーションでこれを行うことができました。


スクリーンショット:

出力についてはスクリーンショットを参照してくださいhttp://morrislgn.brinkster.net/SO/screenshot.jpg


私のコード:

'Text used in this example:
'TextBox1qwertyuiop lkjhgfdsazxcvbnm1234567890 TextBo

'x1qwer tyuioplkjhgfdsazxcvb nm1234567890

'qwe
'End of exmaple text.
            
Me.Textbox2.Text = Me.Textbox1.Text

Dim pobjGraphic As Graphics = Me.Textbox2.Parent.CreateGraphics()
Dim pobjSize As SizeF
            
'Padding values:
Dim piTop As Int32 = 4 'top of text and top of textbox
Dim piBottom As Int32 = 3 'bottom of text and top of textbox
            
Dim piLines As Int32 = 0

'Based on the font size chosen by the user, create a font to perform the calculation with.
Dim piFontSize As Single = 10

If Me.CheckBox1.Checked.Equals(True) Then
    piFontSize = 6
ElseIf Me.CheckBox2.Checked.Equals(True) Then
    piFontSize = 8
ElseIf Me.CheckBox3.Checked.Equals(True) Then
    piFontSize = 12
Else
    piFontSize = 10
End If

Dim pobjFont As New Font("Tahoma", piFontSize, FontStyle.Regular)

'Calculate the height of one line.
pobjSize = pobjGraphic.MeasureString("HELLO WORLD", pobjFont)
'Value of pobjSize returned: {Width = 71.0 Height = 13.0}
            
            
'Calculate the number of lines          
Dim b As Bitmap
b = New Bitmap(1, 1, Imaging.PixelFormat.Format32bppRgb)
            
'Calculate the number of lines required to display the text properly based on the lenght of the text the width of the control.
'Length of text to show divide by the width of the textbox
piLines = Graphics.FromImage(b).MeasureString(Me.Textbox2.Text, pobjFont).Width / Me.Textbox2.Width
'Value of piLines returned: 2

If piLines = 0 Then
    piLines = 1
End If
            
'Calculate the size of the text to be displayed using the margins, height of one line and number of lines.
Me.Textbox2.Height = (pobjSize.Height * piLines) + piTop + piBottom
' value produced: 33 = (13 * 2) + 4 + 3
'set font of text box
Me.Textbox2.Font = pobjFont

最後に、これはp / invokeを使用したCOREDLL.dllの呼び出しを使用して実現できることを知っていますが、これを行うとアプリケーションがクラッシュします。

こんにちはフォークス、

以下は、要求されたピンボークコードです。

    <Runtime.InteropServices.DllImport("coredll.dll")> _
Private Function SendMessage( _
    ByVal hwnd As IntPtr, ByVal msg As Integer, _
    ByVal wParam As Integer, ByVal lParam As Integer) As Integer
End Function

<Runtime.InteropServices.DllImport("coredll.dll")> _
Private Function GetCapture() As IntPtr
End Function

<Runtime.InteropServices.DllImport("coredll.dll")> _
Private Function ReleaseCapture() As Boolean
End Function

Public Function GetNumberOfLines(ByVal ptxtCountBox As TextBox) As Integer
    Try
        Dim hnd As IntPtr = New IntPtr

        ptxtCountBox.Capture = True

        ' Capture the textbox handle.
        hnd = GetCapture()
        ptxtCountBox.Capture = False

        ' Get the count of the lines in the box.
        Dim plCount As Integer = SendMessage(ptxtCountBox.Handle, EM_GETLINECOUNT, 0, 0)

        ' Count the number of return lines as we minus this from the total lines to take.
        plCount = plCount - (CharCount(ptxtCountBox.Text, vbCrLf, False))

        plCount += RemoveNonASCIIReturns(ptxtCountBox)

        ReleaseCapture()

        hnd = Nothing

        ' Return the line count.
        Return plCount
    Catch ex As Exception
        GenerateError(msCLASS_NAME, "GetNumberOfLines", ex.Message.ToString)
    End Try
End Function

ありがとう、

モリス

4

3 に答える 3

0

さて、私はあなたに健全でスマートな解決策を提案します. アルゴリズムは次のとおりです。

  1. 参照用にラベル コントロールを使用します。
  2. 以下を割り当てます。 • Textbox のサイズを Label に割り当てます。• Textbox のフォントを Label に。• Label の Autosize-property をTrueにします。• Textbox' の Label の BorderStyle プロパティ。• Textbox の元のサイズとしての Label の MinimumSize プロパティ。• Label の MaximumSize プロパティは、元の幅と同じ、高さは元の高さの大きな倍数です。

  3. テキストボックスのテキストをラベルのテキストに割り当てます。

  4. ここで、Label の PrefferdHeight プロパティ > Textbox の高さ == True の場合 Textbox の高さを増やし、False になるまで上記の条件を確認します。
  5. ラベルはすぐに処分できます。

MSDN フォーラムにも同様のソリューションを投稿しました。これもチェックアウトできます。 [1]

よろしく。:)

于 2009-04-07T01:16:14.167 に答える
0

同様の質問をしたところ、この件に関する私のニーズを完全に満たす回答が得られました! 私の質問からのstevo3000の回答をチェックしてください: AutoSize for Label / TextBox in .NET Compact Framework

彼は、1 回のスワイプで私の問題を完全に解決したこれらの 2 つのブログ投稿を参照しました。 http://www.mobilepractices.com/2007/12/multi-line-graphicsmeasurestring.html http://www.mobilepractices.com/2008/01/making-multiline-measurestring-work.html

于 2009-04-20T08:55:54.093 に答える
0

私はこれの底に達したと思います:

    Public Function GetNumberOfLines(ByVal pstext As String, ByVal pobjfont As Font, ByVal pobjDimensions As Size) As Decimal
    Dim pslines As String() = Nothing
    'Used to measure the string to be placed into the textbox
    Dim pobjBitMap As Bitmap = Nothing
    Dim pobjSize As SizeF = Nothing

    Try
        Dim psline As String = String.Empty
        Dim pilinecount As Decimal = 0.0
        'Spilt the text based on the number of lines breaks.
        pslines = pstext.Split(vbCrLf)
        For Each psline In pslines
            'Create a graphics image which is used to work out the width of the text.
            pobjBitMap = New Bitmap(1, 1, Imaging.PixelFormat.Format32bppRgb)
            pobjSize = Graphics.FromImage(pobjBitMap).MeasureString(psline, pobjfont)

            'If the width of the text is less than 1.0 then add one to the count. This would incidcate a line break.
            If pobjSize.Width < 1.0 Then
                pilinecount = pilinecount + 1
            Else
                'Based on the dimensions of the text, work out the number of lines. 0.5 is added to round the value to next whole number.
                pilinecount = pilinecount + (Round((pobjSize.Width / pobjDimensions.Width) + 0.5))
            End If
        Next

        'If the line count it less than 1 return one line.
        If pilinecount < 1.0 Then
            Return 1.0
        Else
            Return pilinecount
        End If
    Catch ex As Exception
        Return 1.0
    Finally
        If pslines IsNot Nothing Then
            Array.Clear(pslines, 0, pslines.Length - 1)
            pslines = Nothing
        End If
        If pobjBitMap IsNot Nothing Then
            pobjBitMap.Dispose()
        End If
    End Try
End Function

確かに、ちょっとしたハックですが、現時点では問題なく動作しているようです! これを改善する方法についての観察やコメントは大歓迎です。

また、p/invoke については、問題の根本、または解決策を発見しました。デバイスの .Net fx をアップグレードしたところ、問題が解決したようです。

ありがとう

モリス

于 2009-03-04T12:54:07.990 に答える