Hỏi: Xin hướng dẫn viết một chương trình chạy nền, khi nhấn tổ hợp phím thì chạy một đoạn code và nhấn tổ hợp phím khác thì dừng lại. Khi đang chạy một chương trình khác thì làm sao chương trình của tôi nhận được sự kiện nhấn tổ hợp phím?
Đáp: Để ứng dụng có thể nhận và xử lý sự kiện I/O ở mọi tình huống (ngay cả khi ứng dụng khác đang chạy), bạn phải dùng kỹ thuật câu móc (Hooks) hàm xử lý sự kiện tương ứng vào Windows, hàm được câu móc cấp toàn hệ thống phải được đặt trong thư viện liên kết động *.dll. Như vậy ứng dụng xử lý phím nóng (hot-key) của bạn gồm 2 module:
• file thư viện *.dll chứa hàm xử lý phím nóng và các hàm câu móc/gỡ ra.
• file ứng dụng chứa các hàm chức năng mà sẽ được chạy/dừng khi tổ hợp phím tương ứng được ấn.
Sau đây là qui trình cụ thể để xây dựng 2 module bằng ngôn ngữ VC++ của Microsoft.
Để xây dựng thư viện KeyHook.dll chứa hàm chặn tổ hợp phím nóng mong muốn, bạn hãy tiến hành các bước sau đây:
1. Chạy ứng dụng VC++ (phải cài đặt trước), chọn menu File.New.Projects, chọn loại Win32 Dynamic-link Library, chọn vị trí Location, nhập tên Project là KeyHook, ấn button Ok.
2. Chọn checkbox A simple DLL project rồi chọn button Finish để tạo Project thực sự.
3. Chọn menu File.New.Files, chọn loại C/C++ Header File, nhập tên KeyHook.h vào mục File name, ấn button Ok rồi viết đặc tả 2 hàm câu/gỡ hook như sau vào file KeyHook.h:
Code:
#include "stdafx.h"
//hàm câu móc hàm xử lý phím vào Windows
int FAR InstallHookKeyboard(HWND hWnd);
//hàm gỡ hàm xử lý phím ra khỏi Windows
int FAR UninstallHookKeyboard(void);
Code:
KeyHook.def:
LIBRARY KeyHook
EXETYPE WINDOWS
CODE PRELOAD MOVABLE
DATA PRELOAD SINGLE
HEAPSIZE 8192
STACKSIZE 8192
EXPORTS
InstallHookKeyboard @2
UninstallHookKeyboard @3
Code:
//------------------------------
// Nội dung file KeyHook.cpp
//------------------------------
#include "stdafx.h"
#include "KeyHook.h"
//định nghĩa các message cần dùng
#define WM_MYSTART (WM_USER+1)
#define WM_MYEND (WM_USER+2)
//định nghĩa các biến cần dùng
HWND hMyWnd;
int fHookKeyboard;
HANDLE hHookKeyboard;
HINSTANCE hModuleDll;
//------------------------------
//Hàm khởi động của thư viện,
//hàm này được kích hoạt tự động mỗi khi
//thư viện được link với ứng dụng.
//------------------------------
BOOL APIENTRY DllMain(HINSTANCE hModule, ULONG ulReason, PCONTEXT pctx) {
switch (ulReason) {
case DLL_PROCESS_ATTACH:
if (hModuleDll==0) hModuleDll= hModule;
break;
case DLL_PROCESS_DETACH:
UninstallHookKeyboard();
break;
} return TRUE;
}
//------------------------------
// Hàm xử lý sự kiện phím
//------------------------------
LRESULT FAR PASCAL CALLBACK KeyboardProc (int nCode, WPARAM wParam, LPARAM lParam) {
short FAlt,FControl, FShift;
if (nCode >= 0 && nCode != HC_NOREMOVE && lParam >0) {
//xác định trạng thái các phím điều khiển
FShift = GetKeyState(VK_SHIFT);
FAlt = GetKeyState(VK_MENU);
FControl = GetKeyState(VK_CONTROL);
//kiểm tra tổ hợp phím Ctrl-S
if (FControl < 0 && wParam == 'S') {
//gởi thông báo WM_MYSTART về ứng dụng xử lý
SendMessage(hMyWnd, WM_MYSTART,wParam, (LPARAM)lParam);
return CallNextHookEx((struct HHOOK__ *)hHookKeyboard, nCode, wParam, lParam);
}
if (FControl < 0 && wParam == 'E') {
//gởi thông báo WM_MYEND về ứng dụng xử lý
SendMessage(hMyWnd, WM_MYEND,wParam, (LPARAM)lParam);
return CallNextHookEx((struct HHOOK__ *)hHookKeyboard, nCode, wParam, lParam);
}
}
return CallNextHookEx((struct HHOOK__ *)hHookKeyboard, nCode, wParam, lParam);
}
//------------------------------
// Hàm câu móc hàm xử lý keyboard vào Windows
//------------------------------
int FAR InstallHookKeyboard(HWND hWnd) {
if (fHookKeyboard) return 1;
hHookKeyboard = SetWindowsHookEx(WH_KEYBOARD,(HOOKPROC)KeyboardProc,
hModuleDll, 0);
if (hHookKeyboard == NULL) {
MessageBox(NULL,"Can't set Hook KeyboardProc","Error",MB_OK);
return 0;
}
hMyWnd = hWnd;
fHookKeyboard = 1;
return 1;
}
//------------------------------
// Hàm gỡ hàm xử lý keyboard
//------------------------------
int FAR UninstallHookKeyboard(void) {
if (fHookKeyboard==0) return 0;
fHookKeyboard = 0;
return UnhookWindowsHookEx((struct HHOOK__ *)hHookKeyboard);
}
7. Chọn menu Build.Rebuild All để dịch Project thành file thư viện. Nếu bạn nhập đúng các nội dung trên thì quá trình dịch sẽ không có lỗi, trong Project sẽ có thư mục Release, trong thư mục này sẽ có nhiều file được tạo ra, trong đó bạn hãy quan sát 2 file tên là KeyHook.dll và KeyHook.lib, bạn sẽ copy 2 file này vào thư mục ứng dụng sẽ được viết trong giai đoạn 2 sau đây.
Để xây dựng ứng dụng KeyHookDemo xử lý chức năng theo tổ hợp phím nóng, bạn hãy tiến hành các bước sau đây:
1. Chạy ứng dụng VC++, chọn menu File.New.Projects, chọn loại Win32 Application, chọn vị trí Location, nhập tên Project là KeyHookDemo, ấn button Ok.
2. Chọn checkbox A simple application rồi chọn button Finish để tạo Project thực sự.
3. Copy 2 file đặc tả thư viện KeyHook.dll và KeyHook.lib vào thư mục ứng dụng (do Project hiện hành quản lý).
4. Chọn menu File.New.Files, chọn loại C/C++ Source File, nhập tên KeyHookDemo.cpp vào mục File name, ấn button Ok rồi viết đoạn code xử lý sau vào file KeyHookDemo.cpp:
Code:
#include
//khai báo các hàm trong thư viện KeyHook.dll
int FAR InstallHookKeyboard(HWND hWnd);
int FAR UninstallHookKeyboard(void);
//khai báo các thông báo cần xử lý
#define WM_MYSTART (WM_USER+1)
#define WM_MYEND (WM_USER+2)
//-----------------------------------
//Hàm xử lý cửa sổ của ứng dụng
//-----------------------------------
long FAR PASCAL MainWndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) {
switch (message) {
case WM_CREATE:
//xảy ra 1 lần khi cửa sổ được tạo ra, câu móc hàm hook keyboard
if (InstallHookKeyboard(hWnd)==0)
MessageBox(NULL,"Khong cau moc ham xu ly keyboard duoc!!","Error",MB_OK);
break;
case WM_MYSTART:
//viết đoạn code thực hiện chức năng của bạn vào đây
SetWindowText(hWnd,"Bat dau thuc hien chuc nang cua ban");
break;
case WM_MYEND:
//viet doan code dieu khien dung chuc nang cua ban vao day
SetWindowText(hWnd,"Ket thuc thuc hien chuc nang cua ban");
break;
case WM_ENDSESSION:
case WM_CLOSE:
case WM_DESTROY:
//xảy ra khi ứng dụng dừng
//gở bỏ hàm hook
UninstallHookKeyboard();
PostQuitMessage(0);
break;
default:
return (DefWindowProc(hWnd, message, wParam, lParam));
} // Switch message/
return ((long)NULL);
}
//-----------------------------------
//Hàm khởi động ứng dụng
//-----------------------------------
int InitApplication(HINSTANCE hInstance)
{
WNDCLASS wc;
wc.style = NULL; // Class style(s).
wc.lpfnWndProc = MainWndProc;
// Function to retrieve messages
//for windows of this class.
wc.cbClsExtra = 0;
// No per-class extra data.
wc.cbWndExtra = 0;
// No per-window extra data.
wc.hInstance = hInstance;
// Application that owns the class.
wc.hIcon = LoadIcon(hInstance, "ICON_1");
wc.hCursor = LoadCursor((HINSTANCE)NULL, IDC_ARROW);
wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
wc.lpszMenuName = "KeyHookDemo";
// Name of menu resource in .RC file.
wc.lpszClassName = "KeyHookDemo";
// Name used in call to CreateWindow.
// Register the window class
//and return success/failure code.
return (RegisterClass(&wc));
}
//-----------------------------------
//Hàm khởi động ứng dụng
//-----------------------------------
int InitInstance(HINSTANCE hInstance, short nCmdShow) {
HWND hWnd;
nCmdShow = SW_MINIMIZE;
hWnd = CreateWindow(
"KeyHookDemo",
// See RegisterClass() call.
"KeyHookDemo",
// Text for window title bar.
WS_OVERLAPPED|WS_SYSMENU|WS_MINIMIZEBOX, // Window style.
50,30,400,150,
(HWND)NULL,
// Overlapped windows have no parent.
(HMENU)NULL,
// Use the window class menu.
hInstance,
// This instance owns this window.
NULL);
// Pointer not needed.
// If window could not be created, return "failure"
if (!hWnd) return (FALSE);
// Make the window visible; update its //client area; and return "success"
ShowWindow(hWnd, nCmdShow);
// Show the window
return (TRUE);
// Returns the value from PostQuitMessage
}
//-----------------------------------
//Điểm nhập của ứng dụng
//-----------------------------------
int PASCAL WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {
MSG msg; // message
if (hPrevInstance)
// Other instances of app running?
return (FALSE);
if (!InitApplication(hInstance))
// Initialize shared things
return (FALSE);
// Exits if unable to initialize
if (!InitInstance(hInstance, nCmdShow))
return (FALSE);
//lặp chờ nhận và xử lý thông báo từ Windows
while (GetMessage(&msg,(HWND)NULL,(UINT)NULL,(UINT)NULL)) {
TranslateMessage(&msg);
// Translates virtual key codes
DispatchMessage(&msg);
// Dispatches message to window
}
return (msg.wParam);
// Returns the value from PostQuitMessage
}
6. Chọn menu Build.Rebuild All để dịch Project thành file khả thi. Nếu bạn nhập đúng các nội dung trên thì quá trình dịch sẽ không có lỗi.
7. Chọn menu Build.Execute KeyHookDemo để chạy thử ứng dụng, mỗi lần bạn ấn tổ hợp hot-key Ctrl-S và Ctrl-E, hàm hook sẽ phát hiện được và gửi thông báo tương ứng về ứng dụng. Đoạn code xử lý thông báo của ứng dụng sẽ xử lý các thông báo gửi về theo đúng yêu cầu của bạn.
Lưu ý đoạn code của ứng dụng trên đây chỉ là khung sườn, bạn cần hiệu chỉnh lại đoạn code xử lý 2 thông báo WM_MYSTART và WM_MYEND theo yêu cầu riêng của mình.