Partager via


Didacticiel : hébergement d'objets visuels dans une application Win32

Windows Presentation Foundation (WPF) fournit un environnement riche pour créer des applications. Toutefois, lorsque vous avez écrit une bonne partie du Win32, il peut être plus efficace d'ajouter la fonctionnalité WPF à votre application plutôt que de réécrire le code. Pour fournir le support pour des sous-systèmes Win32 et WPF utilisés concurremment dans une application, WPF fournit un mécanisme pour héberger des objets dans une fenêtre Win32.

Ce didacticiel décrit comment écrire une application d'exemple, Test de positionnement avec interopérabilité Win32, exemple, qui héberge des objets visuels WPF dans une fenêtre Win32.

Cette rubrique comprend les sections suivantes.

  • Configuration requise
  • Création de la fenêtre Win32 hôte
  • Ajout d'objets visuels à la fenêtre Win32 hôte
  • Implémentation du filtre de messages Win32
  • Traitement des messages Win32
  • Rubriques connexes

Configuration requise

Ce didacticiel suppose que avez des connaissances de base en matière de programmation WPF et Win32. Pour une présentation générale de la programmation WPF, consultez Procédure pas à pas : mise en route de WPF. Pour une présentation de la programmation Win32, consultez les divers manuels qui s'y rapportent, notamment à La programmation Windows de Charles Petzold.

RemarqueRemarque

Ce didacticiel inclut des exemples de code de l'exemple associé.Toutefois, il n'inclut pas l'exemple de code complet pour une meilleure lisibilité.Pour l'exemple de code complet, consultez Test de positionnement avec interopérabilité Win32, exemple.

Création de la fenêtre Win32 hôte

La clé d'hébergement d'objets WPF dans une fenêtre Win32 est la classe HwndSource. Cette classe encapsule les objets WPF dans une fenêtre Win32 en leur permettant d'être incorporés dans votre user interface (UI) sous la forme d'une fenêtre enfant.

L'exemple suivant montre le code permettant de créer l'objet HwndSource sous la forme d'une fenêtre de conteneur Win32 pour les objets visuels. Pour définir le style, la position et d'autres paramètres de la Win32, utilisez l'objet HwndSourceParameters.

' Constant values from the "winuser.h" header file.
Friend Const WS_CHILD As Integer = &H40000000, WS_VISIBLE As Integer = &H10000000

Friend Shared Sub CreateHostHwnd(ByVal parentHwnd As IntPtr)
    ' Set up the parameters for the host hwnd.
    Dim parameters As New HwndSourceParameters("Visual Hit Test", _width, _height)
    parameters.WindowStyle = WS_VISIBLE Or WS_CHILD
    parameters.SetPosition(0, 24)
    parameters.ParentWindow = parentHwnd
    parameters.HwndSourceHook = New HwndSourceHook(AddressOf ApplicationMessageFilter)

    ' Create the host hwnd for the visuals.
    myHwndSource = New HwndSource(parameters)

    ' Set the hwnd background color to the form's background color.
    myHwndSource.CompositionTarget.BackgroundColor = System.Windows.Media.Brushes.OldLace.Color
End Sub
// Constant values from the "winuser.h" header file.
internal const int WS_CHILD = 0x40000000,
                   WS_VISIBLE = 0x10000000;

internal static void CreateHostHwnd(IntPtr parentHwnd)
{
    // Set up the parameters for the host hwnd.
    HwndSourceParameters parameters = new HwndSourceParameters("Visual Hit Test", _width, _height);
    parameters.WindowStyle = WS_VISIBLE | WS_CHILD;
    parameters.SetPosition(0, 24);
    parameters.ParentWindow = parentHwnd;
    parameters.HwndSourceHook = new HwndSourceHook(ApplicationMessageFilter);

    // Create the host hwnd for the visuals.
    myHwndSource = new HwndSource(parameters);

    // Set the hwnd background color to the form's background color.
    myHwndSource.CompositionTarget.BackgroundColor = System.Windows.Media.Brushes.OldLace.Color;
}
RemarqueRemarque

