Click here to Skip to main content
Click here to Skip to main content

WPF: ContextMenu Strikes Again. DataContext Not Updated

By , 27 Feb 2011
 

Long Story Short

You probably already know that ContextMenu class is not part of the WPF visual tree (see here and here for some problems that this causes).

Still, ContextMenu acquires DataContext of its parent control (since .NET 3.5?). The problem is, this acquisition happens only once. If parent's data context changes at a later time, the ContextMenu's data context will not be updated. This will cause issues if the view with the context menu is bound to something variable, like a selected item of a listbox.

Workaround

The workaround is to explicitly bind menu's data context to parent's datacontext as follows:

<ContextMenu DataContext="{Binding PlacementTarget.DataContext, 
	RelativeSource={RelativeSource Self}}" >

This magical spell tells WPF to create a permanent binding between the menu's data context and its "placement target" (i.e. parent) data context, which continues to work even after parent's data context is changed. You need this spell only if you expect parent's data context to change during the life of the parent.

Sample

Sample screen shot

I created a simple sample that illustrates the problem. It contains a list box with country names and two user controls: the "good" and the "bad". Both controls show country capital. They also have right click menu to show country language. The good control shows correct language of the selected country. The bad control shows correct language when the menu is first invoked, and then keeps showing that language even if selected country changes.

This happens because the menu gets created first on right click and then acquires the (correct) data context from the parent. On subsequent right clicks, the same menu object is reused (as proved by the "Same menu?" command), and its data context never changes, unless we create an explicit binding for it.

Here are some key pieces of code (certain details, including "Same menu" command were omitted for clarity):

class Country
{
    public string Name { get; set; }
    public string Capital { get; set; }
    public string Language { get; set; }
}

static class Countries
{
    public static readonly Country[] List = new[]
    {
        new Country { Name = "USA", Capital="Washington", Language="English"},
        new Country { Name = "Spain", Capital="Madrid", Language="Spanish"},
        new Country { Name = "France", Capital="Paris", Language="French"},
        new Country { Name = "Brazil", Capital="Brasilia", Language="Portuguese"},
        new Country { Name = "Thailand", Capital="Bangkok", Language="Thai"},
    };
 } 

class LanguageCommand : ICommand
{
    public void Execute(object parameter)
    {
        object safeParameter = parameter ?? "null";
        MessageBox.Show(safeParameter.ToString());
    }
}
<!-- MainWindow.xaml -->
<Window Title="MainWindow" Height="350" Width="525">
    <DockPanel LastChildFill="True">
        <ListBox Name="CountryList" ItemsSource="{x:Static local:Countries.List}" />
        <UniformGrid Rows="2" Columns="1" DataContext="{Binding SelectedItem, 
		ElementName=CountryList}">
            <local:GoodControl />
            <local:BadControl />
        </UniformGrid>
    </DockPanel>
</Window>
        
<!-- GoodControl.xaml -->
<UserControla x:Class="ContextMenuDataContext.GoodControl">
    <UserControl.Resources>
        <local:LanguageCommand x:Key="LanguageCommand" />
    </UserControl.Resources>
    <UserControl.ContextMenu>
        <ContextMenu DataContext="{Binding PlacementTarget.DataContext, 
		RelativeSource={RelativeSource Self}}" >
            <MenuItem Header="Language" Command="{StaticResource LanguageCommand}" 
		CommandParameter="{Binding Language}" />
        </ContextMenu>
    </UserControl.ContextMenu>
    <TextBlock Text="{Binding  Capital}" />
</UserControl>

<!-- BadControl.xaml -->
<UserControl x:Class="ContextMenuDataContext.BadControl">
    <UserControl.Resources>
        <local:LanguageCommand x:Key="LanguageCommand" />
    </UserControl.Resources>
    <UserControl.ContextMenu>
        <ContextMenu>
            <MenuItem Header="Language" Command="{StaticResource LanguageCommand}" 
		CommandParameter="{Binding Language}" />
        </ContextMenu>
    </UserControl.ContextMenu>
    <TextBlock Text="{Binding Capital}" />
</UserControl />

Conclusion

Default mechanism for data context binding works well in most cases, because most views never change data context during their lives. The problems start if the data context changes. The most annoying issue is that even if the data context changes, everything will work right the first time the menu is invoked. However, stale data will be returned for subsequent invocations. This may go unnoticed for quite some time. Explicitly bind context menu's data context to PlacementTarget.DataContext to avoid this bug.

History

  • 27th February, 2011: Initial post

License

This article, along with any associated source code and files, is licensed under The Apache License, Version 2.0

About the Author

Ivan Krivyakov
Architect Sungard Consulting Services
United States United States
Member
Ivan is a hands-on software architect working for SunGard Consulting, in the New York City area. At present I am mostly building complex multi-threaded WPF application for the financial sector, but I am also interested in cloud computing, web development, mobile development, etc.
 
Please visit my web site: www.ikriv.com.

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.
Search this forum  
    Spacing  Noise  Layout  Per page   
GeneralMy vote of 5membermityapo23 Sep '12 - 6:12 
i spent many time
GeneralMany ThanksmemberAndres Hirth14 May '12 - 0:54 
for this elegant solution for that weird behavior.
GeneralThank U very much!!!memberlavige77710 May '12 - 21:23 
!
GeneralThanks!memberRobert Hahn1 Aug '11 - 23:09 
Your article helped me very much to understand this crazy bug/feature.
 
Robert
GeneralGreat articlememberwisodev25 Jul '11 - 11:21 
Thanks for great article, it helped me a lot.
GeneralExcellent!memberMark Cranness16 Mar '11 - 22:27 
I had the same problem, and my solution after some brain storming was a ContextMenu_Opened event handler that set DataContext from PlacementTarget.DataContext.
 
Then I googled for "DataContext PlacementTarget", wondering if anybody had found the same problem and came up with the same solution, and found your article!
 
I have deleted the ContextMenu_Opened and use your method (thank you) which is better.

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

Permalink | Advertise | Privacy | Mobile
Web03 | 2.6.130516.1 | Last Updated 27 Feb 2011
Article Copyright 2011 by Ivan Krivyakov
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid