1. 程式人生 > >【轉】python win32api win32gui win32con 簡單操作教程(視窗控制代碼 傳送訊息 常用方法 鍵盤輸入)

【轉】python win32api win32gui win32con 簡單操作教程(視窗控制代碼 傳送訊息 常用方法 鍵盤輸入)

import win32gui
import win32con
import win32api

# 從頂層視窗向下搜尋主視窗,無法搜尋子視窗
# FindWindow(lpClassName=None, lpWindowName=None)  視窗類名 視窗標題名
handle = win32gui.FindWindow("Notepad", None) 


# 獲取視窗位置
left, top, right, bottom = win32gui.GetWindowRect(handle)
#獲取某個控制代碼的類名和標題
title = win32gui.GetWindowText(handle)     
clsname = win32gui.GetClassName(handle)

# 列印控制代碼
# 十進位制
print(handle)
# 十六進位制
print("%x" %(handle) )


# 搜尋子視窗
# 列舉子視窗
hwndChildList = []     
win32gui.EnumChildWindows(handle, lambda hwnd, param: param.append(hwnd),  hwndChildList)

# FindWindowEx(hwndParent=0, hwndChildAfter=0, lpszClass=None, lpszWindow=None) 父視窗控制代碼 若不為0,則按照z-index的順序從hwndChildAfter向後開始搜尋子窗體,否則從第一個子窗體開始搜尋。 子視窗類名 子視窗標題
subHandle = win32gui.FindWindowEx(handle, 0, "EDIT", None)

# 獲得視窗的選單控制代碼
menuHandle = win32gui.GetMenu(subHandle)
# 獲得子選單或下拉選單控制代碼   
# 引數:選單控制代碼 子選單索引號
subMenuHandle = win32gui.GetSubMenu(menuHandle, 0)
# 獲得選單項中的的標誌符,注意,分隔符是被編入索引的  
# 引數:子選單控制代碼 專案索引號 
menuItemHandle = win32gui.GetMenuItemID(subMenuHandle, 0)
# 傳送訊息,加入訊息佇列,無返回 
# 引數:控制代碼 訊息型別 WParam IParam
win32gui.postMessage(subHandle, win32con.WM_COMMAND, menuItemHandle, 0)


# wParam的定義是32位整型,high word就是他的31至16位,low word是它的15至0位。
# 當引數超過兩個,wParam和lParam不夠用時,可以將wParam就給拆成兩個int16來使用。
# 這種時候在python裡記得用把HIWORD的常數向左移16位,再加LOWORD,即wParam = HIWORD<<16+LOWORD。

# 下選框內容更改
# 引數:下選框控制代碼; 訊息內容; 引數下選框的哪一個item,以0起始的待選選項的索引;如果該值為-1,將從組合框列表中刪除當前選項,並使當前選項為空; 引數
# CB_Handle為下選框控制代碼,PCB_handle下選框父視窗控制代碼
if win32api.SendMessage(CB_handle, win32con.CB_SETCURSEL, 1, 0) == 1:
# 下選框的父視窗命令
# 引數:父視窗控制代碼; 命令; 引數:WParam:高位表示型別,低位表示內容;引數IParam,下選框控制代碼
# CBN_SELENDOK當用戶選擇了有效的列表項時傳送,提示父窗體處理使用者的選擇。 LOWORD為組合框的ID. HIWORD為CBN_SELENDOK的值。
            win32api.SendMessage(PCB_handle, win32con.WM_COMMAND, 0x90000, CB_handle) 
# CBN_SELCHANGE當用戶更改了列表項的選擇時傳送,不論使用者是通過滑鼠選擇或是通過方向鍵選擇都會發送此通知。LOWORD為組合框的ID. HIWORD為CBN_SELCHANGE的值。
            win32api.SendMessage(PCB_handle, win32con.WM_COMMAND, 0x10000, CB_handle) 


# 設定文字框內容,等視窗處理完畢後返回true。中文需編碼成gbk 
# 引數:控制代碼;訊息型別;引數WParam,無需使用; 引數IParam,要設定的內容,字串
win32api.SendMessage(handle, win32con.WM_SETTEXT, 0, os.path.abspath(fgFilePath).encode('gbk'))


# 控制元件點選確定,處理訊息後返回0
# 引數:視窗控制代碼; 訊息型別; 引數WParam HIWORD為0(未使用),LOWORD為控制元件的ID; 引數IParam  0(未使用),確定控制元件的控制代碼
win32api.SendMessage(Mhandle, win32con.WM_COMMAND, 1, confirmBTN_handle)


# 獲取視窗文字不含截尾空字元的長度
# 引數:視窗控制代碼; 訊息型別; 引數WParam; 引數IParam
bufSize = win32api.SendMessage(subHandle, win32con.WM_GETTEXTLENGTH, 0, 0) +1
# 利用api生成Buffer
strBuf = win32gui.PyMakeBuffer(bufSize)
print(strBuf)
# 傳送訊息獲取文字內容
# 引數:視窗控制代碼; 訊息型別;文字大小; 儲存位置
length = win32gui.SendMessage(subHandle, win32con.WM_GETTEXT, bufSize, strBuf)
# 反向內容,轉為字串
# text = str(strBuf[:-1])

address, length = win32gui.PyGetBufferAddressAndLen(strBuf) 
text = win32gui.PyGetString(address, length) 
# print('text: ', text)

