WPF - Guide rapide

WPF signifie Windows Presentation Foundation. C'est un cadre puissant pour créer des applications Windows. Ce didacticiel explique les fonctionnalités que vous devez comprendre pour créer des applications WPF et comment cela apporte un changement fondamental dans les applications Windows.

WPF a été introduit pour la première fois dans la version .NET Framework 3.0, puis de nombreuses autres fonctionnalités ont été ajoutées dans les versions ultérieures du .NET Framework.

Architecture WPF

Avant WPF, les autres frameworks d'interface utilisateur proposés par Microsoft, tels que les formulaires MFC et Windows, n'étaient que des wrappers autour des DLL User32 et GDI32, mais WPF n'utilise que très peu User32. Alors,

  • WPF est plus qu'un simple wrapper.
  • Il fait partie du framework .NET.
  • Il contient un mélange de code managé et non managé.

Les principaux composants de l'architecture WPF sont indiqués dans la figure ci-dessous. La partie de code la plus importante de WPF est:

  • Cadre de présentation
  • Noyau de présentation
  • Milcore

le presentation framework et le presentation core ont été écrits en code managé. Milcore fait partie du code non managé qui permet une intégration étroite avec DirectX (responsable de l'affichage et du rendu). CLR rend le processus de développement plus productif en offrant de nombreuses fonctionnalités telles que la gestion de la mémoire, la gestion des erreurs, etc.

WPF - Avantages

Dans les anciens frameworks GUI, il n'y avait pas de véritable séparation entre l'apparence d'une application et son comportement. L'interface graphique et le comportement ont été créés dans le même langage, par exemple C # ou VB.Net, ce qui nécessiterait plus d'efforts de la part du développeur pour implémenter à la fois l'interface utilisateur et le comportement qui lui sont associés.

Dans WPF, les éléments d'interface utilisateur sont conçus en XAML tandis que les comportements peuvent être implémentés dans des langages procéduraux tels que C # et VB.Net. Il est donc très facile de séparer le comportement du code du concepteur.

Avec XAML, les programmeurs peuvent travailler en parallèle avec les concepteurs. La séparation entre une interface graphique et son comportement peut nous permettre de changer facilement l'apparence d'un contrôle en utilisant des styles et des modèles.

WPF - Caractéristiques

WPF est un framework puissant pour créer une application Windows. Il prend en charge de nombreuses fonctionnalités intéressantes, dont certaines ont été répertoriées ci-dessous -

Fonctionnalité La description
Contrôle à l'intérieur d'un contrôle Permet de définir un contrôle à l'intérieur d'un autre contrôle en tant que contenu.
Liaison de données Mécanisme pour afficher et interagir avec les données entre les éléments de l'interface utilisateur et l'objet de données sur l'interface utilisateur.
Services médias Fournit un système intégré pour créer des interfaces utilisateur avec des éléments multimédias courants tels que des images, du son et de la vidéo.
Modèles Dans WPF, vous pouvez définir l'apparence d'un élément directement avec un modèle
Animations Construire l'interactivité et le mouvement sur l'interface utilisateur
Entrée alternative Prend en charge la saisie multi-touch sur Windows 7 et supérieur.
Direct3D Permet d'afficher des graphiques plus complexes et des thèmes personnalisés

Microsoft fournit deux outils importants pour le développement d'applications WPF.

  • Visual Studio
  • Mélange d'expression

Les deux outils peuvent créer des projets WPF, mais le fait est que Visual Studio est davantage utilisé par les développeurs, tandis que Blend est utilisé plus souvent par les concepteurs. Pour ce didacticiel, nous utiliserons principalement Visual Studio.

Installation

Microsoft fournit une version gratuite de Visual Studio qui peut être téléchargée à partir de VisualStudio .

Téléchargez les fichiers et suivez les étapes ci-dessous pour configurer l'environnement de développement d'applications WPF sur votre système.

  • Une fois le téléchargement terminé, exécutez le installer. La boîte de dialogue suivante s'affiche.

  • Clique le Install bouton et il lancera le processus d'installation.

  • Une fois le processus d'installation terminé, vous verrez la boîte de dialogue suivante.

  • Fermez cette boîte de dialogue et redémarrez votre ordinateur si nécessaire.

  • Maintenant, ouvrez Visual Studio à partir du menu Démarrer qui ouvrira la boîte de dialogue suivante.

  • Une fois que tout est fait, vous verrez la fenêtre principale de Visual Studio.

Vous êtes maintenant prêt à créer votre première application WPF.

Dans ce chapitre, nous développerons une application WPF Hello World simple. Commençons donc la mise en œuvre simple en suivant les étapes ci-dessous.

  • Cliquez sur l'option de menu Fichier> Nouveau> Projet.
  • La boîte de dialogue suivante s'affiche.
  • Sous Modèles, sélectionnez Visual C # et dans le panneau du milieu, sélectionnez Application WPF.

  • Donnez un nom au projet. TypeHelloWorld dans le champ de nom et cliquez sur le bouton OK.

  • Par défaut, deux fichiers sont créés, l'un est le XAML fichier (mainwindow.xaml) et l'autre est le CS fichier (mainwindow.cs)

  • Sur mainwindow.xaml, vous verrez deux sous-fenêtres, l'une est la design window et l'autre est le source (XAML) window.

  • Dans l'application WPF, il existe deux façons de concevoir une interface utilisateur pour votre application. La première consiste simplement à faire glisser et déposer les éléments de l'interface utilisateur de la boîte à outils vers la fenêtre de conception. La deuxième méthode consiste à concevoir votre interface utilisateur en écrivant des balises XAML pour les éléments de l'interface utilisateur. Visual Studio gère les balises XAML lorsque la fonctionnalité glisser-déposer est utilisée pour la conception d'interface utilisateur.

  • Dans le fichier mainwindow.xaml, les balises XAML suivantes sont écrites par défaut.

<Window x:Class = "HelloWorld.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid> 
   </Grid> 
	
</Window>
  • Par défaut, une grille est définie comme le premier élément après la page.
  • Allons dans la boîte à outils et faisons glisser un TextBlock dans la fenêtre de conception.
  • Vous verrez le TextBlock dans la fenêtre de conception.
  • Lorsque vous regardez la fenêtre source, vous verrez que Visual Studio a généré le code XAML du TextBlock pour vous.

  • Modifions la propriété Text de TextBlock dans le code XAML de TextBlock à Hello World.

<Window x:Class = "HelloWorld.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <TextBlock x:Name = "textBlock" HorizontalAlignment = "Left"
         Margin = "235,143,0,0" TextWrapping = "Wrap" Text = "Hello World!"
         VerticalAlignment = "Top" Height = "44" Width = "102" /> 
   </Grid> 
	
</Window>
  • Maintenant, vous verrez également le changement dans la fenêtre de conception.

Lorsque le code ci-dessus est compilé et exécuté, vous verrez la fenêtre suivante.

Toutes nos félicitations! Vous avez conçu et créé votre première application WPF.

L'une des premières choses que vous rencontrerez en travaillant avec WPF est XAML. XAML signifie Extensible Application Markup Language. C'est un langage simple et déclaratif basé sur XML.

  • En XAML, il est très facile de créer, d'initialiser et de définir des propriétés d'objets avec des relations hiérarchiques.

  • Il est principalement utilisé pour concevoir des interfaces graphiques, mais il peut également être utilisé à d'autres fins, par exemple, pour déclarer un flux de travail dans Workflow Foundation.

Syntaxe de base

Lorsque vous créez votre nouveau projet WPF, vous rencontrez une partie du code XAML par défaut dans MainWindow.xaml, comme indiqué ci-dessous.

<Window x:Class = "Resources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
         
   </Grid> 
	
</Window>

Le fichier XAML ci-dessus contient différents types d'informations. Le tableau suivant explique brièvement le rôle de chaque information.

Information La description
<Fenêtre C'est l'élément d'objet d'ouverture ou le conteneur de la racine.
x: Class = "Resources.MainWindow" Il s'agit d'une déclaration de classe partielle qui connecte le balisage au code de classe partiel défini derrière.
xmlns = "http://schemas.microsoft.com/win fx / 2006 / xaml / presentation" Mappe l'espace de noms XAML par défaut pour le client / framework WPF
xmlns: x = "http://schemas.microsoft.com/w infx / 2006 / xaml" Espace de noms XAML pour le langage XAML qui le mappe au préfixe x:
> Fin de l'élément objet de la racine

<Grille>

</Grid>

Il démarre et ferme les balises d'un objet de grille vide.
</Window> Fermer l'élément objet

Les règles de syntaxe pour XAML sont presque similaires à XML. Si vous regardez un document XAML, vous remarquerez qu'il s'agit en fait d'un fichier XML valide, mais qu'un fichier XML n'est pas nécessairement un fichier XAML. En effet, en XML, la valeur des attributs doit être une chaîne tandis qu'en XAML, il peut s'agir d'un objet différent appelé syntaxe d'élément Property.

  • La syntaxe d'un élément Object commence par un crochet angulaire gauche (<) suivi du nom d'un objet, par exemple Button.

  • Définissez certaines propriétés et attributs de cet élément d'objet.

  • L'élément Object doit être fermé par une barre oblique (/) suivie immédiatement d'un crochet à angle droit (>).

Exemple d'objet simple sans élément enfant

<Button/>

Exemple d'élément objet avec quelques attributs

<Button Content = "Click Me" Height = "30" Width = "60" />

Exemple de syntaxe alternative pour définir les propriétés (syntaxe de l'élément Property)

<Button> 
   <Button.Content>Click Me</Button.Content> 
   <Button.Height>30</Button.Height> 
   <Button.Width>60</Button.Width> 
</Button>

Exemple d'objet avec élément enfant: StackPanel contient Textblock comme élément enfant

<StackPanel Orientation = "Horizontal"> 
   <TextBlock Text = "Hello"/> 
</StackPanel>

Pourquoi XAML dans WPF

XAML n'est pas seulement la fonctionnalité la plus connue de WPF, mais c'est aussi l'une des fonctionnalités les plus mal comprises. Si vous êtes exposé à WPF, vous devez avoir entendu parler de XAML; mais prenez note des deux faits moins connus suivants sur XAML -

  • WPF n'a pas besoin de XAML
  • XAML n'a pas besoin de WPF

Ce sont en fait des éléments technologiques séparables. Pour comprendre comment cela peut être, examinons un exemple simple dans lequel un bouton est créé avec certaines propriétés en XAML.

<Window x:Class = "WPFXAMLOverview.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button x:Name = "button" Content = "Click Me" HorizontalAlignment = "Left"  
         Margin = "150" VerticalAlignment = "Top" Width = "75" /> 
   </StackPanel> 
	
</Window>

Si vous choisissez de ne pas utiliser XAML dans WPF, vous pouvez également obtenir le même résultat GUI avec un langage procédural. Regardons le même exemple, mais cette fois, nous allons créer un bouton en C #.

using System.Windows; 
using System.Windows.Controls;  

namespace WPFXAMLOverview { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
			
         // Create the StackPanel 
         StackPanel stackPanel = new StackPanel(); 
         this.Content = stackPanel; 
			
         // Create the Button 
         Button button = new Button();
         button.Content = "Click Me"; 
         button.HorizontalAlignment = HorizontalAlignment.Left; 
         button.Margin = new Thickness(150); 
         button.VerticalAlignment = VerticalAlignment.Top; 
         button.Width = 75; 
         stackPanel.Children.Add(button);  
      } 
   } 
}

Lorsque vous compilez et exécutez le code XAML ou le code C #, vous verrez la même sortie que celle illustrée ci-dessous.

D'après l'exemple ci-dessus, il est clair que ce que vous pouvez faire en XAML pour créer, initialiser et définir des propriétés d'objets, les mêmes tâches peuvent également être effectuées à l'aide de code.

  • XAML n'est qu'un autre moyen simple et facile de concevoir des éléments d'interface utilisateur.

  • Avec XAML, cela ne signifie pas que ce que vous pouvez faire pour concevoir des éléments d'interface utilisateur est le seul moyen. Vous pouvez déclarer les objets en XAML ou les définir à l'aide de code.

  • XAML est facultatif, mais malgré cela, il est au cœur de la conception WPF.

  • L'objectif de XAML est de permettre aux concepteurs visuels de créer directement des éléments d'interface utilisateur.

  • WPF vise à permettre de contrôler tous les aspects visuels de l'interface utilisateur à partir du balisage.

Il existe de nombreuses technologies où les éléments et les composants sont classés dans une arborescence afin que les programmeurs puissent facilement manipuler l'objet et modifier le comportement d'une application. Windows Presentation Foundation (WPF) possède une arborescence complète sous la forme d'objets. Dans WPF, il existe deux façons de conceptualiser une arborescence d'objets complète:

  • Arborescence logique
  • Arborescence visuelle

À l'aide de ces arborescences, vous pouvez facilement créer et identifier la relation entre les éléments de l'interface utilisateur. La plupart du temps, les développeurs et concepteurs WPF utilisent un langage procédural pour créer une application ou conçoivent la partie UI de l'application en XAML en gardant à l'esprit la structure de l'arborescence des objets.

Arborescence logique

Dans les applications WPF, la structure des éléments d'interface utilisateur en XAML représente l'arborescence logique. En XAML, les éléments de base de l'interface utilisateur sont déclarés par le développeur. L'arbre logique dans WPF définit ce qui suit -

  • Propriétés de dépendance
  • Ressources statiques et dynamiques
  • Lier les éléments sur son nom etc.

Jetons un coup d'œil à l'exemple suivant dans lequel un bouton et une zone de liste sont créés.

<Window x:Class = "WPFElementsTree.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button x:Name = "button" Height = "30" Width = "70" Content = "OK" Margin = "20" /> 
		
      <ListBox x:Name = "listBox" Height = "100" Width = "100" Margin = "20"> 
         <ListBoxItem Content = "Item 1" /> 
         <ListBoxItem Content = "Item 2" /> 
         <ListBoxItem Content = "Item 3" /> 
      </ListBox> 
		
   </StackPanel> 
	
</Window>

Si vous regardez le code XAML, vous observerez une structure arborescente, c'est-à-dire que le nœud racine est la fenêtre et à l'intérieur du nœud racine, il n'y a qu'un seul enfant, StackPanel. Mais StackPanel contient deux éléments enfants, un bouton et une zone de liste. La zone de liste a trois autres éléments de zone de liste enfants.

Arborescence visuelle

Dans WPF, le concept de l'arborescence visuelle décrit la structure des objets visuels, représentée par la classe Visual Base. Il signifie tous les éléments de l'interface utilisateur qui sont rendus sur l'écran de sortie.

Lorsqu'un programmeur souhaite créer un modèle pour un contrôle particulier, il rend en fait l'arborescence visuelle de ce contrôle. L'arbre visuel est également très utile pour ceux qui souhaitent dessiner des contrôles de niveau inférieur pour des raisons de performances et d'optimisation.

Dans les applications WPF, l'arborescence visuelle est utilisée pour -

  • Rendu des objets visuels.
  • Rendu des mises en page.
  • Les événements routés voyagent principalement le long de l'arborescence visuelle, pas de l'arborescence logique.

Pour voir l'arborescence visuelle de l'application simple ci-dessus qui contient un bouton et une zone de liste, compilons et exécutons le code XAML et vous verrez la fenêtre suivante.

