Jak pobrać SelectedValue z comboBoxa?

0

Witam
W projekcie stworzyłem combobox o nazwie typComboBox do którego dodaję obiekty typu :

 
 class TypWyposazenia
        {
            public int id_typu { get; set; }
            public string typ { get; set; }
        }

Zmieniam również :

 
typcomboBox.DisplayMember = "typ";
typcomboBox.ValueMember = "id_typu";

Niestety mimo ,że teoretycznie powinienem móc pobrać id_typu wybranego itemu to np. przy poleceniu:

 
typcomboBox.SelectedValue.ToString();

Otrzymuję wyjątek NullReferenceException.

Czy ktoś wie może gdzie popełniam błąd?

0

to zależy jeszcze do którego obiektu rzucany jest ten wyjątek,

  1. Może do ComboBox'a nie załadowałeś obiektów TypWyposażenia
  2. Może załadowany ComboBox jeszcze nie ma żadnego zaznaczonego elementu w sobie i jego własciwość SelectedValue odnosi się do null'a
  3. Może masz jeszcze nie utworzony ComboBox a juz sie do niego odnosisz w kodzie.

3 przypadek jest łatwy do przeoczenia dla okna Form np. tak:

 
        public Form2()   //domyslny konstruktor drugiej formy
        {
            InitializeComponent(); // tu za kulisami jest instrukcja tworzenia kontrolek przenoszonych metodą drag&drop w designerze
        }

        public Form2(object parametr1, object parametr2)   //nasz wlasny konstruktor tego okna, w ktorym przekazujemy do tworzonego okna parametry
        {
            WypelnijComboBoxaDanymi();           //wewnetrzna metoda przypisujaca np. w Twoim przypadku obiekty TypWyposazenia do combo
            //(...) inne Twoje metody
            // ale w tym konstruktorze brakuje na poczatku wywolania metody InitializeComponent() i to moze byc problemem
        }


0

Raczej nie w tym problem.
Do wartości value odwołuje się dla zdarzenia :

 
private void typcomboBox_SelectedIndexChanged(object sender, EventArgs e)
        {
            TypWyposazenia selectedItem = (TypWyposazenia)typcomboBox.SelectedItem;
            string i = selectedItem.id_typu.ToString();
            MessageBox.Show(i);
///////////////////////////////
           label12.Text = typcomboBox.SelectedValue.ToString();
        }

Więc combobox zawiera jakiś wybrany element

Do obiektu mogę się odwołać przez selectedItem, więc na pewno wybrana jest jakaś wartość ,tylko ,że ta metoda nie jest zbyt wygodna. Poza tym combobox ma być finalnie połączony z bazą danych prze binding. W tabeli mam właśnie dwie kolumny, id_typu i typy.
id_typu będzie wrzucana do innej tabeli ,a ten combobox ma "tłumaczyć" id_typu na string typ.
Chcę móc zmieniać programowo wybrany item właśnie korzystając z selectedValue.

0

Nie rozumiem o co Ci chodzi. Kod który podałeś nie działa? Powinien.

0

Również nie rozumiem Twojego problemu, czy w końcu coś rzuca wyjątek czy coś, gdzieś jest niewygodne i na czym polega ta niewygoda?

BindingSource nie jest taki straszny wiec polecam od razu spróbować, wtedy do bieżącego zaznaczonego elementu w ComboBox który jako DataSoruce bedzie miał BindingSource, bedziesz mogł się odwoływać przez BindingSource.Current.

0

To może inaczej. Tak wygląda błąd jaki dostaję:
2015-01-04_23h32_46.png

Oczywiście mogę pobrać tą wartość tą drugą metodą przez rzutowanie Itemu :

TypWyposazenia selectedItem = (TypWyposazenia)typcomboBox.SelectedItem;
            string i = selectedItem.id_typu.ToString();
            MessageBox.Show(i);
           

Ale takie rozwiązanie wydaje się mi bardziej intuicyjne, bo i po co ustawiał bym wcześniej ValueMember :

string i = typcomboBox.SelectedValue.ToString();

Ale pomijając już kwestię odczytywania wybranego itemu ,to jak rozwiązać kwestię pokazywania odpowiedniego itemu po bindowaniu skoro binding będzie zawierał (int) id_typu, a ja chcę wyświetlać (string) typ?
Do combobox'a wkładam obiekty klasy TypWyposazenia, ale w jaki sposób mam bindować ten wybrany item skoro z tabeli pobieram tylko (int) id_typu.

Czyli po koleii:

  1. w pewnej tabeli mam dwie kolumny, id_typu i typ. Używam jej tylko do wypełnienia combobox'a
  2. W combobox chcę wyświetlać wartość z kolumny innej tabeli w której będę miał zapisaną już tylko wartość (int) id_typu, natomiast chcę żeby użytkownik widział w combobox'ie nie numerki typu tylko nazwy do nich przypisane.

Sory jeżeli trochę mieszam. Marny jestem w tłumaczeniu :)

0