# 滑鼠單擊事件
#滑鼠定位到(30,50)
win32api.SetCursorPos([30,150])
#執行左單鍵擊,若需要雙擊則延時幾毫秒再點選一次即可
win32api.mouse_event(win32con.MOUSEEVENTF_LEFTUP | win32con.MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0)
#右鍵單擊
win32api.mouse_event(win32con.MOUSEEVENTF_RIGHTUP | win32con.MOUSEEVENTF_RIGHTDOWN, 0, 0, 0, 0)

def click1(x,y):                #第一種
    win32api.SetCursorPos((x,y))
    win32api.mouse_event(win32con.MOUSEEVENTF_LEFTDOWN,x,y,0,0)
    win32api.mouse_event(win32con.MOUSEEVENTF_LEFTUP,x,y,0,0)

def click2(x,y):               #第二種
    ctypes.windll.user32.SetCursorPos(x,y)
    ctypes.windll.user32.mouse_event(2,0,0,0,0)
    ctypes.windll.user32.mouse_event(4,0,0,0,0)

def click_it(pos):          #第三種
    handle= win32gui.WindowFromPoint(pos)
    client_pos =win32gui.ScreenToClient(handle,pos)
    tmp=win32api.MAKELONG(client_pos[0],client_pos[1])
    win32gui.SendMessage(handle, win32con.WM_ACTIVATE,win32con.WA_ACTIVE,0)
    win32gui.SendMessage(handle, win32con.WM_LBUTTONDOWN,win32con.MK_LBUTTON,tmp)
    win32gui.SendMessage(handle, win32con.WM_LBUTTONUP,win32con.MK_LBUTTON,tmp)

# 傳送回車
win32api.keybd_event(13,0,0,0)
win32api.keybd_event(13,0,win32con.KEYEVENTF_KEYUP,0)


# 關閉視窗
win32gui.PostMessage(win32lib.findWindow(classname, titlename), win32con.WM_CLOSE, 0, 0)


# 檢查視窗是否最小化,如果是最大化
if(win32gui.IsIconic(hwnd)):
#     win32gui.ShowWindow(hwnd, win32con.SW_SHOWNORMAL)
    win32gui.ShowWindow(hwnd, 8)
    sleep(0.5)

# SW_HIDE:隱藏視窗並激活其他視窗。nCmdShow=0。
# SW_MAXIMIZE:最大化指定的視窗。nCmdShow=3。
# SW_MINIMIZE:最小化指定的視窗並且啟用在Z序中的下一個頂層視窗。nCmdShow=6。
# SW_RESTORE:啟用並顯示視窗。如果視窗最小化或最大化,則系統將視窗恢復到原來的尺寸和位置。在恢復最小化視窗時,應用程式應該指定這個標誌。nCmdShow=9。
# SW_SHOW:在視窗原來的位置以原來的尺寸啟用和顯示視窗。nCmdShow=5。
# SW_SHOWDEFAULT:依據在STARTUPINFO結構中指定的SW_FLAG標誌設定顯示狀態,STARTUPINFO 結構是由啟動應用程式的程式傳遞給CreateProcess函式的。nCmdShow=10。
# SW_SHOWMAXIMIZED:啟用視窗並將其最大化。nCmdShow=3。
# SW_SHOWMINIMIZED:啟用視窗並將其最小化。nCmdShow=2。
# SW_SHOWMINNOACTIVE:視窗最小化,啟用視窗仍然維持啟用狀態。nCmdShow=7。
# SW_SHOWNA:以視窗原來的狀態顯示視窗。啟用視窗仍然維持啟用狀態。nCmdShow=8。
# SW_SHOWNOACTIVATE:以視窗最近一次的大小和狀態顯示視窗。啟用視窗仍然維持啟用狀態。nCmdShow=4。
# SW_SHOWNORMAL:啟用並顯示一個視窗。如果視窗被最小化或最大化,系統將其恢復到原來的尺寸和大小。應用程式在第一次顯示視窗的時候應該指定此標誌。nCmdShow=1。


# win32雖然也可控制鍵盤,但不如使用PyUserInput的方便。需要注意在windows和mac下介面引數可能有所不同。
from pymouse import PyMouse
from pykeyboard import PyKeyboard
m = PyMouse()
k = PyKeyboard()

x_dim, y_dim = m.screen_size()
# 滑鼠點選
m.click(x_dim/2, y_dim/2, 1)
# 鍵盤輸入
k.type_string('Hello, World!')

# 按住一個鍵
k.press_key('H')
# 鬆開一個鍵
k.release_key('H')
# 按住並鬆開,tap一個鍵
k.tap_key('e')
# tap支援重複的間歇點選鍵
k.tap_key('l',n=2,interval=5) 
# 傳送判斷文字
k.type_string('123456')

#建立組合鍵
k.press_key(k.alt_key)
k.tap_key(k.tab_key)
k.release_key(k.alt_key)
# 特殊功能鍵
k.tap_key(k.function_keys[5]) # Tap F5
k.tap_key(k.numpad_keys['Home']) # Tap 'Home' on the numpad
k.tap_key(k.numpad_keys[5], n=3) # Tap 5 on the numpad, thrice

# Mac系統
k.press_keys(['Command','shift','3'])
# Windows系統
k.press_keys([k.windows_l_key,'d'])

其中的PyMouseEvent和PyKeyboardEvent還可用於監聽滑鼠和鍵盤事件的輸入