Lorsque l'application est en cours d'exécution, vous pouvez voir l'arborescence visuelle de l'application en cours d'exécution dans la fenêtre Live Visual Tree qui montre la hiérarchie complète de cette application, comme indiqué ci-dessous.

L'arbre visuel est généralement un sur-ensemble de l'arbre logique. Vous pouvez voir ici que tous les éléments logiques sont également présents dans l'arborescence visuelle. Ainsi, ces deux arbres ne sont en réalité que deux vues différentes du même ensemble d'objets qui composent l'interface utilisateur.

  • L'arbre logique laisse de côté beaucoup de détails vous permettant de vous concentrer sur la structure de base de l'interface utilisateur et d'ignorer les détails de la manière exacte dont elle a été présentée.

  • L'arbre logique est ce que vous utilisez pour créer la structure de base de l'interface utilisateur.

  • L'arbre visuel sera intéressant si vous vous concentrez sur la présentation. Par exemple, si vous souhaitez personnaliser l'apparence de n'importe quel élément de l'interface utilisateur, vous devrez utiliser l'arborescence visuelle.

Dans les applications WPF, la propriété de dépendance est un type spécifique de propriété qui étend la propriété CLR. Il tire parti des fonctionnalités spécifiques disponibles dans le système de propriétés WPF.

Une classe qui définit une propriété de dépendance doit être héritée du DependencyObjectclasse. De nombreuses classes de contrôles d'interface utilisateur qui sont utilisées en XAML sont dérivées de laDependencyObject classe et ils prennent en charge les propriétés de dépendance, par exemple, la classe Button prend en charge IsMouseOver propriété de dépendance.

Le code XAML suivant crée un bouton avec certaines propriétés.

<Window x:Class = "WPFDependencyProperty.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFDependencyProperty"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button  Height = "40" Width = "175" Margin = "10" Content = "Dependency Property"> 
         <Button.Style> 
            <Style TargetType = "{x:Type Button}"> 
               <Style.Triggers> 
					
                  <Trigger Property = "IsMouseOver" Value = "True"> 
                     <Setter Property = "Foreground" Value = "Red" /> 
                  </Trigger>
						
               </Style.Triggers>
            </Style> 
         </Button.Style> 
      </Button> 
   </Grid> 
	
</Window>

L'extension de balisage x: Type en XAML a une fonctionnalité similaire à celle de typeof () en C #. Il est utilisé lorsque des attributs sont spécifiés qui prennent le type de l'objet tels que <Style TargetType = "{x: Type Button}">

Lorsque le code ci-dessus est compilé et exécuté, vous obtiendrez ce qui suit MainWindow. Lorsque la souris est sur le bouton, cela changera la couleur de premier plan d'un bouton. Lorsque la souris quitte le bouton, elle reprend sa couleur d'origine.

Pourquoi nous avons besoin de propriétés de dépendance

La propriété de dépendance vous offre toutes sortes d'avantages lorsque vous l'utilisez dans votre application. La propriété de dépendance peut être utilisée sur une propriété CLR dans les scénarios suivants -

  • Si vous souhaitez définir le style
  • Si vous voulez une liaison de données
  • Si vous souhaitez définir avec une ressource (une ressource statique ou dynamique)
  • Si vous souhaitez prendre en charge l'animation

Fondamentalement, les propriétés de dépendance offrent de nombreuses fonctionnalités que vous n'obtiendrez pas en utilisant une propriété CLR.

La principale différence entre dependency properties et autre CLR properties sont énumérés ci-dessous -

  • Les propriétés CLR peuvent directement lire / écrire à partir du membre privé d'une classe en utilisant getter et setter. En revanche, les propriétés de dépendance ne sont pas stockées dans l'objet local.

  • Les propriétés de dépendance sont stockées dans un dictionnaire de paires clé / valeur qui est fourni par la classe DependencyObject. Il économise également beaucoup de mémoire car il stocke la propriété lorsqu'elle est modifiée. Il peut également être lié en XAML.

Propriétés de dépendance personnalisées

Dans le framework .NET, les propriétés de dépendance personnalisées peuvent également être définies. Suivez les étapes ci-dessous pour définir la propriété de dépendance personnalisée en C #.

  • Déclarez et enregistrez votre dependency property avec registre d'appels système.

  • Fournir le setter et getter pour la propriété.

  • Définir un static handler qui gérera tous les changements qui se produisent globalement

  • Définir un instance handler qui gérera toutes les modifications apportées à cette instance particulière.

Le code C # suivant définit une propriété de dépendance pour définir le SetText propriété du contrôle utilisateur.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks; 

using System.Windows; 
using System.Windows.Controls; 
using System.Windows.Data; 
using System.Windows.Documents; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Media.Imaging; 
using System.Windows.Navigation; 
using System.Windows.Shapes;  

namespace WpfApplication3 { 
   /// <summary> 
      /// Interaction logic for UserControl1.xaml 
   /// </summary> 
	
   public partial class UserControl1 : UserControl { 
	
      public UserControl1() { 
         InitializeComponent(); 
      }
		
      public static readonly DependencyProperty SetTextProperty = 
         DependencyProperty.Register("SetText", typeof(string), typeof(UserControl1), new 
            PropertyMetadata("", new PropertyChangedCallback(OnSetTextChanged))); 
				
      public string SetText { 
         get { return (string)GetValue(SetTextProperty); } 
         set { SetValue(SetTextProperty, value); } 
      } 
		
      private static void OnSetTextChanged(DependencyObject d,
         DependencyPropertyChangedEventArgs e) { 
         UserControl1 UserControl1Control = d as UserControl1; 
         UserControl1Control.OnSetTextChanged(e); 
      } 
		
      private void OnSetTextChanged(DependencyPropertyChangedEventArgs e) { 
         tbTest.Text = e.NewValue.ToString(); 
      }  
   } 
}

Voici le fichier XAML dans lequel le TextBlock est défini en tant que contrôle utilisateur et la propriété Text lui sera affectée par la propriété de dépendance SetText.

Le code XAML suivant crée un contrôle utilisateur et initialise son SetText propriété de dépendance.

<Window x:Class = "WpfApplication3.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:views = "clr-namespace:WpfApplication3"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <views:UserControl1 SetText = "Hellow World"/> 
   </Grid> 
	
</Window>

Lançons cette application. Vous pouvez immédiatement observer que dans notre MainWindow, la propriété de dépendance pour le contrôle utilisateur a été utilisée avec succès en tant que texte.

UNE routed eventest un type d'événement qui peut appeler des gestionnaires sur plusieurs écouteurs dans une arborescence d'éléments plutôt que sur l'objet qui a déclenché l'événement. Il s'agit essentiellement d'un événement CLR pris en charge par une instance de la classe Routed Event. Il est enregistré auprès du système d'événements WPF. RoutedEvents a trois stratégies de routage principales qui sont les suivantes -

  • Événement direct
  • Événement bouillonnant
  • Événement de tunnel

Événement direct

Un événement direct est similaire aux événements dans les formulaires Windows qui sont déclenchés par l'élément à l'origine de l'événement.

Contrairement à un événement CLR standard, les événements acheminés directement prennent en charge la gestion des classes et peuvent être utilisés dans les paramètres d'événement et les déclencheurs d'événement dans le style de votre contrôle personnalisé.

Un bon exemple d'un événement direct serait l'événement MouseEnter.

Événement bouillonnant

Un événement bouillonnant commence par l'élément d'où l'événement est originaire. Ensuite, il parcourt l'arborescence visuelle jusqu'à l'élément le plus haut de l'arborescence visuelle. Ainsi, dans WPF, l'élément le plus élevé est probablement une fenêtre.

Événement de tunnel

Les gestionnaires d'événements sur la racine de l'arborescence des éléments sont appelés, puis l'événement se déplace dans l'arborescence visuelle vers tous les nœuds enfants jusqu'à ce qu'il atteigne l'élément à l'origine de l'événement.

La différence entre un événement bouillonnant et un événement tunnel est qu'un événement tunnel commencera toujours par un aperçu.

Dans une application WPF, les événements sont souvent implémentés sous la forme d'une paire tunnel / bullage. Ainsi, vous aurez un aperçu MouseDown puis un événement MouseDown.

Vous trouverez ci-dessous un exemple simple d'événement Routé dans lequel un bouton et trois blocs de texte sont créés avec certaines propriétés et événements.

<Window x:Class = "WPFRoutedEvents.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "450" Width = "604" ButtonBase.Click  = "Window_Click" >
	
   <Grid> 
      <StackPanel Margin = "20" ButtonBase.Click = "StackPanel_Click">
		
         <StackPanel Margin = "10"> 
            <TextBlock Name = "txt1" FontSize = "18" Margin = "5" Text = "This is a TextBlock 1" /> 
            <TextBlock Name = "txt2" FontSize = "18" Margin = "5" Text = "This is a TextBlock 2" /> 
            <TextBlock Name = "txt3" FontSize = "18" Margin = "5" Text = "This is a TextBlock 3" /> 
         </StackPanel> 
			
         <Button Margin = "10" Content = "Click me" Click = "Button_Click" Width = "80"/> 
      </StackPanel> 
   </Grid> 
	
</Window>

Voici le code C # pour l'implémentation des événements Click pour Button, StackPanel et Window.

using System.Windows; 
 
namespace WPFRoutedEvents { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary>
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         txt1.Text = "Button is Clicked"; 
      } 
		
      private void StackPanel_Click(object sender, RoutedEventArgs e) { 
         txt2.Text = "Click event is bubbled to Stack Panel"; 
      } 
		
      private void Window_Click(object sender, RoutedEventArgs e) { 
         txt3.Text = "Click event is bubbled to Window"; 
      }
		
   } 
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante -

Lorsque vous cliquez sur le bouton, les blocs de texte sont mis à jour, comme indiqué ci-dessous.

Si vous souhaitez arrêter l'événement acheminé à un niveau particulier, vous devrez définir e.Handled = true;

Changeons le StackPanel_Click événement comme indiqué ci-dessous -

private void StackPanel_Click(object sender, RoutedEventArgs e) { 
   txt2.Text = "Click event is bubbled to Stack Panel"; 
   e.Handled = true; 
}

Lorsque vous cliquez sur le bouton, vous observerez que l'événement de clic ne sera pas acheminé vers la fenêtre et s'arrêtera au stackpanel et le 3 ème bloc de texte ne sera pas mis à jour.

Événements routés personnalisés

Dans le framework .NET, un événement routé personnalisé peut également être défini. Vous devez suivre les étapes ci-dessous pour définir un événement routé personnalisé en C #.

  • Déclarez et enregistrez votre événement acheminé avec l'appel système RegisterRoutedEvent.

  • Spécifiez la stratégie de routage, c'est-à-dire Bulle, Tunnel ou Direct.

  • Fournissez le gestionnaire d'événements.

Prenons un exemple pour en savoir plus sur les événements routés personnalisés. Suivez les étapes ci-dessous -

  • Créer un nouveau projet WPF avec WPFCustomRoutedEvent

  • Faites un clic droit sur votre solution et sélectionnez Ajouter> Nouvel élément ...

  • La boîte de dialogue suivante s'ouvre, maintenant sélectionnez Custom Control (WPF) et nommez-le MyCustomControl.

  • Clique le Add et vous verrez que deux nouveaux fichiers (Themes / Generic.xaml et MyCustomControl.cs) seront ajoutés dans votre solution.

Le code XAML suivant définit le style du contrôle personnalisé dans le fichier Generic.xaml.

<ResourceDictionary 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomRoutedEvent">
	
   <Style TargetType = "{x:Type local:MyCustomControl}"> 
      <Setter Property = "Margin" Value = "50"/> 
      <Setter Property = "Template"> 
         <Setter.Value> 
            <ControlTemplate TargetType = "{x:Type local:MyCustomControl}">
				
               <Border Background = "{TemplateBinding Background}" 
                  BorderBrush = "{TemplateBinding BorderBrush}" 
                  BorderThickness = "{TemplateBinding BorderThickness}"> 
                  <Button x:Name = "PART_Button" Content = "Click Me" /> 
               </Border> 
					
            </ControlTemplate> 
         </Setter.Value> 
      </Setter> 
   </Style> 
	
</ResourceDictionary>

Ci-dessous, le code C # pour le MyCustomControl class qui hérite de la Control class dans lequel un événement routé personnalisé Click est créé pour le contrôle personnalisé.

using System.Windows; 
using System.Windows.Controls;  

namespace WPFCustomRoutedEvent { 

   public class MyCustomControl : Control { 
	
      static MyCustomControl() { 
         DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), 
            new FrameworkPropertyMetadata(typeof(MyCustomControl))); 
      } 
		
      public override void OnApplyTemplate() { 
         base.OnApplyTemplate();
			
         //demo purpose only, check for previous instances and remove the handler first 
         var button  =  GetTemplateChild("PART_Button") as Button; 
         if (button ! =  null) 
         button.Click + =  Button_Click;  
      } 
		
      void Button_Click(object sender, RoutedEventArgs e) { 
         RaiseClickEvent(); 
      } 
		
      public static readonly RoutedEvent ClickEvent  =  
         EventManager.RegisterRoutedEvent("Click", RoutingStrategy.Bubble, 
         typeof(RoutedEventHandler), typeof(MyCustomControl)); 
			
      public event RoutedEventHandler Click { 
         add { AddHandler(ClickEvent, value); } 
         remove { RemoveHandler(ClickEvent, value); } 
      } 
		
      protected virtual void RaiseClickEvent() { 
         RoutedEventArgs args = new RoutedEventArgs(MyCustomControl.ClickEvent); 
         RaiseEvent(args); 
      }
		
   } 
}

Voici l'implémentation d'événement routé personnalisé en C # qui affichera une boîte de message lorsque l'utilisateur cliquera dessus.

using System.Windows;  

namespace WPFCustomRoutedEvent { 
   // <summary> 
      // Interaction logic for MainWindow.xaml
   // </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void MyCustomControl_Click(object sender, RoutedEventArgs e) { 
         MessageBox.Show("It is the custom routed event of your custom control"); 
      } 
		
   } 
}

Voici l'implémentation dans MainWindow.xaml pour ajouter le contrôle personnalisé avec un événement routé Click.

<Window x:Class = "WPFCustomRoutedEvent.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomRoutedEvent"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <local:MyCustomControl Click = "MyCustomControl_Click" /> 
   </Grid> 
	
</Window>

Lorsque le code ci-dessus est compilé et exécuté, il produira la fenêtre suivante qui contient un contrôle personnalisé.

Lorsque vous cliquez sur le contrôle personnalisé, il produira le message suivant.

Windows Presentation Foundation (WPF) permet aux développeurs de créer et de créer facilement des applications basées sur l'interface utilisateur visuellement enrichies.

  • Les éléments ou contrôles d'interface utilisateur classiques dans d'autres frameworks d'interface utilisateur sont également améliorés dans les applications WPF.

  • Tous les contrôles WPF standard se trouvent dans la boîte à outils qui fait partie de System.Windows.Controls.

  • Ces contrôles peuvent également être créés en langage de balisage XAML.

La hiérarchie d'héritage complète des contrôles WPF est la suivante:

Le tableau suivant contient une liste de contrôles dont nous parlerons dans les chapitres suivants.

Sr. No. Commandes et description
1 Bouton

Un contrôle qui répond à l'entrée de l'utilisateur

2 Calendrier

Représente un contrôle qui permet à un utilisateur de sélectionner une date à l'aide d'un affichage de calendrier visuel.

3 CheckBox

Un contrôle qu'un utilisateur peut sélectionner ou effacer.

4 Boîte combo

Une liste déroulante d'éléments parmi lesquels un utilisateur peut sélectionner.

5 Menu contextuel

Obtient ou définit l'élément de menu contextuel qui doit apparaître chaque fois que le menu contextuel est demandé via l'interface utilisateur (UI) à partir de cet élément.

6 Grille de données

Représente un contrôle qui affiche des données dans une grille personnalisable.

sept Sélecteur de date

Un contrôle qui permet à un utilisateur de sélectionner une date.

8 Dialogues

Une application peut également afficher des fenêtres supplémentaires pour aider l'utilisateur à rassembler ou afficher des informations importantes.

9 GridView

Contrôle qui présente une collection d'éléments dans des lignes et des colonnes qui peuvent défiler horizontalement.

dix Image

Un contrôle qui présente une image.

11 Étiquette

Affiche du texte sur un formulaire. Fournit un support pour les clés d'accès.

12 ListBox

Un contrôle qui présente une liste en ligne d'éléments parmi lesquels l'utilisateur peut sélectionner.

13 Les menus

Représente un contrôle de menu Windows qui vous permet d'organiser hiérarchiquement les éléments associés aux commandes et aux gestionnaires d'événements.

14 PasswordBox

Un contrôle pour entrer les mots de passe.

15 Apparaitre

Affiche le contenu par-dessus le contenu existant, dans les limites de la fenêtre d'application.

16 Barre de progression

Un contrôle qui indique la progression en affichant une barre.

17 Bouton radio

Contrôle qui permet à un utilisateur de sélectionner une seule option dans un groupe d'options.

18 ScrollViewer

Un contrôle conteneur qui permet à l'utilisateur de faire un panoramique et de zoomer sur son contenu.

19 Glissière

Contrôle qui permet à l'utilisateur de sélectionner parmi une plage de valeurs en déplaçant un contrôle Thumb le long d'une piste.

20 TextBlock

Un contrôle qui affiche du texte.

21 Bouton à bascule

Un bouton qui peut être basculé entre 2 états.

22 Info-bulle

Une fenêtre contextuelle qui affiche des informations sur un élément.

23 La fenêtre

La fenêtre racine qui fournit l'option minimiser / agrandir, la barre de titre, la bordure et le bouton de fermeture

24 Contrôles tiers

Utilisez des contrôles tiers dans vos applications WPF.

Nous discuterons de tous ces contrôles un par un avec leur mise en œuvre.

La disposition des contrôles est très importante et critique pour la convivialité de l'application. Il est utilisé pour organiser un groupe d'éléments GUI dans votre application. Il y a certaines choses importantes à considérer lors de la sélection des panneaux de disposition -

  • Positions des éléments enfants
  • Tailles des éléments enfants
  • Superposition d'éléments enfants qui se chevauchent

La disposition fixe des pixels des commandes ne fonctionne pas lorsque l'application doit être placée sur différentes résolutions d'écran. XAML fournit un ensemble complet de panneaux de disposition intégrés pour organiser les éléments de l'interface graphique de manière appropriée. Certains des panneaux de mise en page les plus couramment utilisés et les plus populaires sont les suivants:

Sr. No. Panneaux et description
1 Panneau de pile

Le panneau Stack est un panneau de disposition simple et utile en XAML. Dans le panneau de pile, les éléments enfants peuvent être organisés sur une seule ligne, horizontalement ou verticalement, en fonction de la propriété d'orientation.

2 Panneau Wrap

Dans WrapPanel, les éléments enfants sont positionnés dans un ordre séquentiel, de gauche à droite ou de haut en bas en fonction de la propriété d'orientation.

3 Panneau Dock

DockPanel définit une zone pour organiser les éléments enfants les uns par rapport aux autres, horizontalement ou verticalement. Avec DockPanel, vous pouvez facilement ancrer les éléments enfants en haut, en bas, à droite, à gauche et au centre à l'aide duDock propriété.

4 Panneau de toile

Le panneau Canevas est le panneau de disposition de base dans lequel les éléments enfants peuvent être positionnés explicitement à l'aide de coordonnées relatives au Canvas n'importe quel côté tel que gauche, droite, haut et bas.

5 Panneau de grille

Un panneau de grille fournit une zone flexible qui se compose de lignes et de colonnes. Dans une grille, les éléments enfants peuvent être organisés sous forme de tableau.

L'imbrication de la disposition signifie l'utilisation du panneau de disposition à l'intérieur d'une autre disposition, par exemple définir des panneaux empilés à l'intérieur d'une grille. Ce concept est largement utilisé pour tirer parti des avantages de plusieurs mises en page dans une application. Dans l'exemple suivant, nous utiliserons des panneaux empilés à l'intérieur d'une grille.

Jetons un coup d'œil au code XAML suivant.

<Window x:Class = "WPFNestingLayouts.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFNestingLayouts" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid Background = "AntiqueWhite"> 
      <Grid.RowDefinitions> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "*" /> 
      </Grid.ColumnDefinitions> 
		
      <Label Content = "Employee Info" FontSize = "15"
         FontWeight = "Bold" Grid.Column = "0" Grid.Row = "0"/> 
			
      <StackPanel Grid.Column = "0" Grid.Row = "1" Orientation = "Horizontal"> 
         <Label Content = "Name"  VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtName" Text = "Muhammad Ali" VerticalAlignment = "Center"
            Width = "200">
         </TextBox> 
      </StackPanel>
		
      <StackPanel Grid.Column = "0" Grid.Row = "2" Orientation = "Horizontal"> 
         <Label Content = "ID" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtCity" Text = "421" VerticalAlignment = "Center"
            Width = "50">
         </TextBox> 
      </StackPanel>
		
      <StackPanel Grid.Column = "0" Grid.Row = "3" Orientation = "Horizontal"> 
         <Label Content = "Age" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtState" Text = "32" VerticalAlignment = "Center"
            Width = "50"></TextBox> 
      </StackPanel> 
		
      <StackPanel Grid.Column = "0" Grid.Row = "4" Orientation = "Horizontal"> 
         <Label Content = "Title" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtCountry" Text = "Programmer" VerticalAlignment = "Center"
            Width = "200"></TextBox> 
      </StackPanel> 
		
   </Grid> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante.

Nous vous recommandons d'exécuter l'exemple de code ci-dessus et d'essayer d'autres dispositions d'imbrication.

Windows Presentation Foundation (WPF) fournit une API puissante à l'aide des applications qui peuvent obtenir des entrées à partir de divers périphériques tels que la souris, le clavier et les écrans tactiles. Dans ce chapitre, nous aborderons les types d'entrées suivants qui peuvent être gérés dans les applications WPF -

Sr. No. Entrées et description
1 Souris

Il existe différents types d'entrées de souris telles que MouseDown, MouseEnter, MouseLeave, etc.

2 Clavier

Il existe de nombreux types d'entrées clavier telles que KeyDown, KeyUp, TextInput, etc.

3 ContextMenu ou RoutedCommands

Les RoutedCommands permettent la gestion des entrées à un niveau plus sémantique. Ce sont en fait des instructions simples comme Nouveau, Ouvrir, Copier, Couper et Enregistrer.

4 Multi-touches

Windows 7 et ses versions supérieures ont la capacité de recevoir des entrées de plusieurs appareils tactiles. Les applications WPF peuvent également gérer l'entrée tactile comme une autre entrée, telle que la souris ou le clavier, en déclenchant des événements lorsqu'un contact se produit.

L'argument de ligne de commande est un mécanisme par lequel un utilisateur peut transmettre un ensemble de paramètres ou de valeurs à une application WPF lors de son exécution. Ces arguments sont très importants pour contrôler une application de l'extérieur, par exemple, si vous souhaitez ouvrir un document Word à partir de l'invite de commande, vous pouvez utiliser cette commande "C:\> start winword word1.docx"Et il s'ouvrira word1.docx document.

Les arguments de ligne de commande sont gérés dans la fonction de démarrage. Voici un exemple simple qui montre comment passer des arguments de ligne de commande à une application WPF. Créons une nouvelle application WPF avec le nomWPFCommandLine.

  • Faites glisser une zone de texte de la boîte à outils vers la fenêtre de conception.

  • Dans cet exemple, nous allons passer un chemin de fichier txt à notre application en tant que paramètre de ligne de commande.

  • Le programme lira le fichier txt puis écrira tout le texte dans la zone de texte.

  • Le code XAML suivant crée une zone de texte et l'initialise avec certaines propriétés.

<Window x:Class = "WPFCommandLine.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFCommandLine" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"  
         Height = "180" Margin = "100" TextWrapping = "Wrap" 
         VerticalAlignment = "Top" Width = "300"/> 
   </Grid> 
	
</Window>
  • Maintenant, abonnez-vous à l'événement de démarrage dans le fichier App.xaml comme indiqué ci-dessous.
<Application x:Class = "WPFCommandLine.App" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   xmlns:local = "clr-namespace:WPFCommandLine" 
   StartupUri = "MainWindow.xaml" Startup = "app_Startup"> 
	
   <Application.Resources> 
          
   </Application.Resources>
	
</Application>
  • Vous trouverez ci-dessous l'implémentation de l'événement app_Startup dans App.xaml.cs qui obtiendra les arguments de la ligne de commande.

using System.Windows;
  
namespace WPFCommandLine { 
   /// <summary> 
      /// Interaction logic for App.xaml 
   /// </summary> 
	
   public partial class App : Application { 
      public static string[] Args;
		
      void app_Startup(object sender, StartupEventArgs e) { 
         // If no command line arguments were provided, don't process them 
         if (e.Args.Length == 0) return;
			
         if (e.Args.Length > 0) { 
            Args = e.Args; 
         } 
      } 
   } 
}
  • Maintenant, dans la classe MainWindow, le programme ouvrira le fichier txt et écrira tout le texte sur la zone de texte.

  • S'il y a une erreur trouvée, le programme affichera un message d'erreur sur la zone de texte.

using System; 
using System.IO; 
using System.Windows;  

namespace WPFCommandLine { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         String[] args = App.Args;
			
         try {
            // Open the text file using a stream reader. 
            using (StreamReader sr = new StreamReader(args[0])) { 
               // Read the stream to a string, and write  
               // the string to the text box 
               String line = sr.ReadToEnd(); 
               textBox.AppendText(line.ToString()); 
               textBox.AppendText("\n"); 
            } 
         } 
         catch (Exception e) { 
            textBox.AppendText("The file could not be read:"); 
            textBox.AppendText("\n"); 
            textBox.AppendText(e.Message); 
         } 
      } 
   } 
}
  • Lorsque le code ci-dessus est compilé et exécuté, il produira une fenêtre vide avec une zone de texte car ce programme a besoin d'un argument de ligne de commande. Ainsi, Visual Studio fournit un moyen simple d'exécuter votre application avec des paramètres de ligne de commande.

  • Cliquez avec le bouton droit sur votre projet WPF dans l'explorateur de solutions et sélectionnez les propriétés, il affichera la fenêtre suivante.

  • Sélectionnez l'option Déboguer et écrivez le chemin du fichier dans l'argument de ligne de commande.

  • Créez un fichier txt avec Test.txt et écrivez du texte dans ce fichier et enregistrez-le à n'importe quel emplacement. Dans ce cas, le fichier txt est enregistré sur "D:\" disque dur.

  • Enregistrez les modifications dans votre projet et compilez et exécutez votre application maintenant. Vous verrez le texte dans TextBox que le programme lit dans le fichier Text.txt.

Essayons maintenant de changer le nom du fichier sur votre machine de Test.txt à Test1.txt et exécutez à nouveau votre programme, alors vous verrez ce message d'erreur dans la zone de texte.

Nous vous recommandons d'exécuter le code ci-dessus et de suivre toutes les étapes pour exécuter votre application avec succès.

La liaison de données est un mécanisme dans les applications WPF qui fournit un moyen simple et facile aux applications Windows Runtime d'afficher et d'interagir avec les données. Dans ce mécanisme, la gestion des données est entièrement séparée de la manière dont les données.

La liaison de données permet le flux de données entre les éléments de l'interface utilisateur et l'objet de données sur l'interface utilisateur. Lorsqu'une liaison est établie et que les données ou votre modèle d'entreprise changent, elle reflète automatiquement les mises à jour des éléments de l'interface utilisateur et vice versa. Il est également possible de se lier, non pas à une source de données standard, mais à un autre élément de la page.

La liaison de données est de deux types - one-way data binding et two-way data binding.

Liaison de données unidirectionnelle

Dans la liaison unidirectionnelle, les données sont liées de leur source (c'est-à-dire l'objet qui contient les données) à sa cible (c'est-à-dire l'objet qui affiche les données)

  • Prenons un exemple simple pour comprendre en détail la liaison de données unidirectionnelle. Tout d'abord, créez un nouveau projet WPF avec le nomWPFDataBinding.

  • Le code XAML suivant crée deux étiquettes, deux zones de texte et un bouton et les initialise avec certaines propriétés.

<Window x:Class = "WPFDataBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataBinding" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
	
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "Auto" /> 
         <ColumnDefinition Width = "200" /> 
      </Grid.ColumnDefinitions>
		
      <Label Name = "nameLabel" Margin = "2">_Name:</Label> 
		
      <TextBox Name = "nameText" Grid.Column = "1" Margin = "2" 
         Text = "{Binding Name, Mode = OneWay}"/>  
			
      <Label Name = "ageLabel" Margin = "2" Grid.Row = "1">_Age:</Label> 
		
      <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "2" 
         Text = "{Binding Age, Mode = OneWay}"/>  
			
      <StackPanel Grid.Row = "2" Grid.ColumnSpan = "2"> 
         <Button Content = "_Show..." Click="Button_Click" /> 
      </StackPanel> 
		
   </Grid> 
</Window>
  • Les propriétés de texte des deux zones de texte se lient à «Nom» et «Âge», qui sont des variables de classe de la classe Person illustrée ci-dessous.

  • Dans la classe Person, nous n'avons que deux variables Name et Age, et son objet est initialisé dans MainWindow classe.

  • Dans le code XAML, nous lions à une propriété Name et Age, mais nous n'avons pas sélectionné l'objet auquel appartient cette propriété.

  • Le moyen le plus simple est d'attribuer un objet à DataContext dont nous lions les propriétés dans le code C # suivant dans MainWindowconstructor.

using System.Windows;  
namespace WPFDataBinding { 

   public partial class MainWindow : Window {
	
      Person person = new Person { Name = "Salman", Age = 26 };
		
      public MainWindow() { 
         InitializeComponent(); 
         this.DataContext = person; 
      } 
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         string message = person.Name + " is " + person.Age; 
         MessageBox.Show(message); 
      } 
   } 
	
   public class Person { 
	
      private string nameValue;
		
      public string Name { 
         get { return nameValue; } 
         set { nameValue = value; } 
      } 
		
      private double ageValue; 
		
      public double Age { 
         get { return ageValue; } 
				
         set { 
            if (value != ageValue) { 
               ageValue = value; 
            } 
         } 
      }
		
   } 
}
  • Exécutons cette application et vous pouvez voir immédiatement dans notre MainWindow que nous avons lié avec succès le nom et l'âge de cet objet Person.

