Как определить через UIAutomation, нажата кнопка или нет?
Я хочу узнать, нажата ли кнопка или нет. Это, кажется, не является официальным свойством кнопки (не флажок в стиле кнопки!), Но кажется доступным, например, есть сообщение BM_GETSTATE, которое должно получить желаемый результат.
Проблема в том, что я часто не получаю дескрипторы окон для своих кнопок (они являются лишь частью другой панели инструментов, хотя их можно различить с помощью элемента AutomationElement). И мне понадобится такой дескриптор для функции SendMessage.
Итак... есть ли у меня способ получить доступ к этой собственности? Я знаю, что это доступно, так как я видел это в других программах автоматизации, я просто не знаю, как это сделать.
Я собираюсь использовать C#, но любой код C будет в порядке.
Большое спасибо
1 ответ
(edit: так что мне наконец удалось сделать формат кода здесь правильно - просто вставьте 4 пробела в начале.)
Наслаждайтесь этим, мне потребовалось довольно много времени, чтобы заставить его работать... но теперь я чувствую, что достиг нового уровня.:)
(пожалуйста, скажите мне, как сделать так, чтобы он форматировался правильно - и цитата, и код не были выполнены)
int res;
#region direct method
int hwnd = ae.Current.NativeWindowHandle;
if (hwnd != 0)
{
const UInt32 BM_GETSTATE = 0x00F2;
res = SendMessage(hwnd, BM_GETSTATE, 0, 0);
}
#endregion
else
#region method via toolbar
{
AutomationElement parent = TreeWalker.RawViewWalker.GetParent(ae);
while ((parent != null) && (parent.Current.ControlType != ControlType.ToolBar))
parent = TreeWalker.RawViewWalker.GetParent(ae);
if (parent != null)
{
int toolBarHandle = parent.Current.NativeWindowHandle;
#region defines
const int WM_USER = 0x400;
const int TB_GETSTATE = (WM_USER + 18);
const int TB_GETBUTTON = (WM_USER + 23);
const int TB_BUTTONCOUNT = (WM_USER + 24);
#endregion
#region get correct child number
int numButtons = SendMessage(toolBarHandle, TB_BUTTONCOUNT, 0, 0);
AutomationElement sibling = ae;
int cnt = -1;
while (sibling != null)
{
sibling = TreeWalker.RawViewWalker.GetPreviousSibling(sibling);
++cnt;
}
if (cnt >= numButtons)
cnt = 0; // nonsense value, but pass a valid one
#endregion
#region get command id
TBBUTTON butInfo = new TBBUTTON();
butInfo.idCommand = 1234;
uint pid;
GetWindowThreadProcessId((IntPtr)toolBarHandle, out pid);
IntPtr process = OpenProcess(ProcessAccessFlags.VMOperation | ProcessAccessFlags.VMRead |
ProcessAccessFlags.VMWrite | ProcessAccessFlags.QueryInformation, false, pid);
IntPtr p = VirtualAllocEx(process, IntPtr.Zero, (uint)Marshal.SizeOf(typeof(TBBUTTON)), AllocationType.Commit
, MemoryProtection.ReadWrite);
int _res = SendMessage(toolBarHandle, TB_GETBUTTON, cnt, p.ToInt32());
#region getresult
int read;
IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(TBBUTTON)));
Marshal.StructureToPtr(butInfo, ptr, true);
bool __res = ReadProcessMemory(process, p, ptr, Marshal.SizeOf(typeof(TBBUTTON)), out read);
System.Diagnostics.Debug.Assert(read == Marshal.SizeOf(typeof(TBBUTTON)));
butInfo = (TBBUTTON)Marshal.PtrToStructure(ptr, typeof(TBBUTTON));
#endregion
int commandId = butInfo.idCommand;
VirtualFreeEx(process, p, 0, FreeType.Release);
#endregion
//!define BST_UNCHECKED 0
//!define BST_CHECKED 1
//!define BST_INDETERMINATE 2
//!define BST_PUSHED 4
//!define BST_FOCUS 8
#region get state
res = SendMessage(toolBarHandle, TB_GETSTATE, commandId, 0);
#endregion
}
}
#endregion
РЕДАКТИРОВАТЬ: Здесь http://www.andreas-reiff.de/2011/06/c-speicher-anderen-prozess-befullen-lassen-checken-ob-ein-button-gedruckt/ с читаемым кодом и объяснениями на странном, иностранный язык.. комментарии к коду являются английскими. Надеюсь, что вы найдете ее полезной. Кроме того, я бы не смог решить эту проблему без информации здесь. Почему некоторые элементы управления не имеют дескриптора окна?,