jjzjj

.net - 防止 WebBrowser 控件窃取焦点?

coder 2024-05-12 原文

有没有办法阻止 WebBrowser 控件导致其父窗体将其自身置于最前面?

如果您使用 InvokeScript 方法调用在主父文档中的 iframe 上调用 focus() 的 JavaScript 函数,它将导致窗口直接将其自身带到最前面(或至少导致任务栏图标开始闪烁).有什么办法可以防止这种情况发生吗?

更新:

我找到了我的问题的临时答案。

当 WebBrowser 的父窗体的 Deactive 事件被触发时,我从它的容器中删除 WebBrowser,并在它的旧父窗体再次被激活时重新添加它。

这有点 hacky,但它确实有效。不过,我愿意接受任何更好的建议。

最佳答案

编辑:重写了完整的问题,我误解了原来的问题

让我们概括一下这个问题:您无法控制的控件或组件可以调用 FlashWindow(Win32 API 函数)来引起用户的注意。你不想要那个。

通常有两种解决方案:使用 API hooking 或 Message hooking。由于 API Hook 复杂且涉及面广,我将提供消息 Hook 的解决方案。

闪 window

Microsoft 并没有用太多的文字来解释 FlashWindow 的作用。不幸的是,它不会发送特定消息(比如 WM_FLASH 或类似消息),这样可以更轻松地捕获和取消此行为。相反,FlashWindow 做了三件事:

  1. 它为闪烁间隔设置一个系统计时器
  2. 它为第一个 flash 发送一个 WM_NCACTIVATE 消息
  3. 它在计时器到期时发送一个WM_NCACTIVATE消息(在接收到WM_SYSTIMER时)

根据组件调用 FlashWindow 的方式,这可能是不确定的,直到发生另一个超时,直到它获得焦点或只有一次。每个 WM_NCACTIVATE 消息都会激活或停用 NC 区域(标题栏、任务栏上的按钮)。它不会改变输入的焦点。

挑战

防止闪烁的任何解决方案都有点复杂。主要挑战是:

  1. WM_SYSTIMER 事件与 PostMessage 异步发送,不会被 Form 的 WndProc 方法接收(它只处理同步消息)
  2. WM_NCACTIVATE 消息也会在用户点击标题栏或任务栏按钮设置输入焦点时使用,简单地取消这些消息会产生不需要的副作用
  3. FlashWindow 将始终至少闪烁一次,无论 WM_SYSTIMER 是否触发。

WM_SYSTIMER 消息未记录。它的值为 0x0118,Windows 在内部使用它来计时插入符号的闪烁、菜单打开的延迟等。这里它用于闪烁之间的时间。

解决方案

我在此提供的解决方案是进一步开发的基础。它不是一个完整的解决方案,但在许多情况下可以解决问题。在您的表单代码中放置以下内容:

protected override void WndProc(ref Message m)
{
    bool messageHandled = false;
    if (m.Msg == WM_NCACTIVATE)
    {
        // add logic here to determine user action, losing focus etc and set 
        // messageHandled and m.Result only when user action is not the cause 
        // of triggering WM_NCACTIVATE
        m.Result = IntPtr.Zero;
        messageHandled = true;
    }

    if(!messageHandled)
        base.WndProc(ref m);
}

上面的代码已经完全防止了闪烁。您必须添加一些逻辑来更改标题栏,因为完全忽略 WM_NCACTIVATE 意味着标题栏将始终处于事件状态,即使它不是。

以下代码为您提供了更多控制权。您可以使用它来对闪烁本身使用react。通常,主窗口不会如此频繁地接收 WM_SYSTIMER 事件,但您必须试验是否应该设置异常(exception)。似乎对于 FlashWindowwParam 总是设置为 0xFFF8,但请尝试使用它,因为这在任何地方都没有记录。

public class MyMessageFilter : IMessageFilter
{
    // an application can have many windows, only filter for one window at the time
    IntPtr FilteredHwnd = IntPtr.Zero;

    public MyMessageFilter(IntPtr hwnd)
    {
        this.FilteredHwnd = hwnd;
    }

    public bool PreFilterMessage(ref Message m)
    {
        if (this.FilteredHwnd == m.HWnd && m.Msg == WM_SYSTIMER)
            return true;     // stop handling the message further
        else
            return false;    // all other msgs: handle them
    }
}

