【问题标题】:Default ContextMenu Style - WPF默认 ContextMenu 样式 - WPF
【发布时间】:2009-03-13 06:13:35
【问题描述】:

我正在尝试修改 WPF 中 ContextMenu 的默认样式。

通常,您可以使用编辑控件部件(模板)> 编辑副本菜单选项在 Expression Blend 中创建默认副本。但是我不知道如何使用 ContextMenu 来做到这一点。知道如何获得要修改的默认样式吗?

我正在尝试禁用通常显示图标的上下文菜单的左侧。

谢谢!

更新:也许我不清楚删除图标。例如,如果您有一个没有图标的上下文菜单,那么菜单的整个左侧都是浪费空间。我想修改上下文菜单背景的默认样式以删除它。只是我不知道如何访问这个默认样式。

【问题讨论】:

  • 卢克,你解决了这个问题吗?我对下面所有这些不同的想法有点困惑。感谢您的任何提示!
  • 是的,我使用下面的答案来获取模板并从那里进行修改。

标签: .net wpf contextmenu


【解决方案1】:

对于无法通过 Expression Interface 访问的模板和样式(例如 ContextMenu 模板),您可以使用以下代码提取模板:

Dim sb As System.Text.StringBuilder = New System.Text.StringBuilder
Using Writer As TextWriter = New StringWriter(sb)
    System.Windows.Markup.XamlWriter.Save(ContextMenu.Template, Writer)
End Using
Debug.Write(sb.ToString)

或者在 C# 中

var str = new StringBuilder();
using (var writer = new StringWriter(str))
    XamlWriter.Save(ContextMenu.Template, writer);
Debug.Write(str);

【讨论】:

  • 请注意,为了在 ContextMenu 中获取模板,您必须至少将一个 MenuItem 添加到其 Items 集合中。否则 ContextMenu.Template 为空。
  • 我喜欢 Stack Overflow。非常感谢这些回复,正是我所需要的。
  • 这将为我节省大量时间。多么棒的提示。
  • 我知道你在那里做了什么。不过,这很酷。如果可以的话,我会投票更多。
【解决方案2】:

我找到了在 Blend 中获取 ContextMenu 模板的简单方法:

  1. 我向带有一些菜单项的按钮添加了 ContextMenu。
  2. 在属性窗格的“杂项”下,有一个用于 ContextMenu 的分组项。
  3. 打开这个。您会发现常用的样式和模板属性。
  4. 单击弹出菜单的方块,然后选择转换为新资源...

就是这样。选择您想要放置模板/样式的位置,然后就完成了。

这是我的标记:

<StackPanel x:Name="LayoutRoot">
    <Button Content="Click for ContextMenu" Width="30" HorizontalAlignment="Center" VerticalAlignment="Center">
        <Button.ContextMenu>
            <ContextMenu Template="{DynamicResource ContextMenuControlTemplate1}" Style="{DynamicResource ContextMenuStyle1}">
                <MenuItem Header="File"/>
                <MenuItem Header="Edit"/>
                <MenuItem Header="View"/>
                <MenuItem Header="Recent Files"/>
                    <MenuItem Header="file1.txt"/>
                    <MenuItem Header="file2.txt"/>
            </ContextMenu>
        </Button.ContextMenu>
    </Button>
</StackPanel>

还有我得到的样式/模板:

<Style x:Key="ContextMenuStyle1" TargetType="{x:Type ContextMenu}">
    <Setter Property="Background" Value="{DynamicResource MenuBackgroundBrush}"/>
    <Setter Property="BorderThickness" Value="1"/>
    <Setter Property="BorderBrush" Value="{DynamicResource WindowBorderBrush}"/>
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type ContextMenu}">
                <Border Uid="Border_93">
                    <Border.Style>
                        <Style TargetType="{x:Type Border}">
                            <Setter Property="Tag" Value="{DynamicResource {x:Static SystemParameters.DropShadowKey}}"/>
                            <Style.Triggers>
                                <DataTrigger Binding="{Binding Tag, RelativeSource={RelativeSource Self}}" Value="True">
                                    <Setter Property="Background" Value="Transparent"/>
                                    <Setter Property="Padding" Value="0,0,5,5"/>
                                    <Setter Property="Effect">
                                        <Setter.Value>
                                            <DropShadowEffect BlurRadius="4" Opacity="0.8" ShadowDepth="1"/>
                                        </Setter.Value>
                                    </Setter>
                                </DataTrigger>
                            </Style.Triggers>
                        </Style>
                    </Border.Style>
                </Border>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

希望这会有所帮助。在通常的 MS 彻底性中,找不到默认样式的画笔。 :)

【讨论】:

  • 哈哈!:我找到了在 Blend 中获取 ContextMenu 模板的简单方法 - 这看起来确实是最简单的方法!
  • 似乎不是完整的 XAML。 'ContextMenuControlTemplate1' 的定义在哪里?
【解决方案3】:

试试这个:(将此代码放在 XAML 的资源部分中)这应该会从上下文菜单中删除图标条。

<Style TargetType="{x:Type ContextMenu}">
    <Setter Property="OverridesDefaultStyle" Value="True" />
    <Setter Property="SnapsToDevicePixels" Value="True" />
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type ContextMenu}">
                <Border BorderThickness="1" CornerRadius="4" BorderBrush="Black" x:Name="Border" Background="White">
                    <StackPanel ClipToBounds="True" Orientation="Vertical" IsItemsHost="True" />
                </Border>
                <ControlTemplate.Triggers>
                    <Trigger Property="IsMouseOver" Value="true">
                        <Setter TargetName="Border" Property="Background" Value="White" />
                    </Trigger>
                </ControlTemplate.Triggers>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

【讨论】:

    【解决方案4】:

    实际上空间不是 ContextMenu 的一部分,它是 MenuItem 的一部分。因此,只需将 MenuItem 拖到表达式混合中的窗口并创建控件的副本。希望您的 ContextMenu 声明如下

     <ContextMenu  >
        <MenuItem Header="Copy"/>
        <MenuItem Header="Paste"/>
        <MenuItem Header="Clear"/>
     </ContextMenu>
    

    在您的 MenuItem ControlTemplate 中,您可以看到如下所示的空间。所以删除我在屏幕截图中标记的网格的图标和第一列。

    【讨论】:

    • 嗯,这让我获得了 MenuItem 的默认样式,但它没有让我获得 ContextMenu 样式。 Expression Blend 不允许我向窗口添加 ContextMenu。
    • 如果我将 ContextMenu 添加到窗口,它会显示错误,“ContextMenu 不能有逻辑或视觉父级”。
    • 是的,左侧菜单的 ContextMenu 中还有一些样式需要删除。这是我无法使用 Expression 接口访问的 ContextMenu 样式。我在下面发布了一个使用代码提取模板的解决方案。
    【解决方案5】:

    左侧的多余空间是由于在MenuItem 上将IsCheckableIsChecked 设置为true 时出现的小复选标记。

    复选标记位于MenuItem 的模板中,因此如果您对其进行编辑,您可以将其取出。

    【讨论】:

    • IsCheckable 属性影响用户与菜单项的交互,而不是检查/图标区域是否显示在菜单中。事实上,默认值已经是False。将其设置为 True 只会在用户每次选择该菜单项时自动切换复选标记。
    猜你喜欢
    • 2015-09-05
    • 1970-01-01
    • 2022-01-04
    • 2015-02-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多