linux下python之模擬鼠標鍵盤動作具體實現

  上個月就打算開發個還算好玩的項目,但是一直沒時間。這篇是此項目用到的一部分,

處理好此部分基本還差通信等方面的了。首先模擬鼠標鍵盤按下釋放的動作,本人利用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加函數名即可獲得。

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)。

接口實現如下:

#include <stdio.h>
#include <X11/extensions/XTest.h>
#include <X11/Xlib.h>
Display *dspopen(){  

    Display *dsp = XOpenDisplay(NULL);
    if(!dsp) {
        printf("open display failed\n");
        return NULL;
    }
    return dsp;
}
int presskey(Display *dsp,int s){  //鍵盤按
    if(dsp==NULL)
        return -1;
//    KeySym keysym=XStringToKeysym(s);
    KeyCode key=XKeysymToKeycode(dsp,s);
    if(key==NoSymbol)
        return -1;
    XTestFakeKeyEvent(dsp,key,1,CurrentTime);
    XFlush(dsp);
    return 0;
}
int move(Display *dsp,int x,int y) //鼠標移動
{
    if(0==XTestFakeMotionEvent(dsp,-1,x,y,CurrentTime))
    {
        printf("Cannot move!\n");
        return -1;
    }
    return 0;
}
int buttonpress(Display *dsp,int type) //鼠標按,type=1表示左鍵,3是右鍵,2是中鍵
{
    if(0==XTestFakeButtonEvent(dsp,type,1,CurrentTime))
    {
        printf("press failed\n");
        return -1;
    }
    return 0;
}
int buttonrelease(Display *dsp,int type) //鼠標釋放
{
    if(0==XTestFakeButtonEvent(dsp,type,0,CurrentTime))
    {
        printf("release failed\n");
        return -1;
    }
    return 0;
}
int releasekey(Display *dsp,int s){ //鍵盤release
    if(dsp==NULL)
        return -1;
//    KeySym keysym=XStringToKeysym(s);
    KeyCode key=XKeysymToKeycode(dsp,s);
    if(key==NoSymbol)
        return -1;
    XTestFakeKeyEvent(dsp,key,0,CurrentTime);
    XFlush(dsp);
    return 0;
}
void dspclose( Display *dsp ){
    if(dsp!=NULL){
        XCloseDisplay(dsp);

    }
}
//int main(){     //測試用的會在程序結束後,在光標前輸出c
//    Display *dsp=dspopen();
//    presskey(dsp,'c');
//    releasekey(dsp,'c');
//    dspclose(dsp);
//    return 0;
//}


上面註釋掉的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擴展再分享吧。

利用上面的模擬鍵盤和鼠標的接口可以做一些有意思的事情....

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章