Kader Yildirim
WPF et XAML
Présentation de XAML dans le cadre de WPF
Par Kader Yildirim publié le 30/12/2007 à 23:42
 
WPF introduit la notion de routed event. Un routed event peut invoquer un handler définit à n'importe quel niveau dans la hiérarchie des éléments graphiques (qui héritent de UIElement) d'une application et ceci dans n'importe quel sens (montant ou descendant). Il existe trois types de routed event :
  1. Direct : ce mode correspond à ce qui existe dans le Framework .Net 2.0 et inférieur
  2. Tunneling : dans ce cas il y a tout d'abord appel des handlers de la racine de la hiérarchie puis l'évènement descend vers la source. On le préfixe avec preview par convention.
  3. Bubbling : l'appel a d'abord lieu au niveau du handler de la source puis l'évènement remonte vers la racine de la hiérarchie.
Avec ces informations supplémentaires l'exemple devient :

public partial class Window5 : Window

    {

        Button button1 = new Button();

        Button button2 = new Button();

        string content = "Hello";

        DockPanel dockPanel = new DockPanel();

 

        SolidColorBrush scb = new SolidColorBrush(Color.FromArgb(255, 0, 255, 0));

 

        public Window5()

        {

            InitializeComponent();

            dockPanel.Background = Brushes.Red;

            dockPanel.LastChildFill = false;

            this.AddChild(dockPanel);

 

            button1.Background = scb;

            button1.Height = 50;

            button1.Width = 100;

            button1.Content = content;

            DockPanel.SetDock(button1, Dock.Bottom);

 

            button2.Background = scb;

            button2.Height = 50;

            button2.Width = 100;

            button2.Content = content;

            button2.Click += new RoutedEventHandler(button2_Click);

            DockPanel.SetDock(button2, Dock.Top);

 

            this.dockPanel.Children.Add(button1);

            this.dockPanel.Children.Add(button2);

        }

 

        private void button2_Click(object sender, RoutedEventArgs e)

        {

            MessageBox.Show("coucou");

        }

    }

Soit en XAML :

<Window x:Class="WpfApplication1.Window6"

   xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"

   xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"

   Title="Window6" Height="300" Width="300">

    <Window.Resources>

        <Color A="255" R="0" G="255" B="0" x:Key="scb"/>

    </Window.Resources>

    <DockPanel Name="dockPanel" LastChildFill="False">

        <DockPanel.Children>

            <Button Name="button1" DockPanel.Dock="Bottom">

                <Button.Background >

                    <SolidColorBrush Color="{StaticResource scb}" />

                </Button.Background>

                <Button.Width>100</Button.Width>

                <Button.Height>50</Button.Height>

                <Button.Content>Hello</Button.Content>

            </Button>

            <Button Name="button2" DockPanel.Dock="Top" Click="button2_Click">

                <Button.Background>

                    <SolidColorBrush Color="{StaticResource scb}"/>

                </Button.Background>

                <Button.Width>100</Button.Width>

                <Button.Height>50</Button.Height>

                <Button.Content>Hello</Button.Content>

            </Button>

        </DockPanel.Children>

    </DockPanel>

</Window>

L'exemple précédent représente le cas le plus simple et est fidèle à ce que nous connaissons depuis les premières versions du framework .Net.
Nous allons enrichir cet exemple afin de trapper l'évènement click sur n'importe quel bouton et animer la taille du bouton du bas à cette occasion. Pour cela nous allons ajouter un handler sur l'évènement click au niveau du dockPanel :

dockPanel.AddHandler(Button.ClickEvent, new RoutedEventHandler(button_Click));

Vous noterez que:
  • L'abonnement se fait avec la méthode AddHandler et non += car cela reviendrait à utiliser la mécanique spécifique à .Net 2.0 et inférieur.
  • Qu'un évènement est qualifié par un nom du type NOM_TYPE.EVENEMENT (par exemple Button.Click) lorsqu'il est utilisé dans un élément graphique qui en n'est pas la source.
Pour l'animation nous allons nous appuyer sur les classes natives de .Net 3.5 :

