一、前言

  又临近春节,作为屌丝的我,又要为车票发愁了。记得去年出现了各种12306的插件,最近不忙,于是也着手自己写个抢票插件,当是熟悉一下WinForm吧。小软件还在开发中,待完善后,也写篇博客与大家分享。今天我们的重点不是抢票软件,而是其中的一点功能。我们在买票的时候选站点的自动补全如下图:

这功能在WinForm里用什么控件来实现呢?

一、自带控件  

  WinForm里面的ComBoBox 和TextBox 其实是有自带的自动补全功能的,我们只需要将设置相应的属性:
       1、将  AutoCompleteSource 属性设置为 ListItems 或 CustomerSource (textbox 没有 ListItems)
         2、设置 AutoCompleteMode  自动完成样式属性设置,有三值 Suggest(显示相关下拉)、append(自动补全相关)、suggestappend(前两者的结合),这个可以自行试验下。
         3、然后设置 绑定 控件的 DataSource 或 AutoCompleteCustomSource。
     
当AutoCompleteSource属性设置的是 CustomerSource 的时候我们需要绑定 AutoCompleteCustomSource属性的值,值为一个string类型的数组:
                     this.cbbEndStation.AutoCompleteCustomSource.AddRange(new string[] { "站点1", "站点2", "站点3", "站点4" });

这样ComBoBox 和 TextBox 就有输入提示功能了。至此,不知道大家有没有发现问题,这里绑定的数据只有 显示的值,而没有 实际的值,一般像这种控件,我们都是有一个显示值和一个实际值的。有人可能会说,使用ComBoBox 控件,然后将AutoCompleteSource设置为ListItems,提示的就是DataSource里的值了,而DataSource是可以绑定 集合,设置DisplayMember和ValueMember的。是的,这样可以实现自动提示,并且也能在选中提示的某项时,取到显示的值和实际值。但是这种方式至少有两个缺点:

1、像购票的站点这种,数据量很大,有2k多条吧,你一次全绑定到ComboBox上?数据量太大,它没有提供相应的事件来过滤数据。

2、多种搜索方式怎么办?中文、拼音、实际值、都是是可以用来做输入提示的关键字的。

其实以上两点就是应为 没有提供相应的事件来处理 “搜索”

二、TextBox+ListBox 自定义AutoComplete

其实我可以用 TextBox来获得用户的输入,然后动态控制ListBox。下面就按我做的思路一步步来实现一个自定义AutoComplete。

1、监听 textbox的 keyUp事件,获得用户输入

         /// <summary>
/// 站点文本框 键盘按下松开事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void txtStation_KeyUp(object sender, KeyEventArgs e)
{
TextBox eObj = sender as TextBox; //事件源对象
txtStation_Name = eObj; //当前事件出发对象
if (eObj.Name == "txtStation_S_Name")
{
txtStation_Value = this.txtStation_S_Value; //保存值的textbox
ltb_Stations = this.lb_Start_Stations; //始发站 展示数据的
}
else
{
//到站 控件
txtStation_Value = this.txtStation_E_Value; //保存值的textbox
ltb_Stations = this.lb_End_Stations; //始发站 展示数据的
}
//上下左右
if (e.KeyCode == Keys.Up || e.KeyCode == Keys.Left)
{
if (ltb_Stations.SelectedIndex > )
ltb_Stations.SelectedIndex--;
}
else if (e.KeyCode == Keys.Down || e.KeyCode == Keys.Right)
{
if (ltb_Stations.SelectedIndex < ltb_Stations.Items.Count - )
ltb_Stations.SelectedIndex++;
}
//回车
else if (e.KeyCode == Keys.Enter)
{
StationInfo info = ltb_Stations.SelectedItem as StationInfo;
txtStation_Name.Text = info.StationName_CN;
txtStation_Value.Text = info.StationValue;
ltb_Stations.Visible = false;
}
else
{ if (txtStation_Name.Text != "")
{
IList<StationInfo> dataSource = StationInfo.GetStations(txtStation_Name.Text.Trim());
if (dataSource.Count > )
{
ltb_Stations.DataSource = dataSource;
ltb_Stations.DisplayMember = "StationName_CN";
ltb_Stations.ValueMember = "StationValue";
ltb_Stations.Visible = true;
}
else
ltb_Stations.Visible = false;
}
else
{
ltb_Stations.Visible = false;
}
}
txtStation_Name.Select(txtStation_Name.Text.Length, ); //光标定位到文本框最后
}

2、监听 ListBox 控件的点击事件

         /// <summary>
/// 展示站点列表的listbox的点击事件,为了给textbox赋值
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void ListBox_StationDatas_Click(object sender, EventArgs e)
{
ListBox eObj = sender as ListBox;
StationInfo info = eObj.SelectedItem as StationInfo;
txtStation_Name.Text = info.StationName_CN;
txtStation_Value.Text = info.StationValue;
eObj.Visible = false;
txtStation_Name.Select(txtStation_Name.Text.Length, ); //光标定位到最后
}

3、监听 ListBox 控件的鼠标移动事件

          /// <summary>
/// 展示站点列表的listbox, 鼠标在该控件上移动事件,
/// 为了鼠标移动选项
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void ListBox_StationDatas_MouseMove(object sender, MouseEventArgs e)
{
ListBox eObj = sender as ListBox;
eObj.SelectedIndex = eObj.IndexFromPoint(e.Location);
}