Lorsque vous appuyez sur le Show bouton, il affichera le nom et l'âge sur la boîte de message.

Modifions le nom et l'âge dans la boîte de dialogue.

Si vous cliquez maintenant sur le bouton Afficher, il affichera à nouveau le même message.

En effet, le mode de liaison de données est défini sur unidirectionnel dans le code XAML. Pour afficher les données mises à jour, vous devez comprendre la liaison de données bidirectionnelle.

Liaison de données bidirectionnelle

Dans la liaison bidirectionnelle, l'utilisateur peut modifier les données via l'interface utilisateur et mettre à jour ces données dans la source. Si la source change pendant que l'utilisateur regarde la vue, vous voulez que la vue soit mise à jour.

Prenons le même exemple mais ici, nous allons changer le mode de liaison de One Way à Two Way dans le code XAML.

<Window x:Class = "WPFDataBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataBinding" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
	
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "Auto" /> 
         <ColumnDefinition Width = "200" /> 
      </Grid.ColumnDefinitions> 
		
      <Label Name = "nameLabel" Margin = "2">_Name:</Label> 
      <TextBox Name = "nameText" Grid.Column = "1" Margin = "2" 
         Text = "{Binding Name, Mode = TwoWay}"/>  
      <Label Name = "ageLabel" Margin = "2" Grid.Row = "1">_Age:</Label> 
      <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "2" 
         Text = "{Binding Age, Mode = TwoWay}"/> 
			
      <StackPanel Grid.Row = "2" Grid.ColumnSpan = "2"> 
         <Button Content = "_Show..." Click = "Button_Click" /> 
      </StackPanel>
		
   </Grid>
	
</Window>

Exécutons à nouveau cette application.

Il produira le même résultat -

Modifions maintenant les valeurs Nom et Âge -

Si vous cliquez sur le bouton Afficher maintenant, il affichera le message mis à jour.

Nous vous recommandons d'exécuter le code ci-dessus avec les deux cas pour une meilleure compréhension du concept.

Les ressources sont normalement des définitions liées à un objet que vous prévoyez simplement d'utiliser plus d'une fois. C'est la possibilité de stocker des données localement pour les contrôles ou pour la fenêtre courante ou globalement pour l'ensemble des applications.

Définir un objet comme une ressource nous permet d'y accéder depuis un autre endroit. Cela signifie que l'objet peut être réutilisé. Les ressources sont définies dans des dictionnaires de ressources et tout objet peut être défini comme une ressource, ce qui en fait un actif partageable. Une clé unique est spécifiée pour une ressource XAML et avec cette clé, elle peut être référencée à l'aide d'une extension de balisage StaticResource.

Les ressources peuvent être de deux types -

  • StaticResource
  • DynamicResource

Un StaticResource est une recherche ponctuelle, tandis qu'un DynamicResource fonctionne plus comme une liaison de données. Il se souvient qu'une propriété est associée à une clé de ressource particulière. Si l'objet associé à cette clé change, la ressource dynamique met à jour la propriété cible.

Exemple

Voici une application simple pour la ressource SolidColorBrush.

  • Créons un nouveau projet WPF avec le nom WPFResouces.

  • Faites glisser deux rectangles et définissez leurs propriétés comme indiqué dans le code XAML suivant.

<Window x:Class = "WPFResources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFResources" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Window.Resources> 
      <SolidColorBrush x:Key = "brushResource" Color = "Blue" /> 
   </Window.Resources> 
	
   <StackPanel> 
      <Rectangle Height = "50" Margin = "20" Fill = "{StaticResource brushResource}" /> 
      <Rectangle Height = "50" Margin = "20" Fill = "{DynamicResource brushResource}" /> 
      <Button x:Name = "changeResourceButton"
         Content = "_Change Resource" Click = "changeResourceButton_Click" /> 
   </StackPanel> 
	
</Window>
  • Dans le code XAML ci-dessus, vous pouvez voir qu'un rectangle a StaticResource et l'autre a DynamicResource et la couleur de brushResource est Bisque.

  • Lorsque vous compilez et exécutez le code, il produira le MainWindow suivant.

Lorsque vous cliquez sur le bouton «Modifier la ressource», vous verrez que le rectangle avec DynamicResource changera sa couleur en rouge.

Portée de la ressource

Les ressources sont définies dans resource dictionaries, mais il existe de nombreux endroits où un dictionnaire de ressources peut être défini. Dans l'exemple ci-dessus, un dictionnaire de ressources est défini au niveau de la fenêtre / page. Dans quel dictionnaire une ressource est définie limite immédiatement la portée de cette ressource. Ainsi, la portée, c'est-à-dire où vous pouvez utiliser la ressource, dépend de l'endroit où vous l'avez définie.

  • Définissez la ressource dans le dictionnaire de ressources d'une grille et elle est accessible uniquement par cette grille et par ses éléments enfants.

  • Définissez-le sur une fenêtre / page et il est accessible par tous les éléments de cette fenêtre / page.

  • La racine de l'application se trouve dans le dictionnaire de ressources App.xaml. C'est la racine de notre application, donc les ressources définies ici sont étendues à l'ensemble de l'application.

En ce qui concerne la portée de la ressource, les plus souvent sont le niveau de l'application, le niveau de la page et un niveau d'élément spécifique comme un Grid, StackPanel, etc.

L'application ci-dessus a des ressources dans son niveau Fenêtre / page.

Dictionnaires de ressources

Les dictionnaires de ressources dans les applications XAML impliquent que les dictionnaires de ressources sont conservés dans des fichiers séparés. Il est suivi dans presque toutes les applications XAML. La définition de ressources dans des fichiers séparés peut présenter les avantages suivants:

  • Séparation entre la définition des ressources dans le dictionnaire de ressources et le code lié à l'interface utilisateur.

  • La définition de toutes les ressources dans un fichier séparé tel que App.xaml les rendrait disponibles dans l'application.

Alors, comment définissons-nous nos ressources dans un dictionnaire de ressources dans un fichier séparé? Eh bien, c'est très simple, ajoutez simplement un nouveau dictionnaire de ressources via Visual Studio en suivant les étapes ci-dessous -

  • Dans votre solution, ajoutez un nouveau dossier et nommez-le ResourceDictionaries.

  • Cliquez avec le bouton droit sur ce dossier et sélectionnez Dictionnaire de ressources dans Ajouter un élément de sous-menu et nommez-le DictionaryWithBrush.xaml

Exemple

Prenons maintenant le même exemple, mais ici, nous allons définir le dictionnaire de ressources au niveau de l'application. Le code XAML pour MainWindow.xaml est le suivant -

<Window x:Class = "WPFResources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFResources" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <StackPanel> 
      <Rectangle Height = "50" Margin = "20" Fill = "{StaticResource brushResource}" /> 
      <Rectangle Height = "50" Margin = "20" Fill = "{DynamicResource brushResource}" /> 
      <Button x:Name = "changeResourceButton"
         Content = "_Change Resource" Click = "changeResourceButton_Click" /> 
   </StackPanel> 
	
</Window>

Voici l'implémentation dans DictionaryWithBrush.xaml -

<ResourceDictionary xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"> 
   <SolidColorBrush x:Key = "brushResource" Color = "Blue" /> 
</ResourceDictionary>

Voici l'implémentation dans app.xaml -

<Application x:Class="WPFResources.App" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   StartupUri = "MainWindow.xaml"> 
	
   <Application.Resources> 
      <ResourceDictionary Source = " XAMLResources\ResourceDictionaries\DictionaryWithBrush.xaml"/> 
   </Application.Resources> 
	
</Application>

Lorsque le code ci-dessus est compilé et exécuté, il produira la sortie suivante -

Lorsque vous cliquez sur le bouton Modifier la ressource, le rectangle change sa couleur en rouge.

Nous vous recommandons d'exécuter le code ci-dessus et d'essayer d'autres ressources (par exemple, la couleur d'arrière-plan).

Un modèle décrit l'aspect général et l'aspect visuel d'un contrôle. Pour chaque contrôle, un modèle par défaut lui est associé qui lui donne son apparence. Dans les applications WPF, vous pouvez facilement créer vos propres modèles lorsque vous souhaitez personnaliser le comportement visuel et l'apparence visuelle d'un contrôle.

La connectivité entre la logique et le modèle peut être obtenue par liaison de données. La principale différence entrestyles et templates sont énumérés ci-dessous -

  • Les styles ne peuvent modifier l'apparence de votre contrôle qu'avec les propriétés par défaut de ce contrôle.

  • Avec les modèles, vous pouvez accéder à plus de parties d'un contrôle que dans les styles. Vous pouvez également spécifier le comportement existant et nouveau d'un contrôle.

Il existe deux types de modèles les plus couramment utilisés -

  • Modèle de contrôle
  • Modèle de données

Modèle de contrôle

Le modèle de contrôle définit l'apparence visuelle d'un contrôle. Tous les éléments de l'interface utilisateur ont une sorte d'apparence et de comportement, par exemple, Button a une apparence et un comportement. L'événement de clic ou l'événement de survol de la souris sont les comportements qui sont déclenchés en réponse à un clic et au survol et il existe également une apparence par défaut du bouton qui peut être modifiée par le modèle de contrôle.

Exemple

Prenons un exemple simple. Nous allons créer deux boutons (l'un avec le modèle et l'autre est le bouton par défaut) et les initialiser avec certaines propriétés.

<Window x:Class = "TemplateDemo.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Window.Resources> 
      <ControlTemplate x:Key = "ButtonTemplate" TargetType = "Button">
		
         <Grid> 
            <Ellipse x:Name = "ButtonEllipse" Height = "100" Width = "150" > 
               <Ellipse.Fill> 
                  <LinearGradientBrush StartPoint = "0,0.2" EndPoint = "0.2,1.4"> 
                     <GradientStop Offset = "0" Color = "Red" /> 
                     <GradientStop Offset = "1" Color = "Orange" /> 
                  </LinearGradientBrush> 
               </Ellipse.Fill> 
            </Ellipse> 
				
            <ContentPresenter Content = "{TemplateBinding Content}" 
               HorizontalAlignment = "Center" VerticalAlignment = "Center" /> 
         </Grid> 
			
         <ControlTemplate.Triggers> 
			
            <Trigger Property = "IsMouseOver" Value = "True"> 
               <Setter TargetName = "ButtonEllipse" Property = "Fill" > 
                  <Setter.Value> 
                     <LinearGradientBrush StartPoint = "0,0.2" EndPoint = "0.2,1.4"> 
                        <GradientStop Offset = "0" Color = "YellowGreen" /> 
                        <GradientStop Offset = "1" Color = "Gold" /> 
                     </LinearGradientBrush> 
                  </Setter.Value> 
               </Setter> 
            </Trigger> 
				
            <Trigger Property = "IsPressed" Value = "True"> 
               <Setter Property = "RenderTransform"> 
                  <Setter.Value> 
                     <ScaleTransform ScaleX = "0.8" ScaleY = "0.8" 
                        CenterX = "0" CenterY = "0"  /> 
                  </Setter.Value> 
               </Setter> 
               <Setter Property = "RenderTransformOrigin" Value = "0.5,0.5" /> 
            </Trigger> 
				
         </ControlTemplate.Triggers> 
			
      </ControlTemplate> 
   </Window.Resources> 
	
   <StackPanel> 
      <Button Content = "Round Button!"
         Template = "{StaticResource ButtonTemplate}" 
         Width = "150" Margin = "50" /> 
      <Button Content = "Default Button!" Height = "40" 
         Width = "150" Margin = "5" />
   </StackPanel> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il affichera le MainWindow suivant.

Lorsque vous déplacez la souris sur le bouton avec un modèle personnalisé, sa couleur change comme indiqué ci-dessous.

Modèle de données

Un modèle de données définit et spécifie l'apparence et la structure d'une collection de données. Il offre la flexibilité de formater et de définir la présentation des données sur n'importe quel élément de l'interface utilisateur. Il est principalement utilisé sur les contrôles d'élément liés aux données tels que ComboBox, ListBox, etc.

Exemple

  • Prenons un exemple simple pour comprendre le concept de modèle de données. Créer un nouveau projet WPF avec le nomWPFDataTemplates.

  • Dans le code XAML suivant, nous allons créer un modèle de données en tant que ressource pour contenir les étiquettes et les zones de texte. Il existe également un bouton et une zone de liste pour afficher les données.

<Window x:Class = "WPFDataTemplates.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataTemplates" 
   xmlns:loc = "clr-namespace:WPFDataTemplates" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Window.Resources> 
      <DataTemplate DataType = "{x:Type loc:Person}"> 
		
         <Grid> 
            <Grid.RowDefinitions> 
               <RowDefinition Height = "Auto" /> 
               <RowDefinition Height = "Auto" /> 
            </Grid.RowDefinitions> 
				
            <Grid.ColumnDefinitions> 
               <ColumnDefinition Width = "Auto" /> 
               <ColumnDefinition Width = "200" /> 
            </Grid.ColumnDefinitions>
				
            <Label Name = "nameLabel" Margin = "10"/> 
            <TextBox Name = "nameText" Grid.Column = "1" Margin = "10" 
               Text = "{Binding Name}"/>  
            <Label Name = "ageLabel" Margin = "10" Grid.Row = "1"/> 
            <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "10" 
               Text = "{Binding Age}"/> 
         </Grid> 
			
      </DataTemplate> 
   </Window.Resources> 
	
   <Grid> 
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <ListBox ItemsSource = "{Binding}" />  
      <StackPanel Grid.Row = "1" > 
         <Button Content = "_Show..." Click = "Button_Click" Width = "80" HorizontalAlignment = "Left" Margin = "10"/> 
      </StackPanel> 
		
   </Grid> 
	
</Window>

Voici implementation in C# dans laquelle une liste d'objets Person sont affectés à DataContext, l'implémentation de la classe Person et l'événement de clic de bouton.

using System.Collections.Generic; 
using System.Windows;
  
namespace WPFDataTemplates { 

   public partial class MainWindow : Window { 
	
      Person src = new Person { Name = "Ali", Age = 27 }; 
      List<Person> people = new List<Person>(); 
		
      public MainWindow() { 
         InitializeComponent(); 
         people.Add(src); 
         people.Add(new Person { Name = "Mike", Age = 62 }); 
         people.Add(new Person { Name = "Brian", Age = 12 });  
         this.DataContext = people; 
      } 
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         string message = src.Name + " is " + src.Age; 
         MessageBox.Show(message); 
      } 
   } 
	
   public class Person { 
      private string nameValue; 
		
      public string Name { 
         get { return nameValue; } 
         set { nameValue = value; } 
      }  
		
      private double ageValue; 
		
      public double Age { 
         get { return ageValue; } 
         set { 
            if (value != ageValue) { 
            ageValue = value; 
            } 
         } 
      } 
   } 
	
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante. Il contient une liste et à l'intérieur de la zone de liste, chaque élément de zone de liste contient les données d'objet de classe Person qui sont affichées sur les étiquettes et les zones de texte.

