1. Trang chủ
  2. » Công Nghệ Thông Tin

Giáo trình lập trình C cho winform - 4 pot

10 228 0

Đang tải... (xem toàn văn)

THÔNG TIN TÀI LIỆU

Thông tin cơ bản

Định dạng
Số trang 10
Dung lượng 373,52 KB

Nội dung

Bài 3:Các thiết bị nhập liệu Trần Minh Thái Bài 3: CÁC THIẾT BỊ NHẬP LIỆU Phân bố thời lượng: - Số tiết giảng ở lớp: 15 tiết - Số tiết tự học ở nhà: 15 tiết - Số tiết cài đặt chương trình ở nhà: 30 tiết 1. Bàn phím a. Chương trình điều khiển bàn phím (Keyboard.drv) Windows được nạp Keyboard.drv khi khởi động và xử lý phím. Sau đó keyboard.drv chuyển cho USER biến phím nhấn thành message và đưa vào hàng đợi (Hàng đợi hệ thống và hàng đợi chương trình). b. Cửa sổ có focus  Khi cửa sổ có focus thì phát sinh thông điệp WM_SETFOCUS.  Ngược lại phát sinh WM_KILLFOCUS. c. Thông điệp phím MSG msg; while(GetMessage(&msg,NULL,0,0)) { TranslateMessage(&msg); DispatchMessage(&msg); } Thông điệp Nguyên nhân phát sinh WM_ACTIVATE Thông điệp này cùng được gởi đến các cửa sổ bị kích hoạt và cửa sổ không bị kích hoạt. Nếu các cửa sổ này cùng một hàng đợi nhập liệu, các thông điệp này sẽ được truyền một cách đồng bộ, đầu tiên thủ tục Windows của cửa sổ trên cùng bị mất kích hoạt, sau đó đến thủ tục của cửa sổ trên cùng được kích hoạt. Nếu các cửa sổ này không nằm trong cùng một hàng đợi thì thông điệp sẽ được gởi một cách không đồng bộ, do đó cửa sổ sẽ được kích hoạt ngay lập tức. WM_APPCOMMAND Thông báo đến cửa sổ rằng người dùng đã tạo một sự kiện lệnh ứng dụng, ví dụ khi người dùng kích vào button sử dụng chuột hay đánh vào một kí tự kích hoạt một lệnh của ứng dụng. Bài giảng: Lập trình C for Win Trang 31/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái WM_CHAR Thông điệp này được gởi tới cửa sổ có sự quan tâm khi thông điệp WM_KEYDOWN đã được dịch từ hàm TranslateMessage. Thông điệp WM_CHAR có chứa mã kí tự của phím được nhấn. WM_DEADCHAR Thông điệp này được gởi tới cửa sổ có sự quan tâm khi thông điệp WM_KEYUP đã được xử lý từ hàm TranslateMessage. Thông điệp này xác nhận mã kí tự khi một phím dead key được nhấn. Phím dead key là phím kết hợp để tạo ra kí tự ngôn ngữ không có trong tiếng anh (xuất hiện trong bàn phím hỗ trợ ngôn ngữ khác tiếng Anh). WM_GETHOTKEY Ứng dụng gởi thông điệp này để xác định một phím nóng liên quan đến một cửa sổ. Để gởi thông điệp này thì dùng hàm SendMessage. WM_HOTKEY Thông điệp này được gởi khi người dùng nhấn một phím nóng được đăng kí trong RegisterHotKey. WM_KEYDOWN Thông điệp này được gởi cho cửa sổ nhận được sự quan tâm khi người dùng nhấn một phím trên bàn phím. Phím này không phải phím hệ thống (Phím không có nhấn phím Alt). WM_KEYUP Thông điệp này được gởi cho cửa sổ nhận được sự quan tâm khi người dùng nhả một phím đã được nhấn trước đó.Phím này không phải phím hệ thống (Phím không có nhấn phím Alt). WM_KILLFOCUS Thông điệp này được gởi tới cửa sổ đang nhận được sự quan tâm trước khi nó mất quyền này. WM_SETFOCUS Thông điệp này được gởi tới cửa sổ sau khi cửa sổ nhận được sự quan tâm của Windows WM_SETHOTKEY Ứng dụng sẽ gởi thông điệp này đến cửa sổ liên quan đến phím nóng, khi người dùng nhấn một phím nóng thì cửa sổ tương ứng liên quan tới phím nóng này sẽ được kích hoạt. WM_SYSCHAR Thông điệp này sẽ được gởi tới cửa sổ nhận được sự quan tâm khi hàm TranslateMesage xử lý xong thông điệp WM_SYSKEYDOWN. Bài giảng: Lập trình C for Win Trang 32/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái Thông điệp WM_SYSCHAR chứa mã cửa phím hệ thống. Phím hệ thống là phím có chứa phím Alt và tổ hợp phím khác. WM_SYSDEADCHAR Thông điệp này được gởi tới cửa sổ nhận được sự quan tâm khi một thông điệp WM_SYSKEYDOWN được biên dịch trong hàm TranslateMessage. Thông điệp này xác nhận mã kí tự của phím hệ thống deadkey được nhấn. WM_SYSKEYDOWN Thông điệp này được gởi tới cửa sổ nhận được sự quan tâm khi người dùng nhấn phím hệ thống. d. Ví dụ #define BUFSIZE 65535 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 #define SHIFTED 0x8000 LONG APIENTRY MainWndProc(HWND hwndMain, UINT uMsg, WPARAM wParam, LPARAM lParam) { HDC hdc; // handle to device context TEXTMETRIC tm; // structure for text metrics static DWORD dwCharX; // average width of characters static DWORD dwCharY; // height of characters static DWORD dwClientX; // width of client area static DWORD dwClientY; // height of client area static DWORD dwLineLen; // line length static DWORD dwLines; // text lines in client area static int nCaretPosX = 0; // horizontal position of caret static int nCaretPosY = 0; // vertical position of caret static int nCharWidth = 0; // width of a character static int cch = 0; // characters in buffer static int nCurChar = 0; // index of current character static PTCHAR pchInputBuf; // input buffer int i, j; // loop counters int cCR = 0; // count of carriage returns int nCRIndex = 0; // index of last carriage return int nVirtKey; // virtual-key code TCHAR szBuf[128]; // temporary buffer TCHAR ch; // current character PAINTSTRUCT ps; // required by BeginPaint RECT rc; // output rectangle for DrawText SIZE sz; // string dimensions COLORREF crPrevText; // previous text color Bài giảng: Lập trình C for Win Trang 33/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái COLORREF crPrevBk; // previous background color 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 switch (uMsg) { case WM_CREATE: // Get the metrics of the current font. hdc = GetDC(hwndMain); GetTextMetrics(hdc, &tm); ReleaseDC(hwndMain, hdc); // Save the average character width and height. dwCharX = tm.tmAveCharWidth; dwCharY = tm.tmHeight; // Allocate a buffer to store keyboard input. pchInputBuf = (LPTSTR) GlobalAlloc(GPTR, BUFSIZE * sizeof(TCHAR)); return 0; case WM_SIZE: // Save the new width and height of the client area. dwClientX = LOWORD(lParam); dwClientY = HIWORD(lParam); // Calculate the maximum width of a line and the // maximum number of lines in the client area. dwLineLen = dwClientX - dwCharX; dwLines = dwClientY / dwCharY; break; case WM_SETFOCUS: // Create, position, and display the caret when the // window receives the keyboard focus. CreateCaret(hwndMain, (HBITMAP) 1, 0, dwCharY); SetCaretPos(nCaretPosX, nCaretPosY * dwCharY); ShowCaret(hwndMain); break; case WM_KILLFOCUS: // Hide and destroy the caret when the window loses the // keyboard focus. HideCaret(hwndMain); DestroyCaret(); break; case WM_CHAR: switch (wParam) { case 0x08: // backspace case 0x0A: // linefeed case 0x1B: // escape MessageBeep((UINT) -1); return 0; case 0x09: // tab Bài giảng: Lập trình C for Win Trang 34/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái // Convert tabs to four consecutive spaces. 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 for (i = 0; i < 4; i++) SendMessage(hwndMain, WM_CHAR, 0x20, 0); return 0; case 0x0D: // carriage return // Record the carriage return and position the // caret at the beginning of the new line. pchInputBuf[cch++] = 0x0D; nCaretPosX = 0; nCaretPosY += 1; break; default: // displayable character ch = (TCHAR) wParam; HideCaret(hwndMain); // Retrieve the character's width and output // the character. hdc = GetDC(hwndMain); GetCharWidth32(hdc, (UINT) wParam, (UINT) wParam, &nCharWidth); TextOut(hdc, nCaretPosX, nCaretPosY * dwCharY, &ch, 1); ReleaseDC(hwndMain, hdc); // Store the character in the buffer. pchInputBuf[cch++] = ch; // Calculate the new horizontal position of the // caret. If the position exceeds the maximum, // insert a carriage return and move the caret // to the beginning of the next line. nCaretPosX += nCharWidth; if ((DWORD) nCaretPosX > dwLineLen) { nCaretPosX = 0; pchInputBuf[cch++] = 0x0D; ++nCaretPosY; } nCurChar = cch; ShowCaret(hwndMain); break; } SetCaretPos(nCaretPosX, nCaretPosY * dwCharY); break; case WM_KEYDOWN: switch (wParam) { case VK_LEFT: // LEFT ARROW // The caret can move only to the beginning of Bài giảng: Lập trình C for Win Trang 35/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái // the current line. 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 if (nCaretPosX > 0) { HideCaret(hwndMain); // Retrieve the character to the left of // the caret, calculate the character's // width, then subtract the width from the // current horizontal position of the caret // to obtain the new position. ch = pchInputBuf[ nCurChar]; hdc = GetDC(hwndMain); GetCharWidth32(hdc, ch, ch, &nCharWidth); ReleaseDC(hwndMain, hdc); nCaretPosX = max(nCaretPosX - nCharWidth, 0); ShowCaret(hwndMain); } break; case VK_RIGHT: // RIGHT ARROW // Caret moves to the right or, when a carriage // return is encountered, to the beginning of // the next line. if (nCurChar < cch) { HideCaret(hwndMain); // Retrieve the character to the right of // the caret. If it's a carriage return, // position the caret at the beginning of // the next line. ch = pchInputBuf[nCurChar]; if (ch == 0x0D) { nCaretPosX = 0; nCaretPosY++; } // If the character isn't a carriage // return, check to see whether the SHIFT // key is down. If it is, invert the text // colors and output the character. else { hdc = GetDC(hwndMain); nVirtKey = GetKeyState(VK_SHIFT); if (nVirtKey & SHIFTED) { crPrevText = SetTextColor(hdc, RGB(255, 255, 255)); Bài giảng: Lập trình C for Win Trang 36/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái crPrevBk = SetBkColor(hdc, 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 RGB(0,0,0)); TextOut(hdc, nCaretPosX, nCaretPosY * dwCharY, &ch, 1); SetTextColor(hdc, crPrevText); SetBkColor(hdc, crPrevBk); } // Get the width of the character and // calculate the new horizontal position of the caret. GetCharWidth32(hdc, ch, ch, &nCharWidth); ReleaseDC(hwndMain, hdc); nCaretPosX = nCaretPosX + nCharWidth; } nCurChar++; ShowCaret(hwndMain); break; } break; case VK_UP: // UP ARROW case VK_DOWN: // DOWN ARROW MessageBeep((UINT) -1); return 0; case VK_HOME: // HOME // Set the caret's position to the upper left // corner of the client area. nCaretPosX = nCaretPosY = 0; nCurChar = 0; break; case VK_END: // END // Move the caret to the end of the text. for (i=0; i < cch; i++) { // Count the carriage returns and save the // index of the last one. if (pchInputBuf[i] == 0x0D) { cCR++; nCRIndex = i + 1; } } nCaretPosY = cCR; // Copy all text between the last carriage // return and the end of the keyboard input // buffer to a temporary buffer. Bài giảng: Lập trình C for Win Trang 37/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái for (i = nCRIndex, j = 0; i < cch; i++, j++) 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 szBuf[j] = pchInputBuf[i]; szBuf[j] = TEXT('\0'); // Retrieve the text extent and use it // to set the horizontal position of the // caret. hdc = GetDC(hwndMain); GetTextExtentPoint32(hdc, szBuf, lstrlen(szBuf), &sz); nCaretPosX = sz.cx; ReleaseDC(hwndMain, hdc); nCurChar = cch; break; default: break; } SetCaretPos(nCaretPosX, nCaretPosY * dwCharY); break; case WM_PAINT: if (cch == 0) // nothing in input buffer break; hdc = BeginPaint(hwndMain, &ps); HideCaret(hwndMain); // Set the clipping rectangle, and then draw the text // into it. SetRect(&rc, 0, 0, dwLineLen, dwClientY); DrawText(hdc, pchInputBuf, -1, &rc, DT_LEFT); ShowCaret(hwndMain); EndPaint(hwndMain, &ps); break; // Process other messages. case WM_DESTROY: PostQuitMessage(0); // Free the input buffer. GlobalFree((HGLOBAL) pchInputBuf); UnregisterHotKey(hwndMain, 0xAAAA); break; default: return DefWindowProc(hwndMain, uMsg, wParam, lParam); } return NULL; } 2. Thiết bị chuột a. Kiểm tra thiết bị chuột int GetSystemMetrics( Bài giảng: Lập trình C for Win Trang 38/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái int nIndex // system metric or configuration setting ); fMouse = GetSystemMetrics( SM_MOUSEPRESENT ); Giá trị trả về fMouse là TRUE (1) nếu có thiết bị chuột được cài đặt, và ngược lại bằng FALSE (0) nếu thiết bị chuột không được cài đặt vào máy. b. Trong lớp cửa sổ ta định nghĩa con trỏ chuột cho ứng dụng wndclass.hCursor = LoadCursor ( NULL, IDC_ARROR); wndclass.style = CS_HREDRAW|CS_VREDRAW|CS_DBLCLKS; Với thiết bị chuột ta có thể có các hành động như sau:  Kích chuột : nhấn và thả một nút chuột.  Kích đúp chuột : nhấn và thả chuột nhanh (nhấn 2 lần nhanh).  Kéo : di chuyển chuột trong khi vẫn nắm giữ một nút. c. Thông điệp chuột trong vùng làm việc Nút Nhấn Thả Nhấn đúp Trái WM_LBUTTONDOWN WM_LBUTTONUP WM_LBUTTONDBLCLK Giữa WM_MBUTTONDOWN WM_MBUTTONUP WM_MBUTTONDBLCLK Phải WM_RBUTTONDOWN WM_MBUTTONUP WM_RBUTTONDBLCLK d. Giá trị wParam sẽ cho biết trạng thái của nút nhấn, phím Shift, và phím Ctrl. MK_LBUTTON Nút chuột trái nhấn MK_MBUTTON Nút chuột giữa nhấn MK_RBUTTON Nút chuột phải nhấn MK_SHIFT Phím Shift được nhấn MK_CONTROL Phím Ctrl được nhấn e. Giá trị lParam sẽ cho biết vị trí chuột tại thời điểm phát sinh message.  2 bytes thấp: tọa độ x  2 bytes cao: tọa độ y f. Ví dụ Bài giảng: Lập trình C for Win Trang 39/69 Bài 3:Các thiết bị nhập liệu Trần Minh Thái LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 { HDC hdc; static POINT oldPoint; static int iC; int WIDTH_PEN = 2; HPEN oPen,pen; COLORREF Col [ ] ={ RGB (0, 0, 0) , RGB (255 ,0 ,0), RGB (0, 255, 0), RGB (0, 0, 255), RGB (255, 255, 0)}; POINT point; TCHAR str [255]; switch ( message ) // Xử lý thông điệp { case WM_LBUTTONDOWN: /* Vẽ đường thẳng từ vị trí trước đó đến vị trí chuột hiện tại*/ hdc = GetDC ( hWnd ); pen = CreatePen ( PS_SOLID,WIDTH_PEN,Col [ iC] ); oPen = ( HPEN ) SelectObject ( hdc,pen ); point.x = LOWORD ( lParam ); point.y = HIWORD ( lParam ); MoveToEx ( hdc, oldPoint.x, oldPoint.y, NULL ); LineTo ( hdc, point.x, point.y ); oldPoint = point; /* Chọn lại bút vẽ trước đó và hủy bút vẽ vừa tạo*/ SelectObject ( hdc, oPen ); DeleteObject ( pen ); ReleaseDC ( hWnd, hdc ); break; case WM_RBUTTONDOWN: /* Chuyển index của bảng màu sang vị trí tiếp theo, nếu cuối bảng màu thì quay lại màu đầu tiên*/ iC = ( iC+1 ) % ( sizeof ( Col ) / sizeof ( COLORREF ) ); break; case WM_MOUSEMOVE: /* Xuất toạ độ chuột hiện thời lên thanh tiêu đề*/ sprintf ( str,"Toa do chuot x = %d, To do y = %d", LOWORD(lParam), HIWORD(lParam)); SetWindowText ( hWnd, str ); /* Kiểm tra xem có giữ phím chuột trái hay không*/ if ( wParam & MK_LBUTTON ) { hdc = GetDC ( hWnd ); Bài giảng: Lập trình C for Win Trang 40/69 . liệu, c c thông điệp này sẽ đư c truyền một c ch đồng bộ, đầu tiên thủ t c Windows c a c a sổ trên c ng bị mất kích hoạt, sau đó đến thủ t c của c a sổ trên c ng đư c kích hoạt. Nếu c c cửa. Thái COLORREF crPrevBk; // previous background color 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65. 0; // characters in buffer static int nCurChar = 0; // index of current character static PTCHAR pchInputBuf; // input buffer int i, j; // loop counters int cCR = 0; // count of carriage

Ngày đăng: 13/07/2014, 09:20

TỪ KHÓA LIÊN QUAN

w