CefSharp For WPF自定义右键菜单栏

简介: 原文:CefSharp For WPF自定义右键菜单栏初始化 public MainWindow() { InitializeComponent(); //右键菜单栏 MenuHandler.
原文: CefSharp For WPF自定义右键菜单栏

初始化

<!--浏览器-->
<cefSharpWPF:ChromiumWebBrowser Name="webBrowser" 
                                Grid.Row="0" >
</cefSharpWPF:ChromiumWebBrowser>
public MainWindow()
{
    InitializeComponent();

    //右键菜单栏
    MenuHandler.mainWindow = this;
    webBrowser.MenuHandler = new MenuHandler();
}

MenuHandler.cs

public class MenuHandler : IContextMenuHandler
{
    public static Window mainWindow { get; set; }
    void IContextMenuHandler.OnBeforeContextMenu(IWebBrowser browserControl, IBrowser browser, IFrame frame, IContextMenuParams parameters, IMenuModel model)
    {

    }

    bool IContextMenuHandler.OnContextMenuCommand(IWebBrowser browserControl, IBrowser browser, IFrame frame, IContextMenuParams parameters, CefMenuCommand commandId, CefEventFlags eventFlags)
    {
        return true;
    }

    void IContextMenuHandler.OnContextMenuDismissed(IWebBrowser browserControl, IBrowser browser, IFrame frame)
    {
        //隐藏菜单栏
        var chromiumWebBrowser = (ChromiumWebBrowser)browserControl;

        chromiumWebBrowser.Dispatcher.Invoke(() =>
        {
            chromiumWebBrowser.ContextMenu = null;
        });
    }

    bool IContextMenuHandler.RunContextMenu(IWebBrowser browserControl, IBrowser browser, IFrame frame, IContextMenuParams parameters, IMenuModel model, IRunContextMenuCallback callback)
    {

        //绘制了一遍菜单栏  所以初始化的时候不必绘制菜单栏,再此处绘制即可
        var chromiumWebBrowser = (ChromiumWebBrowser)browserControl;

        chromiumWebBrowser.Dispatcher.Invoke(() =>
        {
            var menu = new ContextMenu
            {
                IsOpen = true
            };

            RoutedEventHandler handler = null;

            handler = (s, e) =>
            {
                menu.Closed -= handler;

                //If the callback has been disposed then it's already been executed
                //so don't call Cancel
                if (!callback.IsDisposed)
                {
                    callback.Cancel();
                }
            };

            menu.Closed += handler;

            menu.Items.Add(new MenuItem
            {
                Header = "最小化",
                Command = new CustomCommand(MinWindow)
            });
            menu.Items.Add(new MenuItem
            {
                Header = "关闭",
                Command = new CustomCommand(CloseWindow)
            });
            chromiumWebBrowser.ContextMenu = menu;

        });

        return true;
    }

    /// <summary>
    /// 关闭窗体
    /// </summary>
    private void CloseWindow()
    {
        //调用线程无法访问此对象,因为另一个线程拥有该对象
        //handler和window是两个线程,WPF做了线程安全。。。so以下
        mainWindow.Dispatcher.Invoke(
            new Action(
                    delegate
                    {
                        Application.Current.Shutdown();
                    }
                ));
    }

    /// <summary>
    /// 最小化窗体
    /// </summary>
    private void MinWindow()
    {
        mainWindow.Dispatcher.Invoke(
            new Action(
                    delegate
                    {
                        mainWindow.WindowState = WindowState.Minimized;
                    }
                ));
    }

    private static IEnumerable<Tuple<string, CefMenuCommand>> GetMenuItems(IMenuModel model)
    {
        var list = new List<Tuple<string, CefMenuCommand>>();
        for (var i = 0; i < model.Count; i++)
        {
            var header = model.GetLabelAt(i);
            var commandId = model.GetCommandIdAt(i);
            list.Add(new Tuple<string, CefMenuCommand>(header, commandId));
        }

        return list;
    }
}

CustomCommand.cs

public class CustomCommand : ICommand
{
    Action _TargetExecuteMethod;
    Func<bool> _TargetCanExecuteMethod;
    public event EventHandler CanExecuteChanged = delegate { };

    public CustomCommand(Action executeMethod)
    {
        _TargetExecuteMethod = executeMethod;
    }

    bool ICommand.CanExecute(object parameter)
    {
        if (_TargetCanExecuteMethod != null)
        {
            return _TargetCanExecuteMethod();
        }
        if (_TargetExecuteMethod != null)
        {
            return true;
        }
        return false;
    }

    public void RaiseCanExecuteChanged()
    {
        CanExecuteChanged(this, EventArgs.Empty);
    }

    void ICommand.Execute(object parameter)
    {
        _TargetExecuteMethod?.Invoke();
    }
}

参考资料

https://github.com/cefsharp/CefSharp/issues/1795

目录
相关文章
|
C# 虚拟化 索引
【WPF】UI虚拟化之------自定义VirtualizingWrapPanel
原文:【WPF】UI虚拟化之------自定义VirtualizingWrapPanel 前言 前几天QA报了一个关于OOM的bug,在排查的过程中发现,ListBox控件中被塞入了过多的Item,而ListBox又定义了两种样式的ItemsPanelTemplate。
2212 0
|
3月前
|
开发框架 缓存 前端开发
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(11) -- 下拉列表的数据绑定以及自定义系统字典列表控件
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(11) -- 下拉列表的数据绑定以及自定义系统字典列表控件
|
C# 数据安全/隐私保护
【WPF】右下角弹出自定义通知样式(Notification)——简单教程
原文:【WPF】右下角弹出自定义通知样式(Notification)——简单教程 1.先看效果 2.实现 1.主界面是MainWindow 上面就只摆放一个Button即可。
3059 1
|
3月前
|
开发框架 前端开发 JavaScript
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(3)--自定义用户控件
循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(3)--自定义用户控件
|
3月前
|
C#
WPF 自定义可拖动标题栏
WPF 自定义可拖动标题栏
51 0
|
3月前
|
开发框架 前端开发 C#
使用WPF开发自定义用户控件,以及实现相关自定义事件的处理
使用WPF开发自定义用户控件,以及实现相关自定义事件的处理
|
前端开发 C# 图形学
【WPF】WPF开发用户控件、用户控件属性依赖DependencyProperty实现双向绑定、以及自定义实现Command双向绑定功能演示
Wpf开发过程中,最经常使用的功能之一,就是用户控件(UserControl)了。用户控件可以用于开发用户自己的控件进行使用,甚至可以用于打造一套属于自己的UI框架。依赖属性(DependencyProperty)是为用户控件提供可支持双向绑定的必备技巧之一,同样用处也非常广泛。
949 0
【WPF】WPF开发用户控件、用户控件属性依赖DependencyProperty实现双向绑定、以及自定义实现Command双向绑定功能演示
|
C#
WPF 4 DataGrid 控件(自定义样式篇)
原文:WPF 4 DataGrid 控件(自定义样式篇)      在《WPF 4 DataGrid 控件(基本功能篇)》中我们已经学习了DataGrid 的基本功能及使用方法。本篇将继续介绍自定义DataGrid 样式的相关内容,其中将涉及到ColumnHeader、RowHeader、Row、Cell 的各种样式设置。
2683 0
|
C# C++ 数据可视化
WPF Calendar 日历控件 样式自定义
原文:WPF Calendar 日历控件 样式自定义 粗略的在代码上做了些注释 blend 生成出来的模版 有的时候 会生成 跟 vs ui界面不兼容的代码 会导致可视化设计界面 报错崩溃掉 但是确不影响 程序的编译运行 这个样式表 在vs 里会提示动画不兼容 Foreground属性 报错 ...
1778 1
|
C#
WPF 控件自定义背景
<!--控件要设置尺寸的话,设置的尺寸必须比下面的图形的尺寸要小,不然显示不开--> <Label Content="直角测试" Width="90" Height="90" HorizontalContentAlignment="Center" Vert...
1018 0