[英]How can I dynamically change auto complete entries in a C# combobox or textbox?
我在 C# 中有一个 combobox 并且我想使用自动完成建议,但是我希望能够将自动完成条目更改为用户类型,因为可能的有效条目太多而无法在启动时填充AutoCompleteStringCollection
。
例如,假设我让用户输入名称。 我有一个可能的名字列表(“Joe”、“John”)和一个姓氏列表(“Bloggs”、“Smith”),但如果我每个都有一千个,那么这将是一百万个可能的字符串 -太多,无法放入自动完成条目。 因此,最初我只想将名字作为建议(“Joe”、“John”),然后一旦用户输入了名字(“Joe”),我想删除现有的自动完成条目并替换他们有一个新的集合,包括选择的名字和可能的姓氏(“Joe Bloggs”、“Joe Smith”)。 为了做到这一点,我尝试了以下代码:
void InitializeComboBox()
{
ComboName.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
ComboName.AutoCompleteSource = AutoCompleteSource.CustomSource;
ComboName.AutoCompleteCustomSource = new AutoCompleteStringCollection();
ComboName.TextChanged += new EventHandler( ComboName_TextChanged );
}
void ComboName_TextChanged( object sender, EventArgs e )
{
string text = this.ComboName.Text;
string[] suggestions = GetNameSuggestions( text );
this.ComboQuery.AutoCompleteCustomSource.Clear();
this.ComboQuery.AutoCompleteCustomSource.AddRange( suggestions );
}
但是,这不能正常工作。 似乎对 Clear() 的调用会导致自动完成机制“关闭”,直到下一个字符出现在组合框中,但是当然当下一个字符出现时,上面的代码再次调用 Clear(),所以用户永远不会实际上看到了自动完成功能。 它还会导致组合框的全部内容被选中,因此在每次按键之间您必须取消选择现有文本,这使其无法使用。 如果我删除对 Clear() 的调用,则自动完成工作,但似乎AddRange()
调用无效,因为我添加的新建议不会出现在自动完成下拉列表中。
我一直在寻找解决方案,并看到了各种建议,但我无法让它们中的任何一个工作 - 自动完成功能似乎被禁用,或者没有出现新字符串。 这是我尝试过的事情的清单:
BeginUpdate()
,之后调用EndUpdate()
。Remove()
而不是 Clear()。AutoCompleteMode
设置为“None”,然后将其设置回“SuggestAppend”。TextUpdate
或KeyPress
事件而不是TextChanged
。AutoCompleteStringCollection
替换现有的AutoCompleteCustomSource
。 这些都没有帮助,即使是各种组合。 Spence建议我尝试覆盖ComboBox
function 以获取要在自动完成中使用的字符串列表。 使用反射器,我在ComboBox
class 中发现了一些看起来很有希望的方法 - GetStringsForAutoComplete()
和SetAutoComplete()
,但它们都是私有的,因此我无法从派生的 ZA2F2ED4F8EBC2CBBD4C21 访问它们。 我不能再进一步了。
我尝试用TextBox
替换ComboBox
,因为自动完成界面是相同的,我发现行为略有不同。 使用TextBox
,它似乎工作得更好,因为自动完成的 Append 部分可以正常工作,但 Suggest 部分不能 - 建议框短暂闪烁,然后立即消失。
所以我想“好吧,我会在没有 Suggest 功能的情况下生活,只使用 Append 代替”,但是当我将AutoCompleteMode
设置为 Append 时,我得到了访问冲突异常。 Suggest 也会发生同样的事情 - 唯一不抛出异常的模式是SuggestAppend
,即使 Suggest 部分的行为不正确。
我认为使用 C# 托管代码时应该不可能出现访问冲突异常。 Avram建议我使用“锁定”来解决此问题,但我不知道我应该锁定什么 - 唯一具有 SyncRoot 成员的是AutoCompleteStringCollection
,并且锁定不会阻止访问冲突异常。 我还尝试锁定ComboBox
或TextBox
,但这也无济于事。 据我了解, lock 只会阻止其他锁,所以如果底层代码没有使用 lock 那么我使用它不会有任何区别。
所有这一切的结果是我目前无法使用具有动态自动完成功能的TextBox
或ComboBox
。 有人对我如何实现这一目标有任何见解吗?
我还没有得到这个工作,但我发现了更多。 也许其中一些会激发其他人提出解决方案。
我尝试用TextBox
替换ComboBox
,因为自动完成界面是相同的,我发现行为略有不同。 使用TextBox
,它似乎工作得更好,因为自动完成的 Append 部分可以正常工作,但 Suggest 部分不能 - 建议框短暂闪烁,然后立即消失。
所以我想“好吧,我会在没有 Suggest 功能的情况下生活,而只使用 Append”,但是当我将AutoCompleteMode
设置为 Append 时,我得到了访问冲突异常。 Suggest 也会发生同样的事情 - 唯一不抛出异常的模式是SuggestAppend
,即使 Suggest 部分的行为不正确。
我认为在使用 C# 托管代码时应该不可能出现访问冲突异常,但无论如何,结果是我目前无法使用具有任何动态自动完成功能的TextBox
或ComboBox
。 有人对我如何实现这一目标有任何见解吗?
在尝试了各种其他事情(例如更改工作线程中的自动完成功能,并使用BeginInvoke()
模拟 PostMessage() 类型的行为之后,我终于放弃了,只是使用列表框实现了我自己的自动完成下拉菜单。 它比内置的响应速度更快,而且我花在这方面的时间比我试图让内置的工作的时间少,所以任何想要这种行为的人的教训是 - 你可能会更好自己实施。
我遇到了同样的问题,并找到了一个非常简单的解决方法。 和这里的其他人一样,我找不到任何方法来控制组件的行为,所以我不得不接受它。
自然的行为是:您不能在用户每次在文本框中键入内容时动态填充列表。 您必须填充一次,然后 AutoComplete 机制取得控制权。 结论是:您应该使用数据库中的每个可能条目填充 AutoCompleteCustomSource 以使其按我们想要的方式工作。
当然,如果您有数百万条记录来填充列表,这是不可行的。 数据传输中的性能问题和自动完成机制本身不允许您这样做。
我找到的折衷解决方案是:每次文本长度恰好达到 N 个字符(在我的情况下为 3 个)时,动态填充 AutoCompleteCustomSource。 这是有效的,因为复杂性大大降低。 从数据库中提取的与这 3 个初始字符匹配的记录数量足够小,可以避免任何性能问题。
主要缺点是:在用户键入第 N 个字符之前,他们不会看到自动完成列表。 但在输入 3 个字符之前,用户似乎并不真正期望一个有意义的自动完成列表。
希望这可以帮助。
这对我addRange
,您不会将范围addRange
到同一个AutoCompleteStringCollection
,而是每次都创建一个新的。
form.fileComboBox.TextChanged += (sender, e) => {
var autoComplete = new AutoCompleteStringCollection();
string[] items = CustomUtil.GetFileNames();
autoComplete.AddRange(items);
form.fileComboBox.AutoCompleteCustomSource = autoComplete;
};
我认为您可能想要退出反射器并查看覆盖组合框本身中的自动完成行为。 我确信自动完成会调用一个访问自动完成列表的函数。 如果你能找到这个函数并覆盖它,你就可以使用任何你想要的行为。
查看您可以在组合框类本身上找到哪些文档。
我没有测试过这个,但它可能值得一试。
不是清除 AutoCompleteCustomSource,而是通过保留两个实例来加倍缓冲。 当文本更改时,调用 GetNameSuggestions() 并为当前未使用的字符串构建字符串,然后将 ComboName.AutoCompleteCustomSource 设置为您刚刚设置的字符串。
我认为它应该看起来像这样。
AutoCompleteCustomSource accs_a;
AutoCompleteCustomSource accs_b;
bool accs_check = true; //true for accs_a, false for accs_b
void InitializeComboBox()
{
ComboName.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
ComboName.AutoCompleteSource = AutoCompleteSource.CustomSource;
accs_a = new AutoCompleteStringCollection();
accs_b = new AutoCompleteStringCollection();
ComboName.AutoCompleteCustomSource = accs_a;
ComboName.TextChanged += new EventHandler( ComboName_TextChanged );
}
void ComboName_TextChanged( object sender, EventArgs e )
{
string text = this.ComboName.Text;
if(accs_check)
{
accs_b.Clear();
accs_b.AddRange(GetNameSuggestions( text ));
accs_check = false;
}
else
{
accs_a.Clear();
accs_a.AddRange(GetNameSuggestions( text ));
accs_check = true;
}
this.ComboQuery.AutoCompleteCustomSource = accs_check? accs_a : accs_b;
}
山姆,你弄明白了吗? 我遇到了同样的情况。 Clear() 似乎导致异常。 我删除了清除呼叫,尽管收藏不断增加,但我收到了正确的建议事件......
此外,关于私有成员:您可以使用反射访问它们:
PropertyInfo[] props = [object].GetType().GetProperties({flags go here});
props[0].SetValue(this, new object[] { 0 });
最好的解决方案是使用组合框的事件处理程序。 通过使用textUpdate KeyDown DropDown和ChangeCommit ,您可以模仿自动完成模式,您可以自定义要搜索的内容以及显示在下拉列表中的内容。
我发现这个答案很有用,但它是用 Visual C++ 编码的,它是 toolstripcombobox 但概念是相同的。 无论如何,.net 中的 c# 和 c++ 有很大的相似性,理解解决方案应该不成问题。
我来这里最初是为了寻找解决方案,但现在找到了自己的解决方案。
诀窍不是在 AutoCompleteCustomSource 上调用 Clear() 而是删除 for 循环中的所有项目,然后用新数据重建列表。 就我而言(图书收藏应用程序),我从数据库中检索作者姓名,其中包含特定的起始字母,而不是全部。 请注意,这仅在组合框的文本框部分为空或已为空时才有效。
private void cboAuthor_KeyDown(object sender, KeyEventArgs e)
{
if (cboAuthor.Text.Length == 0)
{
// Next two lines simple load data from the database in the
// into a collection (var gateway), base on first letter in
// the combobox. This is specific to my app.
var gateway = new AuthorTableGateway();
gateway.LoadByFirstLetter(Char.ConvertFromUtf32(e.KeyValue)[0]);
// Clear current source without calling Clear()
for (int i = 0; i < authorsAutoComplete.Count; i++)
authorsAutoComplete.RemoveAt(0);
// Rebuild with new data
foreach (var author in gateway)
authorsAutoComplete.Add(author.AuthorName);
}
}
我知道这是一个非常古老的问题,但它今天仍然存在。 我的解决方法是将自动完成模式和源属性设置为“无”并手动更新 KeyUp 事件上的项目。
我确定它很笨拙,但无论输入数据的速度如何,它对我来说都是完美的,并且在很长一段时间内都没有问题,而且我的头发开始重新长出额外的好处。
您还可以选择是仅建议,还是建议并附加。 我希望它可以帮助某人。
private void comboBox1_KeyUp(object sender, KeyEventArgs e)
{
if (string.IsNullOrWhiteSpace(comboBox1.Text))
{
e.Handled = true;
return;
}
if (comboBox1.Text.Length < 3)
{
e.Handled = true;
return;
}
if (e.KeyCode == Keys.Down || e.KeyCode == Keys.Up)
{
e.Handled = true;
return;
}
else if (e.KeyCode == Keys.Back)
{
e.Handled = true;
return;
}
string text = comboBox1.Text;
if (e.KeyCode == Keys.Enter)
{
comboBox1.DroppedDown = false;
comboBox1.SelectionStart = text.Length;
e.Handled = true;
return;
}
List<string> LS = Suggestions(comboBox1.Text);
comboBox1.Items.Clear();
comboBox1.Items.AddRange(LS.ToArray());
//If you do not want to Suggest and Append
//comment the following line to only Suggest
comboBox1.Focus();
comboBox1.DroppedDown = true;
comboBox1.SelectionStart = text.Length;
//Prevent cursor from getting hidden
Cursor.Current = Cursors.Default;
e.Handled = true;
}
还没有尝试过,但对于您的具体情况,您可以编写如下代码:
private void txtAutoComplete_KeyUp(object sender, KeyEventArgs e)
{
String text = txtAutoComplete.Text;
if (text.EndsWith(" "))
{
string[] suggestions = GetNameSuggestions( text ); //put [text + " "] at the begin of each array element
txtAutoComplete.AutoCompleteCustomSource.Clear();
txtAutoComplete.AutoCompleteCustomSource.AddRange( suggestions );
}
}
更新:把锁放在这个地方的主要原因是
它的工作:) 在这个技巧消失之后,我曾经拥有的大部分“神秘异常”
private void Form1_Load(object sender, EventArgs e)
{
textBox1.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
textBox1.AutoCompleteSource = AutoCompleteSource.CustomSource;
textBox1.TextChanged+=new EventHandler(textBox1_TextChanged);
col1.AddRange(new string[] { "avi avi", "avram avram" });
col2.AddRange(new string[] { "boria boria", "boris boris" });
textBox1.AutoCompleteCustomSource = col1;
textBox1.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
}
AutoCompleteStringCollection col1 = new AutoCompleteStringCollection();
AutoCompleteStringCollection col2 = new AutoCompleteStringCollection();
object locker = new object();
private void textBox1_TextChanged(object sender, EventArgs e)
{
lock (locker)
{
if (textBox1.Text.StartsWith("a") && textBox1.AutoCompleteCustomSource != col1)
{
textBox1.AutoCompleteCustomSource = col1;
}
if (textBox1.Text.StartsWith("b") && textBox1.AutoCompleteCustomSource != col2)
{
textBox1.AutoCompleteCustomSource = col2;
}
}
}
在尝试了这里提供的所有解决方案(没有成功)之后,我发现了一些对我有用的东西:
private void CellBox_TextChanged(object sender, EventArgs e)
{
((TextBox)sender).TextChanged -= CellBox_TextChanged;
((TextBox)dataGridView1.EditingControl).AutoCompleteMode = AutoCompleteMode.None;
((TextBox)dataGridView1.EditingControl).AutoCompleteCustomSource = null;
aCSC.Clear();
foreach (string value in Autocompletevalues())
{
aCSC.Add(value);
}
((TextBox)dataGridView1.EditingControl).AutoCompleteCustomSource = aCSC;
((TextBox)dataGridView1.EditingControl).AutoCompleteMode = AutoCompleteMode.Suggest;
((TextBox)sender).TextChanged += CellBox_TextChanged;
}
脚步:
我希望它可以帮助某人..
我测试了所有解决方案都没有成功,并且还发生了崩溃(AccessViolation)并开始寻找可行的解决方案。 我找到了一种让它动态工作的方法,并在我的博客中进行了解释:
您需要同时继承 TextBox 和 AutoCompleteStringCollection 。
对我来说,秘密是使用 TextChanged 事件,而没有使用 KeyDown/Up/Press 等。
更新:在动态更改 AutoCompleteCustomSource 遇到其他问题后,我最终放弃了使用内置自动完成功能,并在比我最初浪费在它上面的时间短得多的时间内实现了我自己的功能。 实现 ComboBox 控件的非托管代码中似乎存在一些问题。 具体来说,我在 TextChanged 事件处理程序应该触发时遇到了问题。 我决定在我的自定义实现中只使用 OnKeyDown/Press/Up 处理程序,这似乎更可靠。
if(!textBox3.AutoCompleteCustomSource.Contains(textBox3.Text))
textBox3.AutoCompleteCustomSource.Add(textBox3.Text);
使用此代码
private void dataGridView1_EditingControlShowing(object sender,DataGridViewEditingControlShowingEventArgs e)
{
if (e.Control is DataGridViewComboBoxEditingControl)
{
((ComboBox)e.Control).DropDownStyle = ComboBoxStyle.DropDown;
((ComboBox)e.Control).AutoCompleteSource = AutoCompleteSource.ListItems;
((ComboBox)e.Control).AutoCompleteMode = System.Windows.Forms.AutoCompleteMode.Suggest;
}
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.