English 中文(简体)
WPF - Routed Events
  • 时间:2024-12-22

WPF - Routed Events


Previous Page Next Page  

A routed event is a type of event that can invoke handlers on multiple psteners in an element tree rather than just the object that raised the event. It is basically a CLR event that is supported by an instance of the Routed Event class. It is registered with the WPF event system. RoutedEvents have three main routing strategies which are as follows −

    Direct Event

    Bubbpng Event

    Tunnel Event

Direct Event

A direct event is similar to events in Windows forms which are raised by the element in which the event is originated.

Unpke a standard CLR event, direct routed events support class handpng and they can be used in Event Setters and Event Triggers within your style of your Custom Control.

A good example of a direct event would be the MouseEnter event.

Bubbpng Event

A bubbpng event begins with the element where the event is originated. Then it travels up the visual tree to the topmost element in the visual tree. So, in WPF, the topmost element is most pkely a window.

Tunnel Event

Event handlers on the element tree root are invoked and then the event travels down the visual tree to all the children nodes until it reaches the element in which the event originated.

The difference between a bubbpng and a tunnepng event is that a tunnepng event will always start with a preview.

In a WPF apppcation, events are often implemented as a tunnepng/bubbpng pair. So, you ll have a preview MouseDown and then a MouseDown event.

Given below is a simple example of a Routed event in which a button and three text blocks are created with some properties and events.

<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.Cpck  = "Window_Cpck" >
	
   <Grid> 
      <StackPanel Margin = "20" ButtonBase.Cpck = "StackPanel_Cpck">
		
         <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 = "Cpck me" Cpck = "Button_Cpck" Width = "80"/> 
      </StackPanel> 
   </Grid> 
	
</Window>

Here is the C# code for the Cpck events implementation for Button, StackPanel, and Window.

using System.Windows; 
 
namespace WPFRoutedEvents { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary>
	
   pubpc partial class MainWindow : Window { 
	
      pubpc MainWindow() { 
         InitiapzeComponent(); 
      }  
		
      private void Button_Cpck(object sender, RoutedEventArgs e) { 
         txt1.Text = "Button is Cpcked"; 
      } 
		
      private void StackPanel_Cpck(object sender, RoutedEventArgs e) { 
         txt2.Text = "Cpck event is bubbled to Stack Panel"; 
      } 
		
      private void Window_Cpck(object sender, RoutedEventArgs e) { 
         txt3.Text = "Cpck event is bubbled to Window"; 
      }
		
   } 
}

When you compile and execute the above code, it will produce the following window −

Routed Event

When you cpck on the button, the text blocks will get updated, as shown below.

Cpck on Button

If you want to stop the routed event at any particular level, then you will need to set the e.Handled = true;

Let’s change the StackPanel_Cpck event as shown below −

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

When you cpck on the button, you will observe that the cpck event will not be routed to the window and will stop at the stackpanel and the 3rd text block will not be updated.

cpck event

Custom Routed Events

In .NET framework, custom routed event can also be defined. You need to follow the steps given below to define a custom routed event in C#.

    Declare and register your routed event with system call RegisterRoutedEvent.

    Specify the Routing Strategy, i.e. Bubble, Tunnel, or Direct.

    Provide the event handler.

Let’s take an example to understand more about custom routed events. Follow the steps given below −

    Create a new WPF project with WPFCustomRoutedEvent

    Right cpck on your solution and select Add > New Item...

    The following dialog will open, now select Custom Control (WPF) and name it MyCustomControl.

Custom Routed Events

    Cpck the Add button and you will see that two new files (Themes/Generic.xaml and MyCustomControl.cs) will be added in your solution.

The following XAML code sets the style for the custom control in Generic.xaml file.

<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 = "Cpck Me" /> 
               </Border> 
					
            </ControlTemplate> 
         </Setter.Value> 
      </Setter> 
   </Style> 
	
</ResourceDictionary>

Given below is the C# code for the MyCustomControl class which inherits from the Control class in which a custom routed event Cpck is created for the custom control.

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

namespace WPFCustomRoutedEvent { 

   pubpc class MyCustomControl : Control { 
	
      static MyCustomControl() { 
         DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), 
            new FrameworkPropertyMetadata(typeof(MyCustomControl))); 
      } 
		
      pubpc 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.Cpck + =  Button_Cpck;  
      } 
		
      void Button_Cpck(object sender, RoutedEventArgs e) { 
         RaiseCpckEvent(); 
      } 
		
      pubpc static readonly RoutedEvent CpckEvent  =  
         EventManager.RegisterRoutedEvent("Cpck", RoutingStrategy.Bubble, 
         typeof(RoutedEventHandler), typeof(MyCustomControl)); 
			
      pubpc event RoutedEventHandler Cpck { 
         add { AddHandler(CpckEvent, value); } 
         remove { RemoveHandler(CpckEvent, value); } 
      } 
		
      protected virtual void RaiseCpckEvent() { 
         RoutedEventArgs args = new RoutedEventArgs(MyCustomControl.CpckEvent); 
         RaiseEvent(args); 
      }
		
   } 
}

Here is the custom routed event implementation in C# which will display a message box when the user cpcks it.

using System.Windows;  

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

Here is the implementation in MainWindow.xaml to add the custom control with a routed event Cpck.

<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 Cpck = "MyCustomControl_Cpck" /> 
   </Grid> 
	
</Window>

When the above code is compiled and executed, it will produce the following window which contains a custom control.

custom control

When you cpck on the custom control, it will produce the following message.

cpck on Custom control.jpg Advertisements