• Thứ Ba, 19/06/2007 09:21 (GMT+7)

    Tổ hợp phim và chương trình nền

    Câu 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?



    Trả lời :

    Để ứ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:

    #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);

    4. Chọn menu File.New.Files, chọn loại Text File, nhập tên KeyHook.def vào mục File name, ấn button Ok rồi viết đặc tả thư viện *.dll như sau vào file KeyHook.def:
    LIBRARY   KeyHook
    EXETYPE   WINDOWS
    CODE   PRELOAD MOVABLE
    DATA   PRELOAD SINGLE
    HEAPSIZE  8192
    STACKSIZE 8192
    EXPORTS
       InstallHookKeyboard @2
       UninstallHookKeyboard @3

    5. Chọn tab FileView ở dưới cửa sổ cây Project (thường nằm bên trái màn hình VC++), mở rộng nội dung của nhánh Source Files, nhấn đúp chuột vào file KeyHook.cpp để hiển thị nội dung của nó rồi hiệu chỉnh thành nội dung như sau:

    //------------------------------
    // 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);
    }

    6. Chọn menu Build. Set Active Configuration, chọn mục Win32 Release, ấn button Ok.

    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:

    #include <windows.h>
    //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
    }
    5. Chọn menu Project.Settings, chọn tab Link, nhập thêm tên thư viện KeyHook.lib vào mục Object/Library module (lưu ý không xóa bất kỳ thư viện nào đã có trong mục này) rồi ấn button Ok.

    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.
    Bạn có thể liên hệ với tòa soạn TGVT để copy 2 project trên để tham khảo.

    Chuyên mục: Lập trình