Событие comboBox_TextChanged из ComboBox не всегда срабатывает - PullRequest
1 голос
/ 01 ноября 2019

У меня есть приложение ввода, которое выбирает целочисленное значение из ComboBox. Функция Random имеет начальное значение, поэтому при каждом запуске приложения генерируется одна и та же последовательность значений. Путем ввода TYPING в ComboBox, если первые 2 цифры образуют значение, содержащееся в списке ComboBox.Items, и добавляется третья цифра, чтобы результирующее значение НЕ содержалось в списке ComboBox.Items, обработчик события _availableValuesForSelection_TextChanged отклоняет эту цифру,Первые 2 цифры остаются. Проблема состоит в том, что если одна и та же цифра добавляется несколько раз, в конечном итоге полученное значение принимается. Мой вывод заключается в том, что после нескольких раз событие _availableValuesForSelection_TextChanged из ComboBox не запускается. Попробуйте добавить 231, где значение 23 существует в списке ComboBox.Items, а также 230, но не 231. Код:

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 IntegerSelection
{
    public partial class Form_SelectInteger : Form
    {
        public Form_SelectInteger(List<int> selectableValuesSource)
        {
            InitializeComponent();

            _validValue = "";
            _selectableValuesSource = selectableValuesSource;
            _availableValuesForSelection = new ComboBox();
            // The Source should be set first and then the AutoCompleteMode, otherwise it throws a NotSupportedException.
        _availableValuesForSelection.BindingContext = new BindingContext(); // Create a new context.
        _availableValuesForSelection.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
        _availableValuesForSelection.AutoCompleteSource = AutoCompleteSource.ListItems;
        _availableValuesForSelection.DropDownStyle = ComboBoxStyle.DropDown;
        _availableValuesForSelection.DroppedDown = false;
        _availableValuesForSelection.DrawMode = DrawMode.Normal;
        _availableValuesForSelection.Enabled = true;
        _availableValuesForSelection.BackColor = Color.LightBlue;
        _availableValuesForSelection.ForeColor = Color.DarkBlue;
        _availableValuesForSelection.IntegralHeight = false;
        _availableValuesForSelection.MaxDropDownItems = 1;
        _availableValuesForSelection.MaxDropDownItems = 80;
        _availableValuesForSelection.DropDownHeight = 510;
        _availableValuesForSelection.DropDownWidth = 50;
        _availableValuesForSelection.TabStop = true;
        _availableValuesForSelection.Visible = true;
        _availableValuesForSelection.Width = 100; 
        _availableValuesForSelection.Location = new Point((this.Width - _availableValuesForSelection.Width) / 2, 150);
        _availableValuesForSelection.DataSource = _selectableValuesSource; // DataSource should be the last line.
        _availableValuesForSelection.SelectedIndex = 5;
        this.Controls.Add(_availableValuesForSelection);

        _button_SelectionCompleteWasPressed = false;

        _availableValuesForSelection.TextChanged += new EventHandler(_availableValuesForSelection_TextChanged);  

        _selectedValue = (int)_availableValuesForSelection.Items[_availableValuesForSelection.SelectedIndex];
    }

    void _availableValuesForSelection_TextChanged(object sender, EventArgs e)
    {
            _textChangedEventDisabled = true;
            string resultingText = _availableValuesForSelection.Text;
            textBox1.Text = resultingText;
            if (_availableValuesForSelection.FindString(resultingText) == -1)
            {
                while (_availableValuesForSelection.FindString(resultingText) == -1)
                {
                    resultingText = resultingText.Substring(0, resultingText.Length - 1);
                }
                _availableValuesForSelection.DroppedDown = false;                    
                _availableValuesForSelection.SelectedIndex = _availableValuesForSelection.FindString(resultingText);
                _availableValuesForSelection.Text = resultingText;
                _availableValuesForSelection.SelectionStart = _availableValuesForSelection.Text.Length;
                _availableValuesForSelection.SelectionLength = 0;                    
            }
            _textChangedEventDisabled = false;            
    }

    private ComboBox _availableValuesForSelection;
    private List<int> _selectableValuesSource;

    private int _selectedValue;
    public int SelectedValue
    {
        get
        {
            return _selectedValue;
        }
    }

    private string _validValue;
    bool _textChangedEventDisabled = false;

    private bool _button_SelectionCompleteWasPressed;

    private void ValuesForSelection_SelectedIndexChanged(object sender, EventArgs e)
    {
        if (sender is ComboBox)
        {
            ComboBox displayValuesForSelection = (ComboBox)sender;
            if (_button_SelectionCompleteWasPressed)
            {
                _selectedValue = (int)displayValuesForSelection.Items[displayValuesForSelection.SelectedIndex];
                DialogResult = DialogResult.OK;
            }
        }
    }

    private void button_SelectionComplete_Click(object sender, EventArgs e)
    {
        DialogResult = DialogResult.OK; // This line sets the DialogResult to OK to let the other form know that it wasn't forced closed or canceled.
        _availableValuesForSelection.AutoCompleteMode = AutoCompleteMode.SuggestAppend;
        _availableValuesForSelection.AutoCompleteSource = AutoCompleteSource.ListItems;
        _availableValuesForSelection.DropDownStyle = ComboBoxStyle.DropDown;
        _button_SelectionCompleteWasPressed = true;
        _selectedValue = (int)_availableValuesForSelection.Items[_availableValuesForSelection.SelectedIndex];
        this.Close();
    }
}

}

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 IntegerSelection
{
    public partial class Form_Main : Form
    {
    Random rnd = new Random(0);

    public Form_Main()
    {
        InitializeComponent();

        PopulateValuesForSelection();
        Form_SelectInteger integerSelection = new Form_SelectInteger(_selectableValues);
        DialogResult result = integerSelection.ShowDialog(); // This line calls it as a Modal dialog, and retrieves its response once it's closed.
        if (result == DialogResult.OK) // Test to see if the DialogResult was "OK". 
        {
             _selectedValue = integerSelection.SelectedValue;
             textBox1.ReadOnly = true;
            textBox1.Text = _selectedValue.ToString();
        }
        else
        {
            this.Close();
            Application.Exit();
        }
    }

    private List<int> _selectableValues;
    private int _selectedValue;

    /// <summary>
    /// The values included into _selectableValues must be unique (no duplicates are allowed).
    /// </summary>
    private void PopulateValuesForSelection()
    {
        _selectableValues = new List<int>();
        HashSet<int> uniqueValues = new HashSet<int>();
        int i = 0;
        while (i < 256)
        { 
            int currentValue = rnd.Next(0, 1000);
            List<int> constituentDigits = SplitNumberIntoDigits(currentValue);
            HashSet<int> digits = new HashSet<int>(constituentDigits);
            if (!constituentDigits.Contains(-1) && uniqueValues.Add(currentValue))
            {
                i = uniqueValues.Count;
                _selectableValues.Add(currentValue);
            }
        }
        _selectableValues.Sort();
    }

    private List<int> SplitNumberIntoDigits(int number)
    {
        List<int> constituentDigits = new List<int>();
        while (number > 0)
        {
            int digit = number % 10; // Get the last digit in the number.
            constituentDigits.Insert(0, digit); // Insert the digit in the first location in the list, as it is the digit preceding the digit that is now in the first location in the list.
            number = number / 10; // Shorten the integer by one digit, from the right (the least significant digit).
        }

        return constituentDigits;
    }
}

}

Пожалуйста, помогите. Заранее спасибо.

1 Ответ

1 голос
/ 04 ноября 2019

Событие TextChanged возникает при обновлении значения свойства Text. Изучив это, кажется, что при отклонении значения после ввода недопустимого символа оно все еще сохраняется где-то внутри объекта и не вызывает событие при повторных попытках.

Принимая во внимание, что событие TextUpdate инициируетсякаждый раз, когда текст изменяется, прежде чем он будет показан. Это используется для проверки ввода текста и не запускается, когда свойство Text изменяется программно. Что звучит именно так, как вы хотите.

Посмотрите документацию: Событие TextUpdate

Ваш код будет работать, просто изменив одну строку:

_availableValuesForSelection.TextUpdate += new EventHandler(_availableValuesForSelection_TextChanged);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...