Помогите с GDI + взаимодействие в .Net - PullRequest
0 голосов
/ 07 октября 2010

Я использую некоторый код, чтобы позволить мне отображать повернутый текст с помощью метода TextRenderer.DrawText. (По умолчанию DrawText может копировать только прямое преобразование x и y из графического объекта).

Код (C #): connect.microsoft.com . Ниже приведено преобразование VB.

Код принимает графический объект, создает контекст устройства и копирует матрицу преобразования из графического объекта. Это работает, но я также хотел бы установить TextRenderingHint, поэтому я попытался:

<DllImport("gdiplus.dll", CharSet:=CharSet.Unicode, SetLastError:=True, ExactSpelling:=True)> _
Public Shared Function GdipSetTextRenderingHint(ByVal graphics As HandleRef, ByVal textRenderingHint As System.Drawing.Text.TextRenderingHint) As Integer
End Function

И затем после оператора SetClip я помещаю: GdipSetTextRenderingHint (hDC, someHint)

Это дает мне ошибку нарушения доступа к памяти, поэтому я думаю, что я должен использовать в качестве аргумента что-то отличное от hDC.

Я могу заставить его работать, создав контекст устройства из исходного графического объекта, а затем создав другой графический объект из контекста устройства. Затем я установил подсказку для нового графического объекта. Это кажется немного запутанным, поэтому мне было интересно, возможно ли это через взаимодействие.

Преобразование кода VB.Net:

Friend Class TextRendererDC
    Implements IDeviceContext
    Implements IDisposable

    Private graphics As Graphics
    Private dc As IntPtr

    Private Sub New()
    End Sub

    Public Sub New(ByVal g As Graphics)
        Me.graphics = g
    End Sub

    Public Function GetHdc() As IntPtr Implements System.Drawing.IDeviceContext.GetHdc

        Dim xform As NativeMethods.XFORM
        Dim clipRgn As IntPtr

        Using transf As Matrix = Me.graphics.Transform
            xform = New NativeMethods.XFORM(transf)
        End Using

        Using clip As Region = Me.graphics.Clip
            clipRgn = clip.GetHrgn(Me.graphics)
        End Using

        Me.dc = Me.graphics.GetHdc()

        Dim hDC As New HandleRef(Me, Me.dc)
        Dim hRegion As New HandleRef(Nothing, clipRgn)

        SetTransform(hDC, xform)
        SetClip(hDC, hRegion)
        // The below call creates a memory access violation.
        NativeMethods.GdipSetTextRenderingHint(hDC, System.Drawing.Text.TextRenderingHint.AntiAliasGridFit)

        Return Me.dc
    End Function

    Public Sub ReleaseHdc() Implements System.Drawing.IDeviceContext.ReleaseHdc
        If Me.dc <> IntPtr.Zero Then
            Me.graphics.ReleaseHdc()
            Me.dc = IntPtr.Zero
        End If
    End Sub

    Public Sub Dispose() Implements System.IDisposable.Dispose
        ReleaseHdc()
    End Sub

    Private Shared Sub SetTransform(ByVal hdc As HandleRef, ByVal xform As NativeMethods.XFORM)
        NativeMethods.SetGraphicsMode(hdc, NativeMethods.GM_ADVANCED)
        NativeMethods.SetWorldTransform(hdc, xform)
    End Sub

    Private Shared Sub SetClip(ByVal hdc As HandleRef, ByVal hRegion As HandleRef)
        NativeMethods.SelectClipRgn(hdc, hRegion)
    End Sub

    Private Class NativeMethods

        Public Const GM_ADVANCED As Integer = 2

        <DllImport("Gdi32")> _
        Public Shared Function SetGraphicsMode(ByVal hdc As HandleRef, ByVal mode As Integer) As Integer
        End Function

        <DllImport("Gdi32")> _
        Public Shared Function SetWorldTransform(ByVal hDC As HandleRef, ByVal xform As NativeMethods.XFORM) As Boolean
        End Function

        <DllImport("Gdi32")> _
        Public Shared Function SelectClipRgn(ByVal hDC As HandleRef, ByVal hRgn As HandleRef) As Integer
        End Function

        <DllImport("gdiplus.dll", CharSet:=CharSet.Unicode, SetLastError:=True, ExactSpelling:=True)> _
        Public Shared Function GdipSetTextRenderingHint(ByVal graphics As HandleRef, ByVal textRenderingHint As System.Drawing.Text.TextRenderingHint) As Integer
        End Function

        <StructLayout(LayoutKind.Sequential)> _
        Public Class XFORM

            Public eM11 As Single
            Public eM12 As Single
            Public eM21 As Single
            Public eM22 As Single
            Public eDx As Single
            Public eDy As Single

            Public Sub New()
                Me.eM11 = 1.0!
                Me.eM22 = 1.0!
            End Sub

            Public Sub New(ByVal transform As Matrix)
                Me.eM11 = 1.0!
                Me.eM22 = 1.0!
                Me.eM11 = transform.Elements(0)
                Me.eM12 = transform.Elements(1)
                Me.eM21 = transform.Elements(2)
                Me.eM22 = transform.Elements(3)
                Me.eDx = transform.Elements(4)
                Me.eDy = transform.Elements(5)
            End Sub

        End Class

    End Class

End Class

1 Ответ

2 голосов
/ 07 октября 2010

Ух ты, это соответствует "небольшим знаниям, которые могут быть опасными".Даже нативные программисты C ++ не вызывают напрямую точку входа gdiplus, они используют оболочку C ++ в <gdiplus.h>

Режим сбоя здесь заключается в том, что ваша программа загружает неверную версию gdiplus.dll, тот, что в c: \ windows \ system32.Устаревшая версия.Правильный вариант находится в параллельном кэше Windows, сборка System.Drawing в .NET содержит код, обеспечивающий получение правильной версии DLL из кэша.

Не тот, который вы получаете.Ваш даже не инициализирован, GdiplusStartup никогда не вызывался.Kaboom.

Понятия не имею, чего вы пытаетесь достичь.Класс Graphics имеет свойство TextRenderingHint, нет необходимости в убийстве.

...