Python利用pynput實(shí)現(xiàn)劃詞復(fù)制功能
前言
本文參考了以下代碼
Windows系統(tǒng)環(huán)境下Python腳本實(shí)現(xiàn)全局“劃詞復(fù)制”功能
from pynput import mouse
import time
import threading
__DEBUG = False
def log(msg,debug=False):
if __DEBUG or debug:
print(msg)
class MouseMonitor():
__press_time = 0
__press_double_state = False
__move = (0,0)
def __init__(self,on_selected=None):
if on_selected:
self.on_selected = on_selected
else:
self.on_selected = self.on_selected
self.listener = mouse.Listener(on_move=self.on_move,on_click=self.on_click)
self.listener.start()
self.listener.join()
def on_selected(msg):
print('selected "%s" has been copied.' % (msg,))
def on_move(self,x,y):
if self.__press_time == 0:
self.__move = (x,y)
# log(self.__press_time,time.time())
# log('Pointer moved to {0}'.format((x,y)))
def on_click(self,x,y,button,pressed):
if str(button) == 'Button.left':
if pressed:
self.on_pressed(x,y)
else:
self.on_released(x,y)
def on_pressed(self,x,y):
if self.__press_double_state:
# double click
# self.__press_double_state = False
if not self.check_not_time_out(self.__press_time, time.time(),0.4): # miss double click
log('double1 click timeout and reset then')
self.reset()
self.__press_time = time.time()
else:
# single click
self.__press_time = time.time()
# self.__press_double_state = False
def on_released(self,x,y):
if self.__press_double_state:
# double click
if self.check_not_time_out(self.__press_time, time.time(),0.8):
log('double click: %s' % (self.get_copy()))
self.on_selected(self.get_copy())
self.__press_double_state = False
else:
log('double2 click timeout and reset then')
self.reset()
else:
if self.check_not_time_out(self.__press_time, time.time()):
log('double click maybe')
self.__press_double_state = True
threading.Timer(0.5, self.timeout_handler).start() # wait timeout to reset
elif not self.check_not_time_out(self.__press_time, time.time(),1):
if self.__move != (0,0):
self.on_selected(self.get_copy())
log('selected: %s' % (self.get_copy(),))
self.reset()
else:
log('reset state')
self.reset()
def get_copy(self):
import win32clipboard as wc
import win32con
def trigger_copy():
from pynput.keyboard import Key,Controller
key = Controller()
with key.pressed(Key.ctrl):
key.press('c')
key.release('c')
time.sleep(0.1) # wait for ctrl+c valid
trigger_copy()
msg = ''
try:
wc.OpenClipboard()
msg = wc.GetClipboardData(win32con.CF_UNICODETEXT)
wc.CloseClipboard()
except TypeError:
log('Clipboard Content is TypeError.')
return msg
def reset(self):
self.__press_time = 0
self.__press_double_state = False
self.__move = (0,0)
def timeout_handler(self):
self.reset()
log('timeout reset state')
def check_not_time_out(self,old,new,delta=0.2):
if(new - old > delta): # time delta > 0.2s
return False
else:
return True
def printf(msg):
log('copy content:'+msg,True)
# log('x = {0} , y = {1}'.format(x,y))
if __name__ == '__main__':
mm = MouseMonitor(printf)pynput庫(kù)的官方文檔
實(shí)現(xiàn)代碼
參考的博客實(shí)現(xiàn)了劃詞復(fù)制,但是看了下他的代碼寫(xiě)的有點(diǎn)復(fù)制混亂,監(jiān)聽(tīng)準(zhǔn)確率并不高且不太容易理解。
實(shí)際監(jiān)聽(tīng)鼠標(biāo)的劃詞操作邏輯很簡(jiǎn)單:
記錄下鼠標(biāo)左鍵按下時(shí)的位置,當(dāng)鼠標(biāo)左鍵松開(kāi)時(shí),記錄下鼠標(biāo)左鍵松開(kāi)的位置,如果按下的位置和松開(kāi)的位置不一致,說(shuō)明鼠標(biāo)正在劃詞。
from pynput.mouse import Listener, Button
from pynput.keyboard import Key, Controller
class AutoCopier():
__press_xy = (0, 0) # 私有變量 鼠標(biāo)左鍵按下時(shí)的位置
def __init__(self):
self.keyboard = Controller() # 初始化鍵盤(pán)控制器
self.listener = Listener(on_click=self.on_click) # 初始化鼠標(biāo)監(jiān)聽(tīng)器
self.listener.start() # 開(kāi)啟鼠標(biāo)監(jiān)聽(tīng)器線(xiàn)程
# 點(diǎn)擊函數(shù)
def on_click(self, x, y, button, pressed):
if button == Button.left: # 左鍵點(diǎn)擊
if pressed: # 左鍵按下
self.__press_xy = (x, y) # 記錄當(dāng)前鼠標(biāo)位置
else: # 左鍵松開(kāi)
if self.__press_xy != (x, y): # 按下位置和松開(kāi)位置不一致
self.copy() # 說(shuō)明操作是劃詞,執(zhí)行復(fù)制函數(shù)
# 復(fù)制函數(shù)
def copy(self):
with self.keyboard.pressed(Key.ctrl): # 按下ctrl,with語(yǔ)句結(jié)束后自動(dòng)松開(kāi)
self.keyboard.press('c') # 按下c
self.keyboard.release('c') # 松開(kāi)c
# 等待線(xiàn)程終止
def wait_to_stop(self):
self.listener.join()
# for test
if __name__ == '__main__':
at = AutoCopier()
at.wait_to_stop()知識(shí)點(diǎn)補(bǔ)充
1.pynput是什么
官方說(shuō)法:“他可以控制和監(jiān)聽(tīng)我們的輸入設(shè)備,目前支持鼠標(biāo)和鍵盤(pán)的控制與監(jiān)聽(tīng);因?yàn)槲抑皇褂昧嗽O(shè)備的控制 至于監(jiān)聽(tīng)并沒(méi)作深入了解 文章也不設(shè)計(jì)”
2.使用步驟
安裝pynput模塊
pip install pynput #使用ctrl+v 快捷粘貼時(shí)候用到 pip install pyperclip
鍵盤(pán)控制
如下:
from pynput.keyboard import Key, Controller as c_keyboard
from pynput.mouse import Button, Controller as c_mouse
keyboard = c_keyboard()
#字符與數(shù)字
keyboard.press('a')
keyboard.release('a')
keyboard.press('A')
keyboard.release('A')
keyboard.press('1')
keyboard.release('1')
#非數(shù)字與字母鍵 詳情: https://pynput.readthedocs.io/en/latest/keyboard.html#pynput.keyboard.Key
keyboard.press(Key.enter);
keyboard.release(Key.enter);
#組合
##全選
keyboard.press(Key.ctrl)
keyboard.press('a')
time.sleep(2)
keyboard.release('a')
keyboard.release(Key.ctrl)
###或者
with keyboard .pressed(Key.ctrl):
keyboard.press('a')
keyboard.release('a')
##復(fù)制
seller_nick = 'www.baidu.com/a.php?a=a&b=2'
pyperclip.copy('https://'+seller_nick.replace('amp;',''))
##粘貼
keyboard.press(Key.ctrl)
keyboard.press('v')
time.sleep(1)
keyboard.release('v')
keyboard.release(Key.ctrl)
##回車(chē)
keyboard.press(Key.enter);
keyboard.release(Key.enter);
鼠標(biāo)控制
如下:
from pynput.keyboard import Key, Controller as c_keyboard from pynput.mouse import Button, Controller as c_mouse mouse= c_mouse() #點(diǎn)擊 ##雙擊 mouse.click(Button.left, 2) ##按下右鍵 mouse.press(Button.right) ##釋放右鍵 mouse.release(Button.right) #鼠標(biāo)坐標(biāo) print(mouse.position) ##x軸坐標(biāo) print(mouse.position[0]) ##y軸坐標(biāo) print(mouse.position[1]) #移動(dòng) ##移動(dòng)到絕對(duì)坐標(biāo) mouse.position = (400, 38) ##相對(duì)當(dāng)前坐標(biāo)移動(dòng) mouse.move(300, 2) #滑動(dòng) mouse.press(Button.left) mouse.move(300, 2) mouse.release(Button.left)
到此這篇關(guān)于Python利用pynput實(shí)現(xiàn)劃詞復(fù)制功能的文章就介紹到這了,更多相關(guān)Python pynput劃詞復(fù)制內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python multiprocessing多進(jìn)程原理與應(yīng)用示例
這篇文章主要介紹了Python multiprocessing多進(jìn)程原理與應(yīng)用,結(jié)合實(shí)例形式詳細(xì)分析了基于multiprocessing包的多進(jìn)程概念、原理及相關(guān)使用操作技巧,需要的朋友可以參考下2019-02-02
python標(biāo)識(shí)符命名規(guī)范原理解析
這篇文章主要介紹了python標(biāo)識(shí)符命名規(guī)范原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
Python輕松讀寫(xiě)TOML文件實(shí)現(xiàn)示例
這篇文章主要介紹了Python輕松讀寫(xiě)TOML文件實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
python機(jī)器學(xué)習(xí)基礎(chǔ)決策樹(shù)與隨機(jī)森林概率論
這篇文章主要為大家介紹了python機(jī)器學(xué)習(xí)基礎(chǔ)決策樹(shù)與隨機(jī)森林概率論詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-11-11
Django提示mysql版本過(guò)低:django.db.utils.NotSupportedError: My
這篇文章主要介紹了Django提示mysql版本過(guò)低:django.db.utils.NotSupportedError: MySQL 8 or later is required (found 5.7.26).的解決方法,文中有詳細(xì)的解決方案,具有一定的參考價(jià)值,需要的朋友可以參考下2024-03-03
Python爬蟲(chóng)爬取屬于自己的地鐵線(xiàn)路圖
這篇文章主要介紹了Python爬蟲(chóng)爬取屬于自己的地鐵線(xiàn)路圖,下面文章主要事根據(jù)自己需要對(duì)地鐵路線(xiàn)進(jìn)行爬取的實(shí)現(xiàn)過(guò)程,需要的小伙伴可以參考一下,希望對(duì)你有所幫助2021-12-12
tensorflow如何將one_hot標(biāo)簽和數(shù)字(整數(shù))標(biāo)簽進(jìn)行相互轉(zhuǎn)化
這篇文章主要介紹了tensorflow如何將one_hot標(biāo)簽和數(shù)字(整數(shù))標(biāo)簽進(jìn)行相互轉(zhuǎn)化問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06
Python實(shí)戰(zhàn)項(xiàng)目之MySQL tkinter pyinstaller實(shí)現(xiàn)學(xué)生管理系統(tǒng)
讀萬(wàn)卷書(shū)不如行萬(wàn)里路,只學(xué)書(shū)上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用MySQL、tkinter、 pyinstaller實(shí)現(xiàn)一個(gè)學(xué)生管理系統(tǒng),大家可以通過(guò)案例查缺補(bǔ)漏,提升水平2021-10-10
關(guān)于Python?Tkinter?復(fù)選框?->Checkbutton
這篇文章主要介紹了關(guān)于Python?Tkinter復(fù)選框Checkbutton,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09

