原文: http://zetcode.com/gui/winapi/advancedcontrols/

在 Windows API 教程的这一部分中,我们将详细讨论两个高级 Windows 控件:选项卡控件和列表框控件。

Label控件

选项卡控件将具有相应选项卡的多个窗口合并在一起。

tabcontrol.c

  1. #include <windows.h>
  2. #include <commctrl.h>
  3. #include <wchar.h>
  4. #define ID_TABCTRL 1
  5. #define ID_EDIT 2
  6. #define BTN_ADD 3
  7. #define BTN_DEL 4
  8. #define BTN_CLR 5
  9. #define MAX_TAB_LEN 15
  10. LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
  11. HWND hTab, hEdit;
  12. int WINAPI wWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
  13. PWSTR pCmdLine, int nCmdShow) {
  14. MSG msg ;
  15. WNDCLASSW wc = {0};
  16. wc.lpszClassName = L"Tab control";
  17. wc.hInstance = hInstance;
  18. wc.hbrBackground = GetSysColorBrush(COLOR_3DFACE);
  19. wc.lpfnWndProc = WndProc;
  20. wc.hCursor = LoadCursor(0, IDC_ARROW);
  21. RegisterClassW(&wc);
  22. CreateWindowW(wc.lpszClassName, L"Tab control",
  23. WS_OVERLAPPEDWINDOW | WS_VISIBLE,
  24. 100, 100, 380, 230, 0, 0, hInstance, 0);
  25. while (GetMessage(&msg, NULL, 0, 0)) {
  26. TranslateMessage(&msg);
  27. DispatchMessage(&msg);
  28. }
  29. return (int) msg.wParam;
  30. }
  31. LRESULT CALLBACK WndProc(HWND hwnd, UINT msg,
  32. WPARAM wParam, LPARAM lParam) {
  33. TCITEMW tie;
  34. wchar_t text[4];
  35. LRESULT count, id;
  36. INITCOMMONCONTROLSEX icex;
  37. switch(msg) {
  38. case WM_CREATE:
  39. icex.dwSize = sizeof(INITCOMMONCONTROLSEX);
  40. icex.dwICC = ICC_TAB_CLASSES;
  41. InitCommonControlsEx(&icex);
  42. hTab = CreateWindowW(WC_TABCONTROLW, NULL, WS_CHILD | WS_VISIBLE,
  43. 0, 0, 200, 150, hwnd,(HMENU) ID_TABCTRL, NULL, NULL);
  44. hEdit = CreateWindowW(WC_EDITW, NULL, WS_CHILD | WS_VISIBLE | WS_BORDER,
  45. 250, 20, 100, 25, hwnd, (HMENU) ID_EDIT, NULL, NULL);
  46. SendMessage(hEdit, EM_SETLIMITTEXT, MAX_TAB_LEN, 0);
  47. CreateWindowW(WC_BUTTONW, L"Add", WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON,
  48. 250, 50, 100, 25, hwnd, (HMENU) BTN_ADD, NULL, NULL);
  49. CreateWindowW(WC_BUTTONW, L"Delete", WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON,
  50. 250, 80, 100, 25, hwnd, (HMENU) BTN_DEL, NULL, NULL);
  51. CreateWindowW(WC_BUTTONW, L"Clear", WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON,
  52. 250, 110, 100, 25, hwnd, (HMENU) BTN_CLR, NULL, NULL);
  53. break;
  54. case WM_COMMAND:
  55. switch(LOWORD(wParam)) {
  56. case BTN_ADD:
  57. GetWindowTextW(hEdit, text, 250);
  58. if (lstrlenW(text) != 0 ) {
  59. tie.mask = TCIF_TEXT;
  60. tie.pszText = text;
  61. count = SendMessageW(hTab, TCM_GETITEMCOUNT, 0, 0);
  62. SendMessageW(hTab, TCM_INSERTITEMW, count,
  63. (LPARAM) (LPTCITEM) &tie);
  64. }
  65. break;
  66. case BTN_DEL:
  67. id = SendMessageW(hTab, TCM_GETCURSEL, 0, 0);
  68. if (id != -1) {
  69. SendMessageW(hTab, TCM_DELETEITEM, 0, id);
  70. }
  71. break;
  72. case BTN_CLR:
  73. SendMessageW(hTab, TCM_DELETEALLITEMS, 0, 0);
  74. break;
  75. }
  76. break;
  77. case WM_DESTROY:
  78. PostQuitMessage(0);
  79. break;
  80. }
  81. return(DefWindowProcW(hwnd, msg, wParam, lParam));
  82. }

在我们的例子中,我们使用一个标签控件,一个编辑控件,以及三个按钮。 我们将在选项卡控件上动态创建和删除选项卡。

  1. hTab = CreateWindowW(WC_TABCONTROLW, NULL, WS_CHILD | WS_VISIBLE,
  2. 0, 0, 200, 150, hwnd,(HMENU) ID_TABCTRL, NULL, NULL);

我们使用WC_TABCONTROL窗口类创建一个选项卡控件。

  1. hEdit = CreateWindowW(WC_EDITW, NULL, WS_CHILD | WS_VISIBLE | WS_BORDER,
  2. 250, 20, 100, 25, hwnd, (HMENU) ID_EDIT, NULL, NULL);
  3. SendMessage(hEdit, EM_SETLIMITTEXT, MAX_TAB_LEN, 0);

我们创建一个编辑控件,并通过EM_SETLIMITTEXT消息设置其最大大小。

  1. if (lstrlenW(text) != 0 ) {
  2. tie.mask = TCIF_TEXT;
  3. tie.pszText = text;
  4. count = SendMessageW(hTab, TCM_GETITEMCOUNT, 0, 0);
  5. SendMessageW(hTab, TCM_INSERTITEMW, count,
  6. (LPARAM) (LPTCITEM) &tie);
  7. }

要添加新标签,我们填充TCITEMW结构。 我们提供要设置的数据类型(在我们的示例中为TCIF_TEXT)和实际文本。 然后,我们发送两条消息。 TCM_GETITEMCOUNT消息获取选项卡的数量。 它将在第二条消息中使用。 TCM_INSERTITEMW消息使用count变量和TCITEMW结构在控件中插入新标签。

  1. id = SendMessageW(hTab, TCM_GETCURSEL, 0, 0);
  2. if (id != -1) {
  3. SendMessageW(hTab, TCM_DELETEITEM, 0, id);
  4. }

要删除特定标签,我们需要当前选择的标签。 我们通过将TCM_GETCURSEL消息发送到选项卡控件来解决。 要删除选项卡,我们发送TCM_DELETEITEM消息,并在wParam参数中指定要删除的项目。

  1. SendMessageW(hTab, TCM_DELETEALLITEMS, 0, 0);

要从标签控件中删除所有标签,我们发送TCM_DELETEALLITEMS消息。

Windows API 中的高级控件 - 图1

图:标签控件

列表框

List Box包含一个简单的列表,用户通常可以从中选择一个或多个项目。 所选项目被标记。

