WPF 默认 TextBox ContextMenu 样式

标签 wpf xaml

总结:我的全局“ContextMenu”样式没有应用到文本框和其他控件上的默认上下文菜单。

细节:我的应用程序中有一些没有显式上下文菜单的文本框。因此,当您右键单击它们时,您会看到剪切、复制和粘贴的标准上下文菜单选项。但是,该上下文菜单最终并未使用我在 Generic.xaml 中设置的全局“ContextMenu”样式。 TextBox 上的默认上下文菜单实际上不是 ContextMenu 吗?

如果我显式设置文本框的上下文菜单,则菜单使用我的全局上下文菜单样式。例如,这很好用:

<Style TargetType="{x:Type TextBox}">
    <Setter Property="ContextMenu" Value="{StaticResource StandardContextMenu}"/>
</Style>

<ContextMenu x:Key="StandardContextMenu">
    <MenuItem Header="Cut" Command="ApplicationCommands.Cut"/>
    <MenuItem Header="Copy" Command="ApplicationCommands.Copy"/>
    <MenuItem Header="Paste" Command="ApplicationCommands.Paste"/>
</ContextMenu>

但我真的不想为了让 WPF 应用正确的样式而创建这个完全冗余的 ContextMenu。此外,除了 TextBox 之外,还有其他控件在单击时显示 ContextMenus,并且这些控件也不会采用全局 ContextMenu 样式。

那么,当我右键单击没有明确定义 ContectMenu 的文本框时,实际显示的是什么?那东西不是上下文菜单吗?为什么不使用全局 ContextMenu 样式?

编辑: 使用 Snoop 进一步研究,我发现当我显式添加 ContextMenu 时,它在可视化树中显示为 ContextMenu。但是显示的默认 ContextMenu 在可视化树中显示为 EditorContextMenu。那么,下一个问题是如何全局设置 EditorContextMenu 的样式。

Context Menus

最佳答案

如您所知,我们不能直接在 ResourceDictionary xaml 中针对内部类或私有(private)类重新定义任何样式,但我们可以从代码后面进行。

因此,我们只需要通过反射找到类型,并使用 BasedOn 我们默认的 ContextMenu 和 MenuItem 样式创建新样式。

private void Initialize()
{
  var presentationFrameworkAssembly = typeof(Application).Assembly;
  var contextMenuStyle = FindResource(typeof(ContextMenu)) as Style;
  var editorContextMenuType = Type.GetType("System.Windows.Documents.TextEditorContextMenu+EditorContextMenu, " + presentationFrameworkAssembly);

  if (editorContextMenuType != null)
  {
    var editorContextMenuStyle = new Style(editorContextMenuType, contextMenuStyle);
    Application.Current.Resources.Add(editorContextMenuType, editorContextMenuStyle);
  }

  var menuItemStyle = Application.Current.FindResource(typeof(MenuItem)) as Style;
  var editorMenuItemType = Type.GetType("System.Windows.Documents.TextEditorContextMenu+EditorMenuItem, " + presentationFrameworkAssembly);

  if (editorMenuItemType != null)
  {
    var editorContextMenuStyle = new Style(editorMenuItemType, menuItemStyle);
    Application.Current.Resources.Add(editorMenuItemType, editorContextMenuStyle);
  }
}

此外,我们可以生成自定义 ResourceDictionary 以重新定义默认隐藏样式和虚拟 DefaultHiddenStyle.xaml 以允许将其包含在 MergedDictionaries 中。

<local:DefaultHiddenStyleResourceDictionary
  xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  xmlns:local="clr-namespace:Styles">
  
  <!-- No entries are required -->
    
</local:DefaultHiddenStyleResourceDictionary>
namespace Styles
{
  public class DefaultHiddenStyleResourceDictionary : ResourceDictionary
  {
    public DefaultHiddenStyleResourceDictionary()
    {
      // Run OnResourceDictionaryLoaded asynchronously to ensure other ResourceDictionary are already loaded before adding new entries
      Dispatcher.CurrentDispatcher.BeginInvoke(DispatcherPriority.Normal, new Action(OnResourceDictionaryLoaded));
    }

    private void OnResourceDictionaryLoaded()
    {
      var presentationFrameworkAssembly = typeof(Application).Assembly;

      AddEditorContextMenuDefaultStyle(presentationFrameworkAssembly);
      AddEditorMenuItemDefaultStyle(presentationFrameworkAssembly);
    }

    private void AddEditorContextMenuDefaultStyle()
    {
      var presentationFrameworkAssembly = typeof(Application).Assembly;
      var contextMenuStyle = Application.Current.FindResource(typeof(ContextMenu)) as Style;
      var editorContextMenuType = Type.GetType("System.Windows.Documents.TextEditorContextMenu+EditorContextMenu, " + presentationFrameworkAssembly);

      if (editorContextMenuType != null)
      {
        var editorContextMenuStyle = new Style(editorContextMenuType, contextMenuStyle);
        Add(editorContextMenuType, editorContextMenuStyle);
      }
    }
    
    private void AddEditorMenuItemDefaultStyle(Assembly presentationFrameworkAssembly)
    {
      var menuItemStyle = Application.Current.FindResource(typeof(MenuItem)) as Style;
      var editorMenuItemType = Type.GetType("System.Windows.Documents.TextEditorContextMenu+EditorMenuItem, " + presentationFrameworkAssembly);

      if (editorMenuItemType != null)
      {
        var editorContextMenuStyle = new Style(editorMenuItemType, menuItemStyle);
        Add(editorMenuItemType, editorContextMenuStyle);
      }
    }
  }
}

关于WPF 默认 TextBox ContextMenu 样式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30940939/

相关文章:

c# - WinUI 3 xaml ABI.Microsoft.UI 与 Microsoft.UI

WPF:按可用宽度而不是按其内容扩展 TextBox?

c# - WPF 绑定(bind)到数组/集合

wpf - WPF中的自定义形状边框

c# - 从 Silverlight 中的文件夹加载资源 ".resx"

.net - 将描边应用于 WPF 中的文本 block

c# - UWP DataTemplate 中的列未拉伸(stretch)

wpf - 如何根据条件隐藏wpf xaml中上下文菜单的菜单项

wpf - 混合 MFC 和 WPF : Modal Dialogs

c# - 图片边框控制换行问题