Le framework .NET fournit plusieurs stratégies pour personnaliser et personnaliser l'apparence d'une application. Les styles nous offrent la flexibilité de définir certaines propriétés d'un objet et de réutiliser ces paramètres spécifiques sur plusieurs objets pour un aspect cohérent.

  • Dans les styles, vous ne pouvez définir que les propriétés existantes d'un objet telles que la hauteur, la largeur, la taille de la police, etc.

  • Seul le comportement par défaut d'un contrôle peut être spécifié.

  • Plusieurs propriétés peuvent être ajoutées dans un seul style.

Les styles sont utilisés pour donner un aspect ou une apparence uniforme à un ensemble de contrôles. Les styles implicites sont utilisés pour appliquer une apparence à tous les contrôles d'un type donné et simplifier l'application. Imaginez trois boutons, tous doivent avoir la même apparence, la même largeur et la même hauteur, la même taille de police, la même couleur de premier plan, etc. Jetez un œil au diagramme suivant.

Mais dans les applications réelles, vous en aurez généralement beaucoup plus qui doivent avoir exactement la même apparence. Et pas seulement les boutons, bien sûr, vous souhaiterez généralement que vos blocs de texte, vos zones de texte et vos zones de liste déroulante, etc. aient la même apparence dans toute votre application. Il doit sûrement y avoir une meilleure façon d’y parvenir, et elle est connue sous le nom destyling. Vous pouvez considérer un style comme un moyen pratique d'appliquer un ensemble de valeurs de propriété à plusieurs éléments. Jetez un œil au diagramme suivant.

Exemple

Prenons un exemple simple pour comprendre ce concept. Commencez par créer un nouveau projet WPF.

  • Faites glisser trois boutons de la boîte à outils vers la fenêtre de conception.

  • Le code XAML suivant crée trois boutons et les initialise avec certaines propriétés.

<Window x:Class = "WPFStyle.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace: WPFStyle" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button Content = "Button1" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
      <Button Content = "Button2" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
      <Button Content = "Button3" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
   </StackPanel> 
	
</Window>

Lorsque vous regardez le code ci-dessus, vous verrez que pour tous les boutons, la hauteur, la largeur, la couleur de premier plan, la taille de police et les propriétés de marge sont les mêmes. Maintenant, lorsque le code ci-dessus est compilé et exécuté, la fenêtre suivante s'affiche.

Regardons maintenant le même exemple, mais cette fois, nous allons utiliser style.

<Window x:Class = "XAMLStyle.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"
   xmlns:local = "clr-namespace:XAMLStyle" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Window.Resources> 
      <Style x:Key = "myButtonStyle" TargetType = "Button"> 
         <Setter Property = "Height" Value = "30" /> 
         <Setter Property = "Width" Value = "80" /> 
         <Setter Property = "Foreground" Value = "Blue" /> 
         <Setter Property = "FontSize" Value = "12" /> 
         <Setter Property = "Margin" Value = "10" /> 
      </Style> 
   </Window.Resources> 
	
   <StackPanel> 
      <Button Content = "Button1" Style = "{StaticResource myButtonStyle}" /> 
      <Button Content = "Button2" Style = "{StaticResource myButtonStyle}" /> 
      <Button Content = "Button3" Style="{StaticResource myButtonStyle}" /> 
   </StackPanel> 
	
</Window>

Les styles sont définis dans le dictionnaire de ressources et chaque style a un identificateur de clé unique et un type de cible. À l'intérieur de <style>, vous pouvez voir que plusieurs balises de définition sont définies pour chaque propriété qui sera incluse dans le style.

Dans l'exemple ci-dessus, toutes les propriétés communes de chaque bouton sont maintenant définies dans le style, puis le style est attribué à chaque bouton avec une clé unique en définissant la propriété de style via l'extension de balisage StaticResource.

Lorsque vous compilez et exécutez le code ci-dessus, il affichera la fenêtre suivante (la même sortie).

L'avantage de le faire comme ça est immédiatement évident, nous pouvons réutiliser ce style n'importe où dans sa portée; et si nous devons le changer, nous le changeons simplement une fois dans la définition de style au lieu de sur chaque élément.

À quel niveau un style est défini instantanément, la portée de ce style est limitée. Ainsi, la portée, c'est-à-dire où vous pouvez utiliser le style, dépend de l'endroit où vous l'avez défini. Les styles peuvent être définis aux niveaux suivants -

Sr.Non Niveaux et description
1 Niveau de contrôle

La définition d'un style au niveau du contrôle ne peut être appliquée qu'à ce contrôle particulier. Vous trouverez ci-dessous un exemple de niveau de contrôle où le bouton et TextBlock ont ​​leur propre style.

2 Niveau de mise en page

La définition d'un style à n'importe quel niveau de mise en page le rendra accessible uniquement par cette mise en page et ses éléments enfants.

3 Niveau de la fenêtre

La définition d'un style au niveau d'une fenêtre peut le rendre accessible par tous les éléments de cette fenêtre.

4 Niveau d'application

La définition d'un style au niveau de l'application peut le rendre accessible dans toute l'application. Prenons le même exemple, mais ici, nous allons mettre les styles dans le fichier app.xaml pour le rendre accessible dans toute l'application.

Un déclencheur vous permet essentiellement de modifier les valeurs de propriété ou d'effectuer des actions en fonction de la valeur d'une propriété. Ainsi, il vous permet de modifier dynamiquement l'apparence et / ou le comportement de votre contrôle sans avoir à en créer un nouveau.

Les déclencheurs sont utilisés pour modifier la valeur d'une propriété donnée, lorsque certaines conditions sont satisfaites. Les déclencheurs sont généralement définis dans un style ou à la racine d'un document qui sont appliqués à ce contrôle spécifique. Il existe trois types de déclencheurs -

  • Déclencheurs de propriété
  • Déclencheurs de données
  • Déclencheurs d'événements

Déclencheurs de propriété

Dans les déclencheurs de propriété, lorsqu'une modification se produit dans une propriété, elle entraîne une modification immédiate ou animée d'une autre propriété. Par exemple, vous pouvez utiliser un déclencheur de propriété pour modifier l'apparence d'un bouton lorsque la souris survole le bouton.

L'exemple de code suivant montre comment modifier la couleur de premier plan d'un bouton lorsque la souris survole le bouton.

<Window x:Class = "WPFPropertyTriggers.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Window.Resources> 
      <Style x:Key = "TriggerStyle" TargetType = "Button"> 
         <Setter Property = "Foreground" Value = "Blue" /> 
         <Style.Triggers> 
            <Trigger Property = "IsMouseOver" Value = "True"> 
               <Setter Property = "Foreground" Value = "Green" /> 
            </Trigger> 
         </Style.Triggers> 
      </Style> 
   </Window.Resources> 
	
   <Grid> 
      <Button Width = "100" Height = "70"
         Style = "{StaticResource TriggerStyle}" Content = "Trigger"/> 
   </Grid> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante -

Lorsque la souris survole le bouton, sa couleur de premier plan passe au vert.

Déclencheurs de données

Un déclencheur de données effectue certaines actions lorsque les données liées remplissent certaines conditions. Jetons un coup d'œil au code XAML suivant dans lequel une case à cocher et un bloc de texte sont créés avec certaines propriétés. Lorsque la case est cochée, il changera sa couleur de premier plan en rouge.

<Window x:Class = "WPFDataTrigger.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "Data Trigger" Height = "350" Width = "604">
	
   <StackPanel HorizontalAlignment = "Center"> 
      <CheckBox x:Name = "redColorCheckBox" 
         Content = "Set red as foreground color" Margin = "20"/> 
			
      <TextBlock Name = "txtblock" VerticalAlignment = "Center" 
         Text = "Event Trigger" FontSize = "24" Margin = "20"> 
         <TextBlock.Style> 
            <Style> 
               <Style.Triggers> 
                  <DataTrigger Binding = "{Binding ElementName = redColorCheckBox, Path = IsChecked}" 
                     Value = "true"> 
                     <Setter Property = "TextBlock.Foreground" Value = "Red"/> 
                     <Setter Property = "TextBlock.Cursor" Value = "Hand" /> 
                  </DataTrigger> 
               </Style.Triggers> 
            </Style> 
         </TextBlock.Style> 
      </TextBlock> 
		
   </StackPanel> 
	
</Window>

Lorsque le code ci-dessus est compilé et exécuté, il produira la sortie suivante -

Lorsque vous cochez la case, le bloc de texte change sa couleur de premier plan en rouge.

Déclencheurs d'événements

Un déclencheur d'événement exécute certaines actions lorsqu'un événement spécifique est déclenché. Il est généralement utilisé pour réaliser une animation sur un contrôle tel que DoubleAnumatio, ColorAnimation, etc. Dans l'exemple suivant, nous allons créer un bouton simple. Lorsque l'événement de clic est déclenché, il élargit la largeur et la hauteur du bouton.

<Window x:Class = "WPFEventTrigger.MainWindow"
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button Content = "Click Me" Width = "60" Height = "30">
		
         <Button.Triggers> 
            <EventTrigger RoutedEvent = "Button.Click"> 
               <EventTrigger.Actions> 
                  <BeginStoryboard> 
                     <Storyboard> 
							
                        <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = 
                           "Width" Duration = "0:0:4"> 
                           <LinearDoubleKeyFrame Value = "60" KeyTime = "0:0:0"/> 
                           <LinearDoubleKeyFrame Value = "120" KeyTime = "0:0:1"/> 
                           <LinearDoubleKeyFrame Value = "200" KeyTime = "0:0:2"/> 
                           <LinearDoubleKeyFrame Value = "300" KeyTime = "0:0:3"/> 
                        </DoubleAnimationUsingKeyFrames>
								
                        <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = "Height" 
                           Duration = "0:0:4"> 
                           <LinearDoubleKeyFrame Value = "30" KeyTime = "0:0:0"/> 
                           <LinearDoubleKeyFrame Value = "40" KeyTime = "0:0:1"/> 
                           <LinearDoubleKeyFrame Value = "80" KeyTime = "0:0:2"/> 
                           <LinearDoubleKeyFrame Value = "150" KeyTime = "0:0:3"/> 
                        </DoubleAnimationUsingKeyFrames>
								
                     </Storyboard> 
                  </BeginStoryboard> 
               </EventTrigger.Actions> 
            </EventTrigger> 
         </Button.Triggers> 
			
      </Button> 
   </Grid> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante -

En cliquant sur le bouton, vous remarquerez qu'il commencera à se développer dans les deux dimensions.

Nous vous recommandons de compiler et d'exécuter les exemples ci-dessus et d'appliquer également les déclencheurs à d'autres propriétés.

Il s'agit d'un mécanisme systématique d'identification et de correction des bogues ou des défauts dans un morceau de code qui ne se comportent pas comme prévu. Déboguer une application complexe où les sous-systèmes sont étroitement couplés n'est pas si simple, car la correction de bogues dans un sous-système peut créer des bogues dans un autre sous-système.

Débogage en C #

Dans les applications WPF, les programmeurs utilisent deux langages tels que C # et XAML. Si vous êtes familiarisé avec le débogage dans n'importe quel langage procédural tel que C # ou C / C ++ et que vous connaissez également l'utilisation des points d'arrêt, vous pouvez déboguer facilement la partie C # de votre application.

Prenons un exemple simple pour montrer comment déboguer un code C #. Créer un nouveau projet WPF avec le nomWPFDebuggingDemo. Faites glisser quatre étiquettes, trois zones de texte et un bouton de la boîte à outils. Jetez un œil au code XAML suivant.

<Window x:Class = "WPFDebuggingDemo.Window1" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "Window1" Height = "400" Width = "604"> 
	
   <Grid> 
      <TextBox Height = "23" Margin = "0,44,169,0" Name = "textBox1"  
         VerticalAlignment = "Top" HorizontalAlignment = "Right" Width = "120" /> 
			
      <TextBox Height = "23" Margin = "0,99,169,0" Name = "textBox2"  
         VerticalAlignment = "Top" HorizontalAlignment = "Right" Width = "120" /> 
			
      <TextBox HorizontalAlignment = "Right" Margin = "0,153,169,0"  
         Name = "textBox3" Width = "120" Height = "23" VerticalAlignment = "Top" /> 
			
      <Label Height = "28" Margin = "117,42,0,0" Name = "label1"  
         VerticalAlignment = "Top" HorizontalAlignment = "Left" Width = "120">
         Item 1</Label> 
			
      <Label Height = "28" HorizontalAlignment = "Left"  
         Margin = "117,99,0,0" Name = "label2" VerticalAlignment = "Top" Width = "120">
         Item 2</Label> 
			
      <Label HorizontalAlignment = "Left" Margin = "117,153,0,181"  
         Name = "label3" Width = "120">Item 3</Label>
			
      <Button Height = "23" HorizontalAlignment = "Right" Margin = "0,0,214,127"
         Name = "button1" VerticalAlignment = "Bottom" Width = "75"  
         Click = "button1_Click">Total</Button> 
			
      <Label Height = "28" HorizontalAlignment = "Right"  
         Margin = "0,0,169,66" Name = "label4" VerticalAlignment = "Bottom" Width = "120"/> 
			
   </Grid> 
	
</Window>

Vous trouverez ci-dessous le code C # dans lequel un événement de clic de bouton est implémenté.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text;
 
using System.Windows; 
using System.Windows.Controls; 
using System.Windows.Data; 
using System.Windows.Documents; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Media.Imaging; 
using System.Windows.Navigation; 
using System.Windows.Shapes;  

namespace WPFDebuggingDemo { 
   /// <summary> 
      /// Interaction logic for Window1.xaml 
   /// </summary> 
	
   public partial class Window1 : Window { 
	
      public Window1() {
         InitializeComponent();
      }
		
      private void button1_Click(object sender, RoutedEventArgs e) {
		
         if (textBox1.Text.Length > 0 && textBox2.Text.Length > 0 && textBox2.Text.Length > 0) {
            double total = Convert.ToDouble(textBox1.Text) + 
            Convert.ToDouble(textBox2.Text) + Convert.ToDouble(textBox3.Text); 
            label4.Content = total.ToString(); 
         } 
         else { 
            MessageBox.Show("Enter the value in all field.");
         } 
      } 
   } 
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante. Saisissez maintenant les valeurs dans les zones de texte et appuyez sur le bouton Total. Vous obtiendrez la valeur totale après addition de toutes les valeurs entrées dans les zones de texte.

Si vous essayez d'entrer des valeurs autres que des valeurs réelles, l'application ci-dessus se bloquera. Pour rechercher et résoudre le problème (pourquoi il se bloque), vous pouvez insérer des points d'arrêt dans l'événement de clic de bouton.

Écrivons "abc" dans l'élément 1 comme indiqué ci-dessous.

En cliquant sur le bouton Total, vous verrez que le programme s'arrête au point de rupture

Déplacez maintenant le curseur vers la zone de texte 1.Text et vous verrez que le programme essaie d'ajouter abc value avec les autres valeurs, c'est pourquoi le programme plante.

Débogage en XAML

Si vous vous attendez au même type de débogage en XAML, vous serez surpris de savoir qu'il n'est pas encore possible de déboguer le code XAML comme le débogage de tout autre code de langage procédural. Lorsque vous entendez le terme débogage dans le code XAML, cela signifie essayer de trouver une erreur.

  • Dans la liaison de données, vos données n'apparaissent pas à l'écran et vous ne savez pas pourquoi

  • Ou un problème est lié à des mises en page complexes.

