C#中combobox如何实现模糊查询,并能自动显示下拉列表

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C#中combobox如何实现模糊查询,并能自动显示下拉列表相关的知识,希望对你有一定的参考价值。

1.用一个List<string> listOnit存放初始化数据,用一个List<string> listNew存放输入key之后,返回的数据。

2.用上面的listOnit初始化ComboBox数据源进行绑定。

3.在TextUpdate方法内部,添加实现方法。

首先进入方法,先清除ComboBox的内容,然后将输入的内容去listOnit初始化的数据中比对,找出对应数据,然后放入listNew存放数据,最后将listNew数据重新赋值给ComboBox。

 

后台代码实现:

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Linq;

using System.Text;

using System.Windows.Forms;

namespace TimerDemo

    public partial class Form2 : Form

    

        //初始化绑定默认关键词(此数据源可以从数据库取)

        List<string> listOnit = new List<string>();

        //输入key之后,返回的关键词

        List<string> listNew = new List<string>();

 

        public Form2()

        

            InitializeComponent();

        

 

        private void Form2_Load(object sender, EventArgs e)

        

            //调用绑定

            BindComboBox();

        

        /// <summary>

        /// 绑定ComboBox

        /// </summary>

        private void BindComboBox()

        

            listOnit.Add("张三");

            listOnit.Add("张思");

            listOnit.Add("张五");

            listOnit.Add("王五");

            listOnit.Add("刘宇");

            listOnit.Add("马六");

            listOnit.Add("孙楠");

            listOnit.Add("那英");

            listOnit.Add("刘欢");

 

            /*

             * 1.注意用Item.Add(obj)或者Item.AddRange(obj)方式添加

             * 2.如果用DataSource绑定,后面再进行绑定是不行的,即便是Add或者Clear也不行

             */

            this.comboBox1.Items.AddRange(listOnit.ToArray());

        

 

        private void comboBox1_TextChanged(object sender, EventArgs e)

        

            /*

              

             * 不能用TextChanged操作,当this.comboBox1.DroppedDown为True时,选择项上下键有冲突

              

             */

 

        

 

        private void comboBox1_TextUpdate(object sender, EventArgs e)

        

            //清空combobox

            this.comboBox1.Items.Clear();

            //清空listNew

            listNew.Clear();

            //遍历全部备查数据

            foreach (var item in listOnit)

            

                if (item.Contains(this.comboBox1.Text))

                

                    //符合,插入ListNew

                    listNew.Add(item);

                

            

            //combobox添加已经查到的关键词

            this.comboBox1.Items.AddRange(listNew.ToArray());

            //设置光标位置,否则光标位置始终保持在第一列,造成输入关键词的倒序排列

            this.comboBox1.SelectionStart = this.comboBox1.Text.Length;

            //保持鼠标指针原来状态,有时候鼠标指针会被下拉框覆盖,所以要进行一次设置。

            Cursor = Cursors.Default;

            //自动弹出下拉框

            this.comboBox1.DroppedDown = true;

        

    

   


实现效果截图:

从左到右模糊查询:(例如输入:张)

可以得出正常模糊查询的结果。

从左到右模糊查询(例如输入:三)

实现过程中的问题:

  1.绑定数据一开始用的DataSource方式,但是写到下面重新给ComboBox设置数据源的时候,报错:不能为已经设置DataSource的combobox赋值。

      解决方式:将赋值方式改为:Item.Add(obj)或者Item.AddRange(obj)方式

  2.下拉框的内容一直在增加

      解决方式:当文本框文本改变时,清空下拉框的内容,然后再添加数据。

  3.输入文本改变时,没有自动弹出下拉框显示已经查询好的数据。

      解决方式:设置comboBox的DroppedDown 属性为True。

  4.ComboBox文本框改变事件一开始选择用的是TextChanged事件,但是当在界面用 上 下键盘选择时,出现bug,不能进行选择。

      解决方式:将文本框改变事件换为TextUpdate事件,然后添加实现方法。

  5.当在ComboBox输入内容时,内容文本是倒序输出的,光标位置始终在最前面。

      解决方式:设置光标的显示位置,this.comboBox1.SelectionStart = this.comboBox1.Text.Length;

  6.输入内容改变时,用鼠标选择下拉列表项的时候,鼠标指针消失,被下拉框覆盖掉。

      解决方式:设置鼠标状态为一开始的默认状态,Cursor = Cursors.Default;