this.RegisterName("button1", button1);

            DoubleAnimation widthAnimation = new DoubleAnimation();

            widthAnimation.To = 150;

            widthAnimation.Duration = TimeSpan.FromSeconds(0.5);

            widthAnimation.AutoReverse = true;

            Storyboard.SetTargetName(widthAnimation, "button1");

            Storyboard.SetTargetProperty(widthAnimation, new PropertyPath(Button.WidthProperty));

            clickStoryboard.Children.Add(widthAnimation);

Enfin il ne reste plus qu'à démarrer l'animation lors de l'évènement :

private void button_Click(object sender, RoutedEventArgs e)

        {

            clickStoryboard.Begin(this);

        }

En XAML une version équivalente serait :

<Window x:Class="WpfApplication1.Window6"

   xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"

   xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"

   Title="Window6" Height="300" Width="300">

    <Window.Resources>

        <Color A="255" R="0" G="255" B="0" x:Key="scb"/>

    </Window.Resources>

    <DockPanel Name="dockPanel" LastChildFill="False">

        <DockPanel.Triggers>

            <EventTrigger RoutedEvent="Button.Click">

                <EventTrigger.Actions>

                    <BeginStoryboard>

                        <Storyboard>

                            <DoubleAnimation Storyboard.TargetName="button1"

                                            Storyboard.TargetProperty="Width"

                                            To="150"

                                            Duration="0:0:0.5"

                                            AutoReverse="True" />

                        </Storyboard>

                    </BeginStoryboard>

                </EventTrigger.Actions>

            </EventTrigger>

        </DockPanel.Triggers>

        <DockPanel.Children>

            <Button Name="button1" DockPanel.Dock="Bottom">

                <Button.Background >

                    <SolidColorBrush Color="{StaticResource scb}" />

                </Button.Background>

                <Button.Width>100</Button.Width>

                <Button.Height>50</Button.Height>

                <Button.Content>Hello</Button.Content>

            </Button>

            <Button x:Name="button2" DockPanel.Dock="Top" Click="button2_Click">

                <Button.Background>

                    <SolidColorBrush Color="{StaticResource scb}"/>

                </Button.Background>

                <Button.Width>100</Button.Width>

                <Button.Height>50</Button.Height>

                <Button.Content>Hello</Button.Content>

            </Button>

        </DockPanel.Children>

    </DockPanel>

</Window>

