草庐IT

windows - 在 Windows 上,当我的子窗口的 z 顺序相对于其 sibling 之一发生变化时,它是否应该获得 WM_WINDOWPOSCHANGED?

coder 2024-06-12 原文

我正在尝试将自定义小部件插入到 Internet Explorer 8 网址栏中,位于停止和重新加载按钮旁边。这只是我个人生产力的提升。

IE 框架这部分的“窗口模型”是一个“地址栏根”窗口,它拥有包含 IE8 网址栏的窗口:一个编辑框、一个组合控件以及停止和重新加载按钮。

在另一个进程中,我创建了一个新的 WS_CHILD 窗口(具有自定义类名),它是 IE 的地址栏根窗口的父级,因此使其成为编辑框的同级窗口并停止/重新加载。我用 HWND_TOP 的 hwndInsertAfter 调用 SetWindowPos 以确保它出现在 urlbar 的“上方”(即“在”中)。这很好用,我看到我的窗口最初是在 IE urlbar 中绘制的。

但是,当我激活 IE 窗口时,urlbar 编辑控件跳回到我的窗口前面。我知道这种情况正在发生,因为我仍然看到我的窗口绘制在 urlbar 后面,并且因为当我在计时器上打印 ->GetTopWindow() 到调试控制台时,它变成了 urlbar 编辑控件的 HWND。

如果我更新我的消息循环以在 WM_PAINT 上调用带有 HWND_TOP 的 SetWindowPos,事情会更好——现在当我激活 IE 窗口并移动它时,我的控件正确地保持在编辑控件之上在网址栏中。但是,一旦我在 IE 选项卡之间切换,这会更新 IE 的 urlbar Edit 控件的文本,我的控件就会移回 Edit 控件后面。 (注意:当我最大化或恢复窗口时也会发生这种情况。)

所以我的问题是:

1) 每次您在 IE 中单击选项卡时,IE 是否有意将其 urlbar 编辑控件放回 z 顺序的顶部,或者我对 Windows 绘画和 z- 的理解是否存在差距订购作品?我的理解是,一旦您指定了 子窗口 的 z 顺序(最终用户无法操作),该顺序应保持不变,直到以编程方式更改为止。因此,即使 IE 在选项卡选择时重新绘制其编辑控件,而我没有重新绘制或以其他方式作用于我的窗口,我的窗口仍应牢牢保持在顶部。