参考技术A combobox没有这样的功能,都是自己实现的,比如在onchanged或者textchanged事件里做处理,即时改变combobox的内容,至于自动显示下拉列表,只要把dropped设成true就行了。本回答被提问者和网友采纳 参考技术B 模糊查询 用sql语句控制就行 like '%张%'
自动填充下拉列表这个得自己做算法.在Change()时间中写.实时查询,只加载赋值前几条到列表中
参考技术C 使用正则表达式做判断
public void ThinkInquire(DataSet dataset)

Regex r = new Regex(".*" + be_carNo.Text.Trim() + ".*");//正则表达式
for (int i = 0; i < dataset.Tables[0].Rows.Count; i++)

if (r.IsMatch(dataset.Tables[0].Rows[i]["carno"].ToString()))

be_carNo.Properties.Items.Add(dataset.Tables[0].Rows[i]["carno"]);//绑定下拉框的值




然后用判断做弹出和收回
if (be_carNo.Properties.Items.Equals(null)||be_carNo.Text.Trim()=="")

be_carNo.ClosePopup();//关闭下拉框
be_carNo.Properties.Items.Clear();

else
be_carNo.ShowPopup();//弹出下拉框


记得把控件的自我查询关闭
参考技术D OnChange,OnKeyDown,选一个事件,动态更新弹出的列表。

