如何跟踪 ItemsControl 的哪个元素导致上下文菜单打开?

Posted

技术标签:

【中文标题】如何跟踪 ItemsControl 的哪个元素导致上下文菜单打开?【英文标题】:How can I track which element of ItemsControl caused ContexMenu to open? 【发布时间】:2021-08-28 21:30:28 【问题描述】:

在我的图书编辑器应用程序中,我需要按层次顺序显示主题和子主题,所以我做了这样的标记:

<StackPanel>
      <ScrollViewer>
        <Grid ColumnDefinitions="auto, *" RowDefinitions="auto">
          <ItemsControl Grid.Column="0" Grid.Row="0" Items="Binding Book.Themes">
            <ItemsControl.ItemTemplate>
              <DataTemplate>
                <StackPanel>
                  <Button Content="Binding Name" Command="Binding RelativeSource=RelativeSource FindAncestor, AncestorType=x:Type Window, Path=DataContext.UpdateSubthemesVisibilityCommand" CommandParameter="Binding Name"/>
                  <ItemsControl Items="Binding Subthemes" IsVisible="Binding AreSubthemesVisible">
                    <ItemsControl.ItemTemplate>
                      <DataTemplate>
                        <Button Content="Binding Name"/>
                      </DataTemplate>
                    </ItemsControl.ItemTemplate>
                    <ItemsControl.ContextMenu>
                      <ContextMenu>
                        <MenuItem Header="Rename" />
                        <MenuItem Header="Delete"/>
                      </ContextMenu>
                    </ItemsControl.ContextMenu>
                  </ItemsControl>
                </StackPanel>
              </DataTemplate>
            </ItemsControl.ItemTemplate>
            <ItemsControl.ContextMenu>
              <ContextMenu>
                <MenuItem Header="Rename" Command="Binding RenameThemeCommand" CommandParameter="...How to bind it to button which cause menu to open"/>
                <MenuItem Header="Delete"/>
              </ContextMenu>
            </ItemsControl.ContextMenu>
          </ItemsControl>
        </Grid>
      </ScrollViewer>
    </StackPanel>

也许标记不是很清楚,但关键是我有外部的 ItemsControl 和内部的。外部项目具有结构:按钮(主题名称)+ 内部 ItemsControl,它也将子主题的名称显示为按钮。这是截图:

每个 ItemsControl 都有自己的 ContextMenu,但我无法正确绑定它们。我希望“重命名”MenuItems 绑定到 Button.Content 的 Button 右键单击​​导致菜单打开。我应该怎么做? 我正在使用 MVVM Avalonia,但我希望在 WPF 中它的工作方式相同,以便我可以将 WPF 主题标签添加到这个问题中。

【问题讨论】:

【参考方案1】:

您不需要将 ContextMenu 添加到 ItemsControl。它需要添加到 ItemsControl 中。 这是在 ItemContainerStyle 中完成的。 在这种情况下,在 ContextMenu DataContext 中将包含调用 ContextMenu 的元素。

例子:

    <Window.Resources>
        <spec:StringCollection x:Key="strings">
            <sys:String>First</sys:String>
            <sys:String>Second</sys:String>
            <sys:String>Third</sys:String>
        </spec:StringCollection>
        <Style x:Key="ItemStyle" TargetType="ContentPresenter">
            <Setter Property="ContextMenu">
                <Setter.Value>
                    <ContextMenu>
                        <TextBlock Text="Binding"/>
                    </ContextMenu>
                </Setter.Value>
            </Setter>
        </Style>
    </Window.Resources>
    <ItemsControl ItemsSource="StaticResource strings"
                  ItemContainerStyle="StaticResource ItemStyle"/>

【讨论】:

以上是关于如何跟踪 ItemsControl 的哪个元素导致上下文菜单打开?的主要内容,如果未能解决你的问题,请参考以下文章

使用 ItemsSource 时操作无效。改为使用 ItemsControl.ItemsSource 访问和修改元素

如何正确改善 ItemsControl 加载并避免冻结

ItemsControl - 网格子元素自动调整大小

有没有办法跟踪使用GWT点击哪个元素?

使用 WrapPanel 添加一个 TextBox 作为 ItemsControl 的最后一个元素

WPF 中的 ItemsControl 是不是有一个好的解决方案,可以通过垂直拖放对其元素进行重新排序? [关闭]