问题 如何知道在DropDownButton中点击了什么


这一页,DropDownButton使用ContextMenu来显示ItemsSource。我们如何知道用户点击的内容?按钮上的Click事件不是菜单,而是按钮本身。我看不到其他事件。


7792
2018-04-11 17:46


起源

检查此事件: DropDownOpened/DropDownClosed, ContextMenuOpening/ContextMenuClosing。 - Anatoliy Nikolaev
不要在WPF中使用事件。有可能控制有一个 SelectedItem 属性, - 或 - 您可以分配 Command对于每个项目,就像你在普通项目中所做的那样 Menu。 - Federico Berasategui
@HighCore:DropDownButton是,引用: This control almost the same as SplitButton with few differences: It has no SelectedItem and SelectedIndex properties and also has no SelectionChanged event.。 - Anatoliy Nikolaev
@AnatoliyNikolaev这意味着它不是一个 Selector。我想知道为什么有人会开发一个UI元素,它应该让你选择东西而不是继承 Selector...... S: - Federico Berasategui


答案:


我遇到了这个问题寻找相同的答案。我从来没有在网上找到任何东西,但我自己发现了这个解决方案也许它将在未来帮助某人。

如前所述 DropDownButton 用一个 ContextMenu 显示它 ItemsSource。基本上我正在寻找的是一个来自按钮的“菜单式”下拉菜单。例如,假设你有一个 DropDownButton 说“添加”。也许你想要2个选项,如“添加新”和“添加现有”。所以这就是我做的......

首先,我做了一些对象来保存标题/内容和命令。

public class TitledCommand
{
    public String Title { get; set; }
    public ICommand Command { get; set; }
}

从理论上讲,你会有一个这样的列表绑定到 ItemsSource 的 DropDownButton

public List<TitledCommand> TitledCommmands { get; private set; }

现在我们只为样式容器设置样式 DropDownButton 所以它从我们的对象中获取标题和命令 ItemsSource

包括MahApps:

xmlns:metroControls="http://metro.mahapps.com/winfx/xaml/controls"

这是风格......

<metroControls:DropDownButton Content="Add" ItemsSource="{Binding Path=TitledCommmands}">
    <metroControls:DropDownButton.ItemContainerStyle>        
        <Style TargetType="MenuItem">
            <Setter Property="Header" Value="{Binding Path=Title}"/>
            <Setter Property="Command" Value="{Binding Path=Command}"/>
        </Style>          
    </metroControls:DropDownButton.ItemContainerStyle>
</metroControls:DropDownButton>

14
2018-02-16 00:45



在我的情况下使用 <Setter Property="Header" Value="{Binding Path=Title}"/> 没有工作,DisplayMemberPath属性更好,但无论如何你是一个救星;) - kwesolowski
这个应该是接受的答案。 - Abin Mathew
做得很好!谢谢 - Ravid Goldenberg


您可以覆盖控件的项模板,并可以在其中添加处理程序,如下所示:

<controls:DropDownButton Content="Select me" x:Name="selectMeDropDownButton">
    <controls:DropDownButton.ItemTemplate>
        <DataTemplate>
            <TextBlock Text="{Binding}" MouseDown="selectMeDropDownButton_TextBlock_MouseDown" />
        </DataTemplate>
    </controls:DropDownButton.ItemTemplate>
</controls:DropDownButton>

并在代码隐藏文件中实现事件处理程序,如下所示:

void selectMeDropDownButton_TextBlock_MouseDown(object sender, MouseButtonEventArgs e)
{
    if (e.ChangedButton == MouseButton.Left && this.selectMeDropDownButton.IsExpanded)
    {
        var value = ((TextBlock)e.Source).DataContext;
        // Do something meaningful with the value, it's an item from ItemsSource
    }
}

检查 DropDownButton.IsExpanded 是必要的 ItemTemplate 适用于 Content 在按钮本身。当然你可以用任何替换TextBlock Control/UIElement 你喜欢。


3
2017-11-05 10:08



IMO最好使用 <MenuItem Click="..." /> 代替 <TextBlock /> 在里面 <DataTemplate />。 - xmedeko


创建一个 attached property 到ContextMenu / DropDownButton(你喜欢哪个)。如果你做了下拉菜单,那么得到它正在显示的上下文菜单然后连接 Click 来自它的事件并将价值推回房产。


0
2018-04-11 18:10





使用 SplitButton 而不是DropDownButton。第一个有SelectionChanged事件。


-1
2017-07-15 16:01



是的但点击时他们没有同样的感觉...... - Ravid Goldenberg