Python:将文本编码为WPARAM中的win32api.SendMessage()

时间:2019-05-27 19:42:22

标签: python windows winapi pywin32 sendmessage

我正在尝试通过我的Python应用程序(即发送应用程序)调用win32api.SendMessage()

接收方应用程序的API指出消息的格式为:::SendMessage(<app_name>, <msg_name>, (WPARAM) <value>

但是,value实际上是一个3到4个字符(没有空格)的字符串。

我的问题

使用win32api.SendMessage的正确方法是什么,尤其是对于value而言?

我是否可以简单地将字符串放入,如: win32api.SendMessage(<app_name>, <msg_name>, "ABC")

还是我需要将字符串转换为WPARAM类型(如果是,我该怎么做)?

我一直在使用Linux Python开发,对Windows和C ++的经验很少。不胜感激。

谢谢!

P.s。为了回应评论,接收应用实际上是AmiBroker,API文档中给出的实际消息格式为: ::SendMessage( g_hAmiBrokerWnd, WM_USER_STREAMING_UPDATE, (WPARAM) Ticker, (LPARAM) &recentInfoStructureForGivenTicker ); 我之前提到的'string'是'Ticker',作者说的是string (char*)。最初我没有包括在内,因为我认为实际的消息格式并不重要。

研究:我从this得知WPARAM本质上是整数类型,而this导致我进入win32api。在我读过的许多文章中;他们都没有帮助回答我上面的问题(或者至少是我认为的问题)。

2 个答案:

答案 0 :(得分:1)

如果要发送自定义消息,则可以在WPARAM中发送数据。如果您要发送标准的窗口消息,则应将WPARAM设置为与所发送消息正确的值。

注意WPARAM是一个32位整数,因此如果您不能将字符串适合32位(AKA 4字节),则不能,您不能这样做。请注意,如果您要发送ASCII,这意味着您只能传递4个字符(每个字节一个)。我不了解python,但我想您可以将这4个字节移位,然后将它们加或运算为一个32位整数,以WPARAM的形式发送,也许是这样?

后面是伪代码

Int32 wparam = 0
wparam = wparam | ((Int32)chr[0] << (32 - (8 * 1)))
wparam = wparam | ((Int32)chr[1] << (32 - (8 * 2)))
wparam = wparam | ((Int32)chr[2] << (32 - (8 * 3)))
wparam = wparam | ((Int32)chr[3] << (32 - (8 * 4)))

请参阅Micorsoft网站上的SendMessage函数。

https://docs.microsoft.com/en-us/windows/desktop/api/winuser/nf-winuser-sendmessage

查看有关发送自定义消息的大量其他问题。

Sending a Custom windows message...custom data marshalling

答案 1 :(得分:1)

[Github]: mhammond/pywin32 - Python for Windows (pywin32) Extensions WINAPI 上的 Python 包装,因此被设计为 Python 友好

[ActiveState.Docs]: win32api.SendMessage(我能找到的最佳文档)是[MS.Docs]: SendMessage function的包装。

lParam (最后一个)参数是 LONG_PTR ,这意味着它拥有一个可以指向任何内容的内存地址。通常这是用于传递数据(例如字符串)的数据。

由于我不知道您要发送什么消息,因此花了一些时间直到找到[MS.Docs]: EM_REPLACESEL message

code0.py

#!/usr/bin/env python3

import sys
import win32api
import win32gui
import win32con


is_py2 = sys.version_info.major < 3

if is_py2:
    _input = input
    input = raw_input


def main():
    np_wnd = win32gui.FindWindow(None, "Untitled - Notepad")
    if not np_wnd:
        print("Cound not get Notepad window")
        return
    np_edit_wnd = win32gui.GetWindow(np_wnd, win32con.GW_CHILD)
    if not np_edit_wnd:
        print("Cound not get Notepad child window")
        return
    heading = "After pressing ENTER, "
    #'''
    input("{:s}all text in Notepad will be selected ... ".format(heading))
    # HERE's when the 1st screenshot was taken
    win32api.SendMessage(np_edit_wnd, win32con.EM_SETSEL, 0, -1)
    replaced_text0 = "Replaced\nmultiline text."
    input("{:s}Notepad text will be set (via EM_REPLACESEL) to: \n\"\"\"\n{:s}\n\"\"\" ... ".format(heading, replaced_text0))
    win32api.SendMessage(np_edit_wnd, win32con.EM_REPLACESEL, 0, replaced_text0)  # Regular string
    # HERE's when the 2nd screenshot was taken. It was at the end of the program (at that time), but some stuff was added
    replaced_text1 = "Other\nreplaced\n\nnmultiline text."
    input("\n{:s}Notepad text will be set (via WM_SETTEXT) to: \n\"\"\"\n{:s}\n\"\"\" ... ".format(heading, replaced_text1))
    win32api.SendMessage(np_edit_wnd, win32con.WM_SETTEXT, 0, replaced_text1)
    if not is_py2:
        return
    #'''
    print("\nFor Python 2, also get the text back from Notepad")
    buf_size = 255
    buf = win32gui.PyMakeBuffer(buf_size)
    text_len = win32api.SendMessage(np_edit_wnd, win32con.WM_GETTEXT, buf_size, buf)
    print("    Original text length: {:d}\n    Retrieved text length: {:d}\n    Text: \"\"\"\n{:s}\n    \"\"\"".format(len(replaced_text1), text_len, buf[:text_len]))


if __name__ == "__main__":
    print("Python {:s} on {:s}\n".format(sys.version, sys.platform))
    main()
    print("\nDone.")

结果

  • 初始状态:

    Img0

  • 最终状态:

    Img1

输出

[cfati@CFATI-5510-0:e:\Work\Dev\StackOverflow\q056331657]> "e:\Work\Dev\VEnvs\py_064_02.07.15_test0\Scripts\python.exe" code0.py
Python 2.7.15 (v2.7.15:ca079a3ea3, Apr 30 2018, 16:30:26) [MSC v.1500 64 bit (AMD64)] on win32

After pressing ENTER, all text in Notepad will be selected ...
After pressing ENTER, Notepad text will be set (via EM_REPLACESEL) to:
"""
Replaced
multiline text.
""" ...

After pressing ENTER, Notepad text will be set (via WM_SETTEXT) to:
"""
Other
replaced

nmultiline text.
""" ...

For Python 2, also get the text from Notepad
    Original text length: 32
    Retrieved text length: 32
    Text: """
Other
replaced

nmultiline text.
    """

Done.

如图所示,它可以与普通的 Python 字符串一起使用。

注意:我的 Win 用户具有“超级”管理权限。对于普通用户,某些事情可能无法按预期进行。

您还可以查看[SO]: Keyboard event not sent to window with pywin32 (@CristiFati's answer),以处理类似消息的 WM_CHAR ,并且更重要的是:如何处理子窗口

@ EDIT0

添加:

  • WM_SETTEXT
  • WM_GETTEXT (仅适用于 Python 2 )-展示如何从 SendMessage
  • 取回字符串

但是,由于 WM_USER_STREAMING_UPDATE 超出了 WM_USER btw ,我没有看到任何文档),因此某些事情可能/将不起作用(根据@IInspectable的注释以及 SendMessage 的文档),因此需要进行其他工作(数据封送处理)。

@ EDIT1

我已经注意到您正在尝试使用 AmiBroker (由 Google 使用 WM_USER_STREAMING_UPDATE )。
但是,我找不到该消息的任何(官方)文档,该文档将揭示 WPARAM LPARAM 参数应该包含的内容(例如: [MS.Docs]: WM_SETTEXT message)。
您是要编写插件(意味着您与 AmiBroker 处于同一进程中),还是只是想向其发送消息(就像我在示例中所做的那样: Python -> 记事本)?