通過(guò)C#編寫(xiě)一個(gè)簡(jiǎn)易的Windows截屏增強(qiáng)工具
半年前我開(kāi)源了 DreamScene2 一個(gè)小而快并且功能強(qiáng)大的 Windows 動(dòng)態(tài)桌面軟件。有很多的人喜歡,這使我有了繼續(xù)做開(kāi)源的信心。這是我的第二個(gè)開(kāi)源作品 ScreenshotEx 一個(gè)簡(jiǎn)單易用的 Windows 截屏增強(qiáng)工具。
前言
在使用 Windows 系統(tǒng)的截屏快捷鍵 PrintScreen
截屏?xí)r,如果需要把截屏保存到文件,需要先粘貼到畫(huà)圖工具然后另存為文件。以前我還沒(méi)有覺(jué)得很麻煩,后來(lái)使用了 macOS 系統(tǒng)的截屏工具,我才知道原來(lái)一個(gè)小小的截屏工具也可以這么簡(jiǎn)單易用。于是參考 macOS 系統(tǒng)的截屏工具做了一個(gè) Windows 版的。
功能
自動(dòng)保存截屏到桌面
點(diǎn)擊截屏預(yù)覽可以編輯截屏
實(shí)現(xiàn)原理
如果想在按下系統(tǒng)的截屏快捷鍵后做一些事情,能想到的方法應(yīng)該就是如何監(jiān)聽(tīng)鍵盤(pán)事件。WIN32 API 提供的 SetWindowsHookExA 鉤子函數(shù)剛好可以實(shí)現(xiàn)這個(gè)需求,idHook
參數(shù)設(shè)置成 WH_KEYBOARD_LL
時(shí)是低等級(jí)鍵盤(pán)鉤子可以捕獲鍵盤(pán)消息。
SetWindowsHookExA
函數(shù)定義
HHOOK SetWindowsHookExA( [in] int idHook, // 鉤子類(lèi)型 [in] HOOKPROC lpfn, // 鉤子處理函數(shù) [in] HINSTANCE hmod, // 模塊句柄 [in] DWORD dwThreadId // 線(xiàn)程Id );
鍵盤(pán)處理函數(shù)定義
LRESULT CALLBACK LowLevelKeyboardProc( _In_?int ???nCode, _In_?WPARAM wParam, // 鍵盤(pán)消息 _In_?LPARAM lParam // KBDLLHOOKSTRUCT 結(jié)構(gòu)體指針 );
代碼
C#PInvoke定義
const int HC_ACTION = 0; const int WH_KEYBOARD_LL = 13; const int WM_KEYUP = 0x0101; const int WM_SYSKEYUP = 0x0105; const int VK_SNAPSHOT = 0x2C; [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)] public struct KBDLLHOOKSTRUCT { public uint vkCode; public uint scanCode; public uint flags; public uint time; public UIntPtr dwExtraInfo; } [UnmanagedFunctionPointer(CallingConvention.Winapi)] public delegate IntPtr HookProc(int nCode, IntPtr wParam, ref KBDLLHOOKSTRUCT lParam); [DllImport("User32.dll", SetLastError = true, CharSet = CharSet.Auto)] public static extern IntPtr SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hmod, int dwThreadId); [DllImport("User32.dll", SetLastError = true, ExactSpelling = true)] [return: MarshalAs(UnmanagedType.Bool)] public static extern bool UnhookWindowsHookEx(IntPtr hhk); [DllImport("User32.dll", SetLastError = false, ExactSpelling = true)] public static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, ref KBDLLHOOKSTRUCT lParam); [DllImport("Kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)] public static extern IntPtr GetModuleHandle([Optional] string lpModuleName);
注冊(cè)鍵盤(pán)鉤子
需要注意:因?yàn)?nbsp;SetWindowsHookEx
是非托管函數(shù)第二個(gè)參數(shù)是個(gè)委托類(lèi)型,GC
不會(huì)記錄非托管函數(shù)對(duì) .NET
對(duì)象的引用。如果用臨時(shí)變量保存委托出作用域就會(huì)被 GC
釋放,當(dāng) SetWindowsHookEx
去調(diào)用已經(jīng)被釋放的委托就會(huì)報(bào)錯(cuò)。
SetWindowsHookEx
函數(shù)第一個(gè)參數(shù)傳 WH_KEYBOARD_LL
低等級(jí)鍵盤(pán)鉤子、第二個(gè)參數(shù)傳鍵盤(pán)消息處理函數(shù)的委托、第三個(gè)參數(shù)使用 GetModuleHandle
函數(shù)獲取模塊句柄、第四個(gè)參數(shù)傳 0。
HookProc _hookProc; IntPtr _hhook; void StartHook() { _hookProc = new HookProc(LowLevelKeyboardProc); // 使用成員變量保存委托 _hhook = SetWindowsHookEx(WH_KEYBOARD_LL, _hookProc, GetModuleHandle(null), 0); // 注冊(cè)鍵盤(pán)鉤子,保存返回值卸載鉤子時(shí)用到。GetModuleHandle(null) 獲取當(dāng)前模塊句柄 }
鍵盤(pán)消息處理函數(shù)
在鍵盤(pán)消息處理函數(shù)里面捕獲 PrintScreen
按鍵消息,然后顯示預(yù)覽和保存圖片邏輯
IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, ref KBDLLHOOKSTRUCT lParam) { if (nCode == HC_ACTION) { if (lParam.vkCode == VK_SNAPSHOT) // 捕獲 PrintScreen 按鍵消息 { if ((int)wParam == WM_KEYUP || (int)wParam == WM_SYSKEYUP) // 按鍵釋放時(shí)保存圖片 SaveImage(); else _previewWindow.SetHide(); } } return CallNextHookEx(_hhook, nCode, wParam, ref lParam); }
保存圖片
從系統(tǒng)剪貼板獲取圖片
void SaveImage() { if (Clipboard.ContainsImage()) { if (!Directory.Exists(_settings.SavePath)) Directory.CreateDirectory(_settings.SavePath); string ext = "png"; ImageFormat imageFormat = ImageFormat.Png; switch (_settings.SaveExtension) { case 0: imageFormat = ImageFormat.Png; ext = "png"; break; case 1: imageFormat = ImageFormat.Jpeg; ext = "jpg"; break; case 2: imageFormat = ImageFormat.Bmp; ext = "bmp"; break; } if (_settings.SaveName == 0) { string name = DateTime.Now.ToString("yyyy-MM-dd HH.mm.ss"); _saveFilePath = Path.Combine(_settings.SavePath, $"{PrefixName} {name}.{ext}"); } else { do { _saveFilePath = Path.Combine(_settings.SavePath, $"{PrefixName} {_nameIndex}.{ext}"); _nameIndex++; } while (File.Exists(_saveFilePath)); } Image image = Clipboard.GetImage(); image.Save(_saveFilePath, imageFormat); if (_settings.IsPlaySound) _soundPlayer.Play(); if (_settings.IsShowPreview) _previewWindow.SetImage(_saveFilePath); } }
以上就是通過(guò)C#編寫(xiě)一個(gè)簡(jiǎn)易的Windows截屏增強(qiáng)工具的詳細(xì)內(nèi)容,更多關(guān)于C#截屏增強(qiáng)工具的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C# 實(shí)現(xiàn)把double 存成兩位精度小數(shù)
這篇文章主要介紹了C# 實(shí)現(xiàn)把double 存成兩位精度小數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12C#實(shí)現(xiàn)將數(shù)組內(nèi)元素打亂順序的方法
這篇文章主要介紹了C#實(shí)現(xiàn)將數(shù)組內(nèi)元素打亂順序的方法,涉及C#數(shù)組遍歷及隨機(jī)數(shù)操作的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-08-08c# 獲取照片的經(jīng)緯度和時(shí)間的示例代碼
這篇文章主要介紹了c# 獲取照片的經(jīng)緯度和時(shí)間的示例代碼,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下2020-11-11.NET中保證線(xiàn)程安全的高級(jí)方法Interlocked類(lèi)使用介紹
這篇文章主要介紹了.NET中保證線(xiàn)程安全的高級(jí)方法Interlocked類(lèi)使用介紹,Interlocked類(lèi)可以為為多個(gè)線(xiàn)程共享的變量提供原子操作,需要的朋友可以參考下2014-07-07