Win32 ListBox控件

简介: Win32 ListBox控件

Win32 ListBox控件

创建ListBox控件

  1. 创建窗口函数

    HWND CrateWindowEx(
        DWORD dwExStyle , // 窗口的扩展风格,基本没用
        LPCTSTR lpClassName, // 已经注册的窗口类名称
        LPCTSTR lpWindowName, // 窗口标题栏的名字
        DWORD dwStyle, // 窗口的基本风格
        int x, // 左上角水平坐标
        int y, // 左上角垂直坐标
        int nWidth,
        int nHeight,
        HWND hWndParent, // 窗口的父窗口句柄 --> 如果是子窗口要写这个参数
        HMENU hMenu,     // 窗口菜单句柄
        HINSTANCE hInstance, // 应用程序实例句柄 --> WinMain 第一个参数
        LPVOID lpParam // 窗口创建时附加参数 --> 一般给NULL
    ); // 创建成功返回窗口句柄
  2. 创建ListBox

    #define ID_LIST        1001   // 窗口句柄
    
    HWND hListBox = CreateWindowEx(0, TEXT("ListBox"), NULL, WS_VISIBLE | WS_CHILD | WS_BORDER | LBS_HASSTRINGS,10, 10, 200, 200, hWnd, (HMENU)ID_LIST, ((LPCREATESTRUCT)lParam)->hInstance, 0);
    
    const TCHAR *strArray[] = { TEXT("星期一"), TEXT("星期二"), TEXT("星期三"), 
                              TEXT("星期四"), TEXT("星期五"), TEXT("星期六"), TEXT("星期日")};
    for (int i = 0; i < 7; i++){
        // 添加内容
        SendMessage(hListBox, LB_ADDSTRING, 0, (LPARAM)strArray[i]);
    }
  3. 效果

image.png

控件样式

样式 说明
LBS_COMBOBOX 通知列表框它是组合框的一部分。 这允许两个控件之间的协调,以便它们呈现统一的 UI。 组合框本身必须设置此样式。
LBS_DISABLENOSCROLL 当列表框不包含足够的项目滚动时,显示禁用的水平或垂直滚动条。 如果未指定此样式,则当列表框不包含足够的项目时,滚动条将隐藏。 此样式必须与 WS_VSCROLL 或 WS_HSCROLL 样式一起使用。
LBS_EXTENDEDSEL 允许使用 SHIFT 键和鼠标或特殊键组合选择多个项。
LBS_HASSTRINGS 指定列表框包含由字符串组成的项。 列表框维护字符串的内存和地址,以便应用程序可以使用 LB_GETTEXT 消息检索特定项的文本。 默认情况下,除所有者绘制的列表框之外的所有列表框都具有此样式。 可以创建具有或不带此样式的所有者绘制列表框。对于没有此样式的所有者绘制列表框, LB_GETTEXT 消息检索与项关联的值, (项数据) 。
LBS_MULTICOLUMN 指定水平滚动的多列列表框。 列表框自动计算列的宽度,或者应用程序可以使用 LB_SETCOLUMNWIDTH 消息设置宽度。 如果列表框具有 LBS_OWNERDRAWFIXED 样式,则当列表框发送 WM_MEASUREITEM 消息时,应用程序可以设置宽度。具有 LBS_MULTICOLUMN 样式的列表框不能垂直滚动,它忽略它接收的任何 WM_VSCROLL 消息。不能组合 LBS_MULTICOLUMN 和 LBS_OWNERDRAWVARIABLE 样式。 如果同时指定了两者, 则忽略LBS_OWNERDRAWVARIABLE 。
LBS_MULTIPLESEL 每次用户单击或双击列表框中的字符串时,都会打开或关闭字符串选择。 用户可以选择任意数量的字符串。
LBS_NODATA 指定无数据列表框。 当列表框中的项数超出一千时,请指定此样式。 无数据列表框还必须具有 LBS_OWNERDRAWFIXED 样式,但不得具有 LBS_SORT 或 LBS_HASSTRINGS 样式。无数据列表框类似于所有者描述的列表框,但它不包含项的字符串或位图数据。 用于添加、插入或删除项的命令始终忽略任何指定的项数据;在列表框中查找字符串的请求始终失败。 系统在必须绘制项时将 WM_DRAWITEM 消息发送到所有者窗口。 使用WM_DRAWITEM消息传递的 DRAWITEMSTRUCT 结构的 itemID 成员指定要绘制的项的行号。 无数据列表框不发送 WM_DELETEITEM 消息。
LBS_NOINTEGRALHEIGHT 指定列表框的大小正好是应用程序在创建列表框时指定的大小。 通常,系统调整列表框的大小,以便列表框不显示部分项。对于 具有LBS_OWNERDRAWVARIABLE 样式的列表框,始终强制实施 LBS_NOINTEGRALHEIGHT 样式。
LBS_OWNERDRAWFIXED 指定列表框的所有者负责绘制其内容,并且列表框中的项目高度相同。 创建列表框时,所有者窗口会收到 WM_MEASUREITEM 消息,当列表框的可视方面发生更改时收到 WM_DRAWITEM 消息。
LBS_SORT 按字母顺序对列表框中的字符串进行排序。
LBS_STANDARD 按字母顺序对列表框中的字符串进行排序。 每当用户单击列表框项、双击项目或取消所选内容时,父窗口都会收到通知代码。 列表框具有垂直滚动条,其边框位于右侧。 此样式结合了 LBS_NOTIFY、 LBS_SORT、 WS_VSCROLL和 WS_BORDER 样式。
LBS_USETABSTOPS 允许列表框在绘制其字符串时识别和展开制表符。 可以使用 LB_SETTABSTOPS 消息指定制表位位置。 默认选项卡位置相隔 32 个对话框模板单位。 对话框模板单元是对话框模板中使用的与设备无关的单位。 若要将度量值从对话框模板单位转换为屏幕单位 (像素) ,请使用 MapDialogRect 函数。
LBS_WANTKEYBOARDINPUT 指定每当用户按下键且列表框具有输入焦点时,列表框的所有者将接收 WM_VKEYTOITEM 消息。 这使应用程序能够在键盘输入上执行特殊处理。

