上個月就打算開發個還算好玩的項目,但是一直沒時間。這篇是此項目用到的一部分,
處理好此部分基本還差通信等方面的了。首先模擬鼠標鍵盤按下釋放的動作,本人利用X11
這個庫,所以要了解X11編程;其次,本身用c或者c++就可以實現了,但是由於本人是py
粉,所以總想把代碼搬進python,所以本人就要實現python模塊,本篇用的ctypes,以后會
把python的c擴展模塊附上來的。
1.X11編程
首先簡單的介紹一下X11吧,網上有介紹,本人就不重復了。我們知道X是以server與client
的方式提供服務的,我們想要使用其功能,我們就需要與server通信。使用
Display *XOpenDisplay(char *display_name)獲得一個Display類型的句柄指針就可以了。
display_name可以是DISPLAY環境變量,用echo $DISPLAY輸出是:0(這是本人linux mint輸
出的)。如果display_name為NULL接口默認使用環境變量保存的值。X11編程常用的幾個頭
文件:
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xos.h>
本人用到的是#include <X11/Xlib.h>和 #include <X11/extensions/XTest.h>.
XTest.h有我們模擬鼠標和鍵盤需要的接口XTestFakeButtonEvent、 XTestFakeMotionEvent和
XTestFakeKeyEvent。想了解更多信息只需要在終端上man加函數名即可獲得。
比如XTestFakeMotionEvent接口:
int XTestFakeMotionEvent(display, screen_number, x, y,delay); Display *display; //此值就是從XOpenDisplay獲得 int screen_number; //讓其為-1即可表示當前的屏幕 int x, y; //屏幕位置 unsigned long delay; //延遲毫秒,讓其為CurrentTime表示不延遲
最后我們要關閉Display句柄:XCloseDisplay(Display *display)。
接口實現如下:
1 #include <stdio.h> 2 #include <X11/extensions/XTest.h> 3 #include <X11/Xlib.h> 4 Display *dspopen(){ 5 6 Display *dsp = XOpenDisplay(NULL); 7 if(!dsp) { 8 printf("open display failed\n"); 9 return NULL; 10 } 11 return dsp; 12 } 13 int presskey(Display *dsp,int s){ //鍵盤按 14 if(dsp==NULL) 15 return -1; 16 // KeySym keysym=XStringToKeysym(s); 17 KeyCode key=XKeysymToKeycode(dsp,s); 18 if(key==NoSymbol) 19 return -1; 20 XTestFakeKeyEvent(dsp,key,1,CurrentTime); 21 XFlush(dsp); 22 return 0; 23 } 24 int move(Display *dsp,int x,int y) //鼠標移動 25 { 26 if(0==XTestFakeMotionEvent(dsp,-1,x,y,CurrentTime)) 27 { 28 printf("Cannot move!\n"); 29 return -1; 30 } 31 return 0; 32 } 33 int buttonpress(Display *dsp,int type) //鼠標按,type=1表示左鍵,3是右鍵,2是中鍵 34 { 35 if(0==XTestFakeButtonEvent(dsp,type,1,CurrentTime)) 36 { 37 printf("press failed\n"); 38 return -1; 39 } 40 return 0; 41 } 42 int buttonrelease(Display *dsp,int type) //鼠標釋放 43 { 44 if(0==XTestFakeButtonEvent(dsp,type,0,CurrentTime)) 45 { 46 printf("release failed\n"); 47 return -1; 48 } 49 return 0; 50 } 51 int releasekey(Display *dsp,int s){ //鍵盤release 52 if(dsp==NULL) 53 return -1; 54 // KeySym keysym=XStringToKeysym(s); 55 KeyCode key=XKeysymToKeycode(dsp,s); 56 if(key==NoSymbol) 57 return -1; 58 XTestFakeKeyEvent(dsp,key,0,CurrentTime); 59 XFlush(dsp); 60 return 0; 61 } 62 void dspclose( Display *dsp ){ 63 if(dsp!=NULL){ 64 XCloseDisplay(dsp); 65 66 } 67 } 68 //int main(){ //測試用的會在程序結束后,在光標前輸出c 69 // Display *dsp=dspopen(); 70 // presskey(dsp,'c'); 71 // releasekey(dsp,'c'); 72 // dspclose(dsp); 73 // return 0; 74 //}
上面注釋掉的main函數可以作為測試用的,好了,我們把上面的代碼保存為display.c
編譯成一個共享庫,需要X11和Xtst庫。
gcc -fPIC -shared -o libdisplay.so display.c -lX11 -lXtst
編譯后會生成libdisplay.so 。現在我們ctypes模塊使用這個動態共享庫。
2.ctypes簡單介紹和使用
我們知道python中的類型與c中類型是不一樣的,應該說沒有一樣的,就拿int想來說,
python也是把它看作PyObject類型來處理的。那么我們需要使用ctype提供的接口做類型的
轉換。見:http://docs.python.org/2/library/ctypes.html#fundamental-data-types
此鏈接有張圖詳細的展現類型轉換對應的接口。下面演示具體的操作吧。
我們通過CDLL()接口載入庫:
lc=CDLL("./libdisplay.so")
然后就可以使用庫中提供的接口了,但是上面dspopen()接口的返回值是Display類型的指針,
所以我們需要用c_void_p()轉換一下:
d=c_void_p(lc.dspopen())
之后就可以用d做處理了,代碼如下:
from ctypes import * import time class MOUSE: LEFT=1 MiDDLE=2 RIGHT=3 lc=CDLL("./libdisplay.so") d=c_void_p(lc.dspopen()) time.sleep(5); lc.buttonpress(d,c_int(MOUSE.RIGHT)) lc.buttonrelease(d,c_int(MOUSE.RIGHT)) lc.dspclose(d)
上面的代碼會在5秒后在鼠標指針處打開右鍵菜單。
利用ctypes使用c編寫的庫就講這么多吧。以后會把c代碼的部分寫成python的c擴展再分享吧。
利用上面的模擬鍵盤和鼠標的接口可以做一些有意思的事情....