其实你的问题只是Win32可以解决的小问题:
public class Form1 : Form {
[DllImport("user32")]
private static extern int GetComboBoxInfo(IntPtr hwnd, out COMBOBOXINFO comboInfo);
struct RECT {
public int left, top, right, bottom;
}
struct COMBOBOXINFO {
public int cbSize;
public RECT rcItem;
public RECT rcButton;
public int stateButton;
public IntPtr hwndCombo;
public IntPtr hwndItem;
public IntPtr hwndList;
}
public Form1(){
InitializeComponent();
comboBox1.HandleCreated += (s, e) => {
COMBOBOXINFO combo = new COMBOBOXINFO();
combo.cbSize = Marshal.SizeOf(combo);
GetComboBoxInfo(comboBox1.Handle, out combo);
hwnd = combo.hwndList;
init = false;
};
}
bool init;
IntPtr hwnd;
NativeCombo nativeCombo = new NativeCombo();
//This is to store the Rectangle info of your Icons
//Key: the Item index
//Value: the Rectangle of the Icon of the item (not the Rectangle of the item)
Dictionary<int, Rectangle> dict = new Dictionary<int, Rectangle>();
public class NativeCombo : NativeWindow {
//this is custom MouseDown event to hook into later
public event MouseEventHandler MouseDown;
protected override void WndProc(ref Message m)
{
if (m.Msg == 0x201)//WM_LBUTTONDOWN = 0x201
{
int x = m.LParam.ToInt32() & 0x00ff;
int y = m.LParam.ToInt32() >> 16;
if (MouseDown != null) MouseDown(null, new MouseEventArgs(MouseButtons.Left, 1, x, y, 0));
}
base.WndProc(ref m);
}
}
//DropDown event handler of your comboBox1
private void comboBox1_DropDown(object sender, EventArgs e) {
if (!init) {
//Register the MouseDown event handler <--- THIS is WHAT you want.
nativeCombo.MouseDown += comboListMouseDown;
nativeCombo.AssignHandle(hwnd);
init = true;
}
}
//This is the MouseDown event handler to handle the clicked icon
private void comboListMouseDown(object sender, MouseEventArgs e){
foreach (var kv in dict) {
if (kv.Value.Contains(e.Location)) {
//Show the item index whose the corresponding icon was held down
MessageBox.Show(kv.Key.ToString());
return;
}
}
}
//DrawItem event handler
private void comboBox1_DrawItem(object sender, DrawItemEventArgs e) {
//We have to save the Rectangle info of the item icons
Rectangle rect = new Rectangle(0, e.Bounds.Top, e.Bounds.Height, e.Bounds.Height);
dict[e.Index] = rect;
//Draw the icon
//e.Graphics.DrawImage(yourImage, rect);
}
}
关于发生了什么
ComboBox 有一个附加的下拉列表,可以通过其Handle 访问。我们使用 GetComboBoxInfo win32 api 函数来检索ComboBox 的一些信息,这些信息保存在一个名为 COMBOBOXINFO 的结构中。我们可以通过这个结构中的成员hwndList获取下拉列表的Handle。
在访问 hwndList 之后。我们可以使用自定义的 NativeWindow 类(在示例中为 NativeCombo)挂钩到它的消息循环。这使我们可以轻松地干扰 下拉列表 的消息循环。然后我们可以捕获 WM_LBUTTONDOWN 消息来处理 MouseDown 事件。当然,这不是一个full MouseDown事件,但它只是一个演示,在这种情况下它足以解决问题。 WM_LBUTTONDOWN 与保存在LParam 中的点击点 的一些信息一起发送。点击点在下拉列表的客户区坐标中计算(不是在屏幕坐标中)。我们应该注意到 DrawItem 事件处理程序还有 e.Bounds 参数,该参数也是在 客户区坐标(不是屏幕坐标)中计算的。因此它们在同一个坐标系中。我们使用Rectangle.Contains 方法来了解点击的点 是否包含在某个图标的边界 中。我们将所有 icon Bounds 存储在 Dictionary 中。所以包含点击点的Rectangle对应的Key(存储Item index)让我们知道对应的项目索引,然后我们可以进一步处理。