控件消息

消息码 说明 WPARAM LPARAM 返回值
LB_ADDSTRING 添加字符串到列表项中 0 LPCTSTR Int
LB_DELETESTRING 删除指定字符串项 nIndex 0 Int
LB_GETCOUNT 获得列表项的部数 0 0 Int
LB_GETCURSEL 获得当前选中项的索引 0 0 Int
LB_GETITEMDATA 获得指定项的附加数据 nIndex 0 DWORD_PTR
LB_GETSEL 获得指定项的选中状态 nIndex 0 Int
LB_GETSELCOUNT 获得多选项的总数 0 0 Int
LB_GETSELITEMS 获得选中项的索引数组 nMaxArray(数组大小) lpArray(数组) Int
LB_GETTEXT 获得指定项的字符串 nIndex LPTSTR Int
LB_GETTEXTLEN 获得指定项字符串长度 nIndex 0 Int
LB_GETTOPINDEX 获得第一个可见项的索引 0 0 Int
LB_INSERTSTRING 插入字符串项 nIndex LPCTSTR Int
LB_RESETCONTENT 删除所有的项 0 0 Void
LB_SELITEMRANGE 范围选择项或取消范围选择 BOOL(选择=TRUE,取消=FALSE) DWORD (L=开始选择项的索引,H=结束项的索引) Int
LB_SETCARETINDEX 设置指定项获得焦点,并可见 nIndex BOOL(FALSE为完全可见,TRUE为部分可见) Int
LB_SETCURSEL 设置当前选中的字符串项 nIndex 0 Int
LB_SETITEMDATA 设置指定项的附加数据 nIndex DWORD_PTR Int
LB_SETSEL 设置指定项的选中状态,并获得焦点和可见 BOOL 为真突出显示,为FALSE取消突出显示 nIndex Int
LB_SETTOPINDEX 设置指定项为可见项 nIndex 0 Int

Demo

1. 单击选项弹出消息框显示选择的项
if (LOWORD (wParam) == ID_LIST && HIWORD(wParam) == LBN_SELCHANGE) {
    int iIndex = SendMessage(hListBox, LB_GETCURSEL, 0, 0);
    int iLength = SendMessage(hListBox, LB_GETTEXTLEN, iIndex, 0) + 1;
    TCHAR * pVarName = (TCHAR*)malloc(iLength*sizeof(TCHAR));
    SendMessage(hListBox, LB_GETTEXT, iIndex, (LPARAM)pVarName);
    MessageBox(hWnd, pVarName, TEXT("选择的内容"), MB_OK);
    free(pVarName);
}
2. ListBox重绘

ListBox重绘,实现QQ 消息列表效果

通过指定LBS_OWNERDRAWFIXED样式,可以进行ListBox的重绘。

两个重要消息:WM_DRAWITEMWM_MEASUREITEM

2.1 WM_DRAWITEM 消息

第二个附加参数lParam里包含了控件的相关信息, 其实际类型是DRAWITEMSTRUCT的指针类型

DRAWITEMSTRUCT 原型:

typedef struct tagDRAWITEMSTRUCT {
  UINT      CtlType;
  UINT      CtlID;
  UINT      itemID;
  UINT      itemAction;
  UINT      itemState;
  HWND      hwndItem;
  HDC       hDC;
  RECT      rcItem;
  ULONG_PTR itemData;
} DRAWITEMSTRUCT, *PDRAWITEMSTRUCT, *LPDRAWITEMSTRUCT;
  • CtlType 指定了控件的类型