要激活此消息过滤器,只需在表单加载事件的某处添加以下行:

Application.AddMessageFilter(new MyMessageFilter(this.Handle));

以下常量应放在类级别。它们用于上面的两个代码部分:

public const UInt32 WM_SYSTIMER = 0x0118;
public const UInt32 WM_NCACTIVATE = 0x86;

结论

虽然问题本身是可以解决的,但远非易事。使用上述 handle ,您应该走得很远。使用过滤器来防止闪烁,但第一次“闪烁”仍然会发生。使用 WinProc 重写来防止第一个,但添加一些逻辑来防止您的应用程序表现得太奇怪(即:标题栏始终处于非事件状态,或始终处于事件状态)。您已经有了一些代码,您可以将这些代码与此结合起来以设置一些 bool 标志。

关于.net - 防止 WebBrowser 控件窃取焦点?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1562619/

有关.net - 防止 WebBrowser 控件窃取焦点?的更多相关文章

  1. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  2. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  3. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  4. Get https://registry-1.docker.io/v2/: net/http: request canceled while waiting - 2

    1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里

  5. .net - .NET 将如何影响 Python 和 Ruby 应用程序? - 2

    我很好奇.NET将如何影响Python和Ruby应用程序。用IronPython/IronRuby编写的应用程序是否会非常特定于.NET环境,以至于它们实际上将变得特定于平台?如果他们不使用任何.NET功能,那么IronPython/IronRuby相对于非.NET同类产品的优势是什么? 最佳答案 我不能说任何关于IronRuby的东西,但是大多数Python实现(如IronPython、Jython和PyPy)都试图尽可能忠实于CPython实现。不过,IronPython正在迅速成为这方面的佼佼者之一,并且在PlanetPyth

  6. ruby-on-rails - Rails 3.2 防止使用错误保存对象 - 2

    我有一个ActiveRecord对象,我想在不对模型进行永久验证的情况下阻止它被保存。您过去可以使用errors.add执行类似的操作,但它看起来不再有效了。user=User.lastuser.errors.add:name,"namedoesn'trhymewithorange"user.valid?#=>trueuser.save#=>true或user=User.lastuser.errors.add:base,"myuniqueerror"user.valid?#=>trueuser.save#=>true如何在不修改用户对象模型的情况下防止将用户对象保存在Rails3.2中

  7. ruby - 如何使用 Ruby HTTP::Net 处理 404 错误? - 2

    我正在尝试解析网页,但有时会收到404错误。这是我用来获取网页的代码:result=Net::HTTP::getURI.parse(URI.escape(url))如何测试result是否为404错误代码? 最佳答案 像这样重写你的代码:uri=URI.parse(url)result=Net::HTTP.start(uri.host,uri.port){|http|http.get(uri.path)}putsresult.codeputsresult.body这将打印状态码和正文。

  8. .net - 是否有 Ruby .NET 编译器? - 2

    是否有适用于Ruby语言的.NETFramework编译器?我听说过DLR(动态语言运行时),这是否将使Ruby能够用于.NET开发? 最佳答案 IronRuby是Microsoft支持的项目,建立在动态语言运行时之上。 关于.net-是否有Ruby.NET编译器?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/199638/

  9. ruby - 使用 ruby​​ gem net-ssh-multi 同时在多个服务器上执行 sudo 命令 - 2

    在previousquestion中我想出了如何在多个服务器上启动经过密码验证的sshsession来运行单个命令。现在我需要能够执行“sudo”命令。问题是,net-ssh-multi没有分配sudo需要运行的伪终端(pty),导致以下错误:[127.0.0.1:stderr]sudo:sorry,youmusthaveattytorunsudo根据documentation,可以通过调用channel对象的方法来分配伪终端,但是,以下代码不起作用:它会生成上面的“notty”错误:require'net/ssh'require'net/ssh/multi'Net::SSH::Mul

  10. ruby - 防止SQL注入(inject)/好的Ruby方法 - 2

    Ruby中防止SQL注入(inject)的好方法是什么? 最佳答案 直接使用ruby?使用准备好的语句:require'mysql'db=Mysql.new('localhost','user','password','database')statement=db.prepare"SELECT*FROMtableWHEREfield=?"statement.execute'value'statement.fetchstatement.close 关于ruby-防止SQL注入(inject

随机推荐