以上三步就可以完成一个自定义 AutoComplete的功能了,为什么要叫自定义呢?因为我们在 监听 TextBox 的输入时,可以自定义搜索规则,还有我们可以将ListBox换成 DataGridView都是可以的,灵活性很大,只要按这个思路来就可以。

三、一些第三方控件

当然网络上也有一些非常好的类似AutoComplete的第三方控件,这里我就不一一列出来了,因为我没有找到合适的,呵呵!如果有人用过好的,欢迎在评论中分享,谢谢!

最后附上Demo的源码:点击这里下载!
demo截图:

如发现文中有误,或有更好的建议,欢迎指出!

WinForm AutoComplete 输入提示、自动补全的更多相关文章

  1. angular-ui-bootstrap typeahead 智能提示 自动补全 获取焦点不触发问题的解决

    项目中有一处使用了angular-ui-bootstrap中的typeahead来实现输入框智能提示语自动化补全的功能,存在一个bug, 即输入文字后,当再次点击文本框,其获取焦点后并不会触发智能提示 ...

  2. 【Eclipse】eclipse自动提示+自动补全

    解决代码的自动提示问题: 1.打开 Eclipse -> Window -> Perferences 2.找到Java 下的 Editor 下的 Content Assist , 右边出现 ...

  3. 【学习】eclipse自动提示+自动补全

    解决代码的自动提示问题: 1.打开 Eclipse -> Window -> Perferences 2.找到Java 下的 Editor 下的 Content Assist , 右边出现 ...

  4. PyCharm编辑HTML文件时输入{%不能自动补全

    在PyCharm编辑HTML文件时输入Django模板语言时,发现录入 {% 不能自动补全. 找了一下,发现 setting 里可以设置 Python Template Languages,选择自己使 ...

  5. Codeblock代码提示自动补全(包括结构体成员)

    转:https://blog.csdn.net/haibin8473/article/details/79113650

  6. jQuery搜索框自动补全功能插件实现-autocomplete.js

    最近用nodeclub实现股票的输入关键字自动补全股票信息进行搜索功能,原先用jQuery-ui,结果jQuery-ui库太大,所以考虑用其他插件,最终选择使用autocomplete.js,控件简单 ...

  7. vim下使用YouCompleteMe实现代码提示、补全以及跳转设置

    配置YouCompleteMe 1. 安装vundle vundle是一个管理vim插件的工具,使用vundle安装YouCompleteMe比较方便. 按照作者在https://github.com ...

  8. Vim使用YouCompleteMe达到类似IDE的代码提示、补全,以及其他实用设置

    接触Linux有两年了,vim还是只会简单的操作.最近实在受不了sublime的代码提示,决定花点时间来配置下vim.本文讲自己认为方便的vim配置,称不上完美,只讲究简单实用. 使用 ctags 主 ...

  9. vim自动补全

    Vim 中使用 OmniComplete 为 C/C++ 自动补全 OmniComplete 并不是插件的名字,而是 Vim 众多补全方式中的一种(全能补全).说白了 OmniComplete 其实就 ...

随机推荐

  1. linux 高级编程之库的使用

    一.静态库与动态库 静态库: .a .lib 动态库: .so .dll 差别(静态库中的代码在链接时就已经复制到可执行文件中,执行时不再依赖库,不会自动使用升级后的库,需要重新产生可执行文件. 动态 ...

  2. Android开发环境搭建及常见问题解决方法

    转自: http://www.cnblogs.com/rwxwsblog/p/4769785.html 在移动互联网的时代,Android的份额早已超过了苹果.Android的出现无疑加速了移动互联网 ...

  3. Spring如何处理线程并发

    Spring如何处理线程并发   我们知道Spring通过各种DAO模板类降低了开发者使用各种数据持久技术的难度.这些模板类都是线程安全的,也就是说,多个DAO可以复用同一个模板实例而不会发生冲突.我 ...

  4. SSM三大框架整合详细教程(Spring+SpringMVC+MyBatis)

    使用 SSM ( Spring . SpringMVC 和 Mybatis )已经有三个多月了,项目在技术上已经没有什么难点了,基于现有的技术就可以实现想要的功能,当然肯定有很多可以改进的地方.之前没 ...

  5. [CareerCup] 10.6 Find Duplicate URLs 找重复的URL链接

    10.6 You have 10 billion URLs. How do you detect the duplicate documents? In this case, assume that ...

  6. Jenkins进阶系列之——09配置Linux系统ssh免密码登陆

    ssh认证的完整描述:https://www.ibm.com/developerworks/cn/linux/security/openssh/part1/ 说明:点我去查看 今天我们只说生成ssh的 ...

  7. 学习笔记——Maven settings.xml 配置详解

    文件存放位置 全局配置: ${M2_HOME}/conf/settings.xml 用户配置: ${user.home}/.m2/settings.xml note:用户配置优先于全局配置.${use ...

  8. 如何启动一个已经创建的docker 容器,并进入SHELL 对其操作

    腾讯云使用自己的docker镜像安装后无法启动,下边这个亲测是可用的 sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-keys 36A ...

  9. Android Studio上方便使用butterknife注解框架的偷懒插件Android Butterknife Zelezny

    首先提下ButterKnifey已经更新到版本7.0.1了,现在注解已经不叫@InjectView了,而叫@Bind,感觉更贴合语义.同时注册的方式也从 ButterKnife.inject(this ...

  10. bcd 8421码

    bcd码表: 比如一个字符串 String s = "0200" 按对照表转换成二进制 02 : 0000 0010 00 : 0000 0000 s转换为字节的时候 02和00分 ...