La valeur de la propriété ExtendedWindowStyle ne peut pas être WS_EX_TRANSPARENT.Cela signifie que la fenêtre Win32 hôte ne peut pas être transparente.Pour cette raison, la couleur d'arrière-plan de la fenêtre Win32 hôte a pour valeur la même couleur d'arrière-plan que sa fenêtre parente.

Ajout d'objets visuels à la fenêtre Win32 hôte

Une fois que vous avez créé une fenêtre de conteneur Win32 hôte pour les objets visuels, vous pouvez y ajouter des objets visuels. L'objectif est de veiller à ce que les transformations des objets visuels, telles que les animations, ne s'étendent pas au delà des limites du rectangle englobant de la fenêtre Win32 hôte.

L'exemple suivant montre le code permettant de créer l'objet HwndSource et d'y ajouter des objets visuels.

RemarqueRemarque

La propriété RootVisual de l'objet HwndSource a pour valeur le premier objet visuel ajouté à la fenêtre Win32 hôte.L'objet visuel racine définit le premier nœud de l'arborescence d'objets visuels.Tous les objets visuels suivants ajoutés à la fenêtre Win32 hôte sont ajoutés comme objets enfants.

Public Shared Sub CreateShape(ByVal parentHwnd As IntPtr)
    ' Create an instance of the shape.
    Dim myShape As New MyShape()

    ' Determine whether the host container window has been created.
    If myHwndSource Is Nothing Then
        ' Create the host container window for the visual objects.
        CreateHostHwnd(parentHwnd)

        ' Associate the shape with the host container window.
        myHwndSource.RootVisual = myShape
    Else
        ' Assign the shape as a child of the root visual.
        CType(myHwndSource.RootVisual, ContainerVisual).Children.Add(myShape)
    End If
End Sub
public static void CreateShape(IntPtr parentHwnd)
{
    // Create an instance of the shape.
    MyShape myShape = new MyShape();

    // Determine whether the host container window has been created.
    if (myHwndSource == null)
    {
        // Create the host container window for the visual objects.
        CreateHostHwnd(parentHwnd);

        // Associate the shape with the host container window.
        myHwndSource.RootVisual = myShape;
    }
    else
    {
        // Assign the shape as a child of the root visual.
        ((ContainerVisual)myHwndSource.RootVisual).Children.Add(myShape);
    }
}

Implémentation du filtre de messages Win32

La fenêtre Win32 hôte des objets visuels nécessite une procédure de filtrage des messages de fenêtre pour gérer les messages envoyés à la fenêtre par la file d'attente d'application. La procédure de fenêtre reçoit des messages du système Win32. Il peut s'agir de messages d'entrée ou de messages de gestion de fenêtre. Vous pouvez gérer éventuellement un message dans votre procédure de fenêtre ou passer le message au système pour un traitement par défaut.

L'objet HwndSource que vous avez défini comme parent pour les objets visuels doit faire référence à la procédure de filtre de messages de fenêtre que vous fournissez. Lorsque vous créez l'objet HwndSource, définissez la propriété HwndSourceHook pour faire référence à la procédure de fenêtre.

parameters.HwndSourceHook = New HwndSourceHook(AddressOf ApplicationMessageFilter)
parameters.HwndSourceHook = new HwndSourceHook(ApplicationMessageFilter);

L'exemple suivant montre le code permettant de gérer les messages de relâchement du bouton gauche et du bouton droit de la souris. La valeur de coordonnée de la position de la souris se trouve dans la valeur du paramètre lParam.

' Constant values from the "winuser.h" header file.
Friend Const WM_LBUTTONUP As Integer = &H202, WM_RBUTTONUP As Integer = &H205

Friend Shared Function ApplicationMessageFilter(ByVal hwnd As IntPtr, ByVal message As Integer, ByVal wParam As IntPtr, ByVal lParam As IntPtr, ByRef handled As Boolean) As IntPtr
    ' Handle messages passed to the visual.
    Select Case message
        ' Handle the left and right mouse button up messages.
        Case WM_LBUTTONUP, WM_RBUTTONUP
            Dim pt As New System.Windows.Point()
            pt.X = CUInt(lParam) And CUInt(&HFFFF) ' LOWORD = x
            pt.Y = CUInt(lParam) >> 16 ' HIWORD = y
            MyShape.OnHitTest(pt, message)
    End Select

    Return IntPtr.Zero
