Использование TaskDialogIndirect в Python - PullRequest
1 голос
/ 01 марта 2020

Я хочу вызвать Windows TaskDialogIndirect функцию из Python. Требуется TaskDialogConfig (это довольно большая) структура для передачи в качестве указателя.

Вот мой готовый к запуску пример. Это дает мне «-2147024809» (параметр неверный), и я не могу понять, что не так.

Python 3.7.4 x32, Windows 7 x64

import ctypes
from ctypes.wintypes import *

TDF_ALLOW_DIALOG_CANCELLATION = 8
TDCBF_OK_BUTTON = 1

class TaskDialogConfig(ctypes.Structure):
    class DUMMYUNIONNAME(ctypes.Union):
        _fields_ = [
            ('hMainIcon', HICON)
            , ('pszMainIcon', LPCWSTR)
        ]

    class DUMMYUNIONNAME2(ctypes.Union):
        _fields_ = [
            ('hFooterIcon', HICON)
            , ('sFooterIcon', LPCWSTR)
        ]

    class _TASKDIALOG_BUTTON(ctypes.Structure):
        _fields_ = [
            ('nButtonID', INT)
            , ('pszButtonText', LPCWSTR)
        ]

    _fields_ = [
        ('cbSize', UINT)
        , ('hwndParent', HWND)
        , ('hInstance', HINSTANCE)
        , ('dwFlags', UINT)
        , ('dwCommonButtons', UINT)
        , ('pszWindowTitle', LPCWSTR)
        , ('DUMMYUNIONNAME', DUMMYUNIONNAME)
        , ('pszMainInstruction', LPCWSTR)
        , ('pszContent', LPCWSTR)
        , ('cButtons', UINT)
        , ('pButtons', _TASKDIALOG_BUTTON)
        , ('nDefaultButton', INT)
        , ('cRadioButtons', UINT)
        , ('pRadioButtons', _TASKDIALOG_BUTTON)
        , ('nDefaultRadioButton', INT)
        , ('pszVerificationText', LPCWSTR)
        , ('pszExpandedInformation', LPCWSTR)
        , ('pszExpandedControlText', LPCWSTR)
        , ('pszCollapsedControlText', LPCWSTR)
        , ('DUMMYUNIONNAME2', DUMMYUNIONNAME2)
        , ('pszFooter', LPCWSTR)
        , ('pfCallBack', ctypes.POINTER(None))
        , ('lpCallbackData', LPLONG)
        , ('cxWidth', UINT)
    ]

    def __init__(s):
        s.cbSize = ctypes.sizeof(s)

tdi = ctypes.WinDLL('comctl32.dll').TaskDialogIndirect
tdc = TaskDialogConfig()
tdc.hwndParent = None
tdc.hInstance = None
tdc.dwFlags = TDF_ALLOW_DIALOG_CANCELLATION
tdc.dwCommonButtons = TDCBF_OK_BUTTON
tdc.pszWindowTitle = ctypes.c_wchar_p('Title')
tdc.pszMainInstruction = ctypes.c_wchar_p('Main instruction')
tdc.pszContent = ctypes.c_wchar_p('Content')
print( tdi(ctypes.byref(tdc), None, None, None) )

1 Ответ

2 голосов
/ 01 марта 2020

Два выпуска.

  • В заголовке CommCtrl.h используется 1-байтовая упаковка. Добавьте _pack_ = 1 перед _fields_ определением во всех структурах.
  • Два поля _TASKDIALOG_BUTTON должны иметь тип ctypes.POINTER(_TASKDIALOG_BUTTON).

Я выследил их, используя C программа для печати размера структуры и смещений нескольких полей и печати той же информации в Python:

#include <windows.h>
#include <commctrl.h>
#include <stdio.h>
#include <stdlib.h>

int main()
{
    printf("%zu\n",sizeof(TASKDIALOGCONFIG));
    printf("%zu\n",offsetof(TASKDIALOGCONFIG,pszWindowTitle));
    printf("%zu\n",offsetof(TASKDIALOGCONFIG,pszMainInstruction));
    printf("%zu\n",offsetof(TASKDIALOGCONFIG,pszFooter));
}
tdc = TaskDialogConfig()
print(tdc.cbSize)
print(TaskDialogConfig.pszWindowTitle)
print(TaskDialogConfig.pszMainInstruction)
print(TaskDialogConfig.pszFooter)
...