首页 > 解决方案 > 跟随 html 链接时,如何防止 CefSharp 浏览器打开 2 个窗口?

问题描述

我正在使用 CefSharp 浏览器来显示电子邮件的内容。我从已解析的电子邮件中提取 BodyHtmlText 并使用browser.LoadHtml(BodyHtmlText, "http:\\pixemail.com\");. 我希望 BodyHtmlText 中的链接在系统默认浏览器中打开,所以我实现了 BrowserRequestHandler 并通过设置告诉浏览器使用它browser.RequestHandler = BrowserRequestHandler;

这很好用,除非链接锚标记具有 <a...target="_blank">。当它出现时,单击该链接会打开两个新的浏览器窗口;一个具有预期的链接目标页面,另一个是空的。BrowserRequestHandler.OnBeforeBrowse()只调用一次,即使打开了 2 个窗口。

如何摆脱空窗?

如果我不安装 BrowserRequestHandler,带有 target="_blank" 的链接会在没有额外窗口的新浏览器中打开,但常规链接(没有 target="_blank")会在应用程序的 CefSharp 浏览器窗口中打开,我没有不想要。如果BrowserRequestHandler.OnBeforeBrowse()我可以检测到链接有 target="_blank",我可以让 CefBrowser 处理它并获得我想要的行为。还是有其他方法可以配置和使用 CefSharp 浏览器来获得我想要的行为?

CefSharp.MinimalExample.WinForms 的稍微修改的版本也出现了同样的问题。我的 BrowserRequestHandler 看起来像这样:

public class BrowserRequestHandler : RequestHandler {
    protected override bool OnBeforeBrowse(IWebBrowser browserControl, IBrowser browser, IFrame frame,
      IRequest request, bool userGesture, bool isRedirect) {
        // Open in external default browser (chrome)
        if (!request.Url.StartsWith("file:") && !request.Url.StartsWith("devtools:")
          && !request.Url.StartsWith("chrome-devtools:") && !request.Url.Contains("pixemail.com")) {
            Console.WriteLine($"start new browser at {request.Url}");
            System.Diagnostics.Process.Start(request.Url);
            return true;
        }
        return false;   // let Cef handle the browse request
    }
}

CefSharp.MinimalExample.WinForms.BrowserForm 的变化是:

    string htmlPageFromEmail;

    public BrowserForm() {
        InitializeComponent();
        htmlPageFromEmail = File.ReadAllText("test-email.html");
        //...
        browser = new ChromiumWebBrowser("http://pixemail.com/");   // a dummy web address
        //...
        BrowserRequestHandler = new BrowserRequestHandler();
        browser.IsBrowserInitializedChanged += OnIsBrowserInitializedChanged;
        //...
        browser.Visible = true;
    }

    private void OnIsBrowserInitializedChanged(object sender, EventArgs e) {
        var b = ((ChromiumWebBrowser) sender);
        this.InvokeOnUiThreadIfRequired(() => {
            b.LoadHtml(htmlPageFromEmail, "http://pixemail.com/");
            b.Focus();
        });
    }

我用来模拟电子邮件 BodyHtmlText 字符串的 test-email.html 文件是:

  <html xmlns="http://www.w3.org/1999/xhtml">
    <head></head>
    <body>
      <br><br>Bad link. Opens 2 windows:
      <a href="https://www.youtube.com/watch?v=_7Yw8qMDips&amp;fbclid=IwAR1rGIK9KJukB4_KV5J-E3hqgfFPFZG8scobzKw_aw3jcUnrQWafr4Iw8bc" rel="nofollow" target="_blank" class="enhancr_card_3679327690">Jonny Quest Opening Titles in Stop Motion</a>
      <br><br>Good link. Opens 1 window:
      <a href="https://mewe.com/group/5e35fe0f260e8853cadbe7b7">https://mewe.com/group/5e35fe0f260e8853cadbe7b7</a>
      <br><br>
    </body>
  </html>

我在具有 x64 平台目标的 Windows 10 PC 和 Visual Studio 2019 上使用版本 CefSharp/84。

标签: c#winformscefsharp

解决方案


从内存中 OnBeforePopup 在 OnBeforeBrowse 之前被调用,因此 CEF 在内部创建弹出窗口,然后您取消导航导致一个空窗口。在您的 OnBeforeBrowse 实现中,您应该发现 Browser.IsPopup 在链接具有目标属性的情况下为真。

简而言之,您取消导航为时已晚,弹出窗口已经创建。

有两个选项


推荐阅读