  • Ou un problème d'alignement ou des problèmes de couleur de marge, de superpositions, etc. avec certains modèles étendus tels que ListBox et combo box.

Le débogage d'un programme XAML est quelque chose que vous faites généralement pour vérifier si vos liaisons fonctionnent; et si cela ne fonctionne pas, vérifiez ce qui ne va pas. Malheureusement, la définition de points d'arrêt dans les liaisons XAML n'est pas possible sauf dans Silverlight, mais nous pouvons utiliser la fenêtre Sortie pour vérifier les erreurs de liaison de données. Jetons un coup d'œil au code XAML suivant pour trouver l'erreur dans la liaison de données.

<Window x:Class = "DataBindingOneWay.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <StackPanel Name = "Display"> 
         <StackPanel Orientation = "Horizontal" Margin = "50, 50, 0, 0"> 
            <TextBlock Text = "Name: " Margin = "10" Width = "100"/> 
            <TextBlock Margin = "10" Width = "100" Text = "{Binding FirstName}"/> 
         </StackPanel> 
			
         <StackPanel Orientation = "Horizontal" Margin = "50,0,50,0"> 
            <TextBlock Text = "Title: " Margin = "10" Width = "100"/> 
            <TextBlock Margin = "10" Width = "100" Text = "{Binding Title}" /> 
         </StackPanel> 
			
      </StackPanel> 
   </Grid> 
	
</Window>

Les propriétés de texte de deux blocs de texte sont définies statiquement sur «Nom» et «Titre», tandis que les deux autres blocs de texte Les propriétés de texte sont liées à «FirstName» et «Title», mais les variables de classe sont Name et Title dans la classe Employee qui est illustrée ci-dessous.

Nous avons intentionnellement écrit un nom de variable incorrect afin de comprendre où trouver ce type d'erreur lorsque la sortie souhaitée n'est pas affichée.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks;  

namespace DataBindingOneWay { 

   public class Employee { 
      public string Name { get; set; } 
      public string Title { get; set; }  
		
      public static Employee GetEmployee() { 
		
         var emp = new Employee() { 
            Name = "Ali Ahmed", Title = "Developer" 
         }; 
			
         return emp; 
      }  
   } 
}

Voici l'implémentation de la classe MainWindow en code C #.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace DataBindingOneWay { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         DataContext = Employee.GetEmployee(); 
      } 
   } 
}

Exécutons cette application et vous pouvez voir immédiatement dans notre MainWindow que nous avons lié avec succès au titre de cet objet Employee mais que le nom n'est pas lié.

Pour vérifier ce qui s'est passé avec le nom, regardons dans la fenêtre de sortie où beaucoup de journal est généré.

Le plus facile à trouver une erreur consiste simplement à rechercher l'erreur et vous trouverez l'erreur suivante qui indique «Erreur de chemin BindingExpression: propriété 'FirstName' introuvable sur 'objet' '' Employe»

System.Windows.Data Error: 40 : BindingExpression path error: 'FirstName'
   property not found on 'object' ''Employee' (HashCode=11611730)'.
   BindingExpression:Path = FirstName; DataItem = 'Employee' (HashCode = 11611730);
   target element is 'TextBlock' (Name=''); target property is 'Text' (type 'String')

Ce qui indique clairement que FirstName n'est pas membre de la classe Employee, ce qui permet de résoudre ce type de problèmes dans votre application.

Lorsque vous modifiez à nouveau le prénom en nom, vous verrez la sortie souhaitée.

Outils de débogage de l'interface utilisateur pour XAML

Des outils de débogage de l'interface utilisateur ont été introduits pour XAML avec Visual Studio 2015 afin d'inspecter le code XAML au moment de l'exécution. À l'aide de ces outils, le code XAML est présenté sous la forme d'une arborescence visuelle de votre application WPF en cours d'exécution ainsi que des différentes propriétés d'élément d'interface utilisateur dans l'arborescence. Pour activer ces outils, suivez les étapes ci-dessous.

  • Accédez au menu Outils et sélectionnez Options dans le menu Outils.
  • Cela ouvrira la boîte de dialogue suivante.
  • Accédez aux options générales sous l'élément de débogage sur le côté gauche.
  • Cochez l'option en surbrillance, c'est-à-dire «Activer les outils de débogage de l'interface utilisateur pour XAML» et cliquez sur le bouton OK.

Maintenant, exécutez n'importe quelle application XAML ou utilisez le code XAML suivant.

<Window x:Class = "XAMLTestBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <ComboBox Name = "comboBox"  Margin = "50" Width = "100"> 
         <ComboBoxItem Content = "Green" /> 
         <ComboBoxItem  Content = "Yellow" IsSelected = "True" />
         <ComboBoxItem Content = "Orange" /> 
      </ComboBox> 
		
      <TextBox  Name = "textBox" Margin = "50" Width = "100" Height = "23"
         VerticalAlignment = "Top" Text  =
         "{Binding ElementName = comboBox, Path = SelectedItem.Content, Mode = TwoWay, UpdateSourceTrigger = PropertyChanged}" 
         Background = "{Binding ElementName = comboBox, Path = SelectedItem.Content}"> 
      </TextBox> 
		
   </StackPanel> 
	
</Window>

Lorsque vous exécutez l'application, elle affichera l'arborescence visuelle en direct où tous les éléments sont affichés dans une arborescence.

Cet arbre visuel en direct montre la structure de disposition complète pour comprendre où se trouvent les éléments de l'interface utilisateur. Mais cette option n'est disponible que dans Visual Studio 2015. Si vous utilisez une ancienne option de Visual Studio, vous ne pouvez pas utiliser cet outil, mais il existe un autre outil qui peut être intégré à Visual Studio tel que XAML Spy pour Visual Studio . Vous pouvez le télécharger sur xamlspy

Les applications WPF permettent de créer des contrôles personnalisés, ce qui facilite la création de contrôles riches en fonctionnalités et personnalisables. Les contrôles personnalisés sont utilisés lorsque tous les contrôles intégrés fournis par Microsoft ne remplissent pas vos critères ou que vous ne souhaitez pas payer pour des contrôles tiers.

Dans ce chapitre, vous apprendrez à créer des contrôles personnalisés. Avant de commencer à jeter un coup d'œil aux contrôles personnalisés, examinons d'abord un contrôle utilisateur.

Contrôle utilisateur

Les contrôles utilisateur fournissent un moyen de collecter et de combiner différents contrôles intégrés et de les regrouper dans du XAML réutilisable. Les contrôles utilisateur sont utilisés dans les scénarios suivants -

  • Si le contrôle se compose de contrôles existants, c'est-à-dire que vous pouvez créer un contrôle unique de plusieurs contrôles déjà existants.

  • Si le contrôle n'a pas besoin de support pour la thématisation. Les contrôles utilisateur ne prennent pas en charge la personnalisation complexe, les modèles de contrôle et difficiles à styliser.

  • Si un développeur préfère écrire des contrôles à l'aide du modèle code-behind où une vue, puis un code direct derrière pour les gestionnaires d'événements.

  • Vous ne partagerez pas votre contrôle entre les applications.

Exemple

Passons à un exemple de contrôle utilisateur et suivez les étapes ci-dessous.

  • Créez un nouveau projet WPF, puis cliquez avec le bouton droit sur votre solution et sélectionnez Ajouter> Nouvel élément ...

  • La fenêtre suivante s'ouvrira. Sélectionnez maintenantUser Control (WPF) et nommez-le MyUserControl.

  • Cliquez sur le bouton Ajouter et vous verrez que deux nouveaux fichiers (MyUserControl.xaml et MyUserControl.cs) seront ajoutés dans votre solution.

Voici le code XAML dans lequel un bouton et une zone de texte sont créés avec certaines propriétés dans le fichier MyUserControl.xaml.

<UserControl x:Class = "WPFUserControl.MyUserControl" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"  
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"  
   mc:Ignorable = "d" d:DesignHeight = "300" d:DesignWidth = "300"> 
	
   <Grid> 
      <TextBox Height = "23"  
         HorizontalAlignment = "Left"  
         Margin = "80,49,0,0" Name = "txtBox"  
         VerticalAlignment = "Top" Width = "200" /> 
			
      <Button Content = "Click Me"  
         Height = "23" HorizontalAlignment = "Left"  
         Margin = "96,88,0,0" Name = "button"  
         VerticalAlignment = "Top" Click = "button_Click" />    
   </Grid>
	
</UserControl>

Vous trouverez ci-dessous le code C # pour l'événement de clic de bouton dans le fichier MyUserControl.cs qui met à jour la zone de texte.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace WPFUserControl {
   /// <summary>
      /// Interaction logic for MyUserControl.xaml 
   /// </summary> 
	
   public partial class MyUserControl : UserControl { 
	
      public MyUserControl() { 
         InitializeComponent(); 
      }  
		
      private void button_Click(object sender, RoutedEventArgs e) { 
         txtBox.Text = "You have just clicked the button"; 
      } 
   } 
}

Voici l'implémentation dans MainWindow.xaml pour ajouter le contrôle utilisateur.

<Window x:Class = "XAMLUserControl.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:control = "clr-namespace:WPFUserControl" 
   Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
      <control:MyUserControl/> 
   </Grid> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante.

En cliquant sur le bouton "Cliquez sur moi", vous remarquerez que le texte à l'intérieur de la zone de texte est mis à jour.

Contrôles personnalisés

Un contrôle personnalisé est une classe qui offre son propre style et modèle qui sont normalement définis dans generic.xaml. Les contrôles personnalisés sont utilisés dans les scénarios suivants -

  • Si le contrôle n'existe pas et que vous devez le créer à partir de zéro.

  • Si vous souhaitez étendre ou ajouter des fonctionnalités à un contrôle préexistant en ajoutant une propriété supplémentaire ou une fonctionnalité supplémentaire pour s'adapter à votre scénario spécifique.

  • Si vos commandes doivent prendre en charge le thème et le style.

  • Si vous souhaitez partager votre contrôle entre les applications.

Exemple

Prenons un exemple pour comprendre le fonctionnement des contrôles personnalisés. Créez un nouveau projet WPF, puis cliquez avec le bouton droit sur votre solution et sélectionnez Ajouter> Nouvel élément ...

Cela ouvrira la fenêtre suivante. Sélectionnez maintenantCustom Control (WPF) et nommez-le MyCustomControl.

Cliquez sur le bouton Ajouter et vous verrez que deux nouveaux fichiers (Themes / Generic.xaml et MyCustomControl.cs) seront ajoutés dans votre solution.

Voici le code XAML dans lequel le style est défini pour le contrôle personnalisé dans le fichier Generic.xaml.

<ResourceDictionary 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomControls">  
	
   <Style TargetType = "{x:Type local:MyCustomControl}"
      BasedOn = "{StaticResource {x:Type Button}}"> 
      <Setter Property = "Background" Value = "LightSalmon" /> 
      <Setter Property = "Foreground" Value = "Blue"/> 
   </Style> 
	
</ResourceDictionary>

Voici le code C # pour la classe MyCustomControl qui est hérité de la classe button et dans le constructeur, il remplace les métadonnées.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace WPFCustomControls { 

   public class MyCustomControl : Button { 
	
      static MyCustomControl() { 
         DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new
            FrameworkPropertyMetadata(typeof(MyCustomControl))); 
      } 
		
   } 
}

Voici l'implémentation d'événement de clic de contrôle personnalisé en C # qui met à jour le texte du bloc de texte.

using System; 
using System.Windows; 
using System.Windows.Controls;

namespace WPFCustomControls { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void customControl_Click(object sender, RoutedEventArgs e) { 
         txtBlock.Text = "You have just click your custom control"; 
      }
		
   } 
}

Voici l'implémentation dans MainWindow.xaml pour ajouter le contrôle personnalisé et un TextBlock.

<Window x:Class = "WPFCustomControls.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:control = "clr-namespace:WPFCustomControls" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <control:MyCustomControl x:Name = "customControl"  
         Content = "Click Me" Width = "70" 
         Margin = "10" Click = "customControl_Click"/> 
			
      <TextBlock Name = "txtBlock"  
         Width = "250" Height = "30"/> 
   </StackPanel>
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante avec un contrôle personnalisé qui est un bouton personnalisé.

En cliquant sur le bouton personnalisé, vous verrez que le texte à l'intérieur du bloc de texte est mis à jour.

Une exception est toute condition d'erreur ou un comportement inattendu qui se produit pendant l'exécution d'un programme. Des exceptions peuvent être soulevées pour de nombreuses raisons, certaines d'entre elles sont les suivantes -

  • Défaut dans votre code ou dans le code que vous appelez (comme une bibliothèque partagée),

  • Ressources du système d'exploitation non disponibles,

  • Conditions inattendues rencontrées par un Common Language Runtime (comme du code qui ne peut pas être vérifié)

Syntaxe

Les exceptions ont la capacité de transférer le flux d'un programme d'une partie à une autre. Dans le framework .NET, la gestion des exceptions comporte les quatre mots clés suivants:

  • try - Dans ce bloc, le programme identifie une certaine condition qui soulève une exception.

  • catch- Le mot clé catch indique la capture d'une exception. UNEtry bloc est suivi d'un ou plusieurs catch blocs pour intercepter une exception avec un gestionnaire d'exceptions à l'endroit dans un programme où vous souhaitez gérer le problème.

  • finally- Le bloc finally est utilisé pour exécuter un ensemble d'instructions donné, qu'une exception soit levée ou non. Par exemple, si vous ouvrez un fichier, il doit être fermé, qu'une exception soit déclenchée ou non.

  • throw- Un programme lève une exception lorsqu'un problème apparaît. Ceci est fait en utilisant un mot-clé throw.

La syntaxe pour utiliser ces quatre mots-clés est la suivante -

try { 
   ///This will still trigger the exception 
} 
catch (ExceptionClassName e) { 
   // error handling code 
} 
catch (ExceptionClassName e) { 
   // error handling code
}
catch (ExceptionClassName e) { 
   // error handling code 
} 
finally { 
   // statements to be executed 
}

Plusieurs instructions catch sont utilisées dans les cas où un bloc try peut déclencher plus d'une exception en fonction de la situation d'un flux de programme.

Hiérarchie

Presque toutes les classes d'exception dans le framework .NET sont directement ou indirectement dérivées de la classe Exception. Les classes d'exceptions les plus importantes dérivées de la classe Exception sont -

  • ApplicationException class- Il prend en charge les exceptions générées par les programmes. Lorsque le développeur souhaite définir une exception, la classe doit être dérivée de cette classe.

  • SystemException class- Il s'agit de la classe de base pour toutes les exceptions système d'exécution prédéfinies. La hiérarchie suivante montre les exceptions standard fournies par le runtime.

Le tableau suivant répertorie les exceptions standard fournies par le runtime et les conditions dans lesquelles vous devez créer une classe dérivée.

