Как связать MenuItem.Header со свойством зависимостей Window / UserControl? - PullRequest
4 голосов
/ 25 апреля 2011

Мне интересно, как я могу привязать MenuItem.Header к родительскому свойству зависимостей Window / UserControl?Вот простой пример:

Window1.xaml :

<Window x:Class="WpfApplication1.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="Window1" Height="300" Width="300" x:Name="self">
    <Grid>
        <Grid.ContextMenu>
            <ContextMenu>
                <MenuItem Header="{Binding Path=MenuText, ElementName=self}" />
            </ContextMenu>
        </Grid.ContextMenu>
        <TextBlock Text="{Binding Path=MenuText, ElementName=self}"/>
    </Grid>
</Window>

Window1.xaml.cs :

public partial class Window1 : Window {
    public static readonly DependencyProperty MenuTextProperty = DependencyProperty.Register(
        "MenuText", typeof (string), typeof (Window1), new PropertyMetadata("Item 1"));

    public Window1()
    {
        InitializeComponent();
    }

    public string MenuText {
        get { return (string)this.GetValue(MenuTextProperty); }
        set { this.SetValue(MenuTextProperty, value); }
    }
}

В моем случае в текстовом блоке отображается «Элемент 1», а в контекстном меню отображается пустой элемент.Что я делаю не так?Мне кажется, что я столкнулся с серьезным неправильным пониманием принципов привязки данных WPF.

Ответы [ 2 ]

7 голосов
/ 25 апреля 2011

Вы должны увидеть это в окне вывода Visual Studio:

Ошибка System.Windows.Data: 4: невозможно найти источник для привязки со ссылкой 'ElementName = я'. BindingExpression: Path = MenuText; DataItem = NULL; целевой элемент 'MenuItem' (Name = ''); целевое свойство is 'Header' (тип 'Object')

Это связано с тем, что ContextMenu отключено от VisualTree, поэтому привязку нужно выполнять по-другому.

Один из способов - через ContextMenu.PlacementTarget (который должен быть сеткой), вы можете использовать его DataContext для установления привязки, например ::

<MenuItem Header="{Binding RelativeSource={RelativeSource AncestorType=ContextMenu}, Path=PlacementTarget.DataContext.MenuText}"/>

или настройте DataContext в самом ContextMenu:

<ContextMenu DataContext="{Binding RelativeSource={RelativeSource Self}, Path=PlacementTarget.DataContext}">
    <MenuItem Header="{Binding Path=MenuText}"/>
</ContextMenu>

Если это не вариант (поскольку DataContext Grid не может быть Window / UserControl), вы можете попытаться передать ссылку на Window / UserControl через Tag вашей Grid, например.

<Grid ...
      Tag="{x:Reference self}">
    <Grid.ContextMenu>
        <!-- The DataContext is now bound to PlacementTarget.Tag -->
        <ContextMenu DataContext="{Binding RelativeSource={RelativeSource Self}, Path=PlacementTarget.Tag}">
            <MenuItem Header="{Binding Path=MenuText}"/>
        </ContextMenu>
    ...

В качестве примечания: из-за этого поведения я стремлюсь определить вспомогательный стиль в App.xaml, чтобы все ContextMenus «псевдо-наследовали» DataContext от их родителя:

    <!-- Context Menu Helper -->
    <Style TargetType="{x:Type ContextMenu}">
        <Setter Property="DataContext" Value="{Binding PlacementTarget.DataContext, RelativeSource={RelativeSource Self}}"/>
    </Style>
1 голос
/ 26 апреля 2011

Альтернативой решению HB является следующее прикрепленное поведение: Присоединенное свойство ContextMenuServiceExtensions.DataContext

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...