Binding powinien działać bez problemu z takimi ustawieniami jakie masz, jeśli podepniesz sie do SelectedValue np:

 typcomboBox.DataBindings.Add( new Binding("SelectedValue",...

Wszystko wskazuje na to, że w SelectedValue ustawiłeś wartość, której nie ma na liście Itemów w ComboBoxie.

Na Twoim screenie podejrzane jeszcze jest to co robisz w typTextBox_TextChanged: Skoro w combo są obiekty klasy TypWyposazenia to nie bedziesz mógł wybrać inta.

0
Wielki Szczur napisał(a):

Binding powinien działać bez problemu z takimi ustawieniami jakie masz, jeśli podepniesz sie do SelectedValue np:

 typcomboBox.DataBindings.Add( new Binding("SelectedValue",...

Wszystko wskazuje na to, że w SelectedValue ustawiłeś wartość, której nie ma na liście Itemów w ComboBoxie.

Na Twoim screenie podejrzane jeszcze jest to co robisz w typTextBox_TextChanged: Skoro w combo są obiekty klasy TypWyposazenia to nie bedziesz mógł wybrać inta.

W typTextBox_TextChanged odwołuje się do TextBoxa w którym pojawia mi się string z intem id_typu. To w comboboxie trzymam obiekty.

Podepnołem comboboxa do bindingu :
2015-01-05_13h55_19.png

Niestety zmiana zaznaczonego w DataGridView wiersza nie wpływa na zmianę wybranego w combobox itemu.
Dodatkowo przy próbie wykożystania comboboxa do edycji danych w tabeli dostaję taki błąd:2015-01-05_13h54_53.png
W tabeli faktycznie kolumna typ musi mieć wartość różną od NULL.
Wygląda więc na to ,że coś jest nie tak z tym:

typcomboBox.ValueMember = "id_typu"; 
0

Zamieszczam wszystko co tyczy się tego combobox'a.

Tak wyglada baza danych z której pobieram elementy:
2015-01-05_22h12_50.png

Tak wygląda klasa dla obiektów które umieszczam w comboboxie

class TypWyposazenia
        {
            public int id_typu { get; set; }
            public string typ { get; set; }
        }

Tak wygląda funkcja ,która wypełnia mój combobox:

void wypełnijTypy()
        {
            string query_pokoje = "select * FROM `motel`.`typ_wyposazenia`;";
            MySqlConnection myConn = new MySqlConnection(myConnection);
            MySqlCommand komenda = new MySqlCommand(query_pokoje, myConn);
            MySqlDataReader reader;
            typcomboBox.Items.Clear();



            try
            {
                myConn.Open();
                reader = komenda.ExecuteReader();
                //pobieram liste pokoi

                while (reader.Read())
                {
                    typcomboBox.Items.Add(new TypWyposazenia
                    {
                        id_typu = reader.GetInt32(0),
                        typ = reader.GetString(1)
                    });


                }
                reader.Close();
                myConn.Close();
            }

            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
            typcomboBox.SelectedItem = 1;

        }

To umieściłem przy inicjalizacji okna:

 typcomboBox.DisplayMember = "typ";
            typcomboBox.ValueMember = "id_typu";
            wypełnijTypy();

Teraz problem polega na tym jak ustawiać zaznaczony item posługując się (int) id_typu.

0

Już wiem o co chodzi.
Musisz wypełnić combobox przez DataSource, czyli:

                List<TypWyposazenia> list = new List<TypWyposazenia>();
                 while (reader.Read())
                {
                    list.Add(new TypWyposazenia()
                    {
                        id_typu = reader.GetInt32(0),
                        typ = reader.GetString(1)
                    });
                }
                typcomboBox.DataSource = list;

@Javik lub lepiej użyć bindowania w designerze. Zaznaczasz combobox w prawym górnym rogu comboboxa będziesz miał trójkącik i tam ustawiasz połączenie z bazą, DisplayMember i ValueMember

0
dam1an napisał(a):

Już wiem o co chodzi.
Musisz wypełnić combobox przez DataSource, czyli:

                List<TypWyposazenia> list = new List<TypWyposazenia>();
                 while (reader.Read())
                {
                    list.Add(new TypWyposazenia()
                    {
                        id_typu = reader.GetInt32(0),
                        typ = reader.GetString(1)
                    });
                }
                typcomboBox.DataSource = list;

@Javik lub lepiej użyć bindowania w designerze. Zaznaczasz combobox w prawym górnym rogu comboboxa będziesz miał trójkącik i tam ustawiasz połączenie z bazą, DisplayMember i ValueMember

Dzięki dam1an. Skorzystałem z te drugiej metody i działa dokładnie tak jak chciałem.
A jaki jest najprostszy sposób ,aby pobrać z tego combobox'a selecetedValue skoro jest to Obiekt?
Nie ma prostszego sposobu niż ten?:

 TypWyposazenia selectedItem = (TypWyposazenia)typcomboBox.SelectedItem;
            string i = selectedItem.id_typu.ToString();
            MessageBox.Show(i);
0

Udało się tą metodą:

 Int32 i = (Int32)typcomboBox.SelectedValue;

gdzie Rzutowanie na zwykłego Inta już nie dzialało.
Dziękuję wszystkim za pomoc.

1 użytkowników online, w tym zalogowanych: 0, gości: 1