Dans cette version on voit apparaitre la notion de trigger. Avec WPF il existe plusieurs types de triggers (EventTrigger, DataTrigger, MultiTrigger...) qui se déclenchent quand une ou plusieurs conditions deviennent vraies (dans notre cas la levée de l'évènement Button.Click) afin d'exécuter une action (dans notre cas animer la taille du bouton).

Si vous souhaitez suivre en direct les changements de propriétés et les levées d'évènements dans votre application je vous conseille le logiciel Snoop. Ce dernier espionne toutes les variations qui peuvent apparaitre et vous permet même de modifier les paramètres de vos fenêtres :
 
/content/7BCD5199-24E2-43df-BDD9-58651344179B/Snoop.png
 
Cet outil peut être très utile non seulement pour vous aider à comprendre le fonctionnement de XAML mais aussi pour débugger votre application.

Enfin avec WPF vous entendrez aussi parler de WeakEvent . Ce genre d'évènement est utilisé quand on souhaite découpler la durée de vie de l'évènement et de la source (par exemple dans certains cas il est préférable de lier la durée de vie de l'évènement à celle du contrôle graphique qui l'utilise).

 Commentaires (3) - WPF et XAML 

Discussion démarée par Kader Yildirim le 12/05/2008 à 12:31 , 1 commentaire(s).
Discussion démarée par mehdi.mediouni le 12/05/2008 à 11:05 , 1 commentaire(s).
Discussion démarée par mehdi.mediouni le 12/05/2008 à 10:59 , 1 commentaire(s).

 Dernières Publications      

Exploiter les données CSV via Linq en toute simplicité
  A partir du requêteur dynamique fourni en exemple avec Visual Studio 2008, nous allons essayer de remplir les propriétés d'un ensemble d'objets à partir des données d'un fichier CSV. Nous enrichirons aussi le parseur de nos propres fonctions.
par Frédéric Mélantois posté le 17/05/2008 à 11:41, lu 140 fois, #0
Polymorphisme et contrats de données WCF
  WCF aborde les types polymorphes du point de vue de la sérialisation. En effet, la connaissance du type réel potentiel est rendue nécessaire dès la description du contrat de données. Une fois n'est pas coutume, j'ai réalisé l'exemple en VB.NET.
par Frédéric Colin posté le 14/05/2008 à 08:40, lu 296 fois, #2
A la découverte de BizTalk Server 2006 3/3
  Développer un assembleur pour BizTalk Server 2006 R2
par Kader Yildirim posté le 06/05/2008 à 13:20, lu 147 fois, #0
Chapitre III : Sync Services for ADO.NET et WCF
   Suite des deux premiers chapitres sur la synchronisation avec Sync Services for ADO.NET, voici un nouvel article impliquant WCF dans une synchronisation déconnectée.
Requêtes dynamiques sur les IEnumerable
  A partir d'un exemple fourni avec Visual Studio 2008, initialement prévu pour tout objet Queryable, nous allons présenter comment en ajoutant très peu de code rendre disponible aux IEnumerable un requêteur dynamique.
par Frédéric Mélantois posté le 24/04/2008 à 15:03, lu 842 fois, #0
Développer une visionneuse d'images avec WPF et WCF
  Au travers de cet article, nous allons découvrir comment mettre en place une visionneuse d'images, grâce aux technologies WPF et WCF.
par Thomas Lebrun posté le 22/04/2008 à 22:46, lu 1005 fois, #2
LINQRoleProvider
  L'objectif de cet article est d'implémenter un fournisseur de rôles ASP.NET personnalisé à l'aide de LINQ To Sql tout en faisant un tour d'horizons de la syntaxe des requêtes LINQ.
par Antoine Griffard posté le 13/04/2008 à 22:18, lu 491 fois, #4
WCF : Transfert de messages streamés et sécurisation personnalisée
  Je poursuis ma série d'articles sur WCF en vous présentant cette fois-ci le mode de communication Streamé. Histoire d'aller un petit peu plus loin, j'ai protégé le service de manière personnalisée et utilisé un binding très courant : BasicHttpBinding
par Frédéric Colin posté le 07/04/2008 à 08:12, lu 884 fois, #0

 Dernières Actualités      

Injection de code et API de profiling .NET
  Si vous êtes intéressés par la sécurité du Framework, par le reverse engineering et la manipulation/injection de code .NET et les packers, alors jetez un coup d’œil...
NDepend pour l'analyse statique de code .NET
  Pour ceux qui ne connaissent pas NDepend , il s’agit d’un outil d’analyse statique de code .NET qui permet de remonter des informations à toute une équipe de développement. NDepend aide à travailler sur...
Tags: Outils
Microsoft met à disposition son IoC Container Unity 1.0 en version finale
  Microsoft met à disposition la version 1.0 de son IoC container Unity, sur CodePlex sous la forme d'un Application Block des Enterprise Library. Si vous voulez en savoir plus sur le sujet en .NET je vous...
Tags: Application Block
Microsoft MVP (Most Valuable Professional) sur Tech Head Brothers
  Je voudrais féliciter les nouveaux Microsoft MVP (Most Valuable Professional) du mois d'Avril 2008 qui publient sur Tech Head Brothers! Sans les auteurs le site ne serait rien. Sébastien Pertus - MVP SQL...
Les Webcasts des Microsoft TechDays 2008 sont en ligne
  Si vous avez participé ou non au Microsoft TechDays 2008 vous avez certainement manqué certaines présentation qui vous intéressaient. Voilà enfin votre chance de pouvoir suivre ces présentations en ligne...
Rapide résumé de la conférence MIX08
  La conférence MIX08 s'est achevée la semaine dernière avec peu d'annonces de nouvelles technologies (DeepZoom) mais surtout des releases de produits (et c'est pas un mal...). Nous avons donc eu droit aux...