Type d'exception Type de base La description
Exception Objet Classe de base pour toutes les exceptions.
SystemException Exception Classe de base pour toutes les erreurs générées par l'exécution.
IndexOutOfRangeException SystemException Lancé par le runtime uniquement lorsqu'un tableau est mal indexé.
NullReferenceException SystemException Lancé par le runtime uniquement lorsqu'un objet nul est référencé.
AccessViolationException SystemException Lancé par le runtime uniquement en cas d'accès à une mémoire non valide.
InvalidOperationException SystemException Lancé par des méthodes dans un état non valide.
ArgumentException SystemException Classe de base pour toutes les exceptions d'argument.
ArgumentNullException ArgumentException Lancé par des méthodes qui ne permettent pas à un argument d'être nul.
ArgumentOutOfRangeException ArgumentException Lancé par des méthodes qui vérifient que les arguments sont dans une plage donnée.
ExternalException SystemException Classe de base pour les exceptions qui se produisent ou sont ciblées sur des environnements en dehors du runtime.
SEHException Exception externe Exception encapsulant les informations de gestion des exceptions structurées Win32.

Exemple

Prenons un exemple simple pour mieux comprendre le concept. Commencez par créer un nouveau projet WPF avec le nomWPFExceptionHandling.

Faites glisser une zone de texte de la boîte à outils vers la fenêtre de conception. Le code XAML suivant crée une zone de texte et l'initialise avec certaines propriétés.

<Window x:Class = "WPFExceptionHandling.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFExceptionHandling"
   mc:Ignorable = "d" 
   Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"
         Height = "241" Margin = "70,39,0,0" TextWrapping = "Wrap" 
         VerticalAlignment = "Top" Width = "453"/> 
   </Grid> 
	
</Window>

Voici la lecture du fichier avec la gestion des exceptions en C #.

using System; 
using System.IO; 
using System.Windows;

namespace WPFExceptionHandling { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         ReadFile(0); 
      }
		
      void ReadFile(int index) { 
         string path = @"D:\Test.txt"; 
         StreamReader file = new StreamReader(path); 
         char[] buffer = new char[80]; 
			
         try { 
            file.ReadBlock(buffer, index, buffer.Length); 
            string str = new string(buffer); 
            str.Trim(); 
            textBox.Text = str; 
         }
         catch (Exception e) {
            MessageBox.Show("Error reading from "+ path + "\nMessage = "+ e.Message);
         } 
         finally { 
            if (file != null) { 
               file.Close(); 
            } 
         } 
      } 
   } 
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante dans laquelle un texte est affiché à l'intérieur de la zone de texte.

Lorsqu'une exception est levée ou que vous la lancez manuellement (comme dans le code suivant), une boîte de message contenant une erreur s'affiche.

using System; 
using System.IO; 
using System.Windows;

namespace WPFExceptionHandling {
 
   public partial class MainWindow : Window {
	
      public MainWindow() { 
         InitializeComponent(); 
         ReadFile(0); 
      } 
		
      void ReadFile(int index) { 
         string path = @"D:\Test.txt"; 
         StreamReader file = new StreamReader(path); 
         char[] buffer = new char[80]; 
			
         try { 
            file.ReadBlock(buffer, index, buffer.Length); 
            string str = new string(buffer); 
            throw new Exception(); 
            str.Trim(); 
            textBox.Text = str; 
         }
         catch (Exception e) { 
            MessageBox.Show("Error reading from "+ path + "\nMessage = "+ e.Message); 
         } 
         finally { 
            if (file != null) { 
               file.Close(); 
            } 
         } 
      } 
   } 
}

Lorsqu'une exception est déclenchée lors de l'exécution du code ci-dessus, le message suivant s'affiche.

Nous vous recommandons d'exécuter le code ci-dessus et d'expérimenter ses fonctionnalités.

La localisation est la traduction des ressources de l'application en versions localisées pour les cultures spécifiques prises en charge par l'application.

Lorsque vous développez votre application et que votre application est disponible dans une seule langue, vous limitez le nombre de vos clients et la taille de votre entreprise. Si vous souhaitez augmenter votre clientèle, ce qui augmentera également votre activité, votre produit doit être disponible et accessible à un public mondial. Rentablelocalization de votre produit est l'un des moyens les meilleurs et les plus économiques de toucher plus de clients.

Dans WPF, les applications localisables sont très faciles à créer avec resxfichier qui est la solution la plus simple pour la localisation. Prenons un exemple simple pour comprendre comment cela fonctionne -

  • Créer un nouveau projet WPF avec le nom WPFLocalization.

  • Dans votre explorateur de solutions, vous verrez le fichier Resources.resx sous le dossier Propriétés.

  • Modifiez le modificateur d'accès de interne à public afin qu'il puisse être accessible dans un fichier XAML.

  • Ajoutez maintenant le nom et les valeurs de la chaîne suivante que nous utiliserons dans notre application.

  • Faites deux copies du fichier Resources.resx avec les noms Resources.en.resx et Resources.ru-RU.resx. Il s'agit de conventions de dénomination spécifiques à la langue et au nom du pays / région, et elles peuvent être trouvées sur la référence de l'API National Language Support (NLS) (https://msdn.microsoft.com/en-us/goglobal/bb896001.aspx ) page.

  • Modifiez les valeurs de Resources.ru-RU.resx en mots russes, comme indiqué ci-dessous.

  • Allons à la fenêtre de conception et faisons glisser trois zones de texte, trois étiquettes et trois boutons.

  • Dans le fichier XAML, ajoutez d'abord la déclaration d'espace de noms pour utiliser localize resources xmlns: p = "clr-namespace: WPFLocalization.Properties"

  • Définissez les propriétés de tous les contrôles comme indiqué ci-dessous. Dans cet exemple, nous n'utiliserons pas de chaînes codées en dur pour le contenu des étiquettes, des boutons et du titre de la fenêtre dans le fichier XAML. Nous utiliserons les chaînes définies dans les fichiers * .resx. Par exemple, pour le titre de la fenêtre, nous utilisons la chaîne Title qui est définie dans le fichier * .resx comme ceci "Title =" {x: Static p: Resources.Title} ""

  • Voici le fichier XAML dans lequel les contrôles sont créés et initialisés avec différentes propriétés.

<Window x:Class = "WPFLocalization.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFLocalization" 
   xmlns:p = "clr-namespace:WPFLocalization.Properties"
   Title = "{x:Static p:Resources.Title}" Height = "350" Width = "604">
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,45,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "304"/>
			
      <Label x:Name = "label" Content = "{x:Static p:Resources.Name}"
          HorizontalAlignment = "Left" Margin = "52,45,0,0" VerticalAlignment = "Top" Width = "86"/>
			 
      <TextBox x:Name = "textBox1" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,102,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "304"/> 
			
      <Label x:Name = "label1" Content = "{x:Static p:Resources.Address}" 
         HorizontalAlignment = "Left" Margin = "52,102,0,0" VerticalAlignment = "Top" Width = "86"/>
			
      <TextBox x:Name = "textBox2" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,157,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "80"/>
			
      <Label x:Name = "label2" Content = "{x:Static p:Resources.Age}" 
         HorizontalAlignment = "Left" Margin = "52,157,0,0" VerticalAlignment = "Top" Width = "86"/>
			
      <Button x:Name = "button" Content = "{x:Static p:Resources.OK_Button}" 
         HorizontalAlignment = "Left" Margin = "163,241,0,0" VerticalAlignment = "Top" Width = "75"/> 
			
      <Button x:Name = "button1" Content = "{x:Static p:Resources.Cancel_Button}" 
         HorizontalAlignment = "Left" Margin = "282,241,0,0" VerticalAlignment = "Top" Width = "75"/>
			
      <Button x:Name = "button2" Content = "{x:Static p:Resources.Help_Button}" 
         HorizontalAlignment = "Left" Margin = "392,241,0,0" VerticalAlignment = "Top" Width = "75"/> 
   </Grid> 
	
 </Window>
  • Lorsque le code ci-dessus est compilé et exécuté, vous verrez la fenêtre suivante qui contient différents contrôles.

  • Par défaut, le programme utilise le fichier Resources.resx par défaut. Si vous souhaitez afficher le texte en langue russe qui est défini dans le fichier Resources.ru-RU.resx, vous devrez définir la culture explicitement lorsque le programme démarre dans le fichier App.xaml comme indiqué ci-dessous.

using System.Windows;

namespace WPFLocalization {
   /// <summary> 
      /// Interaction logic for App.xaml 
   /// </summary> 
	
   public partial class App : Application {
	
      App() { 
         System.Threading.Thread.CurrentThread.CurrentUICulture = new System.Globalization.CultureInfo("ru-RU");
         //System.Threading.Thread.CurrentThread.CurrentUICulture = new System.Globalization.CultureInfo("en"); 
      } 
   } 
}

Lorsque vous exécutez votre application, vous verrez tout le texte en langue russe.

Nous vous recommandons d'exécuter le code ci-dessus et de créer également des fichiers resx pour d'autres cultures.

Dans WPF, une interaction montre comment une vue interagit avec les contrôles situés dans cette vue. Les interactions les plus connues sont de deux types -

  • Behaviors
  • Glisser déposer

Comportements

Des comportements ont été introduits avec Expression Blend 3 qui peut encapsuler certaines fonctionnalités dans un composant réutilisable. Pour ajouter des comportements supplémentaires, vous pouvez attacher ces composants aux contrôles. Les comportements offrent plus de flexibilité pour concevoir facilement des interactions utilisateur complexes.

Jetons un coup d'œil à un exemple simple dans lequel un comportement ControlStoryBoardAction est attaché à des contrôles.

  • Créez un nouveau projet WPF avec le nom WPFBehavior.

  • Le code XAML suivant crée une ellipse et deux boutons pour contrôler le mouvement de l'ellipse.

<Window 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFBehaviors" 
   xmlns:i = "http://schemas.microsoft.com/expression/2010/interactivity" 
   xmlns:ei = "http://schemas.microsoft.com/expression/2010/interactions" 
   x:Class = "WPFBehaviors.MainWindow" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Window.Resources> 
      <Storyboard x:Key = "Storyboard1" RepeatBehavior = "Forever" AutoReverse = "True"> 
		
         <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty =
            "(UIElement.RenderTransform).(TransformGroup.Children )[3].(TranslateTransform.X)"
            Storyboard.TargetName = "ellipse"> 
            <EasingDoubleKeyFrame KeyTime = "0:0:1" Value = "301.524"/> 
            <EasingDoubleKeyFrame KeyTime = "0:0:2" Value = "2.909"/> 
         </DoubleAnimationUsingKeyFrames>
			
         <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = 
            "(UIElement.RenderTransform).(TransformGroup.Children )[3].(TranslateTransform.Y)"
            Storyboard.TargetName = "ellipse"> 
            <EasingDoubleKeyFrame KeyTime = "0:0:1" Value = "-0.485"/> 
            <EasingDoubleKeyFrame KeyTime = "0:0:2" Value = "0"/> 
         </DoubleAnimationUsingKeyFrames> 
			
         <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty = "(ContentControl.Content)"
            Storyboard.TargetName = "button"> 
            <DiscreteObjectKeyFrame KeyTime = "0" Value = "Play"/> 
         </ObjectAnimationUsingKeyFrames>
			
         <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty = "(ContentControl.Content)"
            Storyboard.TargetName = "button1"> 
            <DiscreteObjectKeyFrame KeyTime = "0" Value = "Stop"/> 
            <DiscreteObjectKeyFrame KeyTime = "0:0:2" Value = "Stop"/> 
         </ObjectAnimationUsingKeyFrames> 
      </Storyboard> 
   </Window.Resources> 
	
   <Window.Triggers> 
      <EventTrigger RoutedEvent = "FrameworkElement.Loaded"> 
         <BeginStoryboard Storyboard = "{StaticResource Storyboard1}"/> 
      </EventTrigger> 
   </Window.Triggers> 
	
   <Grid> 
      <Ellipse x:Name = "ellipse" Fill = "#FFAAAAC5" HorizontalAlignment = "Left"
         Height = "50.901" Margin = "49.324,70.922,0,0" Stroke = "Black"
         VerticalAlignment = "Top" Width = "73.684" RenderTransformOrigin = "0.5,0.5"> 
         <Ellipse.RenderTransform> 
            <TransformGroup> 
               <ScaleTransform/> 
               <SkewTransform/> 
               <RotateTransform/> 
               <TranslateTransform/> 
            </TransformGroup> 
         </Ellipse.RenderTransform> 
      </Ellipse>
		
      <Button x:Name = "button" Content = "Play" HorizontalAlignment = "Left" Height = "24.238"
         Margin = "63.867,0,0,92.953" VerticalAlignment = "Bottom" Width = "74.654"> 
         <i:Interaction.Triggers> 
            <i:EventTrigger EventName = "Click"> 
               <ei:ControlStoryboardAction Storyboard = "{StaticResource Storyboard1}"/> 
            </i:EventTrigger> 
         </i:Interaction.Triggers> 
      </Button>
		
      <Button x:Name = "button1" Content = "Stop" HorizontalAlignment = "Left" Height = "24.239"
         Margin = "160.82,0,0,93.922" VerticalAlignment = "Bottom" Width = "75.138"> 
         <i:Interaction.Triggers> 
            <i:EventTrigger EventName = "Click"> 
               <ei:ControlStoryboardAction ControlStoryboardOption = "Stop"
                  Storyboard = "{StaticResource Storyboard1}"/> 
            </i:EventTrigger> 
         </i:Interaction.Triggers> 
      </Button> 
		
   </Grid> 
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante qui contient une ellipse et deux boutons.

Lorsque vous appuyez sur le bouton de lecture, il commencera à se déplacer de gauche à droite, puis reviendra à sa position d'origine. Le bouton d'arrêt arrêtera le mouvement de l'ellipse.

Glisser déposer

Le glisser-déposer sur l'interface utilisateur peut considérablement améliorer l'efficacité et la productivité de l'application. Il existe très peu d'applications dans lesquelles les fonctionnalités de glisser-déposer sont utilisées car les gens pensent que c'est difficile à mettre en œuvre. Dans une certaine mesure, il est difficile de gérer une fonctionnalité de glisser-déposer, mais dans WPF, vous pouvez la gérer assez facilement.

Prenons un exemple simple pour comprendre comment cela fonctionne. Nous allons créer une application dans laquelle vous pouvez faire glisser et déposer la couleur d'un rectangle à un autre.

  • Créez un nouveau projet WPF avec le nom WPFDragAndDrop.

  • Faites glisser cinq rectangles vers la fenêtre de conception et définissez les propriétés comme indiqué dans le fichier XAML suivant.

<Window x:Class = "WPFDragAndDrop.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDragAndDrop" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "402.551" Width = "604"> 
	
   <Grid> 
      <Rectangle Name = "Target" Fill = "AliceBlue" HorizontalAlignment = "Left"  
         Height = "345" Margin = "10,10,0,0" Stroke = "Black"  
         VerticalAlignment = "Top" Width = "387" AllowDrop = "True" Drop = "Target_Drop"/> 
			
      <Rectangle Fill = "Beige" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,10,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightBlue" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,80,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightCoral" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,150,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightGray" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,220,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "OliveDrab" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,290,0,-7" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/>  
   </Grid> 
	
</Window>
  • Le premier rectangle est le rectangle cible, de sorte que l'utilisateur peut faire glisser la couleur de l'autre rectangle vers le rectangle cible.

  • Vous trouverez ci-dessous l'implémentation des événements en C # pour le glisser-déposer.

using System.Windows; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Shapes; 
 
namespace WPFDragAndDrop { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void Rect_MLButtonDown(object sender, MouseButtonEventArgs e) { 
         Rectangle rc = sender as Rectangle; 
         DataObject data = new DataObject(rc.Fill); 
         DragDrop.DoDragDrop(rc, data,DragDropEffects.Move); 
      }  
		
      private void Target_Drop(object sender, DragEventArgs e) { 
         SolidColorBrush scb = (SolidColorBrush)e.Data.GetData(typeof(SolidColorBrush)); 
         Target.Fill = scb; 
      } 
   } 
}

