Lập trình windows

Tài liệu Lập trình windows: 9/13/2011 1 TRƯỜNG ĐẠI HỌC B\CH KHOA H[ NỘI VIỆN CÔNG NGHỆ THÔNG TIN V[ TRUYỀN THÔNG Phạm Ngọc Hưng Bộ môn Kỹ thuật Máy tính hungpn@soict.hut.edu.vn LẬP TRÌNH WINDOWS 9/13/2011 2 Lập trình Windows • Mã số: IT4781 • Thời lượng: 2 (2-1-0-4) – Lý thuyết: 30 tiết – Bài tập: 15 tiết • Học phần tiên quyết: – Ngôn ngữ lập trình C, C++ – Lập trình hướng đối tượng 2Lập trình Windows 9/13/2011 3 Nội dung • Chương 1. Tổng quan lập trình Windows (4LT + 1BT) • Chương 2. Lập trình Windows VC++/MFC (15LT + 6BT) • Chương 3. Một số chủ đề nâng cao – Tạo thư viện liên kết động (DLL) (3LT + 2BT) – Luồng (Thread) (3LT + 2BT) – Windows Socket (2LT + 2BT) • Chương 4. X}y dựng ứng dụng Windows Form (3LT + 2BT) • Bài tập lớn 3Lập trình Windows 9/13/2011 4 Tài liệu tham khảo • 1. Ivor Horton’s, Beginning Visual C++ 2010 • 2. Programming Windows with MFC 2nd • 2. VC++ and MFC Programming 2nd • 3. Sams teach yourself VC++ 6 in 21 days • 4. MSDN • 5. 4Lập trình Windo...

pdf312 trang | Chia sẻ: Khủng Long | Lượt xem: 1225 | Lượt tải: 0download
Bạn đang xem trước 20 trang mẫu tài liệu Lập trình windows, để tải tài liệu gốc về máy bạn click vào nút DOWNLOAD ở trên
9/13/2011 1 TRƯỜNG ĐẠI HỌC B\CH KHOA H[ NỘI VIỆN CÔNG NGHỆ THÔNG TIN V[ TRUYỀN THÔNG Phạm Ngọc Hưng Bộ môn Kỹ thuật Máy tính hungpn@soict.hut.edu.vn LẬP TRÌNH WINDOWS 9/13/2011 2 Lập trình Windows • Mã số: IT4781 • Thời lượng: 2 (2-1-0-4) – Lý thuyết: 30 tiết – Bài tập: 15 tiết • Học phần tiên quyết: – Ngôn ngữ lập trình C, C++ – Lập trình hướng đối tượng 2Lập trình Windows 9/13/2011 3 Nội dung • Chương 1. Tổng quan lập trình Windows (4LT + 1BT) • Chương 2. Lập trình Windows VC++/MFC (15LT + 6BT) • Chương 3. Một số chủ đề nâng cao – Tạo thư viện liên kết động (DLL) (3LT + 2BT) – Luồng (Thread) (3LT + 2BT) – Windows Socket (2LT + 2BT) • Chương 4. X}y dựng ứng dụng Windows Form (3LT + 2BT) • Bài tập lớn 3Lập trình Windows 9/13/2011 4 Tài liệu tham khảo • 1. Ivor Horton’s, Beginning Visual C++ 2010 • 2. Programming Windows with MFC 2nd • 2. VC++ and MFC Programming 2nd • 3. Sams teach yourself VC++ 6 in 21 days • 4. MSDN • 5. 4Lập trình Windows 9/13/2011 5 Chương 1. Tổng quan lập trình Windows • 1.1. Cơ bản về lập trình Windows (1T) • 1.2. Cấu trúc của chương trình Windows (2T) • 1.3. Microsoft Foundation Classes (MFC) • 1.4. Lập trình Windows với VC++ 2008 (1LT + 1BT) 5Lập trình Windows 9/13/2011 6 1.1. Cơ bản về lập trình Windows • Windows: đồ họa trực quan, tài nguyên đa dạng. • Ứng dụng thân thiện với người sử dụng thông qua giao diện đồ họa Windows. • Các ứng dụng có giao diện tương tác giống nhau: – Windows, title bar, toolbar, menu bar, status bar – Thanh thực đơn: File, Edit, Tool, Help – Hộp thoại: thường chứa các điều khiển chung: Edit Control, Button Control, Checkbox. 6Lập trình Windows 9/13/2011 7 Các thành phần của một cửa sổ 7Lập trình Windows 9/13/2011 8 Cơ chế lập trình Event-driven 8Lập trình Windows 9/13/2011 9 Thông điệp (Windows Messages) • Hệ thống (Windows) ghi lại mỗi sự kiện xảy ra (event) trong một thông điệp (message) và đặt trong hàng đợi thông điệp (messages queue) • Thông điệp phát sinh: – Từ hệ thống (Windows OS) – Từ ứng dụng • Ví dụ: click/drag chuột, ấn phím, minimize/maximize/close cửa sổ, thay đổi kích thước cửa sổ, 9Lập trình Windows 9/13/2011 10 Windows Message (cont.) • HĐH Windows đảm nhiệm việc truyền tất cả các thông điệp của ứng dụng vào các cửa sổ khác nhau của ứng dụng đó. • Một chương trình Windows phải chứa một hàm đặc biệt để xử lý các thông điệp gọi là hàm xử lý cửa sổ WinProc (hoặc WindowProc): – Được gọi khi có bất kỳ thông điệp được truyền đến cửa sổ (thông qua con trỏ hàm) – Xử lý các thông điệp đó và trả điều khiển về cho hệ thống. • Nhiều thông điệp trả lại cho Windows xử lý, gọi hàm DefWindowProc( ) 10Lập trình Windows 9/13/2011 11 Windows Message (cont.) 11Lập trình Windows 9/13/2011 12 Windows API • Application Programming Interface: Giao diện lập trình ứng dụng • Cung cấp các hàm truy cập tài nguyên trong hệ thống (Windows) chứa trong các thư viện liên kết động DLL. • Các ứng dụng có thể truy cập đến các hàm API • Các hàm GDI (Graphics Device Interface) cho phép ứng dụng làm việc với nhiều kiểu thiết bị đồ họa. 12Lập trình Windows 9/13/2011 13 Windows Data Types (1) • Định nghĩa nhờ từ khóa typedef trong các tập tin (windef.h; winbase.h, winuser.h được include trong windows.h). • Ví dụ: – typedef int INT; – typedef unsigned int UINT; – char * PSTR • WPARAM (UINT) và LPARAM (LONG) đều 32 bit • LRESULT: kiểu LONG • HANDLE: Một gi| trị 32bits không dấu (unsigned) do HĐH tạo ra để l{m định danh cho 1 đối tượng (file, cửa sổ, vùng nhớ, menu, toolbar). Ví dụ : HWND, HINSTANCE, hPrevInstance 13Lập trình Windows 9/13/2011 14 Windows Data Types (2) Kiểu Ý nghĩa HANDLE Số nguyên 32-bit, định danh. HWND Số nguyên 32-bit, định danh. BYTE Giá trị 8-bit không dấu. WORD Số nguyên 16-bit không dấu. DWORD Số nguyên 32-bit không dấu. UINT Số nguyên không dấu 32-bit. LONG long 32-bit. BOOL Kiểu logic True/False LPSTR Con trỏ chuỗi. (char *) LPCSTR Hằng con trỏ chuỗi. WPARAM 32-bit. LPARAM 32-bit. BSTR Giá trị 32-bit trỏ đến kí tự. LPVOID Con trỏ 32-bit đến một kiểu không xác định. LPTSTR Giống như LPSTR nhưng có thể chuyển sang dạng Unicode và DBCS. LPCTSTR Giống như LPCTSTR nhưng có thể chuyển sang dạng Unicode và DBCS. 14Lập trình Windows 9/13/2011 15 Microsoft style (1) • Tất cả c|c kiểu dữ liệu mở rộng của windows , tên class, tên struct, tên hằng số định sẵn đều được viết in hoa Ví dụ : Tên kiểu dữ liệu : UINT , ATOM , HANDLE, HWND, BYTE,. Tên cấu trúc : MSG, PAINSTRUCT, WNDCLASS ,.... Tên hằng số định sẵn : NULL, IDI_APPLICATION, CS_VREDRAW,.. • Tất cả c|c method (c|c h{m, c|c phương thức) đều được viết hoa (pascal notation). Ví dụ : LoadIcon() LoadCursor() SomeOtherName() 15Lập trình Windows 9/13/2011 16 Microsoft Style (2) • Qui ước: bắt đầu (tiền tố) chữ thường thể hiện kiểu dữ liệu biến. • Ví dụ: szCmdLine là một biến lưu chuỗi nhập từ dòng lệnh, sz là thể hiện cho biến kiểu chuỗi kết thúc ký tự 0, • hInstance và hPrevInstance, trong đó h viết tắt cho kiểu handle. 16Lập trình Windows 9/13/2011 17 Microsoft Style (3) – Tiền tố tên biến Tiền tố Kiểu dữ liệu c char, WCHAR, TCHAR by BYTE n short i int x,y biến lưu tọa độ x, y b BOOL w WORD l long dw DWORD s string sz chuỗi kết thúc bởi kí tự 0 h handle p pointer Lpsz con trỏ dài chuỗi ký tự kết thúc kí tự 0 17Lập trình Windows 9/13/2011 18 Microsoft style (4) - Calling convention • Định nghĩa kiểu h{m gọi bởi Windows OS #define CALLBACK __stdcall #define WINAPI __stdcall #define APIENTRY WINAPI #define APIPRIVATE __stdcall #define PASCAL __stdcall • Thêm vào trước tên hàm để chỉ ra cho compiler rằng hàm này có cách tổ chức phục vụ cho việc gọi đến như thế nào. 18Lập trình Windows 9/13/2011 19 1.2. Cấu trúc của chương trình Windows 19Lập trình Windows 9/13/2011 20 1.2. Cấu trúc của chương trình Windows • Chương trình Windows đơn giản (dùng hàm API) gồm 2 hàm WinMain() và hàm xử lý cửa sổ WinProc(). • WinMain() thực hiện các chức năng: – Tell Windows what kind of window the program requires – Create the program window – Initialize the program window – Retrieve Windows messages intended for the program • WinProc(): xử lý các thông điệp liên quan đến cửa sổ của nó. 20Lập trình Windows 9/13/2011 21 1.2.1. Hàm WinMain() • Hàm chính của ứng dụng trên Windows • Program entry point • hInstance, is a handle to the instance of the program you are writing. • hPrevInstance, is used if your program had any previous instance. If not, this argument can be ignored. • szCmdLine, is a string that represents all items used on the command line to compile the application. • nCmdShow, controls how the window you are building will be displayed. int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow) 21Lập trình Windows 9/13/2011 22 Hàm WinMain (cont.) • Tìm hiểu hàm WinMain mẫu: int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow) { static TCHAR szAppName[] = TEXT("HelloWin"); // tên ứng dụng HWND hwnd; MSG msg; WNDCLASS wndclass; // biến để định danh một cửa sổ /* Ðịnh nghĩa kiểu cửa sổ */ wndclass.style = SC_HREDRAW | CS_VREDRAW; wndclass.lpfnWndProc = WndProc; // Hàm thủ tục cửa sổ wndclass.cbClsExtra = 0; wndclass.cbWndExtra = 0; wndclass.hInstance = hInstance; // Ðịnh danh ứng dụng wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION); wndclass.hCursor = LoadCusor (NULL, IDC_ARROW); wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH); wndclass.lpszMenuName = NULL; // Không có menu 22 9/13/2011 23 Tìm hiểu hàm WinMain mẫu (cont.) wndclass.lpszClassName = szAppName; // tên ứng dụng /* Ðăng ký lớp cửa sổ */ if (!RegisterClass(&wndclass)) return 0; /* Tạo thể hiện lớp cửa sổ đã định nghĩa */ hwnd = CreateWindow(szAppName, // Tên cửa sổ "Hello Program", // Tiêu đề WS_OVERLAPPEDWINDOW, // Kiểu cửa sổ CW_USEDEFAULT, // Toạ độ x CW_USEDEFAULT, // Toạ độ y CW_USEDEFAULT, // Chiều rộng CW_USEDEFAULT, // Chiều dài NULL, // Cửa sổ cha NULL, // Không có menu hInstance, // Ðịnh danh ứng dụng NULL); // Tham số bổ sung /* Hiển thị cửa sổ */ ShowWindow (hwnd, iCmdShow); UpdateWindow (hwnd); /* Chu trình xử lý các thông điệp*/ while (GetMessage (&msg, NULL, 0, 0)) { TranslateMessage (&msg); DispatchMessage (&msg); } return msg.wParam; } 23 Lập trình Windows 9/13/2011 24 Định nghĩa lớp cửa sổ WNDCLASS • Windows cung cấp một cấu trúc WNDCLASS (file winuser.h) gọi là lớp cửa sổ. • Lớp này chứa các thuộc tính tạo thành một cửa sổ typedef struct _WNDCLASS { UINT style; WNDPROC lpfnWndProc; int cbClsExtra; HINSTANCE hInstance; HICON hIcon; HCURSOR hCursor; HBRUSH hbrBackground; LPCSTR lpszMenuName; LPCSTR lpszClassName; } WNDCLASS, *PWNDCLASS; 24Lập trình Windows 9/13/2011 25 Định nghĩa lớp cửa sổ WNDCLASS • Ý nghĩa thuộc tính lớp cửa sổ WNDCLASS: Thuộc tính Ý nghĩa Ghi chú style Kiểu lớp Kết hợp nhiều kiểu giá trị khác nhau bằng toán tử OR. lpfnWndProc Con trỏ đến thủ tục xử lý cửa sổ cbClsExtra Số byte được cấp phát thêm sau cấu trúc window-class Mặc định cbWndExtra Số byte được cấp phát thêm sau một instance của window Mặc định hInstance Định danh chứa thủ tục cửa sổ của lớp window hIcon Định danh của biểu tượng Dùng hàm LoadIcon hCursor Định danh của con trỏ chuột Dùng hàm LoadCursor hbrBackground Định danh của chổi tô nền Dùng hàm GetStockObject lpszMenuName Tên thực đơn Tên thực đơn gắn với cửa sổ, thực đơn này được khai báo trong tập tin tài nguyên. lpszClassName Tên lớp 25 9/13/2011 26 Đăng ký lớp cửa sổ • Sau khi định nghĩa lớp cửa sổ phải đăng ký bằng hàm RegisterClass: ATOM RegisterClass(CONST WNDCLASS * lpWndClass); 26Lập trình Windows 9/13/2011 27 Tạo cửa sổ • Lớp cửa sổ thể hiện đặc tính chung cửa sổ. Cho phép tạo nhiều thể hiện cửa sổ của lớp • Dùng hàm CreateWindow, chỉ định các đặc tính riêng của cửa sổ, và phân biệt nó với các cửa sổ khác tạo ra cùng một lớp. HWND CreateWindow(LPCSTR lpClassName, // Tên lớp cửa sổ đã đăng ký LPCSTR lpwindowName, // Tên của cửa sổ DWORD dwStyle, // Kiểu của cửa sổ int x, // Vị trí ngang ban đầu int y, // Vị trí dọc ban đầu int nWidth, // Ðộ rộng ban đầu int nHeight, // Ðộ cao ban đầu HWND hWndParent, // Ðịnh danh của cửa sổ cha MENU hMenu, // Ðịnh dang của thực đơn INSTANCE hInstance, // Ðịnh danh thể hiện ứng dụng PVOID lpParam // Các tham số ban đầu ); 27Lập trình Windows 9/13/2011 28 Hiển thị cửa sổ • Sau khi tạo cửa sổ định danh cửa sổ HWND • Để xuất hiện cửa sổ gọi hàm ShowWindow BOOL ShowWindow(HWND hWnd, // Ðịnh danh của cửa sổ cần thể hiện int iCmdShow // Trạng thái hiển thị ); • Một số trạng thái hiển thị (iCmdShow): SW_HIDE: Ẩn cửa sổ. SW_MAXIMIZE: Phóng cửa sổ ra toàn bộ màn hình. SW_MINIMIZE: Thu nhỏ thành biểu tượng trên màn hình. SW_RESTORE: Hiển thị dưới dạng chuẩn. 28Lập trình Windows 9/13/2011 29 1.2.2. Hàm xử lý thông điệp WinProc • Có thể có nhiều hơn một hàm xử lý thông điệp • Luôn đi kèm cửa sổ của nó • Giao tiếp với hệ thống (Windows), toàn bộ các thông điệp gửi tới được xử lý qua hàm này • Thường khai báo như sau: • Trong đó tham số đầu tiên là định danh của cửa sổ, tham số thứ 2 là định danh thông điệp, và cuối cùng là 2 tham số WPARAM và LPARAM bổ sung thông tin kèm theo thông điệp. LRESULT CALLBACK WinProc (HWND, UINT, WPARAM, LPARAM); 29Lập trình Windows 9/13/2011 30 Message Loop • Windows duy trì một hàng đợi thông điệp cho mỗi chương trình • Khi sự kiện nhập thông tin xuất hiện, Windows dịch sự kiện thành dạng thông điệp và đưa vào hàng đợi thông điệp của ứng dụng tương ứng. • Ứng dụng nhận thông điệp từ hàng đợi bằng đoạn mã: MSG msg; // Windows message structure while (GetMessage(&msg, NULL, 0,0)) { TranslateMessage (&msg); DispatchMessage (&msg); } 30Lập trình Windows 9/13/2011 31 Message Loop • Trong đó msg là một biến cấu trúc kiểu MSG được định nghĩa trong winuser.h typedef struct tagMSG { HWND hwnd; UINT message; WPARAM wParam; LPARAM lParam; DWORD time; POINT pt; } MSG, *PMSG; • POINT là kiểu tọa độ điểm, định nghĩa trong windef.h typedef struct tagPOINT { LONG x; LONG y; } POINT, *PPOINT; 31 9/13/2011 32 Message Loop • Hàm GetMessage sẽ trả về 0 nếu msg chứa thông điệp có định danh WM_QUIT (0x0012), khi đó vòng lặp thông điệp ngưng và ứng dụng kết thúc. • Ngược lại thì hàm sẽ trả về một giá trị khác 0 với các thông điệp khác. 32Lập trình Windows 9/13/2011 33 Multitasking 33Lập trình Windows 9/13/2011 34 Hàm xử lý thông điệp WinProc - Ví dụ • Tìm hiểu một hàm xử lý cửa sổ: LRESULT CALLBACK WinProc (HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { HDC hdc; PAINTSTRUCT ps; RECT rect; /*Xử lý các thông điệp cần thiết với ứng dụng*/ switch (msg){ case WM_CREATE: /*Viết đoạn mã khi tạo cửa sổ*/ return 0; case WM_PAINT: /*Viết đoạn mã khi tô vẽ lại cửa sổ*/ hdc = BeginPaint (hwnd, &ps); GetClientRect (hwnd, &rect); 34Lập trình Windows 9/13/2011 35 • Thông thường chỉ chặn để xử lý các thông điệp cần thiết, còn lại giao cho hàm xử lý mặc địnhDefWindowProc DrawText(hdc, "Hello", -1, &rect, DT_SINGLELINE| DT_CENTER| DT_VCENTER); EndPaint(hwnd, &ps); return 0; case WM_SIZE: /*Viết đoạn mã khi kích thước cửa sổ thay đổi*/ return 0; case WM_DESTROY: /*Cửa sổ bị đóng*/ PostQuitMessage (0); return 0; } return DefWindowProc (hwnd, msg, wParam, lParam); } 35 9/13/2011 36 Tóm tắt 36Lập trình Windows 9/13/2011 37 1.2.3. Chương trình Windows đơn giản - Chương trình: HelloWorld (Sử dụng trực tiếp Windows API) - 2 hàm chính - Hàm WinMain: thực hiện c|c chức năng - Khai b|o lớp cửa sổ - Đăng ký lớp cửa sổ vừa khai b|o - Tạo v{ hiển thị lớp cửa sổ - Vòng lặp nhận thông điệp v{ gửi tới thủ tục xử lý - Hàm WinProc: H{m xử lý thông điệp gửi đến. C|c thông điệp được Windows xử lý mặc định bằng h{m DefWindowProc() 37Lập trình Windows 9/13/2011 38 Win32 Application in Visual C++ Developer Studio 38Lập trình Windows 9/13/2011 39 HelloWorld (1) /* HELLOWORLD.C */ #include LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM); int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow) { static TCHAR szAppName [] = TEXT ("HelloWorld"); HWND hwnd; MSG msg; WNDCLASS wndclass; wndclass.style = CS_HREDRAW|CS_VREDRAW; wndclass.lpfnWndProc = WndProc; wndclass.cbClsExtra = 0; wndclass.cbWndExtra = 0; wndclass.hInstance = hInstance; wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION); wndclass.hCursor = LoadCursor (NULL, IDC_ARROW); wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH); wndclass.lpszMenuName = NULL; wndclass.lpszClassName = szAppName; if (!RegisterClass (&wndclass)) { MessageBox(NULL, TEXT (" The program requires Windows"), szAppName, MB_ICONERROR); return 0; 39 Lập trình Windows 9/13/2011 40 HelloWorld (2) hwnd = CreateWindow(szAppName, // Tên lớp cửa sổ TEXT (" The Hello World Program"), // Tiêu đề cửa sổ WS_OVERLAPPEDWINDOW, // Kiểu cửa sổ CW_USEDEFAULT, // Toạ độ x CW_USEDEFAULT, // Toạ độ y CW_USEDEFAULT, // Chiều ngang CW_USEDEFAULT, // Chiều dọc NULL, // Cửa sổ cha NULL, // Thực đơn hInstance, // Ðịnh danh NULL); // Tham số ShowWindow (hwnd, iCmdShow); UpdateWindow (hwnd); while (GetMessage (&msg, NULL, 0, 0)) { TranslateMessage (&msg); DispatchMessage (&msg); } return msg.wParam; } // End WinMain 40Lập trình Windows 9/13/2011 41 HelloWorld (3) LRESULT CALLBACK WndProc (HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { HDC hdc; PAINTSTRUCT ps; RECT rect; switch (msg) { case WM_CREATE: return 0; case WM_PAINT: hdc = BeginPaint (hwnd, &ps); GetClientRect (hwnd, &rect); DrawText(hdc, TEXT("Hello World"), -1, &rect, DT_SINGLELINE | DT_CENTER | DT_VCENTER); EndPaint (hwnd, &ps); return 0; case WM_DESTROY: PostQuitMessage (0); return 0; } // End switch return DefWindowProc (hwnd, msg, wParam, lParam); } 41Lập trình Windows 9/13/2011 42 HelloWorld (run) 42Lập trình Windows 9/13/2011 43 Ý nghĩa các hàm trong HelloWorld Tên hàm Ý nghĩa LoadIcon Nạp một biểu tượng để sử dụng trong chương trình. LoadCursor Nạp một con trỏ chuột cho chương trình. GetStockObject Nhận một đối tượng đồ họa, trong trường hợp của chương trình thì lấy một chổi tô để tô lại nền của cửa sổ. RegisterClass Đăng ký một lớp cửa sổ cho cửa sổ ứng dụng trong chương trình. MessageBox Hiển thị một thông điệp. CreateWindow Tạo một cửa sổ dựa trên một lớp cửa sổ. ShowWindow Hiển thị cửa sổ lên màn hình. UpdateWindow Yêu cầu cửa sổ vẽ lại chính bản thân nó. GetMesssage Nhận một thông điệp từ hàng đợi thông điệp. TranslateMessage Dịch thông điệp bàn phím. DispatchMessage Gởi thông điệp đến hàm xứ lý cửa sổ. BeginPaint Khởi tạo chức năng vẽ của cửa sổ. GetClientRect Lấy hình chữ nhật lưu vùng làm việc. DrawText Hiển thị một chuỗi văn bản. EndPaint Kết thúc việc vẽ cửa sổ. PostQuitMessage Đưa thông điệp thoát vào hàng đợi thông điệp. DefWindowProc Thực hiện việc xử lý mặc định các thông điệp. 43 9/13/2011 44 1.3. Microsoft Foundation Classes (MFC) • MFC = Microsoft Foundation Classes • Thư viện c|c lớp đóng gói c|c h{m Win32 API • Tập hợp các lớp C++ (methods, members) và hàm toàn cục • Phát triển ứng dụng Windows dễ dàng, rút ngắn thời gian (so với lập trình Windows API). • Program user interface 44Lập trình Windows 9/13/2011 45 MFC và Win32 API • Win32 API: tập hợp các hàm, cấu trúc, thông điệp, macros và interfaces – Cho phép phát triển ứng dụng Win32 platforms. – Program low-level, tiêu tốn thời gian. • MFC: – Tái sử dụng code (reuse) và phát triển các thư viện. – Đơn giản hóa và tăng tốc phát triển ứng dụng bằng cách cung cấp một tập các lớp C++ đóng gói các đặc điểm của Win32 API 45Lập trình Windows 9/13/2011 46 MFC và Win32 API – Ví dụ • Khái niệm Win32 của một cửa sổ (window) được đóng gói bởi lớp CWnd trong MFC. • CWnd chứa 1 biến thành viên kiểu HWND (kiểu dữ liệu định nghĩa Win32 biểu diễn một cửa sổ). • CWnd chứa các phương thức đóng gói gọi đến các hàm Win32 mà dùng HWND như 1 tham số. • Ví dụ: – Hàm: BOOL ShowWindow(HWND hWnd, int nCmdShow); – Được đóng gói trong MFC: BOOL CWnd::ShowWindow(int nCmdShow); 46Lập trình Windows 9/13/2011 47 MFC Application Framework • Ngoài việc đóng gói của Win32 API, MFC còn định nghĩa một nhóm các lớp để biểu biễn các object ứng dụng chung và thiết lập mối quan hệ trong nhóm này để thực thi các hành vi của ứng dụng. • Application Framework: một nền tảng để xây dựng các ứng dụng. • Dùng MFC Wizard để tạo ra tập hợp các lớp dẫn xuất từ các lớp của framework. • Hoặc xây dựng các lớp mới theo yêu cầu sử dụng 47Lập trình Windows 9/13/2011 48 48Lập trình Windows 9/13/2011 49 Cấu trúc ứng dụng MFC • Ứng dụng MFC cung cấp: – Một lớp biểu diễn ứng dụng (Application Class) – Một lớp biểu diễn cửa sổ chính của ứng dụng (Main Window) – Triển khai h{m WinMain() (được ẩn đi) (WinMain() Function) 49Lập trình Windows 9/13/2011 50 1.4. Lập trình Windows với VC++ 2008 • 3 c|ch tạo ứng dụng Windows với VC++: – Ứng dụng sử dụng trực tiếp Windows API (Win32 Project) – Ứng dụng sử dụng MFC (MFC Application) – Ứng dụng Windows Form (CLR/Windows Form Application) 50Lập trình Windows 9/13/2011 51 Môi trường phát triển tích hợp VC++ (2008) IDE: • The Solution (Workspace) • Output Pane • Editor Area • Menu Bar • Rearranging on the IDE 51Lập trình Windows 9/13/2011 52 VC++ 2008 Developer Studio Environment -Solution Explorer -Class View -Resource View Properties Editor Area Output ToolBox 52Lập trình Windows 9/13/2011 53 The Solution • Solution Explorer: allows you to view and navigate all the files that make up your application. • Class View allows you to navigate and manipulate your source code on a C++ class level. • Resource View: allows you to find and edit each of the various resources in your application, including dialog window designs, icons, and menus. 53Lập trình Windows 9/13/2011 54 Output Pane & Editor Area • Output Pane: – Cung cấp thông tin cho người lập trình – Compiler progress statements – Warnings, error messages – Thông tin debugger: – Current values of variables when step through code. (Cửa sổ Watch) • Editor Area: – Soạn thảo m~ lệnh – Thiết kế giao diện (Design mode) – Icon Editor 54Lập trình Windows 9/13/2011 55 Menu Bar + Rearranging IDE • Menu bar + Tool bars • Standard toolbar • Build toolbar • Text Editor • . • Add/remove Toolbars • Kéo thả, bố trí c|c cửa sổ l{m việc • Floating, Dock, Auto hiden, . 55Lập trình Windows 9/13/2011 56 C|c th{nh phần ứng dụng ph|t triển với VC++ 56 9/13/2011 57 Character Sets, và _T macro (1) • Old Windows: Tập ký tự ANSI 8 bit • Windows NT: Tập ký tự Unicode 16 bit (bao trùm ANSI). • Biên dịch ứng dụng VC++ trên Windows NT có thể chọn sử dụng Unicode hoặc không. • Nếu một chuỗi “Hello” thì trình biên dịch sẽ thể hiện dạng ANSI (8bit/1 ký tự). • Nếu chuỗi L”Hello” thì trình biên dịch sẽ thể hiện dạng chuỗi Unicode. 57Lập trình Windows 9/13/2011 58 Character Sets, và _T macro (2) • Dùng macro _T (của MFC) cho chuỗi dạng _T(“Hello”) thì kết quả sẽ được thể hiện dạng Unicode nếu ký hiệu tiền xử lý _UNICODE được định nghĩa (chọn kiểu chuỗi ký tự Unicode cho ứng dụng), nếu không mặc định l{ ký tự ANSI. 58 9/13/2011 59 Character Sets và _T macro (3) • Kiểu TCHAR (dùng thay cho kiểu char): – L{ wchar_t (Unicode 16 bit) nếu _UNICODE được định nghĩa – L{ char (ANSI 8 bit) nếu _UNICODE không được định nghĩa • Tương tự, nên dùng TCHAR* hoặc LPTSTR (con trỏ chuỗi) thay cho wchar_t*, hoặc char*. • LPCTSTR: con trỏ tới chuỗi hằng • Sử dụng sizeof(TCHAR) để x|c định độ rộng ký tự (8 hay 16 bit) • Sử dụng c|c macro trong thư viện tchar.h thay cho c|c h{m xử lý ký tự/chuỗi của thư viện C-runtime. Ví dụ: _tcscpy thay cho h{m strcpy( ) 59 9/13/2011 60 Exercises Chapter 1. • Ex1-1. HelloWorld – Tạo project Win32 Project như ví dụ lý thuyết – Triển khai h{m WinMain, WinProc, cơ chế thông điệp • Ex1-2. HelloMFC – Tìm hiểu sử dụng IDE Visual studio – Tạo project MFC appllication, sử dụng Wizard – Code view, design view – Compile, debug, run 60Lập trình Windows 9/13/2011 61 Create a project Tên ứng dụng 61Lập trình Windows 9/13/2011 62 Sử dụng Wizard 62 Lập trình Windows Chọn kiểu ứng dụng Sử dụng thư viện Unicode 9/13/2011 63 Thiết kế giao diện ứng dụng (Dialog) 63Lập trình Windows 9/13/2011 64 Add code to greeting (1) • Button “Hello”: Right Mouse/Add Event Handler Chọn sự kiện, tên hàm xử lý 64Lập trình Windows 9/13/2011 65 Add code to greeting (2) void CHelloMFCDlg::OnBnClickedButtonHello() { // TODO: Add your control notification handler code here MessageBox(L"Xin chào ! Lập trình Windows", L"First MFC Application"); } Ví dụ gọi hộp thoại thông báo 65Lập trình Windows 9/13/2011 66 Compile, Build, Debug, Run Compile (Ctrl+F7), Build (F6), Debug (F5), Run without Debug (Ctrl+F5) Set Breakpoint (F9) and Run Debug 9/13/2011 67 Chương 2. Lập trình Windows VC++/MFC • 2.1. \nh xạ thông điệp trong MFC • 2.2. Document/View trong MFC • 2.3. Menu và Toolbar • 2.4. Xử lý vẽ (Drawing) • 2.5. Giao tiếp b{n phím/chuột • 2.6. Hộp thoại (Dialogs) • 2.7. Sử dụng các điều khiển (controls) 67Lập trình Windows 9/13/2011 68 2.1. Message Map trong MFC (1) • Lập trình MFC cho phép viết c|c h{m để xử lý c|c thông điệp (messages) mong muốn. (Ngo{i c|c xử lý mặc định được MFC cung cấp). • Gọi l{ c|c message handlers (hoặc handlers). (cũng l{ h{m th{nh viên của lớp v{ được ph}n biệt bằng tiền tố afx_msg). • Cơ chế liên kết giữa thông điệp v{ h{m xử lý được x|c định bởi Message Map (\nh xạ thông điệp) 68Lập trình Windows 9/13/2011 69 2.1. Message Map trong MFC (2) • Mỗi lớp có xử lý thông điệp sẽ triển khai một Message Map. Sử dụng c|c macro: – DECLARE_MESSAGE_MAP() trong file .h – Cặp BEGIN_MESSAGE_MAP() v{ END_MESSAGE_MAP() trong file .cpp (Cặp n{y bao một danh s|ch c|c h{m xử lý tương ứng với c|c thông điệp) 69Lập trình Windows 9/13/2011 70 2.1. Message Map trong MFC (3) • Xét một ví dụ ứng dụng MFC đơn giản: 70Lập trình Windows 9/13/2011 71 Message Map trong MFC (2) 71Lập trình Windows 9/13/2011 72 Qu| trình xử lý thông điệp trong MFC 72Lập trình Windows 9/13/2011 73 Ph}n loại thông điệp trong MFC Message Category Description Windows Messages Thông điệp chuẩn của Windows, bắt đầu với WM_ (Ngoại trừ c|c thông điệp WM_COMMAND) Ví dụ: WM_PAINT, WM_LBUTTONUP, Control Notification Messages Các thông điệp WM_COMMAND được gửi từ c|c điều khiển (Controls) như Button, CheckBox, đến cửa sổ chứa chúng, hoặc từ cửa sổ con đến cửa sổ cha. Tham số đi kèm (IDs) cho biết gửi từ controls n{o Command Messages Cũng l{ c|c thông điệp WM_COMMAND nhưng được từ c|c th{nh phần giao diện người dùng như menu items, toolbar buttons. (Được MFC định nghĩa c|c IDs chuẩn, người dùng có thể tạo thêm) 73Lập trình Windows 9/13/2011 74 Add Message Handlers • Có thể l{m thủ công (Manual) (Viết trong Message Map). • Hoặc chọn Properties/Message 74Lập trình Windows BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWndEx) ON_WM_CREATE() ON_COMMAND(ID_WINDOW_MANAGER, &CMainFrame::OnWindowManager) ON_COMMAND(ID_VIEW_CUSTOMIZE, &CMainFrame::OnViewCustomize) ON_REGISTERED_MESSAGE(AFX_WM_CREATETOOLBAR, &CMainFrame::OnToolbarCreateNew) ON_WM_CHAR() END_MESSAGE_MAP() 9/13/2011 75 2.2. Document/View trong MFC • Document: Tập hợp dữ liệu của ứng dụng (dẫn xuất từ lớp CDocument) • View: Cơ chế hiển thị dữ liệu trong Document (dẫn xuất từ lớp Cview) 75Lập trình Windows 9/13/2011 76 Document/View trong MFC • Có thể lựa chọn Document Interfaces: – SDI (Single Document Interface): Hiển thị một t{i liệu (Document) mỗi lần trên một cửa sổ (view) – MDI (Multiple Document Interface): Cho phép mở nhiều t{i liệu một lúc • Đều được hợp th{nh bởi c|c th{nh phần CDocument và Cview • Có thể tạo ứng dụng MFC kiểu kiến trúc SDI/MDI 76Lập trình Windows 9/13/2011 77 Khung ứng dụng dạng SDI • Chỉ mở một t{i liệu 77Lập trình Windows 9/13/2011 78 Khung ứng dụng dạng MDI • Cho phép mở nhiều t{i liệu 78Lập trình Windows 9/13/2011 79 Ứng dụng MFC based Document/View • Trong qu| trình tạo ứng dụng dạng SDI v{ MDI, c|c class được tạo ra hầu hết được kế thừa (dẫn xuất) từ c|c class như: – Với dạng SDI: CWinApp, CFrameWnd, CDocument, CView – Với dạng MDI: CWinApp, CMDIFrameWnd, CMDIChildWnd, CDocument, CView 79Lập trình Windows 9/13/2011 80 Ứng dụng MFC based Document/View • Kiểu SDI 80 9/13/2011 81 Mô tả vai trò các class 81Lập trình Windows 9/13/2011 82 Cấu trúc Document/View dạng SDI 82Lập trình Windows 9/13/2011 83 Cấu trúc Document/View dạng MDI 83Lập trình Windows 9/13/2011 84 Sự tương tác giữa Document và View • Với cả 2 dạng SDI v{ MDI, sự liên kết thể hiện thông qua 1 biến pointer kiểu đối tượng lớp CDocument (biến n{y thường có tên l{ pDoc). Thông qua biến n{y, có thể điều khiển thao t|c chuyển thông tin (từ lớp CView sang CDocument) hay lấy thông tin về (từ lớp CDocument sang lớp CView). • Thông thường, với cả ứng dụng dạng SDI hay MDI thì trong lớp CView luôn có h{m GetDocument nhằm hỗ trợ việc lấy biến con trỏ n{y. 84Lập trình Windows 9/13/2011 85 Tương tác Document/View • Ví dụ: void CSDIDrawView::OnDraw(CDC* pDC) { CSDIDrawDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); if (!pDoc) return; //Add draw code here } 85Lập trình Windows 9/13/2011 86 Lớp CDocument Một số phương thức quan trọng: • virtual BOOL OnNewDocument(); • virtual BOOL OnOpenDocument(LPCTSTR lpszPathName); • void SetModifiedFlag(BOOL bModified = TRUE); (Đ|nh dấu: clean/dirty) • BOOL IsModified(); • virtual void OnCloseDocument(); • virtual BOOL OnSaveDocument(LPCTSTR lpszPathName); • CDocument::DeleteContents 86Lập trình Windows 9/13/2011 87 Lớp CView • CDocument giữ nội dung t{i liệu. Để hiển thị, lớp CView tạo ra con trỏ đến lớp CDocument – CDocument* GetDocument() const; • OnDraw: Vẽ cửa sổ View • OnInitialUpdate: Được gọi trước khi một view khởi tạo v{ hiển thị. • OnUpdate: Được gọi khi dữ liệu đ~ thay đổi để cập nhật lại View • Xem MSDN 87Lập trình Windows 9/13/2011 88 C|c lớp View cung cấp sẵn • MFC x}y dựng sẵn c|c lớp View hỗ trợ c|c dạng giao diện hiển thị dữ liệu (chọn dẫn xuất từ lớp base tương ứng) 88Lập trình Windows 9/13/2011 89 Ex2-1. Ứng dụng TextEditor • X}y dựng ứng dụng MFC kiến trúc SDI • Có c|c tính năng soạn thảo đơn giản (Tương tự Notepad) 89Lập trình Windows 9/13/2011 90 App Wizard • X}y dựng ứng dụng kiến trúc SDI 90Lập trình Windows 9/13/2011 91 App Wizard 91Lập trình Windows Chọn lớp CView sẽ kế thừa (base class) 9/13/2011 92 Ex2-2. Ứng dụng Sketcher • Tạo ứng dụng MFC kiến trúc MDI • Sử dụng Wizard: – Leave the default option, Multiple documents, but opt out of Tabbed documents. – Select MFC standard as the project style and Windows Native/Default as the Visual style and colors option. – Keep the Use Unicode libraries option. – Specify the file extension as ske to get a filter for *.ske documents defined. – Change the Generated Classes set of options at their default settings so that the base class for the CSketcherView class is CView. 92Lập trình Windows 9/13/2011 93 App Wizard 93Lập trình Windows 9/13/2011 94 2.3. Menu và Toolbar • 2.3.1. Menu – Một v{i kh|i niệm – Tạo lập Menu – Xử lý – B{i tập 94Lập trình Windows 9/13/2011 95 Menu – Một vài khái niệm • Menu bar: bao gồm nhiều drop-down menu và menu items • Drop-down menu: chứa menu item hoặc drop- down menu khác (File, Edit, ) • Menu items: tương ứng với một lệnh của chương trình. Xác định bằng ID riêng biệt, item ID hay command ID (Open, Save, ) • Pop-up menu: giống drop-down menu nhưng có thể xuất hiện ở vị trí bất kỳ (thường chuột phải) • System menu: chứa các lệnh hệ thống điều khiển cửa sổ (Minimize, maximize, Close) 95Lập trình Windows 9/13/2011 96 Tạo lập menu • Thường có 2 cách chính: – Tạo menu ở dạng resource, load vào khi chạy (Dùng menu editor) – Tạo trực tiếp bằng các hàm (sử dụng lớp CMenu) 96 Lập trình Windows 9/13/2011 97 Sử dụng Menu Editor 97Lập trình Windows 9/13/2011 98 Add Handler for menu messages 98Lập trình Windows 9/13/2011 99 Xử lý lệnh của Menu item • Dựa trên message WM_COMMAND • Định nghĩa message map BEGIN_MESSAGE_MAP(CSketcherDoc, CDocument) ON_COMMAND(ID_COLOR_BLACK, &CSketcherDoc::OnColorBlack) ON_COMMAND(ID_COLOR_RED, &CSketcherDoc::OnColorRed) ON_COMMAND(ID_COLOR_GREEN, &CSketcherDoc::OnColorGreen) ON_COMMAND(ID_COLOR_BLUE, &CSketcherDoc::OnColorBlue) END_MESSAGE_MAP() • Viết h{m xử lý: void CSketcherDoc::OnColorBlack() { // TODO: Add your command handler code here } 99Lập trình Windows 9/13/2011 100 Sử dụng lớp CMenu • BOOL CreateMenu(); • BOOL CreatePopupMenu(); • BOOL LoadMenu(LPCTSTR lpszResourceName); • BOOL LoadMenu(UINT nIDResource); • Thao tác trên Menu: – DeleteMenu: Xo| một đối tượng trên menu – TrackPopupMenu: hiển thị menu popup 100Lập trình Windows 9/13/2011 101 Sử dụng lớp CMenu • Thao tác trên Menu item: – AppendMenu: thêm v{o cuối menu một menuitem. – InsertMenu: thêm một đối tượng menu tại một vị trí x|c định – ModifyMenu: chỉnh sửa một đối tượng menu x|c định. – CheckMenuItem: thêm/bỏ dấu check phía trước một menuitem – EnableMenuItem: thiết lập menuitem c|c trạng th|i enable (MF_ENABLED), disable (MF_DISABLED), gray(MF_GRAYED). 101Lập trình Windows 9/13/2011 102 Popup (Context) Menu • 2 cách: – Thiết kế một menu đặc biệt dùng làm context menu (bỏ qua top level) – Sử dụng một thành phần drop-down menu (submenu) từ menu chính 102Lập trình Windows 9/13/2011 103 Tạo pop-up menu • Hàm xử lý CWnd::OnContextMenu • Hiển thị một menu pop-up nổi – BOOL CMenu::TrackPopupMenu(UINT nFlags, int x, int y, CWnd* pWnd, LPCRECT lpRect=0); • nFlags: cờ vị trí trên màn hình (TPM_CENTERALIGN , TPM_LEFTALIGN , TPM_RIGHTALIGN), cờ chỉ phím nhấn của chuột (TPM_LEFTBUTTON, TPM_RIGHTBUTTON) • Viết hàm xử lý: 103 9/13/2011 104 Tạo lập menu cho dialog • Tạo một tài nguyên menu cho ứng dụng • Chọn menu cần gắn vào dialog trong hộp thoại Dialog Properties. • Hoặc sử dụng h{m CMenu::LoadMenu() 104Lập trình Windows 9/13/2011 105 Ex2-3a. Menu • Tạo lập Menu cho ứng dụng Sketcher trong Ex2-2 có c|c th{nh phần: – Element chứa c|c menu items: Line, Rectangle, Circle, Curve – Color chứa c|c menu items: Black, Red, Green, Blue • (Phục vụ việc vẽ c|c hình với m{u sắc kh|c nhau) 105Lập trình Windows 9/13/2011 106 2.3. Menu và Toolbar (cont.) • 2.3.2. Toolbar – Một số dạng Toolbar – Thiết kế Toolbar resource – Thêm Toolbar v{o ứng dụng – Xử lý thông điệp c|c nút lệnh trên Toolbar – Xử lý một số dạng toolbar đặc biệt 106Lập trình Windows 9/13/2011 107 Một số dạng Toolbar 107Lập trình Windows 9/13/2011 108 Thiết kế Toolbar Resource • Với ứng dụng kiến trúc SDI/MDI: tạo sẵn toolbar mặc định. • Có thể sử dụng Toolbar Editor (Mở Resource/Toolbar) để sửa đổi toolbar đ~ có hoặc tạo toolbar mới. • Có thể sử dụng ảnh bitmap (thêm v{o Resource) để thiết kế toolbar. (Sử dụng h{m LoadBitmap( ) của lớp CToolbar để nạp) 108Lập trình Windows 9/13/2011 109 Sử dụng toolbar editor • Resource/Toolbar 109Lập trình Wndows 9/13/2011 110 Thêm Toolbar vào ứng dụng • Sử dụng lớp CToolBar: (Hoặc CMFCToolBar) – Khai b|o đối tượng lớp – Gọi h{m tạo lập cửa sổ Toolbar – Nạp Toolbar resource đ~ thiết kế – Thiết lập thuộc tính cho toolbar – Gắn toolbar v{o cửa sổ chính 110Lập trình Windows 9/13/2011 111 Khai báo đối tượng lớp CToolBar 111Lập trình Windows 9/13/2011 112 Xử lý trong hàm OnCreate – Minh họa 1 112Lập trình Windows 9/13/2011 113 Xử lý trong h{m OnCreate – Minh họa 2 113Lập trình Windows 9/13/2011 114 Toolbar Style 114Lập trình Windows 9/13/2011 115 Xử lý trong h{m Oncreate – Minh họa 2 (tt) 115Lập trình Windows 9/13/2011 116 Toolbar docking style 116Lập trình Windows 9/13/2011 117 H{m xử lý cho c|c nút lệnh trên toolbar • Nếu toolbar là chức năng từ menu item: Chọn ID cho button trùng ID của menu item tương ứng (tham chiếu đến h{m xử lý của menu items) • Nếu không có menu item tương ứng, thêm viết hàm xử lý thông điệp (vào MESSAGE_MAP). Ví dụ ID_BUTTON_BOLD 117 9/13/2011 118 Xử lý một số dạng toolbar đặc biệt • Có nhóm button dạng checkbox • Có nhóm button dạng radio • Có Combo box trên Toolbar 118Lập trình Windows 9/13/2011 119 Nhóm button dạng checkbox và radio • Dạng checkbox: Cho chọn nhiều button trong 1 nhóm • Dạng radio: Chỉ chọn 1 button trong 1 nhóm • Giải ph|p: Đặt c|c biến cờ trạng th|i, phối hợp xử lý sự kiện UPDATE_COMMAND_UI của c|c button tương tự như với menu items 119Lập trình Windows 9/13/2011 120 Ví dụ xử lý button kiểu check box (1) 120Lập trình Windows 9/13/2011 121 Ví dụ xử lý button kiểu check box (2) 121Lập trình Windows 9/13/2011 122 Ví dụ xử lý button kiểu radio (1) 122Lập trình Windows 9/13/2011 123 Ví dụ xử lý button kiểu radio (2) 123Lập trình Windows 9/13/2011 124 Enable/Disable button 124Lập trình Windows 9/13/2011 125 Combo box trên toolbar • Nguyên lý tạo cửa sổ con (Combox box) gắn v{o cửa sổ cha (Toolbar) 125Lập trình Windows 9/13/2011 126 Khai báo đối tượng combo box 126Lập trình Windows 9/13/2011 127 Gắn combo box vào toolbar 127Lập trình Windows 9/13/2011 128 Ex2-3b. Toolbar • X}y dựng toolbar cho ứng dụng Sketcher. Toolbar triển khai c|c lệnh của menu items chọn Element v{ Color tương ứng. • Modify từ Toolbar có sẵn của ứng dụng MDI 128Lập trình Windows 9/13/2011 129 2.4. Xử lý vẽ (Drawing) • 2.4.1. Cơ bản về vẽ trên Windows • 2.4.2. Truy xuất ngữ cảnh thiết bị • 2.4.3. Một số thao t|c vẽ cơ bản • 2.4.4. Sử dụng c|c đối tượng vẽ 129Lập trình Windows 9/13/2011 130 2.4.1. Cơ bản về vẽ trên Windows • Hệ tọa độ cung cấp cho xử lý vẽ trên cửa sổ ? • Vùng ứng dụng của cửa sổ (client area) ? 130Lập trình Windows 9/13/2011 131 GDI ? • Windows cung cấp GDI (Graphics Device Interface) – giao diện thiết bị đồ họa, giúp thực hiện thao tác vẽ dễ dàng hơn, không phụ thuộc vào phần cứng đồ họa của hệ thống. • Một chương trình ứng dụng (WindowsApp) không vẽ trực tiếp ra màn hình, máy in mà chỉ vẽ trên “bề mặt luận lý” thể hiện bởi ngữ cảnh thiết bị (Device Context – DC). • Ngữ cảnh thiết bị chứa các thông tin về hệ thống, ứng dụng và cửa sổ của WindowsApp cũng như các đối tượng đồ họa đang được vẽ trong WindowApp đó. 131Lập trình Windows 9/13/2011 132 Cơ chế vẽ trong MFC • Ứng dụng MFC (kiến trúc Document/View) chứa lớp dẫn xuất từ lớp CView (quản lý việc hiển thị dữ liệu lên vùng l{m việc (client area) của cửa sổ ứng dụng). • C|c xử lý vẽ trên vùng hiển thị n{y được thực hiện trong hàm OnDraw( ) (h{m override đ|p ứng thông điệp WM_PAINT được gọi bất cứ khi n{o vùng cửa sổ cần được vẽ lại). Minh họa: 132 9/13/2011 133 2.4.2. Truy xuất ngữ cảnh thiết bị (1) • Lớp CDC: – Chứa 1 ngữ cảnh thiết bị (device context – dc) và c|c phương thức cần thiết để xử lý vẽ trong ứng dụng. (~ 100 h{m). – Ví dụ: Vẽ đường (LineTo), vẽ hình chữ nhật (Rectangle), vẽ hình Ellipse, – L{ lớp cơ sở 133Lập trình Windows 9/13/2011 134 Truy xuất ngữ cảnh thiết bị (2) • Ngo{i ra MFC cung cấp một số lớp ngữ cảnh thiết bị chuyên biệt (kế thừa từ lớp CDC): – CPaintDC: Sử dụng cho việc vẽ trong vùng ứng dụng của cửa sổ (chỉ thao t|c với sự kiện OnPaint) – CClientDC: Sử dụng cho việc vẽ trong vùng ứng dụng cửa sổ (vẽ bất kỳ nơi n{o nhưng chỉ thao t|c với sự kiện OnPaint) – CWindowDC: (Sử dụng cho việc vẽ trong cửa sổ, bao gồm cả vùng không l{ vùng ứng dụng cửa sổ) – CMetaFileDC: (Sử dụng cho việc vẽ một GDI metafile) 134Lập trình Windows 9/13/2011 135 Truy xuất ngữ cảnh thiết bị (3) • Để lấy ngữ cảnh thiết bị, khai b|o biến đối tượng lớp, dùng: CDC dc(this); • Hoặc biến con trỏ CDC *pDC = GetDC( ); • Để giải phóng ngữ cảnh thiết bị, dùng: ReleaseDC(pDC); delete pDC; 135Lập trình Windows 9/13/2011 136 Truy xuất ngữ cảnh thiết bị (4) • Tọa độ vẽ hiện thời (current position): – Tọa độ điểm trên vùng vẽ – Quan hệ tương đối, có thể thay đổi khi thực hiện vẽ c|c thực thể (line, circle, rectangle, text) – Có thể thiết lập bằng phương thức của lớp CDC: 136Lập trình Windows CPoint MoveTo(int x, int y); // Move to position x,y CPoint MoveTo(POINT aPoint); // Move to position defined by aPoint 9/13/2011 137 2.4.3. Một số thao t|c vẽ cơ bản (1) • Vẽ đường thẳng (Lines): từ vị trí hiện tại đến vị trí (x,y) • Ví dụ: 137Lập trình Windows BOOL LineTo(int x, int y); // Draw a line to position x,y BOOL LineTo(POINT aPoint); // Draw a line to position defined by aPoint CDC *pDC; pDC->MoveTo(50,50); // Set the current position pDC->LineTo(50,200); // Draw a vertical line down 150 units pDC->LineTo(150,200); // Draw a horizontal line right 100 units pDC->LineTo(150,50); // Draw a vertical line up 150 units pDC->LineTo(50,50); // Draw a horizontal line left 100 units 9/13/2011 138 Một số thao t|c vẽ cơ bản (2) • Vẽ đường tròn (circle): • Vẽ cung (Arc): • Ví dụ: BOOL Ellipse(int x1, int y1, int x2, int y2); BOOL Ellipse(LPCRECT lpRect); BOOL Arc(int x1, int y1, int x2, int y2, int x3, int y3, int x4, int y4); BOOL Arc(LPCRECT lpRect, POINT startPt, POINT endPt); CDC *pDC; pDC->Ellipse(50,50,150,150); // Draw the 1st (large) circle // Define the bounding rectangle for the 2nd (smaller) circle CRect rect(250,50,300,100); CPoint start(275,100); // Arc start point CPoint end(250,75); // Arc end point pDC->Arc(&rect, start, end); // Draw the second circle 9/13/2011 139 Một số thao t|c vẽ cơ bản (3) • CDC::Polygon • CDC::Rectangle • CDC::Draw3dRect • CDC::DrawText • 139Lập trình Windows virtual int DrawText( LPCTSTR lpszString, int nCount, LPRECT lpRect, UINT nFormat ); int DrawText( const CString& str, LPRECT lpRect, UINT nFormat ); 9/13/2011 140 2.4.4. Sử dụng c|c đối tượng vẽ (1) • L{m thế n{o để: – Vẽ c|c đường vẽ có m{u sắc, phong c|ch (style) khác nhau – Tô m{u c|c đối tượng vẽ với m{u sắc, kiểu tô khác nhau – Vẽ chữ với c|c m{u sắc, kiểu c|ch kh|c nhau • Cần sử dụng c|c đối tượng vẽ (bút vẽ, chổi tô m{u, font chữ) 140Lập trình Windows 9/13/2011 141 Sử dụng c|c đối tượng vẽ (2) • Bút vẽ (Pen): dùng để vẽ đường • Chổi vẽ/cọ vẽ (Brush): dùng tô m{u • Font • C|c đối tượng n{y đều được quản lý bởi c|c lớp MFC tương ứng. 141Lập trình Windows 9/13/2011 142 Sử dụng bút vẽ (1) • Lớp CPen • Vẽ đường với kiểu, m{u sắc, độ rộng x|c định • Tạo bút vẽ: 142Lập trình Windows CPen( ); CPen( int nPenStyle, int nWidth, COLORREF crColor ); CPen( int nPenStyle, int nWidth, const LOGBRUSH* pLogBrush, int nStyleCount = 0, const DWORD* lpStyle = NULL ); 9/13/2011 143 Sử dụng bút vẽ (2) • Ví dụ như sau tạo bút vẽ mới v{ chọn l{m bút vẽ hiện thời, dùng: 143Lập trình Windows 9/13/2011 144 Sử dụng bút vẽ (3) • Có thể tạo bút vẽ dùng h{m CPen::CreatePen • Ví dụ minh họa c|ch dùng n{y: 144Lập trình Windows // Declare a pen object and initialize it as // a red solid pen drawing a line 2 pixels wide CPen aPen; aPen.CreatePen(PS_SOLID, 2, RGB(255, 0, 0)); CPen* pOldPen = pDC->SelectObject(&aPen); // Select aPen as the pen pDC->Ellipse(50,50,150,150); // Draw the 1st (large) circle // Define the bounding rectangle for the 2nd (smaller) circle CRect rect(250,50,300,100); CPoint start(275,100); // Arc start point CPoint end(250,75); // Arc end point pDC->Arc(&rect,start, end); // Draw the second circle pDC->SelectObject(pOldPen); // Restore the old pen 9/13/2011 145 Sử dụng cọ vẽ (1) • Lớp Cbrush • Có thể tạo cọ vẽ bằng c|ch dùng h{m tạo, hoặc dùng hàm CreateBrush 145Lập trình Windows CBrush( ); CBrush( COLORREF crColor ); CBrush( int nIndex, COLORREF crColor ); explicit CBrush( CBitmap* pBitmap ); 9/13/2011 146 Sử dụng cọ vẽ (2) • Ví dụ tạo cọ vẽ mới v{ chọn l{m cọ vẽ hiện thời: 146Lập trình Windows 9/13/2011 147 GDI Fonts và Lớp CFont • Ngo{i việc sử dụng font chữ mặc định, người dùng còn có thể tạo font chữ trong chế độ đồ họa tùy chọn. 147Lập trình Windows 9/13/2011 148 Ex2-4. Vẽ trong ứng dụng Sketcher • Thực hiện các thao tác vẽ cơ bản trên ứng dụng Sketcher: – Vẽ đường thẳng, hình tròn, hình chữ nhật – (C|c đối tượng vẽ tại vị trí hiện thời) • Thay đổi bút vẽ (CPen), chổi vẽ (CBrush) khác nhau 148Lập trình Windows 9/13/2011 149 2.5. Giao tiếp với chuột, b{n phím • 2.5.1. Giao tiếp với chuột • 2.5.2. Giao tiếp với b{n phím 149Lập trình Windows 9/13/2011 150 2.5.1. Giao tiếp với chuột • Cần xử lý sự kiện chuột gì ? – Phím chuột tr|i/phải được ấn/nhả (down/up) – Giữ phím tr|i v{ di chuột (mousemove) – Phím chuột giữa, phím chuột mở rộng (nếu có) – Ấn phím chuột kèm phím chức năng (Control, Shift) • Xử lý gì ? (phụ thuộc v{o ứng dụng) • Nơi n{o (lớp n{o) nhận xử lý sự kiện chuột ? • Ví dụ: Với ứng dụng sketcher: – Muốn vẽ đối tượng bắt đầu khi ấn chuột tr|i (down), giữ + di chuột (move), ho{n th{nh khi nhả chuột (up). – C|c đối tượng được vẽ trên lớp CSketcherView, cần bắt sự kiện chuột trên lớp n{y. 150Lập trình Windows 9/13/2011 151 Thông điệp từ sự kiện chuột • Chọn lớp cần xử lý thông điệp từ chuột • Chọn mouse messages muốn xử lý (trong Properties) • Chọn h{m xử lý (message handlers) trong cột bên phải tương ứng message muốn xử lý 151Lập trình Windows 9/13/2011 152 Mouse Message Handlers (1) Thông điệp (Messages) Sự kiện (Events) Hàm xử lý (Handlers) WM_LBUTTONDOWN Nút tr|i chuột được nhấn xuống OnLButtonDown WM_LBUTTONUP Nút tr|i chuột được nhả ra OnLButtonUp WM_LBUTTONDBLCLK Nút tr|i chuột được nhấn kép OnLButtonDblClk WM_MBUTTONDOWN Nút giữa chuột được nhấn xuống OnMButtonDown WM_MBUTTONUP Nút giữa chuột được nhả ra OnMButtonUp WM_MBUTTONDBLCLK Nút giữa chuột được nhấn kép OnMButtonDblClk WM_RBUTTONDOWN Nút phải chuột được nhấn xuống OnRButtonDown WM_RBUTTONUP Nút phải chuột được nhả ra OnRButtonUp WM_RBUTTONDBLCLK Nút phải chuột được nhấn kép OnRButtonDblClk WM_MOUSEMOVE Con chuột được di chuyển OnMouseMove 152Lập trình Windows 9/13/2011 153 Mouse Message Handlers (2) • C|c h{m xử lý thông điệp có dạng: • Ví dụ h{m xử lý thông điệp nút chuột tr|i được nhấn trên vùng cửa sổ hiển thị • Lập trình xử lý hoặc gọi đến handlers tương ứng của lớp cơ sở. 153Lập trình Windows void CSketcherView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CView::OnLButtonDown(nFlags, point); } afx_msg void OnMsgName(UINT nFlags, CPoint point) 9/13/2011 154 Mouse Message Handlers (3) • 2 tham số: – CPoint point: Tọa độ chuột lúc xảy ra sự kiện – UINT nFlags: Cờ (32 bit) cho biết nút chuột được nhấn v{ nhận biết c|c phím Control/Shift nếu được bấm kèm. – Có c|c gi| trị hằng mặt nạ (Mask) được định nghĩa phục vụ cho x|c định nFlags n{y. 154Lập trình Windows 9/13/2011 155 nFlags • Để nhận biết gi| trị cờ n{y, dùng phép & (bitwise) với mặt nạ. Ví dụ nhận biết phím Ctrl có được nhấn cùng chuột: 155Lập trình Windows if ((nFlags & MK_CONTROL) == MK_CONTROL) {} 9/13/2011 156 Ví dụ giao tiếp với chuột (1) void CVdMouseView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default MyDrawFunction(point); CView::OnLButtonDown(nFlags, point); } void CVdMouseView::MyDrawFunction(CPoint point) { CDC *pDC = GetDC(); if((m_PrevPoint.x == -1)&&(m_PrevPoint.y == -1)) pDC->MoveTo(point); else { pDC->MoveTo(m_PrevPoint); pDC->LineTo(point); } m_PrevPoint = point; } 9/13/2011 157 Ví dụ giao tiếp với chuột (2) 157Lập trình Windows void CVdMouseView::OnMouseMove(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CDC *pDC = GetDC(); CString szTextOut; szTextOut.Format("Current point(%d, %d)", point.x, point.y); pDC->TextOut(0, 0, szTextOut); if((nFlags & MK_CONTROL)==MK_CONTROL) MyDrawFunction(point); CView::OnMouseMove(nFlags, point); } 9/13/2011 158 2.5.2. Giao tiếp với b{n phím 158Lập trình Windows • Loại sự kiện phím: – Phím thông thường được nhấn xuống/nhả lên (down/up) – Phím chức năng được nhấn xuống/nhả lên – Phím được nhấn v{ giữ (hold), đọc được nhiều lần. – Tổ hợp phím được nhấn (nhấn kèm c|c phím chức năng: Alt, Ctrl, Shift) • Mục tiêu: bắt được c|c sự kiện về phím, đọc m~ phím để có xử lý thích hợp 9/13/2011 159 Xử lý thông điệp từ b{n phím (1) • MFC hỗ trợ giao tiếp b{n phím bởi c|c thông điệp v{ h{m xử lý tương ứng như sau: 159Lập trình Windows Thông điệp (Messages) Sự kiện (Events) Hàm xử lý (Handlers) WM_KEYDOWN Phím thường được nhấn xuống OnKeyDown WM_KEYUP Phím thường được nhả ra OnKeyUp WM_SYSKEYDOWN Phím chức năng được nhấn xuống OnSysKeyDown WM_SYSKEYYP Phím chức năng được nhả ra OnSysKeyUp 9/13/2011 160 Xử lý thông điệp từ b{n phím (2) • C|c h{m xử lý thông điệp phím có dạng: • Ví dụ: 160Lập trình Windows afx_msg void OnMsgName(UINT nChar, UINT nRepCnt, UINT nFlags) void CSketcherView::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags) { // TODO: Add your message handler code here and/or call default CView::OnKeyDown(nChar, nRepCnt, nFlags); } 9/13/2011 161 Xử lý thông điệp từ b{n phím (3) • Ngo{i ra có thể dùng h{m: • C|c tham số: – nChar: x|c định m~ phím được ấn (virtual key code). (Được định nghĩa trong Winuser.h) – nRepCnt: Số lần lặp (khi giữ phím) – nFlags: x|c định m~ scan code, trạng th|i phím trước, phím nhấn/nhả, – Ví dụ: • Bit 13 (Context Code) =1 nếu phím Alt được nhấn kèm • Bit 15 (Key-transition) = 0 nếu l{ nhấn (pressed), =1 nếu l{ nhả (released) 161Lập trình Windows afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags) 9/13/2011 162 nChar (virtual key code) – Ví dụ 162Lập trình Windows 9/13/2011 163 Ví dụ giao tiếp b{n phím • Thực hiện thay đổi con trỏ tương ứng với các phím chọn • Xử lý sự kiện: OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags) 163Lập trình Windows 9/13/2011 164 2.6. Hộp thoại (Dialogs) • 2 loại: – Dialog (Hộp thoại thông thường/tùy biến) – Common Dialog (C|c hộp thoại chuyên dụng) 164Lập trình Windows 9/13/2011 165 2.6.1. Hộp thoại (Dialogs) • Dialog box: – L{ một cửa sổ (window) – Ứng dụng thường dùng dialogs chứa c|c điều khiển (controls) (bản chất cũng l{ cửa sổ) • 2 yếu tố của 1 dialog trong ứng dụng MFC: – Diện mạo vật lý của dialog (được định nghĩa trong file resource) – Đối tượng lớp dialog (dùng để quản lý c|c hoạt động của dialog v{ c|c điều khiển đặt trên nó) • MFC cung cấp lớp cơ sở CDialog, có thể sử dụng lớp dẫn xuất từ nó cho c|c dialog sau khi đ~ tạo ra resource (diện mạo thiết kế). 165Lập trình Windows 9/13/2011 166 Thiết kế dialog (resource) • Mở Resource View, cho phép: – Thiết kế 1 dialog trên vùng l{m việc – Lấy c|c điều khiển (control) từ ToolBox – Tạo dialog mới bằng Insert Dialog • Định danh dialog: IDD_DIALOG 166 9/13/2011 167 Lập trình cho dialog (1) • Gồm 2 khía cạnh cần giải quyết: – Hiển thị (gọi) dialog đó lên – Lập trình xử lý sự kiện cho c|c điều khiển đặt trên dialog đó 167Lập trình Windows 9/13/2011 168 Lập trình cho dialog (2) • Gắn lớp quản lý cho dialog sau khi đ~ thiết kế resource bằng c|ch chọn Add Class (chọn chuột phải trên dialog cần add) 168Lập trình Windows 9/13/2011 169 Lớp quản lý dialog – minh họa 169Lập trình Windows // CPenDialog dialog class CPenDialog : public CDialogEx { DECLARE_DYNAMIC(CPenDialog) public: CPenDialog(CWnd* pParent = NULL); // standard constructor virtual ~CPenDialog(); // Dialog Data enum { IDD = IDD_PENWIDTH_DLG }; protected: virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support DECLARE_MESSAGE_MAP() }; 9/13/2011 170 Modal và Modaless Dialogs • Có 2 loại dialogs: kiểu modal v{ modaless (kh|c nhau về c|ch hoạt động) • Modal dialog: – Khi hiển thị l{m tất cả c|c cửa sổ kh|c của ứng dụng sẽ treo cho đến khi nó được đóng (thường l{ chọn OK, Cancel) – Tạo bằng c|ch khai b|o đối tượng lớp quản lý dialog, v{ gọi h{m DoModal(). • Modeless dialog: (ngược lại) – Khi hiển thị nó, vẫn có thể chọn v{o c|c cửa sổ kh|c của ứng dụng. – Có thể tạo bằng c|ch dùng h{m CDialog::Create 170Lập trình Windows 9/13/2011 171 Hiển thị dialog • X|c định khi n{o muốn hiển thị dialog (phụ thuộc v{o ứng dụng). Ví dụ: khi chọn một menu item, click v{o một nút lệnh, • Viết m~ lệnh gọi hiển thị dialog đ~ chuẩn bị tại nơi đ~ x|c định. Minh họa: • Chú ý: Cần #include header file của lớp quản lý v{o file m~ nguồn chứa khai b|o đối tượng lớp quản lý dialog. 171Lập trình Windows // Handler for the pen width menu item void CSketcherDoc::OnPenWidth() { CPenDialog aDlg; // Create a local dialog object aDlg.DoModal(); // Display the dialog as modal } 9/13/2011 172 Đóng (close) dialog • Nút lệnh OK, Cancel • H{m ảo (virtual) tương ứng: – CDialog::OnOK() trả về ID nút OK (mặc định IDOK) – CDialog::OnCancel() trả về ID nút Cancel (IDCANCEL) • Có thể thiết kế, Override lại h{m xử lý sự kiện click c|c nút n{y để thực hiện những h{nh xử thích hợp 172Lập trình Windows 9/13/2011 173 Hàm MessageBox (1) • nType: what buttons display ? 173Lập trình Windows 9/13/2011 174 Hàm MessageBox (2) • nType: button OR Icon 174Lập trình Windows 9/13/2011 175 2.6.2. Common Dialogs (1) • MFC cung cấp c|c hộp thoại chuyên dụng hỗ trợ những thao t|c phổ biến trong c|c ứng dụng trên Windows: chọn Font, chọn m{u, Open/Save file, In ấn, • C|c loại hộp thoại n{y đều có lớp MFC quản lý 175Lập trình Windows 9/13/2011 176 Common Dialogs (2) • Nguyên tắc gọi các dialogs chuyên dụng: – Tạo đối tượng thuộc lớp Dialog cần dùng (cung cấp các thông tin cơ bản) – Gọi hàm DoModal() để hiển thị – Lấy các giá trị quan tâm từ đối tượng đ~ khai báo (Sau khi tương t|c với người dùng) 176Lập trình Windows 9/13/2011 177 CFileDialog (1) • Lớp đóng gói làm việc với hộp thoại tập tin thông dụng của Windows (Open, Save As, ) • H{m khởi tạo CFileDialog::CFileDialog • Khởi tạo một đối tượng hộp thoại tập tin Windows chuẩn. 177Lập trình Windows CFileDialog( BOOL bOpenFileDialog, LPCTSTR lpszDefExt = NULL, LPCTSTR lpszFileName = NULL, DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, LPCTSTR lpszFilter = NULL, CWnd* pParentWnd = NULL ); 9/13/2011 178 CFileDialog (2) • bOpenFileDialog: – TRUE: Open – FALSE: Save As • lpszDefExt: – Đuôi mở rộng mặc định của tên tập tin. Nếu người dùng không nhập kèm đuôi mở rộng trong ô edit box của File name, phần mở rộng được chỉ định bởi lpszDefExt sẽ được tự động thêm v{o tên của tập tin. Nếu gi| trị n{y l{ NULL, sẽ không có phần mở rộng được thêm v{o. • lpszFileName: – Tên tập tin khởi tạo sẽ được xuất hiện trong ô edit box của File name. Nếu gi| trị n{y l{ NULL, sẽ không có tên tập tin n{o xuất hiện. 178Lập trình Windows 9/13/2011 179 CFileDialog (3) • dwFlags: – Một phép kết hợp của một hay nhiều biến cờ cho phép tùy chỉnh hộp thoại. Khi chỉnh sửa biến th{nh viên cấu trúc m_ofn.Flags, dùng một to|n tử OR trong những thay đổi để giữ những thiết lập mặc định không bị ảnh hưởng. • lpszFilter: – Một d~y c|c cặp chuỗi chỉ định những bộ lọc bạn có thể |p dụng cho tập tin. Nếu bạn chỉ định bộ lọc tập tin, chỉ những tập tin có trong bộ lọc mới xuất hiện trong list box. • pParentWnd: – Một con trỏ trỏ tới đối tượng hộp thoại tập tin l{ cha hay cửa sổ chủ. 179Lập trình Windows 9/13/2011 180 Các phương thức lớp CFileDialog • CFileDialog::DoModal() - H{m trả về IDOK hay IDCANCEL. (tương ứng người dùng chọn nút OK hay nút Cancel). - Khi người dùng click nút OK hay Cancel, hay chọn chức năng Close từ đều khiển menu của hộp thoại, quyền điều khiển được trao lại cho ứng dụng. Sau đó, có thể gọi c|c h{m th{nh viên kh|c để nhận về c|c thiết lập hay c|c thông tin m{ người dùng nhập v{o hộp thoại. - DoModal l{ một h{m ảo kế thừa từ lớp CDialog. 180Lập trình Windows 9/13/2011 181 Các phương thức lớp CFileDialog • GetFileExt: lấy phần mở rộng của tập tin được chọn • GetFileName: lấy tên tập tin được chọn • GetFileTitle: lấy phần tên (không có phần mở rộng) của tập tin được chọn • GetFolderPath: lấy thư mục của tập tin được chọn • GetPathName: lấy đường dẫn của tập tin được chọn 181Lập trình Windows 9/13/2011 182 CFileDialog - Open 182Lập trình Windows TCHAR szFilters[] = _T("Text files(*.txt)¦*.txt¦All files *.*)¦*.*¦¦"); CFileDialog dlg(TRUE,_T("txt"),_T("*.txt"),OFN_FILEMUSTEXIST¦OFN_HIDEREADONLY,z Filters); if(dlg.DoModal() == IDOK) { filename = dlg.GetPathName(); // Open the file and read it. } 9/13/2011 183 CFileDialog – Save As 183Lập trình Windows void CChildView::OnFileSaveAs() { CFileDialog dlg(FALSE,_T("phn"),m_strPathName, OFN_OVERWRITEPROMPT¦OFN_PATHMUSTEXIST ¦OFN_HIDEREADONLY, m_szFilters); if(dlg.DoModal() == IDOK) { if(SaveFile(dlg.GetPathName())) m_strPathName = dlg.GetPathName(); } 9/13/2011 184 CColorDialog • Constructor: – CColorDialog( COLORREF clrInit = 0, DWORD dwFlags = 0, CWnd* pParentWnd = NULL ) • Methods: – virtual int DoModal() – COLORREF GetColor() const – static COLORREF* PASCAL GetSavedCustomColors – protected virtual BOOL OnColorOK() – void SetCurrentColor( COLORREF clr ) 184Lập trình Windows 9/13/2011 185 CColorDialog 185Lập trình Windows CColorDialog dlg(RGB(255, 0, 0), CC_FULLOPEN); if(dlg.DoModal() == IDOK) { /////////////////////////////////// AfxMessageBox("Chon mau", 0, 0); m_oColor = dlg.GetColor(); } 9/13/2011 186 CFontDialog • H{m tạo/Constructor: – CFontDialog( const CHARFORMAT& charformat, DWORD dwFlags = CF_SCREENFONTS, CDC* pdcPrinter = NULL, CWnd* pParentWnd = NULL ); 186Lập trình Windows 9/13/2011 187 CFontDialog • Parameters: – lplfInitial: A pointer to a LOGFONT data structure that allows you to set some of the font's characteristics. – charFormat: A pointer to a CHARFORMAT data structure that allows you to set some of the font's characteristics in a rich edit control. – dwFlagsSpecifies one or more choose-font flags. One or more preset values can be combined using the bitwise OR operator. If you modify the m_cf.Flags structure member, be sure to use a bitwise OR operator in your changes to keep the default behavior intact. For details on each of these flags, see the description of the CHOOSEFONT structure in the Platform SDK. – pdcPrinter: A pointer to a printer-device context. If supplied, this parameter points to a printer-device context for the printer on which the fonts are to be selected. – pParentWnd: A pointer to the font dialog box's parent or owner window. 187Lập trình Windows 9/13/2011 188 2.7. C|c điều khiển thông dụng • 2.7.1. Các điều khiển thông dụng • 2.7.2. Thiết kế c|c điều khiển trên hộp thoại • 2.7.3. C|c vấn đề chính với c|c điều khiển • 2.7.4. Lớp MFC của c|c điều khiển • 2.7.5. B{i tập 188Lập trình Windows 9/13/2011 189 2.7.1. Các điều khiển thông dụng • Static text • Edit box • Command button • Check box • Radio button • Drop-down list box (combo box) • Sliders, tree and list controls, progress bars, Click & Drag/Drop 189Lập trình Windows 9/13/2011 190 2.7.2. Thiết kế c|c điều khiển trên hộp thoại (1) • Lựa chọn điều khiển muốn dùng trong Toolbox. Thao t|c click hoặc kéo thả. • Thiết kế c|c điều khiển đặt trên dialog (sắp xếp, căn chỉnh) 190Lập trình Windows 9/13/2011 191 Thiết kế điều khiển trên hộp thoại (2) • Thiết lập c|c thuộc tính cho điều khiển tại thời điểm thiết kế. • Cửa sổ Properties của điều khiển 191Lập trình Windows 9/13/2011 192 • Sử dụng Tab Order: – Format/Tab Order (Ctrl+D): Thiết lập thứ tự chọn (focus) bằng phím Tab cho các điều khiển – Click chuột để chọn thứ tự. 192 Thiết kế c|c điều khiển trên hộp thoại (3) 9/13/2011 193 • Sử dụng Mnemonics: – Dùng để thao t|c với điều khiển bằng phím tắt (Alt + ?) – Ký tự gạch chân trong caption của điều khiển • Ví dụ: – Button: “Show &Message” Alt+M • Kiểm tra: Format/Check Mnemonic 193Lập trình Windows Thiết kế c|c điều khiển trên hộp thoại (3) 9/13/2011 194 2.7.3. C|c vấn đề chính với c|c điều khiển (1) • Dữ liệu được lưu trên c|c điều khiển là gì ? (Cần lấy/input hay hiện/output dữ liệu) • Lấy/hiện dữ liệu của c|c điều khiển bằng cách nào ? • Dữ liệu lấy v{o từ điều khiển có hợp lệ ? • Loại sự kiện gì cần dùng khi tương t|c với điều khiển ? (h{m xử lý sự kiện cần dùng) • Có nhu cầu thao t|c vật lý đối với điều khiển đó không ? (Thay đổi chữ, di chuyển, enable/disable, show/hidden, ) 194Lập trình Windows 9/13/2011 195 C|c vấn đề chính với c|c điều khiển (2) • Để lấy/đưa dữ liệu từ/đến c|c điều khiển cần thông qua biến quản lý (loại Value) được đăng ký (gắn) với điều khiển. • Cần sử dụng h{m cho phép đồng bộ dữ liệu giữa gi| trị biến nắm giữ v{ nội dung hiển thị trên điều khiển (khi tương t|c với người dùng) • Để thực hiện thao t|c vật lý với điều khiển cần gọi c|c phương thức thích hợp (của lớp cửa sổ). Thông qua biến loại Control được đăng ký với điều khiển. • Khai b|o v{ viết h{m xử lý sự kiện cho điều khiển 195Lập trình Windows 9/13/2011 196 Đăng ký biến cho điều khiển (1) • Có 2 loại biến có thể gắn với điều khiển tùy mục đích: – Để tham chiếu đến điều khiển (thay đổi diện mạo) Dùng biến loại Control – Để nhận/đưa dữ liệu từ/đến điều khiển Dùng biến loại Value • Thực hiện: – Chọn điều khiển muốn đăng ký biến – Format/Add variable Hoặc Right Mouse/Add variable 196Lập trình Windows 9/13/2011 197 Đăng ký biến cho điều khiển (2) Chọn loại biến: Control/Value Chú thích xuất hiện trong code Tên, kiểu dữ liệu của biến 197Lập trình Windows Tên biến 9/13/2011 198 Đồng bộ dữ liệu giữa điều khiển/biến • Mục đích: – Đồng bộ giá trị của một điều khiển với biến loại giá trị của nó – Cập nhật khi cần nhận dữ liệu từ điều khiển hoặc đưa dữ liệu ra điều khiển. • Thực hiện bằng h{m: – bSaveAndValidate = TRUE (1) : Dữ liệu trên điều khiển được cập nhật v{o biến. – bSaveAndValidate = FALSE (0) : Dữ liệu lưu giữ trong biến được đưa ra điều khiển. BOOL UpdateData(BOOL bSaveAndValidate) 198Lập trình Windows 9/13/2011 199 Đăng ký hàm xử lý sự kiện (1) • Có nhiều sự kiện khác nhau với c|c điều khiển – Mouse down/up, DBClick, Move, – Key down/up, pressed, – Init, create, paint, • Thêm h{m xử lý sự kiện mong muốn – C1: Right Mouse/ Add Event Handler – C2: Properties Window/ Control Events 199Lập trình Windows 9/13/2011 200 Đăng ký h{m xử lý sự kiện (2) • Ví dụ: Button ShowMSG Add Event Handler Chọn sự kiện 200Lập trình Windows Chọn lớp nhận thông điệp tương ứng Tên h{m xử lý được gợi ý 9/13/2011 201 Thao t|c vật lý với c|c điều khiển • Cho phép/không cho phép (enable/disable) tương t|c với điều khiển. BOOL CWnd::EnableWindow(BOOL bEnable) • Ẩn/Hiện (Hidden/Show) điều khiển BOOL CWnd::ShowWindow(int iCmdShow) • Hàm: CWnd* CWnd::GetDlgItem(int nID) //Lấy con trỏ đến điều khiển/cửa sổ con trong cửa sổ cha. Ví dụ 201Lập trình Windows CEdit* pBoxOne; pBoxOne = (CEdit*)GetDlgItem(IDC_MYEDIT); 9/13/2011 202 2.7.4. Lớp MFC của c|c điều khiển • MFC cung cấp lớp đại diện cho c|c điều khiển. • Một số lớp ví dụ: • Bản chất c|c điều khiển (controls) cũng l{ c|c cửa sổ đặc biệt. C|c lớp của điều khiển dẫn xuất từ lớp cơ sở CWnd. Do đó có thể sử dụng đến c|c h{m thích hợp của lớp n{y. • Ngo{i tạo điều khiển lúc thiết kế (Resource), có thể tạo điều khiển bằng h{m Create (lớp cơ sở CWnd) lúc thực thi. 202Lập trình Windows 9/13/2011 203 2.7.4.1. Lớp CEdit (1) – Khởi tạo • Quản lý hộp nhập, xử lý hoạt động nhập liệu, thông tin về tình hình nhập liệu cho cửa sổ cha. • H{m tạo: CEdit() • H{m Create: Khởi tạo đối tượng hộp nhập với c|c tham số. 203Lập trình Windows BOOL Create ( DWORD dwStyle, // kiểu hộp nhập (ES_MULTILINE, ES_PASSWORD, ES_READONLY) const RECT& rect, // Tọa độ v{ kích thước hộp nhập CWnd* pParentWnd, // Con trỏ tới đối tượng cửa sổ cha UINT nID // Định danh hộp nhập ); 9/13/2011 204 Lớp CEdit (2) – Hàm thành viên • 3 nhóm hàm thành viên: – CEdit Attributes: Nhóm h{m về thuộc tính của hộp nhập. (kích thước, vị trí, số dòng, undo, ) – CEdit Operations: Nhóm h{m về hoạt động của hộp nhập – CEdit Clipboard Operations: Nhóm h{m về thao t|c soạn thảo trên hộp nhập (Copy, paste, cut, clear, undo) • Ví dụ một số h{m: – void GetRect ( LPRECT lpRect); – BOOL GetModify( ); – void SetModify(BOOL bModified = TRUE); – BOOL SetReadOnly( BOOL bReadOnly = TRUE ); – void SetPasswordChar (TCHAR ch); – (Xem thêm MSDN) 204Lập trình Windows 9/13/2011 205 Lớp CEdit (3) – Xử lý sự kiện • C|c thông điệp sự kiện trên hộp nhập liệu được gửi đến cửa sổ cha. • Mục xử lý thông được được thêm v{o |nh xạ thông điệp (Message Map) trong lớp cửa sổ cha có dạng: – ON_Notification (ID, memberFxn ) – Trong đó: ID x|c định định danh hộp nhập; memberFxn tên h{m xử lý thông điệp tương ứng. – Ví dụ: ON_EN_UPDATE(IDC_EDIT1, &CMyDialogDlg::OnEnUpdateEdit1) 205Lập trình Windows 9/13/2011 206 Lớp CEdit (4) – Xử lý sự kiện • C|c thông điệp trên hộp nhập: • (EN = Event Notification) 206Lập trình Windows 9/13/2011 207 Lớp CEdit (5) – Ví dụ: 207Lập trình Windows // dynamically create an edit control CEdit* pEdit = new CEdit; pEdit->Create(ES_MULTILINE | WS_CHILD | WS_VISIBLE | WS_TABSTOP | WS_BORDER, CRect(10, 10, 100, 100), this, 1); //get text from edit box. CString yourname; pEdit->GetWindowText( yourname ); MessageBox ( yourname, "Hello !", MB_OK | MB_ICONEXCLAMATION ); 9/13/2011 208 2.7.4.2. Lớp CButton (1) – Khởi tạo • Quản lý c|c điều khiển dạng nút chọn (Button, radio, checkbox, ) (xem button styles) • H{m tạo CButton() • Hàm Create() 208Lập trình Windows BOOL Create ( LPCTSTR lpszCaption, //Nội dung thông b|o DWORD dwStyle, //dạng nút (ví dụ BS_PUSHBUTTON) const RECT& rect, // Tọa độ, kích thước CWnd* pParentWnd, //Con trỏ cửa sổ cha UINT nID // Định danh nút chọn ); 9/13/2011 209 Lớp CButton (2) – Hàm thành viên • Các hàm thành viên (xem MSDN) • Một số h{m: – HBITMAP SetBitmap(HBITMAP hBitmap) – HICON SetIcon(HICON hIcon); – virtual void DrawItem( LPDRAWITEMSTRUCT lpDrawItemStruct ); – int GetCheck() const; – void SetCheck(int nCheck); 209Lập trình Windows 9/13/2011 210 Lớp CButton (3) – Xử lý sự kiện • Thông điệp trên nút chọn 210Lập trình Windows ON_BN_CLICKED(IDOK, &CMyDialogDlg::OnBnClickedOk) 9/13/2011 211 Lớp CButton (4) – Ví dụ 211Lập trình Windows CButton myButton1, myButton2, myButton3, myButton4; //Khai báo trong .h //Thường tạo điều khiển trong OnCreate của lớp cha // Create a push button. myButton1.Create(_T(“Push button"), WS_CHILD|WS_VISIBLE|BS_PUSHBUTTON, CRect(10,10,100,30), pParentWnd, 1); // Create a radio button. myButton2.Create(_T(“Radio button"), WS_CHILD|WS_VISIBLE|BS_RADIOBUTTON, CRect(10,40,100,70), pParentWnd, 2); // Create an auto 3-state button. myButton3.Create(_T(“3-state button"), WS_CHILD|WS_VISIBLE|BS_AUTO3STATE, CRect(10,70,100,100), pParentWnd, 3); // Create an auto check box. myButton4.Create(_T(“Check box button"), WS_CHILD|WS_VISIBLE| BS_AUTOCHECKBOX, CRect(10,100,100,130), pParentWnd, 4); 9/13/2011 212 2.7.4.3. Lớp CListBox (1) • Lớp quản lý hộp danh s|ch c|c phần tử • H{m tạo CListBox() • Hàm Create() • ListBox styles: LBS_XXX (LBS_MULTILESEL, LBS_MULTCOLUMN, ) (Xem MDSN) 212Lập trình Windows virtual BOOL Create( DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 213 Lớp CListBox (2) – Hàm thành viên • Chia thành các nhóm: – General Operations: • int GetCount( ) trả về số mục chọn • GetText, GetSel, – Single-selection Operations: • int GetCurSel() lấy chỉ số mục được chọn, • int SetCurSel(int nSelect) Ấn định mục chọn – Multiple-selection Operations: GetSelCount, GetSelItems, – String Operations: AddString, DeleteString, InsertString, ResetContent, • Chi tiết: MSDN 213Lập trình Windows 9/13/2011 214 Lớp CListBox (3) – Xử lý sự kiện • Sự kiện trên listbox v{ h{m xử lý tương ứng 214Lập trình Windows ON_LBN_SELCHANGE(IDC_LIST1, &CControlDemoDlg::OnLbnSelchangeList1) 9/13/2011 215 Lớp CListBox (4) – Ví dụ 215Lập trình Windows int CControlDemoDlg::OnCreate( LPCREATESTRUCT lpCreateStruct ) { if (CWnd::OnCreate(lpCreateStruct) == -1) return -1; //Tạo một listbox m_listbox- >Create ( WS_CHILD | WS_VISIBLE | WS_BORDER | LBS_NOTIFY | WS_VSCROLL, CRect( 200, 10, 280, 80 ), this, IDC_LISTBOX); //ID định nghĩa trong resource.h m_listbox- >AddString( "Ong A" ); m_listbox- >AddString( "Ba B" ); m_listbox- >AddString( "Co C" ); //Tạo một edit box (dùng để hiện phần tử được chọn từ listbox m_edit->Create(WS_CHILD | WS_VISIBLE | WS_TABSTOP | WS_BORDER, CRect(200, 100, 280, 120), this, 1010); return 0; } 9/13/2011 216 Lớp CListBox – Ví dụ (tiếp) 216Lập trình Windows void CControlDemoDlg::OnLbnSelchangeListBox() { CString info; int iSel = m_listbox->GetCurSel(); if (iSel == LB_ERR) // Không có lựa chọn iSel = 0; // Thì chọn mục đầu tiên m_listbox->GetText(iSel, info ); m_edit->SetWindowText( info); } 9/13/2011 217 2.7.4.4. Lớp CComboBox (1) • Lớp quản lý combo box (hộp nhập v{ phần hỗ trợ nhập với danh s|ch c|c phần tử cho trước) • H{m tạo CComboBox() • Hàm Create() • dwStyle: C|c dạng Combo Box kh|c nhau: – CBS_DROPDOWNLIST: Gi| trị nhập chỉ được chọn từ danh s|ch. – CBS_SIMPLE : Như Drop list m{ danh s|ch hiển thị thường trực – CBS_DROPDOWN: Như drop list v{ có thể nhập nội dung mới 217Lập trình Windows BOOL Create ( DWORD dwStyle, // Kiểu comboBox const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 218 Lớp CComboBox (2) – Hàm thành viên • Nhiều h{m tương tự CListBox • 2 nhóm: – General Operations: • int GetCount( ) trả về số mục chọn • GetCurSel, SetCurSel, • GetLBText() Lấy text của phần tử – String Operations: AddString, DeleteString, InsertString, ResetContent, • Xem MSDN 218Lập trình Windows 9/13/2011 219 Lớp CComboBox (3) – Xử lý sự kiện • Sự kiện trên ComboBox v{ h{m xử lý 219Lập trình Windows ON_CBN_SELCHANGE(IDC_COMBO1, &CControlDemoDlg::OnCbnSelchangeCombo1) 9/13/2011 220 Lớp CComboBox (4) – Ví dụ 220Lập trình Windows m_pComboBox->Create( WS_CHILD|WS_VISIBLE|WS_VSCROLL|CBS_DROPDOWNLIST, CRect(10,10,200,100), pParentWnd, IDC_COMBO1); void CControlDemoDlg::OnCbnSelchangeCombo1() { CString info; int iSel = m_Combobox.GetCurSel( ); if (iSel == LB_ERR) // no selection iSel = 0; m_pComboBox- >GetLBText( iSel, info ); m_edit->SetWindowText( info ); } 9/13/2011 221 2.7.4.5. Lớp CSpinButtonCtrl (1) • Lớp quản lý điều khiển cho phép lựa chọn gi| trị (tăng/giảm) trong một khoảng đ~ c{i đặt. • Thường đi kèm với một edit box (để hiện gi| trị muốn chọn) • Hàm Create() 221Lập trình Windows virtual BOOL Create( DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 222 CSpinButtonCtrl (2) – Hàm thành viên • CWnd* SetBuddy (CWnd* pWndBuddy); //Ấn định đối tượng hộp nhập (edit box) m{ điều khiển n{y hỗ trợ • CWnd* GetBuddy( ); • int SetBase(int nBase); //thiết lập bước tăng giảm • void SetRange ( int nLower, int nUpper); //Ấn định phạm vi gi| trị biến thiên của nút tăng giảm. • void GetRange ( int &nLower, int &nUpper); • int SetPos( int nPos ); //Đặt gi| trị nút tăng giảm • int GetPos( int &nPos); 222Lập trình Windows 9/13/2011 223 CSpinButtonCtrl (4) – Ví dụ 223Lập trình Windows int CControlDemoDlg::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CWnd::OnCreate(lpCreateStruct) == -1) return -1; m_editAge.Create ( WS_CHILD | WS_VISIBLE | WS_BORDER | ES_NUMBER, CRect( 135, 45, 190, 65 ), this, IDC_EDIT); static CSpinButtonCtrl m_spinAge; m_spinAge.Create ( WS_CHILD | WS_VISIBLE | UDS_ARROWKEYS | UDS_SETBUDDYINT | UDS_ALIGNRIGHT, CRect(0,0,1,1), this, IDC_SPIN); m_spinAge.SetBase(1); //Bước tăng giảm m_spinAge.SetRange( 15, 80 ); m_spinAge.SetBuddy ( &m_editAge ); return 0; } 9/13/2011 224 2.7.4.6. Lớp CSliderCtrl (1) • Lớp quản lý thanh trượt • Tương tự Spin • Thuộc tính: – Tick Marks – Point: Kiểu con chạy – Orientation: Horizontal & vertical • Hàm Create() 224Lập trình Windows virtual BOOL Create( DWORD dwStyle, //TBS_HORZ, TBS_VERT, const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 225 Lớp CSliderCtrl (2) – Hàm thành viên • void SetRange(int nMin, int nMax, BOOL bRedraw = FALSE); • void SetPos(int nPos); int GetPos(), . • CWnd* SetBuddy (CWnd* pWndBuddy); • CWnd* GetBuddy( ); • int GetTic(); //Lấy gi| trị bước con chạy • int SetTic(int nTic); //Thiết lập bước con chạy • void SetRangeMin(int nMin, BOOL bRedraw = FALSE); GetRangeMin(); • void SetTicFreq(int nFreq); • Xem MSDN 225Lập trình Windows 9/13/2011 226 2.7.4.7. Lớp CProgressCtrl (1) • Lớp quản lý thanh tiến độ xử lý • H{m tạo CProgressCtrl() • Hàm Create() – dwStyle: • PBS_VERTICAL : Thanh tiến trình nằm đứng • PBS_SMOOTH : Chỉ độ tiến độ m{u liên tục 226Lập trình Windows virtual BOOL Create( DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 227 CProgressCtrl (2) – Hàm thành viên • void SetRange ( int nLower, int nUpper); //Ấn định phạm vi gi| trị biến thiên của thanh tiến độ • void GetRange ( int &nLower, int &nUpper); • int SetPos( int nPos ); //Đặt gi| trị vị trị tiến độ • int GetPos( int &nPos); • int SetStep(int nStep); //Thiết lập bước tăng • int GetStep(); • int OffsetPos( int nPos ); //Ấn định mức tăng giảm trên một đơn vị th{nh tiến độ bằng nPos, đồng thời cập nhật thanh tiến độ theo gi| trị ấn định mới. • int StepIt(); //Thay đổi chỉ mục tiến độ một bước 227Lập trình Windows 9/13/2011 228 CProgressCtrl (3) – Ví dụ 228Lập trình Windows static CProgressCtrl myCtrl; // Create a smooth child progress control. myCtrl.Create(WS_CHILD|WS_VISIBLE|PBS_SMOOTH, CRect(10,10,200,30), this, 1012); // Set the range to be 0 to 100. myCtrl.SetRange(0, 100); // Set the position to be half, 50. myCtrl.SetPos(50); // Offset the position by one-fourth of the total range. int nLower, nUpper; myCtrl.GetRange(nLower, nUpper); myCtrl.OffsetPos((nUpper-nLower)/4); // Advance the position to the next step. myCtrl.StepIt(); 9/13/2011 229 2.7.4.8. Lớp CScrollBar • Lớp quản lý điều khiển thanh cuộn • Hàm Create() – dwStyle: SBS_HORZ, SBS_VERT, . • Các hàm thành viên: – void SetScrollRange ( int nLower, int nUpper); – void GetScrollRange ( int &nLower, int &nUpper); – int SetScrollPos( int nPos ); – int GetScrollPos( int &nPos); – void ShowScrollBar ( BOOL bShow = TRUE ); //Bật/tắt thanh cuộn 229Lập trình Windows virtual BOOL Create( DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID ); 9/13/2011 230 Xử lý sự kiện với c|c điều khiển dạng cuộn • Sự kiện thường dùng – CWnd::OnHScroll() nếu điều khiển dạng đứng, – CWnd::OnVScroll() nếu điều khiển dạng ngang • Sự kiện n{y được gửi đến lớp cửa sổ cha chứa điều khiển (kế thừa từ lớp CWnd) • 2 h{m xử lý sự kiện l{ chung cho c|c điều khiển có dạng cuộn (Spin, Progress, slider, scroll), cần x|c định điều khiển n{o g}y ra bằng c|ch kiểm tra ID 230Lập trình Windows 9/13/2011 231 2.7.5. Bài tập – Common Controls (1) • Ex2-7a. Sử dụng một số điều khiển thông dụng 231Lập trình Windows 9/13/2011 232 Thiết lập thuộc tính cho các control • Ví dụ: Button “Show Message” – ID: IDC_BT_SHOWMSG – Caption: Show &Message 232Lập trình Windows 9/13/2011 233 Thiết lập thuộc tính cho các control • Ví dụ: Combo box – ID: IDC_CB_PROG – Data: Notepad;Spider;Solitaire 233Lập trình Windows 9/13/2011 234 B{i tập Common Controls (2) • Ex2-7b. UserProfile : – Combo box – Listbox – Radio button – Spin – Slider 234Lập trình Windows 9/13/2011 235 Chương 3. Một số chủ đề nâng cao • 3.1. DLL • 3.2. Thread • 3.3. Windows Socket 235Lập trình Windows 9/13/2011 236 3.1. Dynamic Link Libraries - DLLs • 3.1.1. Giới thiệu DLL ? • 3.1.2. Ph}n loại DLLs • 3.1.3. C|ch thức gọi DLL trong ứng dụng • 3.1.4. Viết DLL • 3.1.5. Ứng dụng sử dụng DLL • B{i tập 236Lập trình Windows 9/13/2011 237 3.1.1. Giới thiệu DLL • DLL (Dynamic Link Library): Thư viện liên kết động. • L{ một file nhị ph}n chứa một hoặc nhiều h{m thực thi. Cho phép ứng dụng kh|c sẽ có thể gọi đến. • Một DLL có thể được nhiều ứng dụng sử dụng đồng thời. • C|c h{m trong DLL được chia l{m 2 loại: – Export (h{m xuất: c|c h{m cho phép ứng dụng bên ngo{i sử dụng). – Internal (h{m nội bộ: những h{m chỉ sử dụng trong nội bộ của DLL) 237Lập trình Windows 9/13/2011 238 Static Library • Thư viện liên kết tĩnh (statically link library) ? • Ví dụ: h{m ceil() thư viện to|n học cmath.h 238Lập trình Windows 9/13/2011 239 How DLL Works ? • DLL chỉ được nạp v{o bộ nhớ khi chương trình thực thi. 239Lập trình Windows 9/13/2011 240 Ưu điểm sử dụng DLL • L{m kích thước ứng dụng nhỏ. • Nhiều ứng dụng có thể dùng chung 1 DLL, tiết kiệm bộ nhớ. (C|c ứng dụng có bộ dữ liệu riêng, nhưng có thể chia sẻ m~ lệnh). • Khi không còn sử dụng, có thể giải phóng DLL khỏi bộ nhớ • Khi cần n}ng cấp, chỉ cần thay thế file DLL, c|c file chương trình kh|c không bị ảnh hưởng. • Nhược điểm: Lập trình xử lý nhiều thao t|c phức tạp hơn. 240Lập trình Windows 9/13/2011 241 Nội dung DLL • M~ lệnh của c|c h{m (functions) • C|c lớp (Classes) • Biến to{n cục (global variables) • Resource (bitmap, font) • Ví dụ: Solitaire game sử dụng Cards.dll chứa cards images + functions • Chỉ những thành phần được export của DLL mới có thể được truy xuất từ bên ngoài. 241Lập trình Windows 9/13/2011 242 Hàm DllMain() • DLL không thực thi như một chương trình độc lập, nhưng có chứa h{m main đặc biệt gọi l{ DllMain(). • Được gọi bởi Windows khi nó được nạp v{o bộ nhớ lần đầu tiên. Để thực hiện khởi tạo trước khi sử dụng nội dung của DLL. • Cũng được gọi khi giải phóng khỏi bộ nhớ để thực hiện những giải phóng cần thiết. 242Lập trình Windows 9/13/2011 243 3.1.2. Ph}n loại DLLs • Có 3 loại: (trong MFC) – DLL mở rộng (MFC Extension DLL) – DLL thông thường với liên kết tĩnh MFC (Regular DLL with MFC statically linked). – DLL thông thường với liên kết động MFC (Regular DLL with MFC dynamically linked) 243Lập trình Windows 9/13/2011 244 MFC Extension DLL • Cho phép sử dụng c|c lớp (trong DLL) dẫn xuất từ c|c lớp MFC. • Cho phép ứng dụng dùng DLL có thể sử dụng c|c lớp của nó. • MFC Extension DLL liên kết với c|c DLL của MFC theo dạng liên kết động. • C|c DLLs dạng n{y chỉ sử dụng cho ứng dụng MFC có tính năng “Use MFC in a Shared DLL” • Không dùng được bởi c|c ứng dụng sử dụng liên kết tĩnh đến thư viện MFC (Statically linked to MFC) 244Lập trình Windows 9/13/2011 245 Regular DLL – Statically linked to MFC • L{ c|c DLL sử dụng c|c lớp của MFC dưới dạng liên kết tĩnh. (c|c đoạn m~ trong DLL của MFC được chèn v{o Regular DLL mỗi khi h{m tương ứng được sử dụng đến). L{m cho Regular DLL kiểu n{y có kích thước lớn. • Có thể được sử dụng bởi c|c ứng dụng MFC hoặc Win32 Application (không cần môi trường MFC) • Chỉ cho export c|c h{m, không cho export c|c lớp có trong DLL. 245Lập trình Windows 9/13/2011 246 Regular DLL – Dynamically linked to MFC • L{ DLL sử dụng c|c lớp MFC dạng liên kết động (không thêm m~ lệnh của lớp MFC v{o DLL n{y, chỉ gọi đến khi được dùng). • Kích thước giảm so với kiểu Regular DLL liên kết tĩnh MFC. • Có thể được sử dụng bởi ứng dụng MFC hoặc bất kỳ ứng dụng Win32 (nhưng cần có DLL của MFC tương ứng đi kèm) 246Lập trình Windows 9/13/2011 247 3.1.3. C|ch gọi DLL trong ứng dụng • 2 kiểu: – Gọi lúc ứng dụng được nạp (load time dynamic linking hay early binding). – Gọi lúc thực thi (runtime dynamic linking hay late binding) 247Lập trình Windows 9/13/2011 248 Gọi lúc ứng dụng được nạp (load time) • DLL được nạp v{o bộ nhớ khi ứng dụng sử dụng nó được nạp v{o bộ nhớ để thực thi. • DLL được nạp ngay từ đầu bất kể ứng dụng yêu cầu nạp nó có sử dụng đến c|c h{m của nó trong qu| trình thực thi hay không. • Liên kết đến c|c h{m DLL muốn dùng được thiết lập ngay từ đầu. • Có thể bất tiện: Tốn bộ nhớ nạp DLL, ứng dụng có thể không hoạt động nếu DLL yêu cầu nạp không tồn tại. 248Lập trình Windows 9/13/2011 249 Gọi lúc thực thi (run time) • Lập trình viên quyết định khi n{o cần nạp DLL lên bộ nhớ. (Được nạp khi có một h{m của DLL được ứng dụng yêu cầu sử dụng). • Khó khăn: Tốn công sức lập trình cho việc nạp DLL, gọi h{m DLL. – Cần dùng h{m API LoadLibrary() để nạp DLL khi nó được yêu cầu. – Hàm GetProcAddress() để lấy địa chỉ h{m (con trỏ h{m) của h{m DLL cần dùng. – Hàm FreeLibrary() để deattach DLL khi ứng dụng không còn dùng đến. (DLL sẽ được giải phóng khi không còn ứng dụng n{o dùng đến) 249Lập trình Windows 9/13/2011 250 250Lập trình Windows Minh họa gọi DLL lúc thực thi ứng dụng 9/13/2011 251 3.1.4. Viết DLL • Minh họa x}y dựng một Regular DLL (dạng liên kết động với MFC) • DLL n{y chứa 2 h{m tính to|n (cộng v{ nh}n) cho phép ứng dụng kh|c gọi đến. • Tạo Project MFC/MFC DLL • Chọn kiểu Regular DLL using shared MFC DLL 251Lập trình Windows 9/13/2011 252 Minh họa viết DLLRegular 252Lập trình Windows 9/13/2011 253 Khai b|o h{m xuất 253Lập trình Windows 9/13/2011 254 • Khai b|o h{m xuất đặt trong tệp tin .cpp hoặc .h • H{m nội bộ của DLL khai b|o v{ triển khai như h{m của ứng dụng thông thường 254Lập trình Windows 9/13/2011 255 Thực thi h{m xuất • Đặt macro AFX_MANGE_STATE đầu tiên trong thân hàm. 255Lập trình Windows //TODO: If this DLL is dynamically linked against the MFC DLLs, // any functions exported from this DLL which call into // MFC must have the AFX_MANAGE_STATE macro added at the // very beginning of the function. // // For example: // // extern "C" BOOL PASCAL EXPORT ExportedFunction() // { // AFX_MANAGE_STATE(AfxGetStaticModuleState()); // // normal function body here // } 9/13/2011 256 Thực thi h{m xuất (minh họa) • Phần thực thi h{m cộng: • Phần thực thi h{m nh}n: 256Lập trình Windows 9/13/2011 257 Biên dịch DLL • Như biên dịch ứng dụng thông thường • Tập tin kết quả: .dll v{ .lib • C|c ứng dụng sử dụng dll n{y cần dùng đến 2 tập tin trên. 257Lập trình Windows 9/13/2011 258 3.1.5. Minh họa ứng dụng sử dụng DLL • Gọi DLLRegular đ~ viết lúc nạp ứng dụng (load time) • Tạo ứng dụng đơn giản (kiểu dialog base) • Chứa tập tin .dll v{ .lib cùng thư mục project • Add tập tin .lib v{o Project (Project/Add Existing Items) 258Lập trình Windows 9/13/2011 259 Khai b|o c|c h{m import trong ứng dụng • Muốn sử dụng h{m xuất (export) n{o từ DLL, ứng dụng sẽ khai b|o import cho h{m đó. • Dạng khai b|o tổng qu|t 259Lập trình Windows 9/13/2011 260 Viết m~ thực thi cho c|c chức năng • Viết m~ xử lý nút cộng, nh}n gọi h{m cong, nhan tương ứng của DLL • Ví dụ: H{m xử lý sự kiện click chuột nút cộng 260Lập trình Windows 9/13/2011 261 Minh họa ứng dụng gọi DLL kiểu runtime • C|ch thức nạp h{m từ DLL: – Nạp DLL bằng h{m LoadLibrary. Kiểm tra xem DLL có tồn tại không bằng c|ch xem kết quả trả về có kh|c NULL hay không – Nạp h{m cần thiết bằng h{m GetProcAddress. Kiểm tra việc gọi h{m th{nh công hay không. – Giải phóng DLL khỏi bộ nhớ bằng h{m FreeLibrary. 261Lập trình Windows 9/13/2011 262 Ứng dụng gọi DLL kiểu Run time • Khai b|o con trỏ h{m cho h{m muốn import v{o ứng dụng. (Phụ thuộc v{o h{m export trong DLL) • Với ví dụ DLLRegular, h{m xuất có dạng: • Định nghĩa con trỏ h{m cho 2 h{m n{y: 262Lập trình Windows 9/13/2011 263 C|ch thức nạp h{m từ DLL 263Lập trình Windows 9/13/2011 264 Ví dụ: Gọi h{m nh}n 264Lập trình Windows 9/13/2011 265 Thử nghiệm cho c|ch gọi kiểu run time • Không chép file .dll v{o cùng thư mục ứng dụng sử dụng dll. Chạy ứng dụng được không ? Có thực thi được c|c chức năng (cộng, nh}n) của ứng dụng không ? • Không tắt ứng dụng, chép file .dll v{o cùng thư mục ứng dụng. Thực thi được c|c chức năng của ứng dụng không ? 265Lập trình Windows 9/13/2011 266 Ex-3.? DLL • Viết một Extension DLL cho ứng dụng Sketcher 266Lập trình Windows 9/13/2011 267 3.2. Thread (Tiểu trình/Luồng) • 3.2.1. Một số kh|i niệm • 3.2.2. Lập trình đa luồng • 3.2.3. Tạo Worker Thread • 3.2.4. Tạo UI Thread • 3.2.5. Kết thúc một Thread • 3.2.6. Đồng bộ c|c Thread • 3.2.7. Ví dụ minh họa 267Lập trình Windows 9/13/2011 268 3.2.1. Một số kh|i niệm (1) • Trên Windows, mỗi ứng dụng đang chạy l{ một tiến trình (Process) • Một tiến trình có thể chứa một hoặc nhiều tiểu trình/hay luồng (thread) thực thi. (path of execution) 268Lập trình Windows • Main Thread/ Primary Thread: là thread chính/mặc định của một ứng dụng. • Secondary Thread: các thread kh|c (nếu chương trình được lập trình tạo thêm thread) 9/13/2011 269 Một số kh|i niệm (2) • Multithreading: Nhiều thread cùng chạy đồng thời (Được Windows lập lịch - schedule) • Hệ điều h{nh ph}n chia thời gian CPU cho c|c thread – cơ chế đa nhiệm (multitasking) • Tại sao cần đa luồng ? – Ứng dụng muốn thực hiện nhiều nhiệm vụ đồng thời. – Thường c|c nhiệm vụ xử lý được chạy “hậu cảnh” (background) trong khi c|c xử lý tương t|c người dùng chạy “tiền cảnh” (foreground) – Ví dụ: ứng dụng MS Word với thread check spelling grammar. 269Lập trình Windows 9/13/2011 270 3.2.2. Lập trình đa luồng • Lập trình đa luồng (multithreading): – Multithreading in pure C – Multithreading in Win32 API – Multithreading in MFC • Lớp CWinThread: lớp cơ sở đóng gói c|c hoạt động thực thi của thread. • MFC ph}n biệt 2 loại thread: – Worker Thread (Tiểu trình xử lý nội): hỗ trợ thực hiện c|c xử lý bên trong, “hậu cảnh” (background). – User Interface Thread (Tiểu trình giao diện): trực tiếp tiếp nhận yêu cầu người dùng (cơ chế Windows Messages) 270Lập trình Windows 9/13/2011 271 Lớp CWinThread • Data members – m_hThread: The current thread handle – m_nThreadID: The current thread ID – m_bAutoDelete: (TRUE/FALSE) thread được thiết lập tự động giải phóng hoặc không. • Data Functions: – CreateThread: Bắt đầu thực thi thread – SuspendThread: Tạm dừng/ngưng một thread (tăng biến đếm số lần suspend) – ResumeThread: Tiếp tục một thread bị tạm dừng (giảm biến đếm số lần suspend). – SetThreadPriority: Thiết lập mức độ ưu tiến của thread (tương đối so với mức độ ưu tiên của process) ( LOW, BELOW LOW or HIGH) – GetThreadPriority: Lấy mức độ ưu tiên của thread. 271Lập trình Windows CObject CCmdTarget CWinThread CWinApp 9/13/2011 272 3.2.3. Tạo Worker Thread • H{m to{n cục AfxBeginThread, bắt đầu một thread trong MFC • Có 2 dạng cho 2 kiểu thread. • Cú ph|p dạng tạo Worker Thread: – 2 tham số đầu: quan trọng – 4 tham số sau: tùy chọn (có thể sử dụng hoặc không) 272Lập trình Windows CWinThread* AfxBeginThread( AFX_THREADPROC ThreadProc, LPVOID pParam, int nPriority = THREAD_PRIORITY_NORMAL, UINT nStackSize = 0, DWORD dwCreateFlags = 0, LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL ); 9/13/2011 273 Tạo Worker Thread (2) 273Lập trình Windows • Ý nghĩa c|c tham số h{m AfxBeginThread Các tham số Mô tả ThreadProc controlling function, cannot be NULL. This function must be declared as follows: UINT MyControllingFunction( LPVOID pParam ); pParam Parameter to be passed to the controlling function as shown in the parameter to the function declaration in pfnThreadProc. nPriority Thread priority. nStackSize Specifies the size in bytes of the stack for the new thread. If 0, the stack size defaults to the same size stack as the creating thread. dwCreateFlags additional flag and can be any of these two CREATE_SUSPENDED or 0. CREATE_SUSPENDED starts the thread with a suspend count of one. The thread will not execute until ResumeThread is called. 0 Start the thread immediately after creation. lpSecurityAttrs Points to a SECURITY_ATTRIBUTES structure. See SECURITY_ATTRIBUTES for more details. 9/13/2011 274 Thread Function (1) • H{m thực thi của một thread (Thread Function/ThreadProc) • L{ h{m callback (Được gọi bởi hệ điều h{nh), cần khai b|o static hoặc to{n cục (bên ngo{i c|c lớp). • Có dạng: • pParam l{ con trỏ đến 1 cấu trúc dữ liệu của thread do người lập trình định nghĩa. Được truyền cho h{m thông qua tham số thứ 2 của h{m AfxBeginThread 274Lập trình Windows UINT ThreadFunc (LPVOID pParam) 9/13/2011 275 Thread Function (2) • Ví dụ: 275Lập trình Windows CwinThread *pThread = AfxBeginThread( ThreadFunction, &data); UINT ThreadFunction(LPVOID param) { UINT nIterations = (UINT) pParam; for (int i=0; i<nIterations; i++); //do something return 0; } 9/13/2011 276 Thread Priorities (1) • C|c tiến trình (process) được hệ điều h{nh lập lịch thực thi với cấp độ ưu tiên kh|c nhau (chiếm thời gian CPU kh|c nhau). Bao gồm: REALTIME_PRIORITY_CLASS HIGH_PRIORITY_CLASS NORMAL_PRIORITY_CLASS IDLE_PRIORITY_CLASS • Cấp độ ưu tiên của tiến trình được thiết lập tương đối với c|c độ ưu tiên của process nó thuộc về. • Có thể thiết lập khi gọi h{m AfxBeginThread( ), hoặc dùng h{m: CWinThread::SetThreadPriority 276Lập trình Windows 9/13/2011 277 Thread Priorities (2) 277Lập trình Windows THREAD_PRIORITY_TIME_CRITICAL Normal for HIGH, NORMAL, or IDLE. Double for REALTIME. THREAD_PRIORITY_HIGHEST = Process Priority + 2. THREAD_PRIORITY_ABOVE_NORMAL = Process Priority +1. THREAD_PRIORITY_NORMAL = Process Priority. THREAD_PRIORITY_BELOW_NORMAL = Process Priority -1. THREAD_PRIORITY_LOWEST = Process Priority -2. THREAD_PRIORITY_IDLE Normal for REALTIME, 1 for HIGH, NORMAL, 9/13/2011 278 Suspending and Resuming Threads • Mỗi thread có thể chuyển đổi qua c|c nhiều trạng th|i hoạt động như sơ đồ. • Tạm ngừng: H{m SuspendThread • Khôi phục: H{m ResumeThread • Trạng th|i “ngủ”: H{m Sleep 278Lập trình Windows CWinThread* pThread = AfxBeginThread (ThreadFunc, &threadInfo, THREAD_PRIORITY_NORMAL, 0, CREATE_SUSPENDED); pThread->ResumeThread (); // Start the thread 9/13/2011 279 3.2.4. Tạo User Interface Thread (1) • UI Thread có giao diện GUI với vòng lặp thông điệp (message loop). • C|c bước chính tạo UI Thread: – Tạo lớp dẫn xuất từ CWinThread v{ override c|c h{m: 279Lập trình Windows Function name Purpose InitInstance Perform thread instance initialization. Must be overridden. Run Controlling function for the thread. Contains the message pump. Rarely overridden. OnIdle Perform thread-specific idle-time processing. Not usually overridden. PreTranslateMessage Filter messages before they are dispatched to TranslateMessage and DispatchMessage. Not usually overridden. ProcessWndProcExcept ion Intercept unhandled exceptions thrown by the thread's message and command handlers. Not usually overridden. ExitInstance Perform cleanup when thread terminates. Usually overridden. 9/13/2011 280 Tạo User Interface Thread (2) – Gọi h{m AfxBeginThread cho UI Thread – Tham số đầu tiên l{ con trỏ lớp Thread dẫn xuất đ~ tạo. – C|c tham số còn lại l{ tùy chọn (có thể dùng hoặc không) 280Lập trình Windows CWinThread* AfxBeginThread( CRuntimeClass* pThreadClass, int nPriority = THREAD_PRIORITY_NORMAL, UINT nStackSize = 0, DWORD dwCreateFlags = 0, LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL ); 9/13/2011 281 3.2.5. Kết thúc một Thread • Với Worker Thread: 2 trường hợp kết thúc: – Khi h{m thực thi của thread (ThreadFunction) kết thúc (trả về return) – Khi có h{m ở bất kỳ đầu gọi h{m kết thúc thread AfxEndThread() (h{m to{n cục) 281Lập trình Windows CwinThread *pThread = AfxBeginThread(ThreadFunction, &data); UINT ThreadFunction(LPVOID param) { DWORD result =0 ; // do somthing AfxEndThread(exitCode); return result; } 9/13/2011 282 Kết thúc một Thread (2) • Với UI Thread: – Kết thúc khi có thông điệp WM_QUIT gửi đến h{ng đợi thông điệp của nó. – Hoặc khi bản th}n thread gọi h{m AfxEndThread • Khi kết thúc trả về m~ (exit code 32 bit), có thể kiểm tra m~ n{y: 282Lập trình Windows DWORD dwExitCode; ::GetExitCodeThread (pThread->m_hThread, &dwExitCode); if (dwExitCode == STILL_ACTIVE) { // The thread is still running. } else { // The thread has terminated. Delete the CWinThread object. delete pThread; } 9/13/2011 283 3.2.6. Đồng bộ c|c thread • Thread Synchronization • Vấn đề ? • Giải ph|p ? • Windows cung cấp 4 cơ chế đồng bộ c|c thread: – Critical sections – Mutexes – Events – Semaphores • T{i liệu: – Chapter 17, Programming Windows with MFC – MSDN 283Lập trình Windows 9/13/2011 284 3.2.7. Ví dụ Thread • C|c ví dụ đơn giản minh họa sử dụng thread dùng Win32 API, MFC • Minh họa Worker Thread – Tiểu trình xử lý nội (Worker Thread) c{i đặt một h{m có nhiệm vụ thực hiện chạy chữ trong dòng thông b|o cho đến khi có tín hiệu ngừng. – Tiểu trình chính của ứng dụng thực hiện giao tiếp người dùng. Nút Start/Stop, Close 284Lập trình Windows 9/13/2011 285 Ex3-?. Prime Sieve • X}y dựng ứng dụng s{ng số nguyên tố (Prime Sieve) với tiểu trình xử lý nội (Worker Thread) cho việc xử lý giải thuật s{ng số nguyên tố. 285Lập trình Windows 9/13/2011 286 3.3 Lập trình Windows Socket • 3.3.1. Một số kh|i niệm • 3.3.2. Lập trình Socket với WINSOCK • 3.3.3. Lập trình Socket với MFC • 3.3.4. Ví dụ minh họa 286Lập trình Windows 9/13/2011 287 3.3.1. Một số kh|i niệm (1) • Winsock (Windows Socket): – Xuất xứ từ BSD (Berkeley Software Distribution – UNIX) – Giao diện c|c dịch vụ x}y dựng trên nền giao thức truyền thông TCP(UDP)/IP • Port: – Số hiệu cổng dịch vụ, ph}n biệt giữa c|c tiến trình trên cùng một host đồng thời sử dụng giao thức TCP/IP. 287Lập trình Windows 9/13/2011 288 Một số kh|i niệm (2) • Socket ? (= IP + Port) – Điểm truyền thông đầu cuối cho phép ứng dụng gửi v{ nhận c|c gói tin qua đường truyền mạng. – Cung cấp giao diện để một chương trình kết nối, trao đổi dữ liệu với một chương trình trên một m|y tính kh|c trong mạng 288Lập trình Windows 9/13/2011 289 Một số kh|i niệm (3) • C|c loại Socket: – Datagram socket – sử dụng UDP • Ít tin cậy • Không gửi lại • Không kết nối (Connectionless) – Stream socket – using TCP • Tin cậy • Có gửi lại • Hướng kết nối (Connection- oriented) – Raw and others 289Lập trình Windows 9/13/2011 290 Lập trình Socket trên Windows • Ứng dụng Win 32: Sử dụng thư viện Winsock • Ứng dụng MFC: sử dụng lớp CAsyncSocket (đóng gói c|c h{m Winsock API) 290Lập trình Windows 9/13/2011 291 3.3.2. Lập trình Socket với Winsock • Khai b|o thư viện winsock2.h v{ file .lib (ws2_2.lib) • Khởi tạo Winsock 291Lập trình Windows #include #pragma comment (lib,"ws2_32.lib") int WSAStartup(WORD wVersionRequested, LPWSADATA lpWSAData); 9/13/2011 292 Ví dụ khởi tạo Winsock 292Lập trình Windows #include "stdafx.h" using namespace std; int _tmain(int argc, _TCHAR* argv[]) { WSADATA SData; int iResult = WSAStartup(0x0202,&SData); if (iResult!=0){ cout << "KHONG THE KHOI DONG WINSOCK"; return 1; } cout << "KHOI TAO SOCKET THANH CONG: \n"; cout << "Phien ban: "<< SData.wVersion << "\n"; cout << "Phien ban co the ho tro: "<< SData.wHighVersion << "\n"; cout << "Ghi chu: " << SData.szDescription << "\n"; cout << "Thong tin cau hinh: " << SData.szSystemStatus << "\n"; WSACleanup(); return 0; } 9/13/2011 293 Giao tiếp Client – Server với TCP 293 9/13/2011 294 9/13/2011 295 Giao tiếp Client – Server với UDP socket() bind() recvfrom() sendto() UDP Server socket() UDP Client sendto() recvfrom() close() blocks until datagram received from a client data request data reply 9/13/2011 296 9/13/2011 297 Các hàm Winsock (1) • SOCKET create a new socket • BIND attach a local address to a socket • LISTEN announce a willingness to accept connections • ACCEPT block the caller process until a connection attempt arrives • CONNECT actively attempt to establish a connection • SEND send some data over the connection • RECEIVE receive some data from the connection • CLOSE release the connection (the port) 297Lập trình Windows 9/13/2011 298 Hàm socket( ) • Tạo Socket bằng h{m: • Ví dụ: 298Lập trình Windows SOCKET socket ( int af, //giao thức kết nối (IF_NET cho TCP/IP) int type, //giao thức vận chuyển (SOCK_DIAGRAM, SOCK_STREAM, SOCK_RAW) int protocol //chỉ định rõ lại giao thức ); SOCKET s = socket(AF_INET,SOCK_STREAM,IPPROTO_IP); 9/13/2011 299 Hàm bind( ) int bind( SOCKET s, const struct sockaddr FAR* name, int namelen ); Tác dụng dụng của BIND là sẽ giúp cho SOCKET của SERVER biết rằng nó sẽ chờ đợi kết nối và nhận dữ liệu trên IP nào và PORT bao nhiêu? Hàm Bind gồm có 3 thông số: - SOCKET s: Socket được thiết lập - sockaddr name: Cấu trúc ADDR bao gồm địa chỉ IP và PORT - int namelen: Kích thước của cấu trúc sockaddr 299Lập trình Windows 9/13/2011 300 Hàm listen( ) • int listen( SOCKET s, int backlog ); • Kể từ khi gọi hàm này thì SERVER sẽ bắt đầu lắng nghe kết nối của mình. Hàm LISTEN gồm có 2 thông số: - SOCKET s: Socket đ~ được thiết lập IP và PORT. - int backlog: Số kết nối cho phép chờ trong hàng đợi khi Server chưa chấp nhận kết nối. (vì đôi lúc có thể có tới 2 hay 3 client kết nối tới cùng 1 lúc). Giá trị tốt nhất là khoảng từ 5 – 10. 300Lập trình Windows 9/13/2011 301 Connect • int connect( SOCKET s, struct sockaddr *serv_addr, int addrlen ); • Hàm được gọi từ CLIENT nếu nó muốn kết nối tới SERVER - SOCKET s: Socket đ~ được khởi tạo. - sockaddr *serv_addr: IP và PORT của Server. - int addrlen: Sizeof của cấu trúc sockaddr. 301Lập trình Windows 9/13/2011 302 Accept • Chấp nhận một kết nối đến • SOCKET accept( SOCKET s, struct sockaddr FAR* addr, int FAR* addrlen ); • Tham số: - SOCKET s: Socket lắng nghe của SERVER. - sockaddr addr: Là cấu trúc sockaddr lưu địa chỉ IP và PORT của CLIENT kết nối tới SERVER. - int addrlen: Kích thước cấu trúc địa chỉ IP này. • Trả về: 1 SOCKET mới Socket mới được tạo này đại diện cho 1 Connection (kết nối) mới giữa Server và Client. Sau khi đ~ truyền dữ liệu thì ta phải đóng SOCKET này lại bằng hàm close như closesocket(connect); 302Lập trình Windows 9/13/2011 303 REVC/SEND • Nhận dữ liệu trên giao thức TCP • int recv( SOCKET s, char FAR* buf, int len, int flags ); • Gửi dữ liệu giao thức TCP • int send( SOCKET s, const char FAR * buf, int len, int flags ); - SOCKET s: Là SOCKET được tạo ra khi Server chấp nhận kết nối từ CLIENT - char FAR* buf: Là dữ liệu (dạng BYTE – char) nhận hay gửi. - int len: Kích thước của dữ liệu. - int flags: Một số cờ hiệu đi kèm (thông thường là 0). 303Lập trình Windows 9/13/2011 304 REVCFROM/SENDTO • Được sử dụng trên giao thức UDP • int recvfrom( SOCKET s, char FAR* buf, int len, int flags, struct sockaddr *from, int *fromlen); • Các thông số: - SOCKET s: Là SOCKET được tạo ra ban đầu. - char FAR* buf: Là dữ liệu (dạng BYTE – char) nhận - int len: Kích thước của dữ liệu nhận. - int flags: Một số cờ hiệu đi kèm (thông thường là 0). - sockaddr *from: IP và PORT từ bên gửi. - int *fromlen: Sizeof cấu trúc addr. 304Lập trình Windows 9/13/2011 305 REVCFROM/SENDTO • int sendto( SOCKET s, const char FAR *buf, int len, int flags, const struct sockaddr* to, int tolen ); • - SOCKET s: Là SOCKET được tạo ra ban đầu. - char FAR* buf: Là dữ liệu (dạng BYTE – char) gửi - int len: Kích thước của dữ liệu gửi. - int flags: Một số cờ hiệu đi kèm (thông thường là 0). - sockaddr *to: IP và PORT tới bên nhận. - int tolen: Sizeof cấu trúc addr. 305Lập trình Windows 9/13/2011 306 Close/Shutdown • int shutdown( SOCKET s, int how ); • int closesocket (SOCKET s); Hủy SOCKET sau một kết nối hoặc kết thúc chương trình. Tham số how của shutdown: - SD_RECEIVE: Đóng SOCKET, không cho phép NHẬN nhưng cho phép GỬI. - SD_SEND: Đóng SOCKET, không cho phép GỬI nhưng cho phép NHẬN. - SD_BOTH: Không cho GỬI và NHẬN (giống gọi hàm closesocket). 306Lập trình Windows 9/13/2011 307 3.3.3. Lập trình Socket với MFC • Thư viện sử dụng AfxSock.h • Khởi động Winsock: – BOOL AfxSocketInit ( WSADATA* lpwsaData = NULL ); – Thực hiện trong InitInstance( ) – Đảm nhận thực hiện WSAStartup khi bắt đầu ứng dụng v{ WSACleanup khi ứng dụng kết thúc 307Lập trình Windows 9/13/2011 308 Lớp CAsyncSocket • Lớp quản lý Socket • Đóng gói c|c dịch vụ Winsock API • H{m tạo: CAsyncSocket() • Hàm Create() 308Lập trình Windows BOOL Create ( UINT nSocketPort = 0, int nSocketType = SOCK_STREAM, long lEvent = FD_READ | FD_WRITE | FD_OOB | FD_ACCEPT | FD_CONNECT | FD_CLOSE, LPCTSTR lpszSocketAddress = NULL ); 9/13/2011 309 Lớp CAsyncSocket - Các hàm thành viên • Bind( ) • SetSockOpt( ) • Listen( ) • Connect( ) • Accept( ) • Send( )/SendTo( )

Các file đính kèm theo tài liệu này:

  • pdftailieu.pdf
Tài liệu liên quan