2) 鉴于我的窗口的 z 顺序显然正在改变,它不应该收到 WM_WINDOWPOSCHANGING/WM_WINDOWPOSCHANGED 吗?如果是这样,我至少可以响应该事件并让自己处于编辑控件的顶部。但是,即使当我单击选项卡时我可以看到我的窗口在 urlbar Edit 控件后面绘制,即使我的调试窗口输出确认地址栏根目录的 GetTopWindow() 成为当我单击选项卡时 Edit 控件的 HWND ,即使我在单击选项卡时看到 WM_WINDOWPOSCHANGING/WM_WINDOWPOSCHANGED 被发送到带有 HWND_TOP 的 hwndInsertAfter 的编辑控件,我自己的窗口也没有收到任何消息,这将使我保持 z 顺序不变。这对我来说似乎是错误的,解决它会迫使我在 IE 的进程中运行并 Hook 所有发送到它的编辑控件的消息,只是为了有一个事件来响应:(

感谢您的帮助!

最佳答案

  1. 当您更改选项卡时,IE 很可能会调整控件的 Z 顺序。在 IE9 中,URL 栏和选项卡有一个共同的父级。当您选择一个新选项卡时,它会激活 URL 栏(并且激活通常会将窗口带到其本地 Z 顺序的顶部)。

  2. 没有。当 SetWindowPos 函数作用于您的窗口时,您会得到 WM_WINDOWPOSCHANGED。如果某些 sibling 的 z 顺序发生变化,您不会收到消息。没有人在您的窗口上调用 SetWindowPos。您可以通过编写一个调整某些子窗口的 z 顺序的测试程序来了解这一点。

这是有道理的,因为可能有任意数量的同级窗口,并且通知所有这些窗口的开销可能是无限的。考虑到一些 sibling 可能会通过进一步打乱 z 顺序来使用react,因此几乎不可能提出一套一致的规则来将这些消息传递给所有 sibling 。还没有收到第一个通知的 sibling 现在有两个待处理的通知吗?他们会立即发布或发送吗?如果队列越来越大直到溢出怎么办?

这与 WM_KILLFOCUS/WM_SETFOCUS 通知的不同之处在于它最多影响两个窗口。这对通知的数量进行了合理的限制。即使因为失去控制而试图窃取焦点而出现失控的无限循环,队列也不会溢出,因为对于每个 WM_KILLFOCUS 传递只有一个 SetFocus 调用。

此外,Windows 可能需要对失去焦点使用react,这是合理的。窗口 C 不太可能需要知道 B 现在位于 A 之上,而不是相反,那么为什么要设计系统来发送大量不必要的消息?

破解不受您控制且没有明确定义的 API 来执行您想要执行的操作的应用程序的用户界面在任何地方都是困难甚至不可能的,而且它总是很脆弱。提供工具栏和浏览器自定义功能的小组雇用的人数可能比您预期的要多,而且他们一天中的大部分时间都在使用 Spy++ 进行探索和试验。这是天生的黑客行为。

关于windows - 在 Windows 上,当我的子窗口的 z 顺序相对于其 sibling 之一发生变化时,它是否应该获得 WM_WINDOWPOSCHANGED?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6348743/

有关windows - 在 Windows 上,当我的子窗口的 z 顺序相对于其 sibling 之一发生变化时,它是否应该获得 WM_WINDOWPOSCHANGED?的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - Chef 执行非顺序配方 - 2

    我遵循了教程http://gettingstartedwithchef.com/,第1章。我的运行list是"run_list":["recipe[apt]","recipe[phpap]"]我的phpapRecipe默认Recipeinclude_recipe"apache2"include_recipe"build-essential"include_recipe"openssl"include_recipe"mysql::client"include_recipe"mysql::server"include_recipe"php"include_recipe"php::modul

  3. ruby - 无法在 60 秒内获得稳定的 Firefox 连接 (127.0.0.1 :7055) - 2

    我使用的是Firefox版本36.0.1和Selenium-Webdrivergem版本2.45.0。我能够创建Firefox实例,但无法使用脚本继续进行进一步的操作无法在60秒内获得稳定的Firefox连接(127.0.0.1:7055)错误。有人能帮帮我吗? 最佳答案 我遇到了同样的问题。降级到firefoxv33后一切正常。您可以找到旧版本here 关于ruby-无法在60秒内获得稳定的Firefox连接(127.0.0.1:7055),我们在StackOverflow上找到一个类

  4. ruby - 在 Windows 机器上使用 Ruby 进行开发是否会适得其反? - 2

    这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby​​-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub

  5. Vscode+Cmake配置并运行opencv环境(Windows和Ubuntu大同小异) - 2

    之前在培训新生的时候,windows环境下配置opencv环境一直教的都是网上主流的vsstudio配置属性表,但是这个似乎对新生来说难度略高(虽然个人觉得完全是他们自己的问题),加之暑假之后对cmake实在是爱不释手,且这样配置确实十分简单(其实都不需要配置),故斗胆妄言vscode下配置CV之法。其实极为简单,图比较多所以很长。如果你看此文还配不好,你应该思考一下是不是自己的问题。闲话少说,直接开始。0.CMkae简介有的人到大二了都不知道cmake是什么,我不说是谁。CMake是一个开源免费并且跨平台的构建工具,可以用简单的语句来描述所有平台的编译过程。它能够根据当前所在平台输出对应的m

  6. 深度学习部署:Windows安装pycocotools报错解决方法 - 2

    深度学习部署:Windows安装pycocotools报错解决方法1.pycocotools库的简介2.pycocotools安装的坑3.解决办法更多Ai资讯:公主号AiCharm本系列是作者在跑一些深度学习实例时,遇到的各种各样的问题及解决办法,希望能够帮助到大家。ERROR:Commanderroredoutwithexitstatus1:'D:\Anaconda3\python.exe'-u-c'importsys,setuptools,tokenize;sys.argv[0]='"'"'C:\\Users\\46653\\AppData\\Local\\Temp\\pip-instal

  7. ruby - 为什么当我调用类的实例方法时,初始化不显示为方法? - 2

    我正在写一篇关于在Ruby中几乎一切都是对象的博客文章,我试图通过以下示例来展示这一点:classCoolBeansattr_accessor:beansdefinitialize@bean=[]enddefcount_beans@beans.countendend所以从类中我们可以看出它有4个方法(当然,除非我错了):它可以在创建新实例时初始化一个默认的空bean数组它可以计算它有多少个bean它可以读取它有多少个bean(通过attr_accessor)它可以向空数组写入(或添加)更多bean(也通过attr_accessor)但是,当我询问类本身它有哪些实例方法时,我没有看到默认

  8. ruby-on-rails - 在 RSpec 中,如何以任意顺序期望具有不同参数的多条消息? - 2

    RSpec似乎按顺序匹配方法接收的消息。我不确定如何使以下代码工作:allow(a).toreceive(:f)expect(a).toreceive(:f).with(2)a.f(1)a.f(2)a.f(3)我问的原因是a.f的一些调用是由我的代码的上层控制的,所以我不能对这些方法调用添加期望。 最佳答案 RSpecspy是测试这种情况的一种方式。要监视一个方法,用allowstub,除了方法名称之外没有任何约束,调用该方法,然后expect确切的方法调用。例如:allow(a).toreceive(:f)a.f(2)a.f(1)

  9. ruby - 按数字(从大到大)然后按字母(字母顺序)对对象集合进行排序 - 2

    我正在构建一个小部件来显示奥运会的奖牌数。我有一个“国家”对象的集合,其中每个对象都有一个“名称”属性,以及奖牌计数的“金”、“银”、“铜”。列表应该排序:1.首先是奖牌总数2.如果奖牌相同,按类型分割(金>银>铜,即2金>1金+1银)3.如果奖牌和类型相同,则按字母顺序子排序我正在用ruby​​做这件事,但我想语言并不重要。我确实找到了一个解决方案,但如果感觉必须有更优雅的方法来实现它。这是我做的:使用加权奖牌总数创建一个虚拟属性。因此,如果他们有2个金牌和1个银牌,加权总数将为“3.020100”。1金1银1铜为“3.010101”由于我们希望将奖牌数排序为最高的,因此列表按降序排

  10. ruby - 如何在 Ruby 中执行 Windows CLI 命令? - 2

    我在目录“C:\DocumentsandSettings\test.exe”中有一个文件,但是当我用单引号编写命令时`C:\DocumentsandSettings\test.exe(我无法在此框中显示),用于在Ruby中执行命令,我无法这样做,我收到的错误是找不到文件或目录。我尝试用“//”和“\”替换“\”,但似乎没有任何效果。我也使用过系统、IO.popen和exec命令,但所有的努力都是徒劳的。exec命令还使程序退出,这是我不想发生的。提前致谢。 最佳答案 反引号环境就像双引号,所以反斜杠用于转义。此外,Ruby会将空格解

随机推荐