Lorsque vous exécutez votre application, elle produira la fenêtre suivante.

Si vous faites glisser une couleur du rectangle sur le côté droit et la déposez sur le grand rectangle de gauche, vous verrez immédiatement son effet.

Tirons le 4 ème du côté droit.

Vous pouvez voir que la couleur du rectangle cible a changé. Nous vous recommandons d'exécuter le code ci-dessus et d'expérimenter ses fonctionnalités.

WPF fournit une large gamme de graphiques 2D qui peuvent être améliorés selon les exigences de votre application. WPF prend en charge les objets Drawing et Shape utilisés pour dessiner du contenu graphique.

Formes et dessin

  • La classe Shape est dérivée de la classe FrameworkElement, les objets Shape peuvent être utilisés à l'intérieur des panneaux et de la plupart des contrôles.

  • WPF fournit des objets de forme de base dérivés de la classe Shape tels que Ellipse, Line, Path, Polygon, Polyline et Rectangle.

  • Les objets de dessin, en revanche, ne dérivent pas de la classe FrameworkElement et fournissent une implémentation plus légère.

  • Les objets de dessin sont plus simples que les objets Shape. Ils ont également de meilleures caractéristiques de performance.

Exemple

Prenons un exemple simple pour comprendre comment utiliser différents objets de formes.

  • Créer un nouveau projet WPF avec le nom WPF2DGraphics.

  • Le code suivant crée différents types de formes.

<Window x:Class = "WPF2DGraphics.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF2DGraphics" 
   xmlns:PresentationOptions = "http://schemas.microsoft.com/winfx/2006/xaml/present ation/options"  
   mc:Ignorable = "PresentationOptions" Title = "MainWindow" Height = "400" Width = "604"> 
	
   <StackPanel> 
      <Ellipse Width = "100" Height = "60" Name = "sample" Margin = "10"> 
         <Ellipse.Fill> 
            <RadialGradientBrush> 
               <GradientStop Offset = "0" Color = "AliceBlue"/> 
               <GradientStop Offset = "1" Color = "Gray"/> 
               <GradientStop Offset = "2" Color = "Red"/> 
            </RadialGradientBrush> 
         </Ellipse.Fill> 
      </Ellipse>
		
      <Path Stroke = "Red" StrokeThickness = "5" Data = "M 10,70 L 200,70"
         Height = "42.085" Stretch = "Fill" Margin = "140.598,0,146.581,0" /> 
      <Path Stroke = "BlueViolet" StrokeThickness = "5" Data = "M 20,100 A 100,56 42 1 0 200,10"
         Height = "81.316" Stretch = "Fill" Margin = "236.325,0,211.396,0" /> 
			
      <Path Fill = "LightCoral" Margin = "201.424,0,236.325,0"
         Stretch = "Fill" Height = "124.929"> 
         <Path.Data> 
            <PathGeometry> 
               <PathFigure StartPoint = "50,0" IsClosed = "True"> 
                  <LineSegment Point = "100,50"/> 
                  <LineSegment Point = "50,100"/> 
                  <LineSegment Point = "0,50"/> 
               </PathFigure> 
            </PathGeometry> 
         </Path.Data> 
      </Path> 
		
   </StackPanel> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira une ellipse, une ligne droite, un arc et un polygone.

Exemple

Jetons un coup d'œil à un autre exemple qui montre comment peindre une zone avec un dessin.

  • Créer un nouveau projet WPF avec le nom WPF2DGraphics1.

  • Le code XAML suivant montre comment peindre différemment avec le dessin d'image.

<Window x:Class = "WPF2DGraphics1.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:PresentationOptions = "http://schemas.microsoft.com/winfx/2006/xaml/present ation/options"  
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   mc:Ignorable = "PresentationOptions" 
   xmlns:local = "clr-namespace:WPF2DGraphics1" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Border BorderBrush = "Gray" BorderThickness = "1"  
         HorizontalAlignment = "Left" VerticalAlignment = "Top" 
         Margin = "20"> 
			
         <Image Stretch = "None">
            <Image.Source> 
               <DrawingImage PresentationOptions:Freeze = "True"> 
					
                  <DrawingImage.Drawing> 
                     <DrawingGroup> 
                        <ImageDrawing Rect = "300,100,300,180" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "0,100,250,100" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "150,0,25,25" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "0,0,75,75" ImageSource = "Images\DSC_0104.JPG"/> 
                     </DrawingGroup> 
                  </DrawingImage.Drawing> 
						
               </DrawingImage> 
            </Image.Source> 
         </Image> 
			
      </Border> 
   </Grid> 
	
</Window>

Lorsque vous exécutez votre application, elle produira la sortie suivante -

Nous vous recommandons d'exécuter le code ci-dessus et d'essayer d'autres formes et dessins 2D.

Windows Presentation Foundation (WPF) fournit une fonctionnalité pour dessiner, transformer et animer des graphiques 3D selon les besoins de votre application. Il ne prend pas en charge le développement de jeux 3D à part entière, mais à un certain niveau, vous pouvez créer des graphiques 3D.

En combinant des graphiques 2D et 3D, vous pouvez également créer des contrôles riches, fournir des illustrations complexes de données ou améliorer l'expérience utilisateur de l'interface d'une application. L'élément Viewport3D héberge un modèle 3D dans notre application WPF.

Exemple

Prenons un exemple simple pour comprendre comment utiliser les graphiques 3D.

  • Créer un nouveau projet WPF avec le nom WPF3DGraphics.

  • Le code XAML suivant montre comment créer un objet 2D à l'aide de la géométrie 3D.

<Window x:Class = "WPF3DGraphics.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF3DGraphics" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "500" Width = "604"> 
	
   <Grid> 
      <Viewport3D> 
		
         <Viewport3D.Camera> 
            <PerspectiveCamera Position = "2,0,10" LookDirection = "0.2,0.4,-1"
               FieldOfView = "65" UpDirection = "0,1,0" /> 
         </Viewport3D.Camera> 
			
         <ModelVisual3D> 
            <ModelVisual3D.Content> 
               <Model3DGroup> 
                  <AmbientLight Color = "Bisque" /> 
						
                  <GeometryModel3D> 
                     <GeometryModel3D.Geometry> 
                        <MeshGeometry3D Positions = "0,0,0 0,8,0 10,0,0 8,8,0"
                           Normals = "0,0,1 0,0,1 0,0,1 0,0,1" TriangleIndices = "0,2,1 1,2,3"/> 
                     </GeometryModel3D.Geometry>
							
                     <GeometryModel3D.Material> 
                        <DiffuseMaterial Brush = "Bisque" /> 
                     </GeometryModel3D.Material> 
                  </GeometryModel3D>
						
               </Model3DGroup> 
            </ModelVisual3D.Content> 
         </ModelVisual3D> 
			
      </Viewport3D> 
   </Grid> 
	
</Window>

Lorsque vous compilez et exécutez le code ci-dessus, il produira un objet 2D en 3D.

Exemple

Jetons un coup d'œil à un autre exemple qui montre un objet 3D.

  • Créer un nouveau projet WPF avec le nom WPF3DGraphics1

  • Le code XAML suivant crée un objet 3D et un curseur. À l'aide du curseur, vous pouvez faire pivoter cet objet 3D.

<Window x:Class = "WPF3DGraphics1.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF3DGraphics1" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid>
      <Viewport3D Name="viewport3D1">
		
         <Viewport3D.Camera> 
            <PerspectiveCamera x:Name = "camMain" Position = "6 5 4" LookDirection = "-6 -5 -4"> 
            </PerspectiveCamera> 
         </Viewport3D.Camera>
		
         <ModelVisual3D> 
            <ModelVisual3D.Content> 
               <DirectionalLight x:Name = "dirLightMain" Direction = "-1,-1,-1"> 
               </DirectionalLight> 
            </ModelVisual3D.Content> 
         </ModelVisual3D>
		
         <ModelVisual3D x:Name = "MyModel"> 
            <ModelVisual3D.Content> 
               <GeometryModel3D>
				
                  <GeometryModel3D.Geometry>
                     <MeshGeometry3D x:Name = "meshMain"
                        Positions = "0 0 0  1 0 0  0 1 0  1 1 0  0 0 1  1 0 1  0 1 1  0 1 1"  
                        TriangleIndices = "2 3 1  3 1 0  7 1 3  7 5 1  6 5 7  6 4 5  6 2 0
                        2 0 4  2 7 3  2 6 7  0 1 5  0 5 4"> 
                     </MeshGeometry3D> 
                  </GeometryModel3D.Geometry> 
					
                  <GeometryModel3D.Material> 
                     <DiffuseMaterial x:Name = "matDiffuseMain">
                        <DiffuseMaterial.Brush> 
                           <SolidColorBrush Color = "Bisque"/> 
                        </DiffuseMaterial.Brush> 
                     </DiffuseMaterial> 
                  </GeometryModel3D.Material> 
					
               </GeometryModel3D>
            </ModelVisual3D.Content>
			 
            <ModelVisual3D.Transform>                         
               <RotateTransform3D> 
                  <RotateTransform3D.Rotation> 
                     <AxisAngleRotation3D x:Name = "rotate" Axis = "1 2 1"/> 
                  </RotateTransform3D.Rotation> 
               </RotateTransform3D>
            </ModelVisual3D.Transform> 
			 
         </ModelVisual3D> 
      </Viewport3D>
		
      <Slider Height = "23" HorizontalAlignment = "Left"  
         Margin = "145,271,0,0" Name = "slider1" 
         VerticalAlignment = "Top" Width = "269"  
         Maximum = "360" 
         Value = "{Binding ElementName = rotate, Path=Angle}" /> 
			
   </Grid> 
	
</Window>

Lorsque vous exécutez votre application, elle produira un objet 3D et un curseur sur votre fenêtre.

Lorsque vous faites glisser le curseur, l'objet de votre fenêtre pivote également.

Nous vous recommandons d'exécuter le code ci-dessus et d'essayer plus de géométrie 3D.

Les applications WPF prennent en charge la vidéo et l'audio en utilisant MediaElement. Il vous permet d'intégrer l'audio et la vidéo dans une application. La classe MediaElement fonctionne de la même manière que la classe Image. Vous le dirigez simplement vers les médias et il le rend. La principale différence est que ce sera une image animée, mais si vous la pointez vers le fichier qui ne contient que de l'audio et aucune vidéo comme un MP3, il la lira sans rien afficher à l'écran.

WPF prend en charge tous les types de format vidéo / audio en fonction de la configuration de la machine. Si un fichier multimédia lit un lecteur multimédia, il fonctionnera également dans WPF sur la même machine.

Exemple

Prenons un exemple pour comprendre comment intégrer le multimédia dans votre application.

  • Créer un nouveau projet WPF avec le nom WPFMultimedia.

  • Le code XAML suivant crée un élément multimédia et trois boutons et les initialise avec certaines propriétés.

<Window x:Class = "WPFMultimedia.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFMultimedia" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <StackPanel HorizontalAlignment = "Center" VerticalAlignment = "Center"> 
         <MediaElement Name = "myMedia" Source = "D:\MicrosoftMVA.mp4" 
            LoadedBehavior = "Manual" Width = "591" Height = "274" /> 
         <StackPanel Orientation = "Horizontal" Margin = "0,10,0,0">
            <Button Content = "Play" Margin = "0,0,10,0" Padding = "5" Click = "mediaPlay" /> 
            <Button Content = "Pause" Margin = "0,0,10,0" Padding = "5" Click = "mediaPause" />
            <Button x:Name = "muteButt" Content = "Mute" Padding = "5" Click = "mediaMute" /> 
         </StackPanel> 
      </StackPanel>
   </Grid> 
	
</Window>

Voici l'implémentation des événements Click en C # pour différents boutons.

using System; 
using System.Windows; 
 
namespace WPFMultimedia { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         myMedia.Volume = 100; 
         myMedia.Play(); 
      } 
		
      void mediaPlay(Object sender, EventArgs e) { 
         myMedia.Play(); 
      }  
		
      void mediaPause(Object sender, EventArgs e) { 
         myMedia.Pause();
      } 
		
      void mediaMute(Object sender, EventArgs e) { 
		
         if (myMedia.Volume == 100) { 
            myMedia.Volume = 0; 
            muteButt.Content = "Listen"; 
         } 
         else { 
            myMedia.Volume = 100; 
            muteButt.Content = "Mute"; 
         } 
      } 
   } 
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante. Vous pouvez lire la vidéo et contrôler sa lecture avec les trois boutons.

Avec les boutons, vous pouvez mettre en pause, couper le son et lire la vidéo.

Synthétiseur vocal

WPF a des fonctionnalités pour convertir le texte en parole. Cette API est incluse dans l'espace de noms System.Speech.SpeechSynthesizer classe transforme le texte en mots parlés.

Exemple

Jetons un coup d'œil à un exemple simple.

  • Créer un nouveau projet WPF avec le nom WPFTextToSpeech.

  • Nous aurons besoin de l'assemblage System.Speech à ajouter comme référence pour SpeechSynthesizer classe au travail.

  • Cliquez avec le bouton droit sur Références et sélectionnez Ajouter une référence.

  • La boîte de dialogue Gestionnaire de références s'ouvre. Maintenant, cochez la case System.Speech

  • Cliquez sur le bouton Ok. Vous pouvez voir l'assembly System.Speech dans vos références.

  • Faites maintenant glisser un bouton et une zone de texte dans la fenêtre de conception à partir de la boîte à outils.

  • Le code XAML suivant crée un bouton et une zone de texte et les initialise avec certaines propriétés.

<Window x:Class = "WPFTextToSpeech.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFTextToSpeech"
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button x:Name = "button" Content = "Speak"  
         HorizontalAlignment = "Left" Margin = "218,176,0,0"  
         VerticalAlignment = "Top" Width = "75"/> 
			
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"  
         Height = "23" Margin = "60,104,0,0" TextWrapping = "Wrap"  
         VerticalAlignment = "Top" Width = "418"/> 
   </Grid> 
	
</Window>
  • Voici l'implémentation simple en C # qui convertira le texte à l'intérieur de la zone de texte en mots prononcés.

using System.Speech.Synthesis; 
using System.Windows; 
 
namespace WPFTextToSpeech { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      } 
		
      private void button_Click(object sender, RoutedEventArgs e) { 
		
         if (textBox.Text != "") {
            SpeechSynthesizer speechSynthesizer = new SpeechSynthesizer(); 
            speechSynthesizer.Speak(textBox.Text);  
         } 
         else { 
            MessageBox.Show("Write some thing in the textbox!");
         } 
      } 
   }
}

Lorsque vous compilez et exécutez le code ci-dessus, il produira la fenêtre suivante. Maintenant, tapez Hello World dans la zone de texte et cliquez sur le bouton Parler.

Il produira le son "Hello World". Si vous ne tapez rien dans la zone de texte, le message suivant clignotera.

Nous vous recommandons d'exécuter les exemples ci-dessus.