Swing JComboBox im AbstractTableModel

Melamed

Mitglied
Ziel:
Ich lese aus einer Datenbank Werte in eine JTable. Zwei dieser Werte hole ich aus anderen Tabellen und soll sie mit einer JComboBox auswählen können. Einer der Werte ist Boolean zum ankreuzeln.
Insgesamt hat die in der JTable auszugebende Tabelle 6 Spalten.

Durchführung:
Hab mir aus einem AbstractTableModel alles zusammengebaut, den Renderer für die beiden JComboboxen definiere ich im Frame selbst. Der Code ist jetzt nicht außergewöhnlich, aber ich poste ihn für alle Fälle:
Java:
    private TableColumn tc_land, tc_zweck;
    private JComboBox jcb_land, jcb_zweck;

private void fuelleColumnModels() throws SQLException {
        try {
            
            DBConnection dbconn = new DBConnection(dbkonfig.getVerbindung() + konfiguration.getDatenbankdatei(), dbkonfig.getKlasse());

            ComboTableCellRenderer renderer = new ComboTableCellRenderer();
            jcb_land = new JComboBox(dbconn.getAbfragenVektor("SELECT Land FROM Land"));
            jcb_zweck = new JComboBox(dbconn.getAbfragenVektor("SELECT Zweck FROM Zweck"));
            jcb_land.setRenderer(renderer);
            jcb_zweck.setRenderer(renderer);

            tc_land = jTable.getColumnModel().getColumn(4);
            tc_zweck = jTable.getColumnModel().getColumn(5);
            tc_land.setCellRenderer(renderer);
            tc_zweck.setCellRenderer(renderer);
            tc_land.setCellEditor(new DefaultCellEditor(jcb_land));
            tc_zweck.setCellEditor(new DefaultCellEditor(jcb_zweck));
            
        } catch (ClassNotFoundException ex) {
            Logger.getLogger(DBJTable.class.getName()).log(Level.SEVERE, null, ex);
        }

Problem:
Prinzipiell läuft alles. Ich kann die JComboBoxen anklicken und etwas auswählen. Die Änderungen werden dann auch erfolgreich in die Datenbank übertragen.
Seltsamerweise funktioniert das Anklicken exakt 1 Mal.
Wenn ich die Land-Box anklicke kann ich auswählen, was ich brauche, die Änderung wird übertragen, aber danach funktioniert keine der beiden Boxen. Es gibt keine Fehlermeldung und die normal editierbaren Zellen lassen Änderungen erfolgreich zu.

Ich vermute, dass die JComboBox überschrieben wird, deswegen habe ich mir die API vom AbstractTableModel durchgesehen, konnte jedoch keine Funktionen finden, die den Zellen oder Spaltentypen definieren (die Überschreibung der Methode getColumnClass bewirkt ebenfalls überhaupt nichts)

Der Code im AbstractTableModel sieht folgendermaßen aus:
Java:
   /**
     * Tabelle:
     * (ID INTEGER, Datum DATE, Uhrzeit BOOLEAN, Anzahl INTEGER, LandID INTEGER, ZweckID INTEGER);")
     * @param value
     * @param row
     * @param col
     */
    @Override
    public void setValueAt(Object value, int row, int col) {

        DBConnection dbconn = new DBConnection(currentURL, this.dbkonf.getKlasse());
        int rowcount = dbconn.getRowCount("Besuchergruppe");
        //Zuerst überprüfen, ob schon etwas vorhanden ist.
        //Wenn nicht - Ärger umgehen und einfach sofort anlegen, was sich finden lässt.
        //Es wird übrigens auch aufgerufen, falls die neue Zeile bearbeitet wird
        

        if ((rowcount == 0) || row >= rowcount) {
            try {
                System.out.println("insert");
                initDB();
                // Prepare a statement to insert a record
                //(ID INTEGER, Datum DATE, Uhrzeit BOOLEAN, Anzahl INTEGER, LandID INTEGER, ZweckID INTEGER)
                String sql = "INSERT INTO Besuchergruppe("
                        + "ID,"
                        + "Datum,"
                        + "Uhrzeit,"
                        + "Anzahl,"
                        + "LandID,"
                        + "ZweckID) "
                        + "VALUES(?,?,?,?,?,?)";
                System.out.println("Statement: " + sql);
                PreparedStatement ps = db.prepareStatement(sql);

                System.out.println("prepstat - good");
                // Set the values
                //Gefüllt wird ja nur eine einzige Zelle - Plus: ID bekannt
                //Col ist die gefüllte Zelle
                ps.setInt(1, dbconn.getNewID("Besuchergruppe", "ID"));
                System.out.println(dbconn.getNewID("Besuchergruppe", "ID"));
                System.out.println("col: " + col);
                switch (col) {
                    case 1:
                        try {
                            // set the preparedstatement parameters
                            ps.setDate(2, (Date) ddmmyyyy.parse((String) value)); //Datum
                        } catch (ParseException ex) {
                            Logger.getLogger(MyTableModel.class.getName()).log(Level.SEVERE, null, ex);
                        }
                        break;
                    case 2:
                        ps.setBoolean(3, Boolean.getBoolean((String) value)); //Uhrzeit
                        break;
                    case 3:
                        ps.setInt(4, Integer.parseInt((String) value));
                        break;
                    case 4:
                        ps.setInt(5, Integer.parseInt(dbconn.getIDByInformation("LandID", (String) value, "Land", "Land")));
                        break;
                    case 5:
                        ps.setInt(6, Integer.parseInt(dbconn.getIDByInformation("ZweckID", (String) value, "Zweck", "Zweck")));
                        break;
                }
                System.out.println("Execute erfolgreich:" + ps.execute());


            } catch (SQLException e) {
                System.out.println("SQL-Fehler beim insert");
                System.out.println("Code: " + e.getErrorCode());
                System.out.println(e.getMessage());
            } finally {
                closeDB();
            }
//Hier gehe ich davon aus, dass eine vorhandene Besuchergruppe verändert wird
        } else {
            try {
                PreparedStatement ps = null;

                switch (col) {
                    case 1:
                        ps = db.prepareStatement("UPDATE Besuchergruppe SET Datum = ? WHERE id = ?");
                        try {
                            // set the preparedstatement parameters
                            ps.setDate(1, (Date) ddmmyyyy.parse((String) value)); //Datum
                        } catch (ParseException ex) {
                            Logger.getLogger(MyTableModel.class.getName()).log(Level.SEVERE, null, ex);
                        }
                        break;
                    case 2:
                        ps = db.prepareStatement("UPDATE Besuchergruppe SET Uhrzeit = ? WHERE id = ?");
                        ps.setBoolean(1, Boolean.getBoolean((String) value)); //Uhrzeit
                        break;
                    case 3:
                        ps = db.prepareStatement("UPDATE Besuchergruppe SET Anzahl = ? WHERE id = ?");
                        ps.setInt(1, Integer.parseInt((String) value));
                        break;
                    case 4:
                        ps = db.prepareStatement("UPDATE Besuchergruppe SET LandID = ? WHERE id = ?");
                        ps.setInt(1, Integer.parseInt(dbconn.getIDByInformation("LandID", (String) value, "Land", "Land")));
                        break;
                    case 5:
                        ps = db.prepareStatement("UPDATE Besuchergruppe SET ZweckID = ? WHERE id = ?");
                        ps.setInt(1, Integer.parseInt(dbconn.getIDByInformation("ZweckID", (String) value, "Zweck", "Zweck")));
                        break;
                }
                System.out.println("UPDATE Besuchergruppe SET Anzahl = " + cache.get(row)[3] + " WHERE id = " + cache.get(row)[0]);
                ps.setString(2, cache.get(row)[0]);
                // call executeUpdate to execute our sql update statement
                initDB();
                ps.executeUpdate();
                ps.close();
                
            } catch (SQLException ex) {
                Logger.getLogger(MyTableModel.class.getName()).log(Level.SEVERE, null, ex);
            } finally {
                closeDB();
            }
        }
        setQuery();
        fireTableCellUpdated(row, col);
    }

// All the real work happens here; in a real application,
// we'd probably perform the query in a separate thread.
    public void setQuery() {
        initDB();
        cache = new Vector<String[]>();
        DBConnection dbconn = new DBConnection(currentURL, this.dbkonf.getKlasse());
        rs = null;

        try {
            // Execute the query and store the result set and its metadata
            try {
                rs = statement.executeQuery("SELECT ID, Datum, Uhrzeit, Anzahl, Land, Zweck FROM Besuchergruppe, Land, Zweck WHERE Besuchergruppe.LandID=Land.LandID AND Besuchergruppe.ZweckID = Zweck.ZweckID");

            } catch (SQLException sqe) {
                System.err.println(sqe.getSQLState());

            }
            ResultSetMetaData meta = rs.getMetaData();
            colCount = meta.getColumnCount();

            // Now we must rebuild the headers array with the new column names
            headers = new String[colCount];

            for (int h = 1; h <= colCount; h++) {
                headers[h - 1] = meta.getColumnName(h);

            } // and file the cache with the records from our query. This would
            // not be
            // practical if we were expecting a few million records in response
            // to our
            // query, but we aren't, so we can do this.
            while (rs.next()) {
                String[] record = new String[colCount];

                for (int i = 0; i < colCount; i++) {
                    
                            record[i] = rs.getString(i + 1);

                }
                cache.addElement(record);
            }
            fireTableChanged(null); // notify everyone that we have a new table.

        } catch (Exception e) {
            cache = new Vector(); // blank it out and keep going.
            e.printStackTrace();
        }
    }

Hat jemand eine Idee? Für jede Hilfestellung bin ich sehr dankbar
 

HoaX

Top Contributor
So auf den ersten Blick sehe ich nichts. Bau doch mal Logausgaben ein/benutze den Debugger. Dann siehst du was wann ausgeführt wird und welche Werte verarbeitet werden.
 

DerEisteeTrinker

Bekanntes Mitglied
Schau mal nach, ob du bei der Auswahl bei den ComboBoxen nicht iwo nen Listener entfernst oder die Comboboxen zwar aktualisierst aber nicht ausreichend.
 

Melamed

Mitglied
Ich habe jetzt ein Bisschen herumgespielt, um zu sehen, was passiert.
Wenn ich die Methode getColumnClass überschreibe, um zu sagen, dass Spalte 4+5 JComboBoxen sind, werden sie nach der ersten Überschreibung uneditierbar (habe aber nichts dergleichen definiert).

Die Methode ist inzwischen wieder entfernt.
Derzeit habe ich als Länder Deutschland, Österreich und Polen eingespeichert. Deutschland ist ausgewählt, ich wähle um auf Polen. Keine Auswahl mehr möglich, normale Bearbeitung aber schon (wieder).
Ich editiere Polen nach Österreich. Starte das Programm neu. Österreich ist der eingetragene Wert.

@HoaX:
Es gibt wie gesagt keine Fehlermeldung oder Ähnliches ???:L
Deswegen sitze ich bei dem Problem ja auch fest. Was ich beim Debugger finden kann weiß ich zwar nicht, aber probiere es gleich aus.

@DerEisteeTrinker:
Wie stelle ich das fest?
 

Melamed

Mitglied
Noch etwas Interessantes, das vielleicht hilfreich ist. Die Reihenfolge, in der die Tabelle aufgebaut wird läuft folgendermaßen (kurz zusammengefasst):

1. DB-Verbindung und Füllung der Tabelle mit Werten (TableModel)
2. Spalten 4+5 werden im Frame als Combo-Boxen definiert

- Update -
3. Benutzer wählt einen Eintrag aus der ComboBox
4. Ein TableChange wird abgefeuert, die Tabelle in der Datenbank auf den neuen Stand gebracht und die komplette Tabelle neu geladen
5. Bei der Aktualisierung werden natürlich die aktuellen Werte in die Spalten 4+5 geladen. Die Werte, nicht die JComboBoxen.

Bis jetzt habe ich noch keinen Weg gefunden, wie ich eine JComboBox im AbstractTableModel unterbringe.

Ideen?
 

Michael...

Top Contributor
Wenn ich die Methode getColumnClass überschreibe, um zu sagen, dass Spalte 4+5 JComboBoxen sind, werden sie nach der ersten Überschreibung uneditierbar (habe aber nichts dergleichen definiert).
Eine ComboBox hat in einem TableModel nichts verloren. Ein TableModel enthält Daten - eine JTable inkl. eventueller ComboBoxes repräsentiert diese.
 

Melamed

Mitglied
So, inzwischen hab ich die Lösung herausgefunden...

Im Frame gebe ich zuerst den Frame selbst mit
Java:
mtm = new MyTableModel(dbkonfig, konfiguration.getDatenbankdatei(),this);

Setze die Füllen-Methode auf public:
Java:
public void fuelleColumnModels()

Und rufe ebendiese bei jeder Änderung dann im Model auf:
Java:
@Override
    public void fireTableChanged(TableModelEvent e) {
        try {
            super.fireTableChanged(e);
            this.dbjtable.fuelleColumnModels();
        }
        catch (SQLException ex) {
            Logger.getLogger(MyTableModel.class.getName()).log(Level.SEVERE, null, ex);
        }
    }

Hätte schöner werden können, aber jetzt klappt's ;)
Wer den Code schönen will, darf sich gern bedienen ^^
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
G Problem mit AbstractTableModel und JComboBox AWT, Swing, JavaFX & SWT 8
hannibalstgt Fehler bei JCOMBOBOX GUI Anzeige AWT, Swing, JavaFX & SWT 4
L jComboBox Actionlistener wird beim erstmaligen Befüllen getriggert AWT, Swing, JavaFX & SWT 7
N Keylistener & JComboBox AWT, Swing, JavaFX & SWT 5
R JComboBox abfragen AWT, Swing, JavaFX & SWT 1
Esquid If Befehl bei JComboBox AWT, Swing, JavaFX & SWT 3
kodela Swing Element einer JComboBox ausgrauen AWT, Swing, JavaFX & SWT 3
D JComboBox Strings aus JTextFields zuordnen AWT, Swing, JavaFX & SWT 2
F JComboBox und Model AWT, Swing, JavaFX & SWT 10
I Editierbare JComboBox, die nur Ganzzahlen als Eingabewerte zulässt AWT, Swing, JavaFX & SWT 3
Drachenbauer Swing Wie ändere ich die Farbe der Konturen von jButton und jCombobox? AWT, Swing, JavaFX & SWT 18
Drachenbauer Swing Wie ändere ich den Hintergrund vom Anzeigefeld einer JComboBox, die nicht zum Hineinschreiben offen steht? AWT, Swing, JavaFX & SWT 32
Hatsi09 JCombobox default wert AWT, Swing, JavaFX & SWT 6
D Swing JCombobox Aktion löscht Grafik AWT, Swing, JavaFX & SWT 1
cezary Choice, JComboBox oder JList AWT, Swing, JavaFX & SWT 5
R Swing Java9 NullPointerException bei JComboBox AWT, Swing, JavaFX & SWT 13
ralfb1105 Swing JComboBox update der Daten AWT, Swing, JavaFX & SWT 8
D Swing JCombobox Weiße Leerzeilen AWT, Swing, JavaFX & SWT 10
G JComboBox: Arrow-Button permanent anzeigen AWT, Swing, JavaFX & SWT 4
D Swing Java JComboBox Example AWT, Swing, JavaFX & SWT 16
K Swing JComboBox mit ArrayList füllen AWT, Swing, JavaFX & SWT 4
D Swing "blaues" Element aus JComboBox erhalten AWT, Swing, JavaFX & SWT 6
F JComboBox und Einträge AWT, Swing, JavaFX & SWT 3
gamillton Swing JComboBox mit extra Inhalt + breiteres Popupmenü AWT, Swing, JavaFX & SWT 0
T JPanelsteuerung mit JComboBox AWT, Swing, JavaFX & SWT 5
N Swing Duplikate im Jcombobox vermeiden AWT, Swing, JavaFX & SWT 15
L Swing JComboBox kann nicht erstellt werden! AWT, Swing, JavaFX & SWT 2
C JComboBox Objecte übergeben und Eintrag auswählen AWT, Swing, JavaFX & SWT 3
M Swing JComboBox mit Laufwerksbuchstaben (Windows) AWT, Swing, JavaFX & SWT 6
D Swing Größe einer JComboBox im GridBagLayout aufgrund der maximalen Länge der enthaltenen Daten AWT, Swing, JavaFX & SWT 7
D Swing JComboBox (DefaultComboBoxModel) überschreibt Eintrag beim erstellen AWT, Swing, JavaFX & SWT 0
R Swing Durch JComboBox-Item eine TextArea aktualisieren AWT, Swing, JavaFX & SWT 2
3 JComboBox - Action - Auslesen & Umwandeln AWT, Swing, JavaFX & SWT 9
M Alle Schriftarten in JComboBox AWT, Swing, JavaFX & SWT 5
M Swing JComboBox eigenes Design AWT, Swing, JavaFX & SWT 3
Sin137 JComboBox mit Generics AWT, Swing, JavaFX & SWT 14
R JComboBox erweitern AWT, Swing, JavaFX & SWT 5
S Swing Eigene JComboBox Problem! AWT, Swing, JavaFX & SWT 1
V Editierbare JComboBox AWT, Swing, JavaFX & SWT 2
D Swing JCombobox in einem JTable vorbelegen AWT, Swing, JavaFX & SWT 4
F neuen Menüpunkt in jcombobox einfügen AWT, Swing, JavaFX & SWT 1
F Swing Jcombobox mit manueller Rückgabe AWT, Swing, JavaFX & SWT 4
W JComboBox auffristen nach Listenänderung AWT, Swing, JavaFX & SWT 20
S JComboBox mit mehreren Spalten? AWT, Swing, JavaFX & SWT 6
J JComboBox + ItemListener AWT, Swing, JavaFX & SWT 12
T Swing jComboBox auslesen mit Datenbankanbindung AWT, Swing, JavaFX & SWT 3
S JComboBox aus anderer Klasse füllen (Java-Swing) AWT, Swing, JavaFX & SWT 0
D Swing Erstes Item einer JCombobox in JTable anzeigen AWT, Swing, JavaFX & SWT 2
N JComboBox in JTable [Swing] -> totaler UI-Einsteiger AWT, Swing, JavaFX & SWT 3
H Swing jComboBox Ausgabe -1/null AWT, Swing, JavaFX & SWT 4
C Arrow Farbe bei JComboBox AWT, Swing, JavaFX & SWT 2
H Swing JList/JTable mit JButton, JTextField, Image, JComboBox und JLable AWT, Swing, JavaFX & SWT 2
O JTextfield und JComboBox Wert in SQL Database speichern AWT, Swing, JavaFX & SWT 4
S JComboBox UTF-8 Codierung zuweisen AWT, Swing, JavaFX & SWT 5
A JComboBox mit Array aktualisieren AWT, Swing, JavaFX & SWT 7
S JComboBox nach auswahl erneuern AWT, Swing, JavaFX & SWT 4
S Swing JComboBox mit Listener aktualisieren AWT, Swing, JavaFX & SWT 13
A Swing per JComboBox gewünschtes Attribut auswählen und Komponenten passen sich an AWT, Swing, JavaFX & SWT 7
I JComboBox gibt bei ausgewählten ITem NullPointer an... AWT, Swing, JavaFX & SWT 3
D Swing JCombobox - Tooltip Location ändern AWT, Swing, JavaFX & SWT 4
J JComboBox Dropdown Icon AWT, Swing, JavaFX & SWT 2
M Swing Mix JComboBox - JButton? AWT, Swing, JavaFX & SWT 6
A Swing bei Auswahl und klick eines JComboBox buttons die Klasse eines anderen Projekts aufrufen AWT, Swing, JavaFX & SWT 3
F Swing jComboBox auslesen AWT, Swing, JavaFX & SWT 6
O Swing TableCellRenderer und JComboBox AWT, Swing, JavaFX & SWT 4
F Swing JComboBox - Frage zur Größe AWT, Swing, JavaFX & SWT 11
C Swing JComboBox, ItemListener gibt 2x aus AWT, Swing, JavaFX & SWT 2
O JComboBox - autom. Selektieren AWT, Swing, JavaFX & SWT 6
J JComboBox - wie getSelectedItem() überschreiben? AWT, Swing, JavaFX & SWT 8
S Swing Jcombobox und array AWT, Swing, JavaFX & SWT 6
D Swing JComboBox aktualisieren AWT, Swing, JavaFX & SWT 4
E JComboBox AWT, Swing, JavaFX & SWT 8
N Swing JComboBox Frage AWT, Swing, JavaFX & SWT 5
S Swing Bild auf jPanel nach Änderung von JComboBox zeichnen AWT, Swing, JavaFX & SWT 4
H Swing Element aus JComboBox auswählen AWT, Swing, JavaFX & SWT 2
H JCombobox inhalt löschen AWT, Swing, JavaFX & SWT 17
N Swing JCombobox - PopupMenu-Inhalt mit KSKB AWT, Swing, JavaFX & SWT 2
O JComboBox mit ArrayList füllen AWT, Swing, JavaFX & SWT 3
S Swing JComboBox mit Datenbank füllen AWT, Swing, JavaFX & SWT 16
S Inhalt einer JComboBox aktualisieren AWT, Swing, JavaFX & SWT 6
F Swing JComboBox in JTable AutoComplete + Tab AWT, Swing, JavaFX & SWT 4
O JComboBox getSelectedItem AWT, Swing, JavaFX & SWT 4
M JComboBox Hintergrundfarbe des gewählten Items AWT, Swing, JavaFX & SWT 3
B Swing Problem: Horizontaler Scrollbalken in JComboBox hinzufügen AWT, Swing, JavaFX & SWT 4
M JCombobox mit ID und Text AWT, Swing, JavaFX & SWT 4
A JComboBox-Inhalt durch neues Array ersetzen AWT, Swing, JavaFX & SWT 2
C Swing Dynamische JComboBox (Filter) AWT, Swing, JavaFX & SWT 28
M JComboBox Item-Auswahl in JTable AWT, Swing, JavaFX & SWT 2
L Jbutton + jcombobox mit vorhandenen frame verknüfen AWT, Swing, JavaFX & SWT 8
D JCombobox mit Linien Styles AWT, Swing, JavaFX & SWT 4
H Unterschiedliche JComboBox je JTable Zeile AWT, Swing, JavaFX & SWT 4
B jCombobox addItem funktioniert nicht AWT, Swing, JavaFX & SWT 9
G JComboBox mit CellRenderer (Auswahl) AWT, Swing, JavaFX & SWT 11
G Swing JComboBox anpassen AWT, Swing, JavaFX & SWT 6
M Item in JComboBox umbenennen AWT, Swing, JavaFX & SWT 5
M Swing Busy Waiting Problem (JComboBox) AWT, Swing, JavaFX & SWT 11
C JComboBox Renderer Problem AWT, Swing, JavaFX & SWT 7
N Renderer Editoren und die JCombobox AWT, Swing, JavaFX & SWT 2
T Wie ist das "Lookup-Verhalten" von JList, JCombobox änderbar? AWT, Swing, JavaFX & SWT 4
N JCombobox und Actionlistener Aktion nur ausführen, wenn Useraktion ihn auslöst AWT, Swing, JavaFX & SWT 4

Ähnliche Java Themen

Neue Themen


Oben