WPF ComboBox with IsEditable=“True” - How can I indicate that no match was found?

前端 未结 3 734
无人共我
无人共我 2021-01-05 12:57

Using the following simple text box as an example:


    Angus/ComboBoxItem         


        
相关标签:
3条回答
  • 2021-01-05 13:29

    As a starter, you might want to let the user see if they are typing in one of the available options.

    1) Search "autocomplete combobox" online.

    2) Check these out:

    http://weblogs.asp.net/okloeten/archive/2007/11/12/5088649.aspx

    http://www.codeproject.com/KB/WPF/WPFCustomComboBox.aspx

    3) Also try this:

        <ComboBox IsEditable="true" TextSearch.TextPath="Content">
            <ComboBoxItem Content="Hello"/>
            <ComboBoxItem Content="World"/>
        </ComboBox>
    

    The above code snippet is a primite way to provide that "visual indication" you're looking for. If the user types in 'h', then 'hello' will appear in the input textbox. However, this on its own won't have a mechanism to stop the user from typing in an illegal character.

    4) This is a more advanced version:

        <ComboBox Name="myComboBox" IsEditable="true" KeyUp="myComboBox_KeyUp">
            <ComboBoxItem Content="Hello"/>
            <ComboBoxItem Content="World"/>
            <ComboBoxItem Content="WPF"/>
            <ComboBoxItem Content="ComboBox"/>
        </ComboBox>
    

    Code-behind:

        private void myComboBox_KeyUp(object sender, KeyEventArgs e)
        {
            // Get the textbox part of the combobox
            TextBox textBox = myComboBox.Template.FindName("PART_EditableTextBox", myComboBox) as TextBox;
    
            // holds the list of combobox items as strings
            List<String> items = new List<String>();
    
            // indicates whether the new character added should be removed
            bool shouldRemove = true;
    
            for (int i = 0; i < myComboBox.Items.Count; i++)
            {
                items.Add(((ComboBoxItem)myComboBox.Items.GetItemAt(i)).Content.ToString());
            }
    
            for (int i = 0; i < items.Count; i++)
            {
                // legal character input
                if(textBox.Text != "" && items.ElementAt(i).StartsWith(textBox.Text))
                {
                    shouldRemove = false;
                    break;
                }
            }
    
            // illegal character input
            if (textBox.Text != "" && shouldRemove)
            {
                textBox.Text = textBox.Text.Remove(textBox.Text.Length - 1);
                textBox.CaretIndex = textBox.Text.Length;
            }
        }
    

    Here, we don't let the user continue typing in once we detect that no combobox item starts with the text in the textbox. We remove the character added and wait for another character.

    0 讨论(0)
  • 2021-01-05 13:49

    This is a good solution until you have a lot of records in the combo box. I would do it this way:

    Declare this at the top of the file

    List<String> items = new List<String>(); 
    
    private void myComboBox_KeyUp(object sender, KeyEventArgs e) 
    { 
      TextBox textBox = myComboBox.Template.FindName("PART_EditableTextBox", myComboBox) as TextBox; 
    
         // indicates whether the new character added should be removed 
        bool shouldRemove = true; 
    
        // this way you don't fill the list for every char typed
        if(items.Count <= 0)
        { 
           for (int i = 0; i < myComboBox.Items.Count; i++) 
           { 
               items.Add(((ComboBoxItem)myComboBox.Items.GetItemAt(i)).Content.ToString()); 
           } 
        }
        // then look in the list
        for (int i = 0; i < items.Count; i++) 
        { 
            // legal character input 
            if(textBox.Text != "" && items.ElementAt(i).StartsWith(textBox.Text)) 
            { 
                shouldRemove = false; 
                break; 
            } 
        } 
    
        // illegal character input 
        if (textBox.Text != "" && shouldRemove) 
        { 
            textBox.Text = textBox.Text.Remove(textBox.Text.Length - 1); 
            textBox.CaretIndex = textBox.Text.Length; 
        } 
    } 
    

    unless the binding is continue adding records to the combo box, I think is a more efficient lookup

    0 讨论(0)
  • 2021-01-05 13:52

    This solution is based on user1234567's answer, with a couple changes. Instead of searching the Items list it simply checks the ComboBox's SelectedIndex for a value >= 0 to see if a match was found and it resolves RB's concern about holding down a key inserting more than one character. It also adds an audible feedback when it rejects characters.

    private int _lastMatchLength = 0;
    
    private void myComboBox_GotFocus(object sender, RoutedEventArgs e)
    {
        _lastMatchLength = 0;
    }
    
    private void myComboBox_KeyUp(object sender, KeyEventArgs e)
    {
        ComboBox cBox = sender as ComboBox;
    
        TextBox tb = cBox.Template.FindName("PART_EditableTextBox", cBox) as TextBox;
        if (tb != null)
        {
            if (cBox.SelectedIndex >= 0)
            {
                _lastMatchLength = tb.SelectionStart;
            }
            else if (tb.Text.Length == 0)
            {
                _lastMatchLength = 0;
            }
            else
            {
                System.Media.SystemSounds.Beep.Play();
                tb.Text = tb.Text.Substring(0, _lastMatchLength);
                tb.CaretIndex = tb.Text.Length;
            }
        }
    }
    
    0 讨论(0)
提交回复
热议问题