I want to create a context menu where one of the menuItem would be a submenu with a choice among enum values.
I do not want to hard code any of the values from my enum into xaml because I want that any enum value changes would be automtically reflected in the UI without any intervention.
I want my menu to be a regular context menu without any artifact (I mean the appearance should be as a regular ContextMenu).
I've tried many ways without success. Each of my trial always misses something but mainly it seems that the main missing part is a converterParamter that could be bound to something.
I red:
This is my many trials and related code:
<Window x:Class="WpfContextMenuWithEnum.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:wpfContextMenuWithEnum="clr-namespace:WpfContextMenuWithEnum"
xmlns:system="clr-namespace:System;assembly=mscorlib"
xmlns:converter="clr-namespace:WpfContextMenuWithEnum.Converter"
Title="MainWindow" Height="350" Width="525"
Name="MyWindow">
<Window.DataContext>
<wpfContextMenuWithEnum:MainWindowModel></wpfContextMenuWithEnum:MainWindowModel>
</Window.DataContext>
<Window.Resources>
<ObjectDataProvider x:Key="EnumChoiceProvider" MethodName="GetValues" ObjectType="{x:Type system:Enum}">
<ObjectDataProvider.MethodParameters>
<x:Type TypeName="wpfContextMenuWithEnum:EnumChoice"/>
</ObjectDataProvider.MethodParameters>
</ObjectDataProvider>
<converter:EnumToBooleanConverter x:Key="EnumToBooleanConverter"></converter:EnumToBooleanConverter>
<converter:MultiBind2ValueComparerConverter x:Key="MultiBind2ValueComparerConverter"></converter:MultiBind2ValueComparerConverter>
</Window.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="Auto"></RowDefinition>
</Grid.RowDefinitions>
<TextBox Text="Right click me">
<TextBox.ContextMenu>
<ContextMenu ItemsSource="{Binding Source={StaticResource EnumChoiceProvider}}">
<ContextMenu.ItemTemplate>
<DataTemplate>
<MenuItem IsCheckable="True" Header="{Binding Path=.}">
<MenuItem.IsChecked>
<MultiBinding Converter="{StaticResource MultiBind2ValueComparerConverter}">
<Binding Path="DataContext.ModelEnumChoice" RelativeSource="{RelativeSource FindAncestor, AncestorType={x:Type Window}}" />
<Binding Path="." Mode="OneWay"></Binding>
</MultiBinding>
</MenuItem.IsChecked>
</MenuItem>
</DataTemplate>
</ContextMenu.ItemTemplate>
</ContextMenu>
</TextBox.ContextMenu>
</TextBox>
</Grid>
</Window>
Enum:
using System.ComponentModel;
namespace WpfContextMenuWithEnum
{
public enum EnumChoice
{
[Description("Default")]
ChoiceDefault = 0, // easier if the default have value = 0
[Description("<1>")]
Choice1 = 1,
[Description("<2>")]
Choice2 = 2,
}
}
Converters:
using System;
using System.Windows;
using System.Windows.Data;
namespace WpfContextMenuWithEnum.Converter
{
public class ConverterWrapperWithDependencyParameterConverter : DependencyObject, IValueConverter
{
public static readonly DependencyProperty ParameterProperty = DependencyProperty.Register("Parameter",
typeof(object), typeof(ConverterWrapperWithDependencyParameterConverter));
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (parameter != null)
{
throw new ArgumentException("The parameter should be set directly as a property not into the Binding object.");
}
return Converter.Convert(value, targetType, Parameter, culture);
}
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (parameter != null)
{
throw new ArgumentException("The parameter should be set directly as a property not into the Binding object.");
}
return Converter.ConvertBack(value, targetType, Parameter, culture);
}
public object Parameter
{
get { return GetValue(ParameterProperty); }
set { SetValue(ParameterProperty, value); }
}
public IValueConverter Converter { get; set; }
}
}
using System;
using System.Windows.Data;
namespace WpfContextMenuWithEnum.Converter
{
public class EnumToBooleanConverter : IValueConverter
{
// **********************************************************************
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
return value.Equals(parameter);
}
// **********************************************************************
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
return value.Equals(true) ? parameter : Binding.DoNothing;
}
// **********************************************************************
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Data;
namespace WpfContextMenuWithEnum.Converter
{
public class MultiBind2ValueComparerConverter : IMultiValueConverter
{
public object Convert(object[] values, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (values.Length != 2)
{
throw new ArgumentException("Can compare only 2 values together fo equality");
}
return (values[0].Equals(values[1]));
}
public object[] ConvertBack(object value, Type[] targetTypes, object parameter, System.Globalization.CultureInfo culture)
{
// if ((bool)value == true)
throw new NotImplementedException();
}
}
}
Trial 1: MultiBindConverter ConvertBack can't work, it misses information.
<ContextMenu ItemsSource="{Binding Source={StaticResource EnumChoiceProvider}}">
<ContextMenu.ItemTemplate>
<DataTemplate>
<MenuItem IsCheckable="True" Header="{Binding Path=.}">
<MenuItem.IsChecked>
<MultiBinding Converter="{StaticResource MultiBind2ValueComparerConverter}">
<Binding Path="DataContext.ModelEnumChoice" RelativeSource="{RelativeSource FindAncestor, AncestorType={x:Type Window}}" />
<Binding Path="."></Binding>
</MultiBinding>
</MenuItem.IsChecked>
</MenuItem>
</DataTemplate>
</ContextMenu.ItemTemplate>
</ContextMenu>
Trial 2: My ConverterParameter bind did not work at all. It never received any value
<ContextMenu ItemsSource="{Binding Source={StaticResource EnumChoiceProvider}}">
<ContextMenu.ItemTemplate>
<DataTemplate>
<MenuItem IsCheckable="True" Header="{Binding Path=.}">
<MenuItem.IsChecked>
<Binding Path="DataContext.ModelEnumChoice" RelativeSource="{RelativeSource FindAncestor, AncestorType={x:Type Window}}">
<Binding.Converter>
<converter:ConverterWrapperWithDependencyParameterConverter Converter="{StaticResource EnumToBooleanConverter}"
Parameter="{Binding Path=.}"/>
</Binding.Converter>
</Binding>
</MenuItem.IsChecked>
</MenuItem>
</DataTemplate>
</ContextMenu.ItemTemplate>
</ContextMenu>
Trial 3:
With a listBox using template and SelectedItem but the UI is not as standard as it should be (an additional frame appears).
Enumeration (or enum) is a value data type in C#. It is mainly used to assign the names or string values to integral constants, that make a program easy to read and maintain.
So you want to be able to
Enum
to ContextMenu
and display it's Description
attributeEnum
, only one can be "active" at any given timeSomething like the following?
MainWindow.xaml
<Window x:Class="WpfApplication1.View.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:viewModel="clr-namespace:WpfApplication1.ViewModel"
xmlns:local="clr-namespace:WpfApplication1"
Title="MainWindow"
Height="300"
Width="250">
<!-- Set data context -->
<Window.DataContext>
<viewModel:MainViewModel />
</Window.DataContext>
<!-- Converters -->
<Window.Resources>
<local:EnumDescriptionConverter x:Key="EnumDescriptionConverter" />
<local:EnumCheckedConverter x:Key="EnumCheckedConverter" />
</Window.Resources>
<!-- Element -->
<TextBox Text="Right click me">
<!-- Context menu -->
<TextBox.ContextMenu>
<ContextMenu ItemsSource="{Binding EnumChoiceProvider}">
<ContextMenu.ItemTemplate>
<DataTemplate>
<!-- Menu item header bound to enum converter -->
<!-- IsChecked bound to current selection -->
<!-- Toggle bound to a command, setting current selection -->
<MenuItem
IsCheckable="True"
Width="150"
Header="{Binding Path=., Converter={StaticResource EnumDescriptionConverter}}"
Command="{Binding DataContext.ToggleEnumChoiceCommand, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType=ContextMenu}}"
CommandParameter="{Binding}">
<MenuItem.IsChecked>
<MultiBinding Mode="OneWay"
NotifyOnSourceUpdated="True"
UpdateSourceTrigger="PropertyChanged"
Converter="{StaticResource EnumCheckedConverter}">
<Binding Path="DataContext.SelectedEnumChoice"
RelativeSource="{RelativeSource Mode=FindAncestor, AncestorType=ContextMenu}" />
<Binding Path="."></Binding>
</MultiBinding>
</MenuItem.IsChecked>
</MenuItem>
</DataTemplate>
</ContextMenu.ItemTemplate>
</ContextMenu>
</TextBox.ContextMenu>
</TextBox>
</Window>
MainViewModel.cs
namespace WpfApplication1.ViewModel
{
public class MainViewModel : ViewModelBase // where base implements INotifyPropertyChanged
{
private EnumChoice? _selectedEnumChoice;
public MainViewModel()
{
EnumChoiceProvider = new ObservableCollection<EnumChoice>
(Enum.GetValues(typeof(EnumChoice)).Cast<EnumChoice>());
ToggleEnumChoiceCommand = new RelayCommand<EnumChoice>
(arg => SelectedEnumChoice = arg);
}
// Selections
public ObservableCollection<EnumChoice> EnumChoiceProvider { get; set; }
// Current selection
public EnumChoice? SelectedEnumChoice
{
get
{
return _selectedEnumChoice;
}
set
{
_selectedEnumChoice = value != _selectedEnumChoice ? value : null;
RaisePropertyChanged();
}
}
// "Selection changed" command
public ICommand ToggleEnumChoiceCommand { get; private set; }
}
}
EnumChoice.cs
namespace WpfApplication1
{
public enum EnumChoice
{
[Description("Default")]
ChoiceDefault,
[Description("<1>")]
Choice1,
[Description("<2>")]
Choice2
}
}
EnumDescriptionConverter.cs
namespace WpfApplication1
{
// Extract enum description
public class EnumDescriptionConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
MemberInfo[] memberInfos = value.GetType().GetMember(value.ToString());
if (memberInfos.Length > 0)
{
object[] attrs = memberInfos[0].GetCustomAttributes(typeof (DescriptionAttribute), false);
if (attrs.Length > 0)
return ((DescriptionAttribute) attrs[0]).Description;
}
return value;
// or maybe just
//throw new InvalidEnumArgumentException(string.Format("no description found for enum {0}", value));
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}
}
EnumCheckedConverter.cs
namespace WpfApplication1
{
// Check if currently selected
public class EnumCheckedConverter : IMultiValueConverter
{
public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
{
return !values.Contains(null) && values[0].ToString().Equals(values[1].ToString(), StringComparison.OrdinalIgnoreCase);
}
public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With