Windows注冊鼠標鉤子,獲取用戶選中的文本

注冊鼠標鉤子

// 注冊鼠標鉤子
HHOOK hMouseHook;
hMouseHook = SetWindowsHookEx(WH_MOUSE_LL, MouseProc, GetModuleHandle(NULL), 0);// 取消鼠標鉤子
UnhookWindowsHookEx(hMouseHook);
hMouseHook = nullptr;

上述代碼中MouseProc方法用于處理系統的鼠標消息

處理鼠標消息

LRESULT MouseHook::MouseProc(int nCode, WPARAM wParam, LPARAM lParam)
{static QPoint pos;static qint64 lastTriggerTime = 0;if (nCode >= 0) {if (wParam == WM_LBUTTONDOWN) {pos = QCursor::pos();           }else if (wParam == WM_LBUTTONUP) {if (pos != QCursor::pos()) {timer->start(280);  //拖拽}qint64 currentTime = QDateTime::currentMSecsSinceEpoch();if (currentTime - lastTriggerTime <= 500) {timer->start(280); //雙擊}lastTriggerTime = currentTime;}}return CallNextHookEx(hMouseHook, nCode, wParam, lParam);
}

這段代碼中,我們使用一個QTimer來處理雙擊或拖拽選中文本的操作。

文本選中后的處理方法

void MouseHook::processMouseUp()
{auto textGetter = new TextGetter(this);connect(textGetter, &TextGetter::resultReady, this, &MouseHook::onTextReady, Qt::ConnectionType::QueuedConnection);textGetter->start();
}

TextGetter是一個繼承自QThread的類,我們將在一個新線程中去獲取用戶選中的文本。

獲取選中文本的第一種情況

QString TextGetter::getSelectedTextByUIAutomation() {try {auto hr = CoInitialize(NULL);if (FAILED(hr)){CoUninitialize();return "";}CComPtr<IUIAutomation> automation;hr = CoCreateInstance(CLSID_CUIAutomation, nullptr, CLSCTX_INPROC_SERVER, IID_IUIAutomation,(void**)(&automation));if (FAILED(hr)){CoUninitialize();return "";}CComPtr<IUIAutomationElement> focusedElement;hr = automation->GetFocusedElement(&focusedElement);if (FAILED(hr) || !focusedElement){CoUninitialize();return "";}CComPtr<IUIAutomationTextPattern> textPattern;hr = focusedElement->GetCurrentPatternAs(UIA_TextPatternId, IID_PPV_ARGS(&textPattern));if (FAILED(hr) || !textPattern){CoUninitialize();return "";}CComPtr<IUIAutomationTextRangeArray> selection;hr = textPattern->GetSelection(&selection);if (FAILED(hr) || !selection){CoUninitialize();return "";}CComPtr<IUIAutomationTextRange> range;hr = selection->GetElement(0, &range);if (FAILED(hr) || !range){ CoUninitialize();return "";}CComBSTR text;range->GetText(-1, &text);std::wstring ws(text, SysStringLen(text));CoUninitialize();return QString::fromStdWString(ws);}catch (std::exception& e) {return "";}       
}

這段代碼使用 Microsoft UI Automation (UIA) API 從當前具有焦點的 UI 元素中獲取選中文本的方法。但有的時候這種方法獲取不到想要的文本(老式窗口中的文本)

獲取選中文本的第二種情況

當第一種情況獲取到的文本是空時,就要嘗試第二種情況

auto hwnd = getCurrentHwnd();
if (!hwnd) {return "";
}
auto cache = cacheClipboard();
sendCtrlC();
str = getClipboardText();
if (str.isEmpty()) {CloseClipboard();return "";
}
restoreClipboard(cache);
CloseClipboard();

這種情況,先獲取系統當前聚焦的窗口,然后緩存當前剪切板,然后發送Ctrl+C復制此窗口選中的文本,然后獲取剪切板內的文本,然后把之前緩存的內容存入剪切板。

下面我們看看這些實現代碼:

獲取系統當前聚焦的窗口

HWND TextGetter::getCurrentHwnd()
{HWND hwnd = GetForegroundWindow();DWORD threadId = GetWindowThreadProcessId(hwnd, NULL);AttachThreadInput(GetCurrentThreadId(), threadId, TRUE);hwnd = GetFocus();AttachThreadInput(GetCurrentThreadId(), threadId, FALSE);POINT pt;GetCursorPos(&pt);RECT rect;GetWindowRect(hwnd, &rect);if (pt.x<rect.left || pt.y<rect.top || pt.x>rect.right || pt.y>rect.bottom) {return nullptr;}return hwnd;
}

如果聚焦的窗口與鼠標所在位置的窗口不是一個窗口,那么我們取消任務。

緩存剪切板的內容

ClipboardData TextGetter::cacheClipboard()
{OpenClipboard(nullptr);ClipboardData cache;UINT format = 0;while ((format = EnumClipboardFormats(format)) != 0) {HANDLE hData = GetClipboardData(format);if (hData) {SIZE_T size = GlobalSize(hData);HGLOBAL hCopy = GlobalAlloc(GMEM_MOVEABLE, size);if (hCopy) {void* pDest = GlobalLock(hCopy);void* pSource = GlobalLock(hData);if (pDest && pSource) {memcpy(pDest, pSource, size);}GlobalUnlock(hData);GlobalUnlock(hCopy);cache.push_back({ format, hCopy });}}}EmptyClipboard();CloseClipboard();return cache;
}

發送Ctrl+C按鍵消息

void TextGetter::sendCtrlC()
{INPUT inputs[4] = { 0 };inputs[0].type = INPUT_KEYBOARD;inputs[0].ki.wVk = VK_CONTROL;inputs[1].type = INPUT_KEYBOARD;inputs[1].ki.wVk = 'C';inputs[2].type = INPUT_KEYBOARD;inputs[2].ki.wVk = 'C';inputs[2].ki.dwFlags = KEYEVENTF_KEYUP;inputs[3].type = INPUT_KEYBOARD;inputs[3].ki.wVk = VK_CONTROL;inputs[3].ki.dwFlags = KEYEVENTF_KEYUP;SendInput(ARRAYSIZE(inputs), inputs, sizeof(INPUT));QThread::msleep(360);
}

按鍵發送成功后需要等待360毫秒

獲取剪切板的內容

QString TextGetter::getClipboardText()
{OpenClipboard(nullptr);if (!IsClipboardFormatAvailable(CF_UNICODETEXT)) {CloseClipboard();return "";}HANDLE hData = GetClipboardData(CF_UNICODETEXT);if (hData == nullptr) {CloseClipboard();return "";}LPCWSTR pText = static_cast<LPCWSTR>(GlobalLock(hData));if (pText == nullptr) {CloseClipboard();return "";}QString result = QString::fromWCharArray(pText);GlobalUnlock(hData);CloseClipboard();return result;
}

不要懷疑發送按鍵Ctrl+C這個方案是否可行,有道詞典就是這么干的。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/75885.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/75885.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/75885.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

flink cdc的source數據流如何配置事件時間,如何設置時間語義,分配時間戳并生成水位線

在 Flink CDC 中為 Source 數據流配置事件時間需要結合時間語義設置、時間戳分配和水位線生成三個核心步驟。以下是具體配置方法及注意事項&#xff1a; 1. 設置時間語義 Flink 默認使用處理時間&#xff08;Processing Time&#xff09;&#xff0c;需顯式指定事件時間語義&a…

C++ 指針類型轉換全面解析與最佳實踐

文章目錄 C 指針類型轉換全面解析與最佳實踐1. 隱式轉換基類和派生類指針 2. 顯式轉換(1) static_cast(2) dynamic_cast(3) reinterpret_cast(4) const_cast 3. C 風格轉換4. 常見問題與注意事項5. 總結最佳實踐 C 指針類型轉換全面解析與最佳實踐 在 C 中&#xff0c;指針類型…

批量將 txt/html/json/xml/csv 等文本拆分成多個文件

我們的文本文件太大的時候&#xff0c;我們通常需要對文本文件進行拆分&#xff0c;比如按多少行一個文件將一個大的文本文件拆分成多個小的文本文件。這樣我們在打開或者傳輸的時候都比較方便。今天就給大家介紹一種同時對多個文本文件進行批量拆分的方法&#xff0c;可以快速…

ARM 匯編啟動代碼詳解:從中斷向量表到中斷處理

ARM 匯編啟動代碼詳解&#xff1a;從中斷向量表到中斷處理 引言 在嵌入式系統開發中&#xff0c;ARM 處理器&#xff08;如 Cortex-A 系列&#xff09;的啟動代碼是系統初始化和運行的基礎。啟動代碼通常包括中斷向量表的創建、初始化硬件狀態&#xff08;如關閉緩存和 MMU&a…

4.7學習總結 可變參數+集合工具類Collections+不可變集合

可變參數&#xff1a; 示例&#xff1a; public class test {public static void main(String[] args) {int sumgetSum(1,2,3,4,5,6,7,8,9,10);System.out.println(sum);}public static int getSum(int...arr){int sum0;for(int i:arr){sumi;}return sum;} } 細節&#xff1a…

2023年藍橋杯第十四屆CC++大學B組真題及代碼

目錄 1A&#xff1a;日期統計 解析代碼_暴力_正解 2B&#xff1a;01串的熵 解析代碼_暴力_正解 3C&#xff1a;冶煉金屬 解析代碼_暴力_正解 4D&#xff1a;飛機降落 解析代碼_暴力dfs_正解 5E&#xff1a;接龍數列 解析代碼_dp_正解 6F&#xff1a;島嶼個數 解析代…

rom定制系列------小米10pro機型定制解鎖固件 原生安卓15批量線刷固件 操作解析與界面預覽

注意;固件用于自己機型忘記密碼或者手機號注銷等出現設備鎖 過保修期 售后無視的機型&#xff0c;勿用于非法途徑 目前有粉絲聯系&#xff0c;自己的機型由于手機號注銷導致手機更新系統后出現設備鎖界面。另外也沒有解鎖bl。目前無法使用手機。經過詢問是小米10pro機型。根據…

信息學奧賽一本通 1861:【10NOIP提高組】關押罪犯 | 洛谷 P1525 [NOIP 2010 提高組] 關押罪犯

【題目鏈接】 ybt 1861&#xff1a;【10NOIP提高組】關押罪犯 洛谷 P1525 [NOIP 2010 提高組] 關押罪犯 【題目考點】 1. 圖論&#xff1a;二分圖 2. 二分答案 3. 種類并查集 【解題思路】 解法1&#xff1a;種類并查集 一個囚犯是一個頂點&#xff0c;一個囚犯對可以看…

我的NISP二級之路-01

目錄 一.SSE-CMM系統安全工程-能力成熟度模型(Systems Security Engineering - Capability Maturity Model) 二.ISMS 即信息安全管理體系(Information Security Management System),是一種基于風險管理的、系統化的管理體系 三.Kerberos協議 1. 用戶登錄與 AS 請求 2…

WEB安全--內網滲透--利用Net-NTLMv2 Hash

一、前言 在前兩篇文章中分析了NTLM協議中Net-NTLMv2 Hash的生成、如何捕獲Net-NTLMv2 Hash&#xff0c;現在就來探討一下在內網環境中&#xff0c;如何利用Net-NTLMv2 Hash進行滲透。 二、Net-NTLM Hash的破解 工具&#xff1a;hashcat 原理&#xff1a;利用其內部的字典對…

如何正確使用 `apiStore` 進行 API 管理

在現代前端開發中&#xff0c;API 管理是一個非常重要的環節。apiStore 是一個基于 Pinia 的狀態管理工具&#xff0c;它可以幫助我們更高效地管理和調用 API。本文將詳細介紹如何正確使用 apiStore&#xff0c;包括如何創建 API 配置文件、在組件中使用 apiStore 以及如何配置…

瓦片數據合并方法

影像數據 假如有兩份影像數據 1.全球底層影像0-5級別如下&#xff1a; 2.局部高清影像數據級別9-14如下&#xff1a; 合并方法 將9-14文件夾復制到全球底層0-5的目錄下 如下&#xff1a; 然后合并xml文件 使得Tileset設置到最高級&#xff08;包含所有級別&#xff09;&…

C++中的類和對象(上)

1 類的定義 1.1 類定義的格式 1 class為定義類的關鍵字&#xff0c;Stack為類的名字&#xff0c;{}中為類的主體&#xff0c;注意類定義結束時后面分號不能省 略》。類體中內容稱為類的成員&#xff1a;類中的變量稱為類的屬性或成員變量; 類中的函數稱為類的方法或者成員函數…

【Tauri2】013——前端Window Event與創建Window

前言 【Tauri2】012——on_window_event函數-CSDN博客https://blog.csdn.net/qq_63401240/article/details/146909801?spm1001.2014.3001.5501 前面介紹了on_window_event&#xff0c;這個在Builder中的方法&#xff0c;里面有許多事件&#xff0c;比如Moved&#xff0c;Res…

【問題處理】webpack4升webpack5,報錯Uncaught ReferrnceError: process is not defined

問題 正在做webpack4升webpack5&#xff0c;項目構建項目成功后在瀏覽器打開時報錯 Uncaught ReferrnceError: process is not defined。 原因 webpack 5 不再自動 polyfill Node.js 的核心模塊。 如果你在瀏覽器運行的代碼中使用它&#xff0c;需要從 NPM 中安裝兼容模塊…

軟件工程師減肥計劃

一、目標設定 在 3 個月內減輕體重 5-7kg&#xff0c;改善身體代謝水平和體脂率&#xff0c;增強身體活力和精神狀態&#xff0c;以更好地適應工作強度。 二、飲食調整 &#xff08;一&#xff09;基本原則 控制熱量攝入&#xff0c;保證每天攝入熱量低于消耗熱量 500-800 …

即時訪問成為降低風險的關鍵

云計算和軟件即服務 (SaaS) 解決方案的廣泛采用從根本上重塑了企業的數字格局。 不同行業的組織越來越多地利用云固有的可擴展性和成本效益來推動創新和簡化運營。 這種向基于云的環境的轉變也帶來了一系列新的復雜安全挑戰&#xff0c;需要仔細考慮并制定強有力的緩解策略。…

[環境配置] 1. 開發環境搭建

開發環境搭建 本文檔將詳細介紹如何搭建深度學習開發環境&#xff0c;包括 Python 環境配置、IDE 選擇與配置以及虛擬環境管理。 也會介紹一下最近比較流行的 uv 工具。它是一個用 Rust 編寫的極其快速的 Python 包和項目管理工具。 uv 是一個非常強大的工具&#xff0c;它可…

rust 同時處理多個異步任務,并在一個任務完成退出

use std::thread; use tokio::{sync::mpsc,time::{sleep, Duration}, };async fn check_for_one() {// 該函數會每秒打印一次 "write"loop {println!("write");sleep(Duration::from_secs(1)).await;} }async fn start_print_task() -> Result<(), (…

“群芳爭艷”:CoreData 4 種方法計算最大值的效率比較(上)

概覽 在 CoreData 支持的 App 中&#xff0c;一種常見操作就是計算數據庫表中指定字段的最大值&#xff08;或最小值&#xff09;。就是這樣一種看起來“不足掛齒”的任務&#xff0c;可能稍不留神就會“馬失前蹄”。 在實際的代碼中&#xff0c;我們怎樣才能既迅速又簡潔的…