End Function
// Constant values from the "winuser.h" header file.
internal const int WM_LBUTTONUP = 0x0202,
                   WM_RBUTTONUP = 0x0205;

internal static IntPtr ApplicationMessageFilter(
    IntPtr hwnd, int message, IntPtr wParam, IntPtr lParam, ref bool handled)
{
    // Handle messages passed to the visual.
    switch (message)
    {
        // Handle the left and right mouse button up messages.
        case WM_LBUTTONUP:
        case WM_RBUTTONUP:
            System.Windows.Point pt = new System.Windows.Point();
            pt.X = (uint)lParam & (uint)0x0000ffff;  // LOWORD = x
            pt.Y = (uint)lParam >> 16;               // HIWORD = y
            MyShape.OnHitTest(pt, message);
            break;
    }

    return IntPtr.Zero;
}

Traitement des messages Win32

Le code dans l'exemple suivant montre comment un test de positionnement est effectué par rapport à la hiérarchie d'objets visuels contenue dans la fenêtre Win32 hôte. Vous pouvez déterminer si un point se trouve dans la géométrie d'un objet visuel en utilisant la méthode HitTest pour spécifier l'objet visuel racine et la valeur de coordonnée à utiliser pour le test de positionnement. Dans ce cas, l'objet visuel racine est la valeur de la propriété RootVisual de l'objet HwndSource.

        ' Constant values from the "winuser.h" header file.
        Public Const WM_LBUTTONUP As Integer = &H0202, WM_RBUTTONUP As Integer = &H0205

        ' Respond to WM_LBUTTONUP or WM_RBUTTONUP messages by determining which visual object was clicked.
        Public Shared Sub OnHitTest(ByVal pt As System.Windows.Point, ByVal msg As Integer)
            ' Clear the contents of the list used for hit test results.
            hitResultsList.Clear()

            ' Determine whether to change the color of the circle or to delete the shape.
            If msg = WM_LBUTTONUP Then
                MyWindow.changeColor = True
            End If
            If msg = WM_RBUTTONUP Then
                MyWindow.changeColor = False
            End If

            ' Set up a callback to receive the hit test results enumeration.
            VisualTreeHelper.HitTest(MyWindow.myHwndSource.RootVisual, Nothing, New HitTestResultCallback(AddressOf CircleHitTestResult), New PointHitTestParameters(pt))

            ' Perform actions on the hit test results list.
            If hitResultsList.Count > 0 Then
                ProcessHitTestResultsList()
            End If
        End Sub
// Constant values from the "winuser.h" header file.
public const int WM_LBUTTONUP = 0x0202,
                 WM_RBUTTONUP = 0x0205;

// Respond to WM_LBUTTONUP or WM_RBUTTONUP messages by determining which visual object was clicked.
public static void OnHitTest(System.Windows.Point pt, int msg)
{
    // Clear the contents of the list used for hit test results.
    hitResultsList.Clear();

    // Determine whether to change the color of the circle or to delete the shape.
    if (msg == WM_LBUTTONUP)
    {
        MyWindow.changeColor = true;
    }
    if (msg == WM_RBUTTONUP)
    {
        MyWindow.changeColor = false;
    }

    // Set up a callback to receive the hit test results enumeration.
    VisualTreeHelper.HitTest(MyWindow.myHwndSource.RootVisual,
                             null,
                             new HitTestResultCallback(CircleHitTestResult),
                             new PointHitTestParameters(pt));

    // Perform actions on the hit test results list.
    if (hitResultsList.Count > 0)
    {
        ProcessHitTestResultsList();
    }
}

Pour plus d'informations sur le test de positionnement par rapport aux objets visuels, consultez Test de positionnement dans la couche visuelle.

Voir aussi

Référence

HwndSource

Concepts

Test de positionnement dans la couche visuelle

Autres ressources

Test de positionnement avec interopérabilité Win32, exemple (page éventuellement en anglais)