python之模擬鼠標(biāo)鍵盤動(dòng)作具體實(shí)現(xiàn)
上個(gè)月就打算開發(fā)個(gè)還算好玩的項(xiàng)目,但是一直沒時(shí)間。這篇是此項(xiàng)目用到的一部分,
處理好此部分基本還差通信等方面的了。首先模擬鼠標(biāo)鍵盤按下釋放的動(dòng)作,本人利用X11
這個(gè)庫,所以要了解X11編程;其次,本身用c或者c++就可以實(shí)現(xiàn)了,但是由于本人是py
粉,所以總想把代碼搬進(jìn)python,所以本人就要實(shí)現(xiàn)python模塊,本篇用的ctypes,以后會(huì)
把python的c擴(kuò)展模塊附上來的。
1.X11編程
首先簡(jiǎn)單的介紹一下X11吧,網(wǎng)上有介紹,本人就不重復(fù)了。我們知道X是以server與client
的方式提供服務(wù)的,我們想要使用其功能,我們就需要與server通信。使用
Display *XOpenDisplay(char *display_name)獲得一個(gè)Display類型的句柄指針就可以了。
display_name可以是DISPLAY環(huán)境變量,用echo $DISPLAY輸出是:0(這是本人linux mint輸
出的)。如果display_name為NULL接口默認(rèn)使用環(huán)境變量保存的值。X11編程常用的幾個(gè)頭
文件:
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xos.h>
本人用到的是#include <X11/Xlib.h>和 #include <X11/extensions/XTest.h>.
XTest.h有我們模擬鼠標(biāo)和鍵盤需要的接口XTestFakeButtonEvent、 XTestFakeMotionEvent和
XTestFakeKeyEvent。想了解更多信息只需要在終端上man加函數(shù)名即可獲得。
比如XTestFakeMotionEvent接口:
int XTestFakeMotionEvent(display, screen_number, x, y,delay);
Display *display; //此值就是從XOpenDisplay獲得
int screen_number; //讓其為-1即可表示當(dāng)前的屏幕
int x, y; //屏幕位置
unsigned long delay; //延遲毫秒,讓其為CurrentTime表示不延遲
最后我們要關(guān)閉Display句柄:XCloseDisplay(Display *display)。
接口實(shí)現(xiàn)如下:
#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) //鼠標(biāo)移動(dòng)
{
if(0==XTestFakeMotionEvent(dsp,-1,x,y,CurrentTime))
{
printf("Cannot move!\n");
return -1;
}
return 0;
}
int buttonpress(Display *dsp,int type) //鼠標(biāo)按,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) //鼠標(biāo)釋放
{
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è)試用的會(huì)在程序結(jié)束后,在光標(biāo)前輸出c
// Display *dsp=dspopen();
// presskey(dsp,'c');
// releasekey(dsp,'c');
// dspclose(dsp);
// return 0;
//}
上面注釋掉的main函數(shù)可以作為測(cè)試用的,好了,我們把上面的代碼保存為display.c
編譯成一個(gè)共享庫,需要X11和Xtst庫。
gcc -fPIC -shared -o libdisplay.so display.c -lX11 -lXtst
編譯后會(huì)生成libdisplay.so ?,F(xiàn)在我們ctypes模塊使用這個(gè)動(dòng)態(tài)共享庫。
2.ctypes簡(jiǎn)單介紹和使用
我們知道python中的類型與c中類型是不一樣的,應(yīng)該說沒有一樣的,就拿int想來說,
python也是把它看作PyObject類型來處理的。那么我們需要使用ctype提供的接口做類型的
轉(zhuǎn)換。見:http://docs.python.org/2/library/ctypes.html#fundamental-data-types
此鏈接有張圖詳細(xì)的展現(xiàn)類型轉(zhuǎn)換對(duì)應(yīng)的接口。下面演示具體的操作吧。
我們通過CDLL()接口載入庫:
lc=CDLL("./libdisplay.so")
然后就可以使用庫中提供的接口了,但是上面dspopen()接口的返回值是Display類型的指針,
所以我們需要用c_void_p()轉(zhuǎn)換一下:
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)
上面的代碼會(huì)在5秒后在鼠標(biāo)指針處打開右鍵菜單。
利用ctypes使用c編寫的庫就講這么多吧。以后會(huì)把c代碼的部分寫成python的c擴(kuò)展再分享吧。
利用上面的模擬鍵盤和鼠標(biāo)的接口可以做一些有意思的事情....
相關(guān)文章
python multiprocessing模塊用法及原理介紹
這篇文章主要介紹了python multiprocessing模塊用法及原理介紹,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08詳解Python中open()函數(shù)指定文件打開方式的用法
well,我們這里所指的文件打開方式并不是指調(diào)用什么應(yīng)用程序去打開某個(gè)文件,而是只讀只寫或者二進(jìn)制等的打開方式,這里我們就來詳解Python中open()函數(shù)指定文件打開方式的用法2016-06-06python爬蟲獲取小區(qū)經(jīng)緯度以及結(jié)構(gòu)化地址
這篇文章主要為大家詳細(xì)介紹了python爬蟲獲取小區(qū)經(jīng)緯度,以及結(jié)構(gòu)化的地址,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12詳解Python和Rust中內(nèi)存管理機(jī)制的實(shí)現(xiàn)與對(duì)比
Python和Rust都采用了垃圾收集(Garbage?Collection)機(jī)制來管理內(nèi)存,但它們各自的實(shí)現(xiàn)方式有很大的不同,下面就跟隨小編一起來深入了解下二者的區(qū)別吧2024-03-03Python預(yù)測(cè)分詞的實(shí)現(xiàn)
本文將結(jié)合實(shí)例代碼,介紹Python預(yù)測(cè)分詞的實(shí)現(xiàn),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-06-06Python使用Pandas對(duì)csv文件進(jìn)行數(shù)據(jù)處理的方法
這篇文章主要介紹了Python使用Pandas對(duì)csv文件進(jìn)行數(shù)據(jù)處理的方法,本文通過實(shí)例代碼相結(jié)合給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-08-08