防止 System.Windows.Forms.ComboBox 的自动选择行为(C#)

【中文标题】防止 System.Windows.Forms.ComboBox 的自动选择行为(C#)【英文标题】:Prevent AutoSelect behavior of a System.Window.Forms.ComboBox (C#) 【发布时间】:2014-10-30 04:07:06 【问题描述】:

背景:

我有一个Forms.ComboBox 和一个DropDownStyle = DropDown

我不使用AutoComplete,但我实现了类似的东西,它不仅过滤文本的开头,而且使用正则表达式并显示与输入的文本匹配的所有项目。这很好用。

但是,当我键入匹配项的第一个字母时,ComboBox 回退到其原始行为并设置DroppedDown = true 并自动选择第一个条目并完成文本以匹配所选项目(类似于@ 987654328@ 附加)。我想要的是没有自动选择和自动完成。

到目前为止,我发现,我必须以某种方式阻止 SendMessage()CB_FINDSTRING 被调用,并将 CB_FINDSTRING 替换为 CB_FINDSTRINGEXACT (MSDN Link)。

我想我必须扩展 ComboBox 类,但我不确定我必须重写哪些方法。我正在使用 C# .NET Framework v3.5。

问题:

如何扩展 Windows.Forms.ComboBox 以防止自动选择行为?

链接:

How can I prevent auto-select in ComboBox on drop-down except for exact matches?(没有帮助我)

【问题讨论】:

【参考方案1】:

试试这个:

using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using System.Windows.Forms;

using Opulos.Core.Win32;

namespace Opulos.Core.UI 

// Extension class to disable the auto-select behavior when a combobox is in DropDown mode.
public static class ComboBoxAutoSelectEx 

    public static void AutoSelectOff(this ComboBox combo) 
        Data.Register(combo);
    

    public static void AutoSelectOn(this ComboBox combo) 
        Data data = null;
        if (Data.dict.TryGetValue(combo, out data)) 
            data.Dispose();
            Data.dict.Remove(combo);
        
    

    private class Data 
        // keep a reference to the native windows so they don't get disposed
        internal static Dictionary<ComboBox, Data> dict = new Dictionary<ComboBox, Data>();

        // a ComboBox consists of 3 windows (combobox handle, text edit handle and dropdown list handle)
        ComboBox combo;
        NW nwList = null; // handle to the combobox's dropdown list
        NW2 nwEdit = null; // handle to the edit window

        internal void Dispose() 
            dict.Remove(this.combo);
            this.nwList.ReleaseHandle();
            this.nwEdit.ReleaseHandle();
        

        public static void Register(ComboBox combo) 
            if (dict.ContainsKey(combo))
                return; // already registered

            Data data = new Data()  combo = combo ;
            Action assign = () => 
                if (dict.ContainsKey(combo))
                    return; // already assigned

                COMBOBOXINFO info = COMBOBOXINFO.GetInfo(combo); // new COMBOBOXINFO();
                //info.cbSize = Marshal.SizeOf(info);
                //COMBOBOXINFO2.SendMessageCb(combo.Handle, 0x164, IntPtr.Zero, out info);

                dict[combo] = data;
                data.nwList = new NW(combo, info.hwndList);
                data.nwEdit = new NW2(info.hwndEdit);
            ;

            if (!combo.IsHandleCreated)
                combo.HandleCreated += delegate  assign(); ;
            else
                assign();

            combo.HandleDestroyed += delegate 
                data.Dispose();
            ;
        
    

    private class NW : NativeWindow 
        ComboBox combo;
        public NW(ComboBox combo, IntPtr handle) 
            this.combo = combo;
            AssignHandle(handle);
        

        private const int LB_FINDSTRING = 0x018F;
        private const int LB_FINDSTRINGEXACT = 0x01A2;

        protected override void WndProc(ref Message m) 
            if (m.Msg == LB_FINDSTRING) 
                m.Msg = LB_FINDSTRINGEXACT;
            

            base.WndProc(ref m);

            if (m.Msg == LB_FINDSTRINGEXACT) 
                String find = Marshal.PtrToStringAuto(m.LParam);
                for (int i = 0; i < combo.Items.Count; i++) 
                    Object item = combo.Items[i];
                    if (item.Equals(find)) 
                        m.Result = new IntPtr(i);
                        break;
                    
                
            
        
    

    private class NW2 : NativeWindow 
        public NW2(IntPtr handle) 
            AssignHandle(handle);
        

        private const int EM_SETSEL = 0x00B1;
        private const int EM_GETSEL = 0x00B0;

        protected override void WndProc(ref Message m) 
            if (m.Msg == EM_SETSEL) 
                // if this code is not here, then the entire combobox text is selected
                // which looks ugly, especially when there are multiple combo boxes.
                //
                // if this method returns immediately, then the caret position is set
                // to (0, 0). However, it seems that calling EM_GETSEL has a side effect
                // that the caret position is mostly maintained. Sometimes it slips back
                // to (0, 0).
                SendMessage(Handle, EM_GETSEL, IntPtr.Zero, IntPtr.Zero);
                //int selStart = (sel & 0x00ff);
                //int selEnd = (sel >> 16) & 0x00ff;
                //Debug.WriteLine("EM_GETSEL: " + selStart + " nEnd: " + selEnd);
                return;
            
            base.WndProc(ref m);
        

        [DllImportAttribute("user32.dll", SetLastError=true)]
        private static extern int SendMessage(IntPtr hWnd, int msg, IntPtr wParam, IntPtr lParam);
    



[StructLayout(LayoutKind.Sequential)]
public struct COMBOBOXINFO 
    public Int32 cbSize;
    public RECT rcItem;
    public RECT rcButton;
    public int buttonState;
    public IntPtr hwndCombo;
    public IntPtr hwndEdit;
    public IntPtr hwndList;

    public static COMBOBOXINFO GetInfo(ComboBox combo) 
        COMBOBOXINFO info = new COMBOBOXINFO();
        info.cbSize = Marshal.SizeOf(info);
        SendMessageCb(combo.Handle, 0x164, IntPtr.Zero, out info);
        return info;
    

    [DllImport("user32.dll", EntryPoint = "SendMessageW", CharSet = CharSet.Unicode)]
    private static extern IntPtr SendMessageCb(IntPtr hWnd, int msg, IntPtr wp, out COMBOBOXINFO lp);


//[StructLayout(LayoutKind.Sequential)]
//public struct RECT 
//  public int Left;
//  public int Top;
//  public int Right;
//  public int Bottom;
//


【讨论】:

这正是我想要的。完美运行。非常感谢! 这对我也很有效。谢谢讨厌! 你如何使用这个?这对我来说并不明显。当我尝试时,我得到“尝试读/写受保护的内存”。也许我做错了 @PandaWood 您可能需要将构建选项从x86 更改为Any CPU。见***.com/questions/4074585/… RECT 从何而来? VS解决不了...

以上是关于C#中combobox如何实现模糊查询,并能自动显示下拉列表的主要内容,如果未能解决你的问题,请参考以下文章

c# 实现ComboBox自动模糊匹配

C# combobox 模糊搜索。

ComboBox如何实现对数据库模糊查询!!!

miniui-ComboBox如何实现对数据库模糊查询!!!

怎样在Winform内实现ComboBox模糊查询

EXCEL VBA combobox 模糊查询触发后 退格键功能改变