c# 调用WinAPI 设置Combobox选择项时手动触发CBN_SELCHANGE事件的方法

c# 调用WinAPI 设置Combobox选择项时手动触发CBN_SELCHANGE事件的方法

选中包含指定文本的Combobox选项

最近因为开发一个桌面系统插件,需要根据指定的文本内容自动选中窗口上的Combobox。逻辑上不复杂,先获取到Combobox的句柄,SendMessage发送CB_GETCOUNT获取Combobox中的选项数,再遍历发送CB_GETLBTEXTLEN获取每个选项的文本长度,发送CB_GETLBTEXT读取每个选项的文本内容,查询是否包含指定的文本,如果包含,那么就发送CB_SETCURSEL选中该选项。
关键代码如下:

if (Handler != IntPtr.Zero)
   {
       int itemCount = SendMessage(Handler, CB_GETCOUNT, 0, 0);

       List<string> items = new List<string>();
       for (int i = 0; i < itemCount; i++)
       {
           var len = SendMessage(Handler, CB_GETLBTEXTLEN, i, 0);
           if (len > 0)
           {
               var sb = new StringBuilder(len);
               SendMessage(Handler, CB_GETLBTEXT, i, sb);
               items.Add(sb.ToString());
           }
       }
       var item = items.FirstOrDefault(x => x.Contains(text));
       if (item != null)
       {
           SendMessage(Handler, CB_SETCURSEL, items.IndexOf(item), 0);
       }
   }

触发SelectChange

但是实际情况比较复杂,窗体上有多个Combobox,并且其他的Combobox可选择的项目会随着第一个Combobox的选中项目而发生变化(类似省、市、区的级联选择)。使用以上代码的话,第一个Combobox选中指定的项目后,并没有触发其他Combobox选择项的同步变化,也就是没有触发SelectChange事件!

搜索了一下,也有一些解决方法,但大多是C++或者其他语言的实现,没有找到C#的。最后
最终解决方法是再手动发送WM_COMMAND到父窗口。代码如下:

if(Handler != IntPtr.Zero)
   {
       SendMessage(GetParent(Handler), WM_COMMAND, GetDlgCtrlID(Handler)+(CBN_SELCHANGE << 16), Handler);
   }

每次设置CB_SETCURSEL后立即再发送一次CBN_SELCHANGE,问题解决!

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
你可以使用 Windows API 通过 C# 将字符串写入剪贴板。以下是一个示例代码: ```csharp using System.Runtime.InteropServices; public class ClipboardHelper { [DllImport("user32.dll", SetLastError = true)] public static extern bool OpenClipboard(IntPtr hWndNewOwner); [DllImport("user32.dll", SetLastError = true)] public static extern bool CloseClipboard(); [DllImport("user32.dll", SetLastError = true)] public static extern bool EmptyClipboard(); [DllImport("user32.dll", SetLastError = true)] public static extern IntPtr SetClipboardData(uint uFormat, IntPtr hMem); [DllImport("kernel32.dll", SetLastError = true)] public static extern IntPtr GlobalAlloc(uint uFlags, UIntPtr dwBytes); [DllImport("kernel32.dll", SetLastError = true)] public static extern IntPtr GlobalLock(IntPtr hMem); [DllImport("kernel32.dll", SetLastError = true)] public static extern bool GlobalUnlock(IntPtr hMem); [DllImport("kernel32.dll", SetLastError = true)] public static extern UIntPtr GlobalSize(IntPtr hMem); private const uint CF_UNICODETEXT = 13; public static void SetText(string text) { if (!OpenClipboard(IntPtr.Zero)) { throw new System.ComponentModel.Win32Exception(); } try { // 清空剪贴板 if (!EmptyClipboard()) { throw new System.ComponentModel.Win32Exception(); } // 分配全局内存,并锁定内存区域 int byteCount = (text.Length + 1) * 2; // 两个字节表示一个 Unicode 字符 IntPtr hGlobal = GlobalAlloc(0x2000, (UIntPtr)byteCount); // 0x2000 表示 GMEM_MOVEABLE if (hGlobal == IntPtr.Zero) { throw new System.ComponentModel.Win32Exception(); } try { IntPtr lpMem = GlobalLock(hGlobal); if (lpMem == IntPtr.Zero) { throw new System.ComponentModel.Win32Exception(); } try { // 将字符串复制到内存区域 byte[] bytes = System.Text.Encoding.Unicode.GetBytes(text); Marshal.Copy(bytes, 0, lpMem, bytes.Length); // 将内存区域设置为剪贴板数据 if (SetClipboardData(CF_UNICODETEXT, hGlobal) == IntPtr.Zero) { throw new System.ComponentModel.Win32Exception(); } // 在 Unlock 前不要释放内存,因为 SetClipboardData 后剪贴板会接管内存的控制权 } finally { GlobalUnlock(lpMem); } } catch { GlobalFree(hGlobal); throw; } } finally { CloseClipboard(); } } } ``` 在上述代码中,SetText() 方法使用 CF_UNICODETEXT 格式将字符串写入剪贴板。你可以调用 SetText("abced") 将字符串 "abced" 写入剪贴板。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值