listbox.c

  1. #include <windows.h>
  2. #include <commctrl.h>
  3. #include <strsafe.h>
  4. #define IDC_LIST 1
  5. #define IDC_STATIC 2
  6. LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
  7. typedef struct {
  8. wchar_t name[30];
  9. wchar_t job[20];
  10. int age;
  11. } Friends;
  12. Friends friends[] = {
  13. {L"Lucy", L"waitress", 18},
  14. {L"Thomas", L"programmer", 25},
  15. {L"George", L"police officer", 26},
  16. {L"Michael", L"producer", 38},
  17. {L"Jane", L"steward", 28},
  18. };
  19. int WINAPI wWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
  20. PWSTR pCmdLine, int nCmdShow) {
  21. MSG msg ;
  22. WNDCLASSW wc = {0};
  23. wc.lpszClassName = L"MyListBox";
  24. wc.hInstance = hInstance;
  25. wc.hbrBackground = GetSysColorBrush(COLOR_3DFACE);
  26. wc.lpfnWndProc = WndProc;
  27. wc.hCursor = LoadCursor(0, IDC_ARROW);
  28. RegisterClassW(&wc);
  29. CreateWindowW(wc.lpszClassName, L"List Box",
  30. WS_OVERLAPPEDWINDOW | WS_VISIBLE,
  31. 100, 100, 340, 200, 0, 0, hInstance, 0);
  32. while (GetMessage(&msg, NULL, 0, 0)) {
  33. TranslateMessage(&msg);
  34. DispatchMessage(&msg);
  35. }
  36. return (int) msg.wParam;
  37. }
  38. LRESULT CALLBACK WndProc(HWND hwnd, UINT msg,
  39. WPARAM wParam, LPARAM lParam) {
  40. static HWND hwndList, hwndStatic;
  41. wchar_t buf[128];
  42. switch(msg) {
  43. case WM_CREATE:
  44. hwndList = CreateWindowW(WC_LISTBOXW , NULL, WS_CHILD
  45. | WS_VISIBLE | LBS_NOTIFY, 10, 10, 150, 120, hwnd,
  46. (HMENU) IDC_LIST, NULL, NULL);
  47. hwndStatic = CreateWindowW(WC_STATICW , NULL, WS_CHILD | WS_VISIBLE,
  48. 200, 10, 120, 45, hwnd, (HMENU) IDC_STATIC, NULL, NULL);
  49. for (int i = 0; i < ARRAYSIZE(friends); i++) {
  50. SendMessageW(hwndList, LB_ADDSTRING, 0, (LPARAM) friends[i].name);
  51. }
  52. break;
  53. case WM_COMMAND:
  54. if (LOWORD(wParam) == IDC_LIST) {
  55. if (HIWORD(wParam) == LBN_SELCHANGE) {
  56. int sel = (int) SendMessageW(hwndList, LB_GETCURSEL, 0, 0);
  57. StringCbPrintfW(buf, ARRAYSIZE(buf), L"Job: %ls\nAge: %d",
  58. friends[sel].job, friends[sel].age);
  59. SetWindowTextW(hwndStatic, buf);
  60. }
  61. }
  62. break;
  63. case WM_DESTROY:
  64. PostQuitMessage(0);
  65. break;
  66. }
  67. return (DefWindowProcW(hwnd, msg, wParam, lParam));
  68. }

在此示例中,我们显示一个列表框控件和一个静态文本控件。 通过从列表框中选择一个人,我们可以在静态控件中显示他的工作和年龄。

  1. hwndList = CreateWindowW(WC_LISTBOXW , NULL, WS_CHILD
  2. | WS_VISIBLE | LBS_NOTIFY, 10, 10, 150, 120, hwnd,
  3. (HMENU) IDC_LIST, g_hinst, NULL);

WC_LISTBOXW窗口类用于创建列表框控件。 每当用户单击列表框项目(LBN_SELCHANGE),双击项目(LBN_DBLCLK)或取消选择(LBN_SELCANCEL时,LBS_NOTIFY标志都会使列表框将通知代码发送到父窗口 ])。

  1. for (int i = 0; i < ARRAYSIZE(friends); i++) {
  2. SendMessageW(hwndList, LB_ADDSTRING, 0, (LPARAM) friends[i].name);
  3. }

通过发送多个LB_ADDSTRING消息,列表框中填充了数据。

  1. if (HIWORD(wParam) == LBN_SELCHANGE) {
  2. int sel = (int) SendMessageW(hwndList, LB_GETCURSEL, 0, 0);
  3. StringCbPrintfW(buf, ARRAYSIZE(buf), L"Job: %ls\nAge: %d",
  4. friends[sel].job, friends[sel].age);
  5. SetWindowTextW(hwndStatic, buf);
  6. }

如果我们从列表框中选择一个项目,则窗口过程会收到LBN_SELCHANGE消息。 首先,我们通过向列表框中发送LB_GETCURSEL消息来确定当前选择的项目。 然后,我们将工作名称和年龄从好友结构复制到buf数组中。 最后,我们通过SetWindowTextW()函数调用来设置静态文本。

Windows API 中的高级控件 - 图2

图:列表框

在 Windows API 教程的这一部分中,我们介绍了两个更高级的 Windows 控件。