image.png

  • CtlID

    组合框、列表框、按钮或静态控件的标识符。 此成员不用于菜单项。

  • itemID

    菜单项的菜单项标识符或列表框或组合框中项的索引。 对于空列表框或组合框,此成员可以是 -1

    这允许应用程序仅在 rcItem 成员指定的坐标处绘制焦点矩形,即使控件中没有项也是如此。 这向用户指示列表框或组合框是否具有焦点。

    itemAction 成员中设置位的方式决定了矩形是否与列表框或组合框具有焦点一样绘制

  • itemAction

    所需的绘图操作。 此成员可以是一个或多个值。

image.png

  • itemState

    当前绘图操作发生后项的视觉状态。 此成员可以是下表中显示的值的组合。

image.png

  • hwndItem

    组合框、列表框、按钮和静态控件的控件的句柄。 对于菜单,此成员是包含该项的菜单的句柄。

  • hDC

    设备上下文的句柄;在控件上执行绘图操作时,必须使用此设备上下文。

  • rcItem

    一个矩形,用于定义要绘制的控件的边界。 此矩形位于 hDC 成员指定的设备上下文中。 系统会自动剪辑所有者窗口在组合框、列表框和按钮的设备上下文中绘制的任何内容,但不会剪辑菜单项。 绘制菜单项时,所有者窗口不得在 rcItem 成员定义的矩形边界之外绘制。

  • itemData

    与菜单项关联的应用程序定义值。 对于控件,此参数指定LB_SETITEMDATA或CB_SETITEMDATA消息最后分配给列表框或组合框的值。 如果列表框或组合框具有 LBS_HASSTRINGS或 CBS_HASSTRINGS样式,则此值最初为零。 否则,此值最初是传递给下列消息之一的 lParam 参数中的列表框或组合框的值:

    • CB_ADDSTRING
    • CB_INSERTSTRING
    • LB_ADDSTRING
    • LB_INSERTSTRING

    如果 CtlTypeODT_BUTTONODT_STATIC, 则itemData 为零。

2.2 WM_MEASUREITEM消息

第二个附加参数lParam描述了相关项的绘制信息,其实际类型是MEASUREITEMSTRUCT的指针类型

typedef struct tagMEASUREITEMSTRUCT {
  UINT      CtlType;
  UINT      CtlID;
  UINT      itemID;
  UINT      itemWidth;
  UINT      itemHeight;
  ULONG_PTR itemData;
} MEASUREITEMSTRUCT, *PMEASUREITEMSTRUCT, *LPMEASUREITEMSTRUCT;

大部分字段同DRAWITEMSTRUCT, 两个重要字段itemWidth, itemHeight 指定列表项的宽度和高度

image.png

2.3 重要代码
void SetListBoxItemRect(LPARAM lParam) {
    PMEASUREITEMSTRUCT pmr = (PMEASUREITEMSTRUCT)lParam;
    if (pmr->CtlID == ID_LIST) {
        pmr->itemHeight = 58;
    }
}

void DrawListBoxItem(LPARAM lParam) {
    TCHAR* pVarName;
    PDRAWITEMSTRUCT pdr = (PDRAWITEMSTRUCT)(lParam);
    COLORREF nColor = RGB(66, 66, 66);
    HBRUSH hBursh = NULL;   
    if(pdr->itemState & ODS_SELECTED ) // 选中状态
        hBursh = CreateSolidBrush(RGB(255, 128, 0));
    else
        hBursh = CreateSolidBrush(RGB(255, 255, 255));
    FillRect(pdr->hDC, &pdr->rcItem, hBursh);
    DeleteObject(hBursh);
    int itemLength = SendMessage(hListBox, LB_GETTEXTLEN, pdr->itemID, 0) + 1;
    pVarName = (TCHAR*)malloc(itemLength * sizeof(TCHAR));
    SendMessage(hListBox, LB_GETTEXT, pdr->itemID, (LPARAM)pVarName);

    if (pVarName == NULL) return;
    HBITMAP hBmp = LoadBitmap(hInst, (char*)info_arr[atoi(pVarName)].bitmapId);
    HDC Bhdc = CreateCompatibleDC(pdr->hDC);
    HGDIOBJ oldBMP = SelectObject(Bhdc, hBmp);
    StretchBlt(pdr->hDC, pdr->rcItem.left + 1, pdr->rcItem.top + 1, 56, 56, Bhdc, 0, 0, 100, 100, SRCCOPY);
    HFONT font1 = CreateFont(20,0,0,0,900,NULL,NULL,NULL,GB2312_CHARSET,NULL,NULL,NULL,NULL,"仿宋");
    HFONT font2 = CreateFont(20,0,0,0,800,NULL,NULL,NULL,GB2312_CHARSET,NULL,NULL,NULL,NULL,"方正舒体");
    SetBkMode(pdr->hDC, TRANSPARENT);
    HGDIOBJ oldfont = SelectObject(pdr->hDC, font1);
    nColor = RGB(0, 0, 0);
    SetTextColor(pdr->hDC, nColor);
    TextOut(pdr->hDC, pdr->rcItem.left + 65, pdr->rcItem.top + 5, info_arr[atoi(pVarName)].userName, strlen(info_arr[atoi(pVarName)].userName));
    SelectObject(pdr->hDC, font2);
    nColor = RGB(133, 193, 246);
    SetTextColor(pdr->hDC, nColor);
    TextOut(pdr->hDC, pdr->rcItem.left + 65, pdr->rcItem.top + 30, info_arr[atoi(pVarName)].newData, strlen(info_arr[atoi(pVarName)].newData));
    SelectObject(pdr->hDC, oldfont);
    SelectObject(Bhdc, oldBMP);
    DeleteObject(hBmp);
    DeleteObject(font1);
    DeleteObject(font2);
    DeleteDC(Bhdc);
    free(pVarName);
}
2.4 实现效果

image.png

参考

  1. 基本使用:https://blog.csdn.net/qq_31178679/article/details/125904122
  2. ListBox风格、消息说明:https://www.cnblogs.com/H-R-J/p/11882224.html
  3. ListBox重绘: https://www.freesion.com/article/3893191680/
  4. ListBox简单使用:https://www.codenong.com/cs109127861/
  5. 微软文档:https://learn.microsoft.com/zh-cn/windows/win32/msi/listbox-control
  6. DRAWITEMSTRUCT 结构: https://learn.microsoft.com/zh-cn/windows/win32/api/winuser/ns-winuser-drawitemstruct

项目地址

相关实践学习
部署高可用架构
本场景主要介绍如何使用云服务器ECS、负载均衡SLB、云数据库RDS和数据传输服务产品来部署多可用区高可用架构。
负载均衡入门与产品使用指南
负载均衡(Server Load Balancer)是对多台云服务器进行流量分发的负载均衡服务,可以通过流量分发扩展应用系统对外的服务能力,通过消除单点故障提升应用系统的可用性。 本课程主要介绍负载均衡的相关技术以及阿里云负载均衡产品的使用方法。
相关文章
|
1月前
|
C# 数据库 开发者
44.c#:combobox控件
44.c#:combobox控件
20 1
|
1月前
|
C# 数据库 虚拟化
43.c#:listbox控件
43.c#:listbox控件
16 1
|
C#
WPF中实现多选ComboBox控件
原文:WPF中实现多选ComboBox控件 在WPF中实现带CheckBox的ComboBox控件,让ComboBox控件可以支持多选。 将ComboBox的ItemsSource属性Binding到一个Book的集合, public class Book { ...
3429 0
WPF 点击 Datagrid 中的TextBox 控件获取其所在行的数据
WPF 点击 Datagrid 中的TextBox 控件获取其所在行的数据
|
索引
QT ComboBox控件 全面详解
本文详细的介绍了QComboBox控件的各种操作,例如:下拉框添加内容、默认显示、获取下拉框总行数、获取选中索引、获取当前内容、清除列表、重绘下拉框等操作。 本系列QT全面详解文章目前共有十五篇,本系列文章较为详细的讲述了QT控件的基础操作和使用,也谢谢大家的关注、点赞、收藏。
666 2
QT ComboBox控件 全面详解
|
C#
WPF TextBox自动滚动到最户一行
原文:WPF TextBox自动滚动到最户一行 textBox经常用来显示程序的运行状态或者消息,如何让他自动滚动呢? 在显示消息代码下加一条自动滚动到底部的语句即可:  TextBox1.ScrollToEnd(); (如果要显示垂直滚动条设置VerticalScrollBarVisibility="Auto",如果不显示设置为Hidden) 我用的程序代码如下:   this.
1987 0
|
C#
【WPF】拖拽ListBox中的Item
原文:【WPF】拖拽ListBox中的Item 整理了两个关于WPF拖拽ListBox中的Item的功能。项目地址 https://github.com/Guxin233/WPF-DragItemInListBox 需求一: 两个ListBox,拖拽其中一个ListBox的Item,放置到另一个ListBox中。
1538 0
|
C#
潜移默化学会WPF(转载篇)--屏幕显示Label,鼠标移上去变成textBox
原文:潜移默化学会WPF(转载篇)--屏幕显示Label,鼠标移上去变成textBox ...
1033 0

热门文章

最新文章