Click here to Skip to main content
Licence CPOL
First Posted 1 Nov 2007
Views 47,083
Downloads 811
Bookmarked 45 times

How to Build Dynamic Menus and Toolbars in WPF?

By | 1 Nov 2007 | Article
This article explains how to use build in features of WPF to insert menus, toolbars or any other content dynamically into your application

Introduction

WPF is a very dynamic language. It contains very smart objects, enables you to load, parse and display content dynamically. More than this, you can even create custom event handlers (commands) and bind them into loose XAML files that will be loaded in runtime, whenever you need it without recompiling application code.

Background

The challenge is as follows:

  • I want to have external XAML files with menus and toolbars
  • I do not want to recompile a project if I want to change something in those menus and toolbars
  • I want to provide commands for those menus and toolbars without recompiling
  • I want to write least amount of code to provide such functionality

Using the Code

Let's create our menu in the Menu.xaml file. This is a loose XAML file and it does not contain C# code behind.

<Menu xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 
  <MenuItem Header="Edit"> 
    <MenuItem Command="ApplicationCommands.Cut"/> 
    <MenuItem Command="ApplicationCommands.Copy"/> 
    <MenuItem Command="ApplicationCommands.Paste"/> 
  </MenuItem> 
</Menu>

I put into this file everything I'll use in the application. In this case, those are three common clipboard commands: Cut, Copy and Paste. ApplicationCommand will take care of those commands and provide me with requested functionality.

Now, let's create a toolbar in the ToolBar1.xaml file.

<ToolBar xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 
  <Button Command="ApplicationCommands.Cut"> 
    <Image Source="pack://siteoforigin:,,,/cut.png"/> 
  </Button> 
    <Button Command="ApplicationCommands.Copy"> 
      <Image Source="pack://siteoforigin:,,,/copy.png"/> 
    </Button> 
    <Button Command="ApplicationCommands.Paste"> 
      <Image Source="pack://siteoforigin:,,,/paste.png"/> 
    </Button> 
</ToolBar>

And last but not least, XAML code for my main Window, named Window1.xaml. Here, I'll put custom style to assure that once any of the Images or its parent controls become disabled, the images become semi-transparent.

<Window x:Class="DynamicMenu.Window1" 
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 
    Title="DynamicMenu" Height="300" Width="300" 
    > 
  <Window.Resources> 
    <Style TargetType="Image"> 
      <Style.Triggers> 
        <Trigger Property="IsEnabled" Value="False"> 
          <Setter Property="Opacity" Value="0.5"/> 
        </Trigger> 
      </Style.Triggers> 
    </Style> 
  </Window.Resources> 
    <StackPanel Name="main"> 
      <TextBox/> 
      <TextBox/> 
    </StackPanel> 
</Window>

Now, when we have all XAMLs, we have to load my dynamic menu and toolbar and add this into the application. To perform this action, we'll use XamlReader that will read and parse my loose XAML and add it into the application.

public Window1() 
       { 
           InitializeComponent(); 
           using (FileStream s = new FileStream("Menu1.xaml", FileMode.Open)) 
           { 
               Menu menu = XamlReader.Load(s, new ParserContext()) as Menu; 
               if (menu != null) 
               { 
                   main.Children.Insert(0, menu); 
               } 
           } 
           using (FileStream s = new FileStream("ToolBar1.xaml", FileMode.Open)) 
           { 
               ToolBar tool = XamlReader.Load(s, new ParserContext()) as ToolBar; 
               if (tool != null) 
               { 
                   main.Children.Insert(1, tool); 
               } 
           } 
       } 

Points of Interest

Isn't it nice that you have not recompiled anything in order to achieve very rich functionality?

History

  • 1st November, 2007: Initial post

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Tamir Khason

Architect
Better Place
Israel Israel

Member

Hello! My name is Tamir Khason, and I am software architect, project manager, system analyst and [of course] programmer. In addition to writing big amount of documentation, I also write code, a lot of code. I used to work as a freelance architect, project manager, trainer, and consultant here, in Israel, but recently join the company with extremely persuasive idea - to make a world better place. I have very pretty wife and 3 charming kids, but unfortunately almost no time for them.
 
To be updated within articles, I publishing, visit my blog or subscribe RSS feed. Also you can follow me on Twitter to be up to date about my everyday life.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralReal-world example? Pinmemberpaul13075:18 27 May '11  
QuestionWPF Project Type Pinmembercyberjp8:07 7 Nov '07  
AnswerRe: WPF Project Type PinmemberTamir Khason19:58 7 Nov '07  
GeneralThis would work for built-in commands but... Pinmemberkemetokara15:03 1 Nov '07  
GeneralRe: This would work for built-in commands but... PinmemberTamir Khason20:29 1 Nov '07  
With custom commands, you should do the exact same things
Assign custom namespace e.g. l:myNamespace and then use l:MyCommands.Command as attribute
 
Want dot.NET? Just ask:
Please, www.dotNET.us

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web01 | 2.5.120529.1 | Last Updated 1 Nov 2007
Article Copyright 2007 by Tamir Khason
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid