JavaFx TableView mit CheckBox

J90

Aktives Mitglied
Hallo,

da der Aufbau eines TableViews sich änderte beim Wechsel von Swing auf JavaFX, benutze ich ein relativ simples Prinzip für die Standardmäßige Wiederverwendung eines Tables (d.h. TableView<Object[]> TableView),d.h. in diesem Fall arbeite ich sehr wenig Objektorientiert.

Da ich nur rudimentär mit der Oberflächenprogrammierung vertraut bin habe ich mich damals für diese funktionierende Variante entschieden.
Mein Problem besteht nun darin, dass ich bei der Verwendung von Boolean.class in einer Column zwar die CheckBox reinbekomme, mein
Listener aber nicht updatet. Bei String.class-Columns verwende ich aktuell:

Code:
       TC.setCellFactory(TextFieldTableCell.forTableColumn());
                TC.setOnEditCommit(new EventHandler<CellEditEvent<Object[], String>>() {
                    @Override
                    public void handle(CellEditEvent<Object[], String> Edit) {
                        TablePosition<Object[], String> TP = Edit.getTablePosition();
                        Object Wert = Edit.getNewValue();
                        int Row = TP.getRow(), Col = TP.getColumn();
                        setValueAt(String.valueOf(Wert), Row, Kopfzeile[Col], true);
                    }
                });

Das funktioniert. Die Methode setValueAt(..) überschreibt das jeweilige item an der Stelle und Zeile (ObservableList<Object[]> items = this.TableView.getItems()).

Die gleiche Variante für die Boolean.class mit

Code:
  TC.setCellFactory(new CheckBoxCellFactory());
                TC.setOnEditCommit(new EventHandler<CellEditEvent<Object[], String>>() {
                    @Override
                    public void handle(CellEditEvent<Object[], String> Edit) {

                        TablePosition<Object[], String> TP = Edit.getTablePosition();
                        Object Wert = Edit.getNewValue();
                        int Row = TP.getRow(), Col = TP.getColumn();
                        setValueAt(String.valueOf(Wert), Row, Kopfzeile[Col], true);
                    }
                });

reagiert nicht, d.h. die handle Methode wird nicht ausgeführt beim Klicken der CheckBox im Table.

Hierzu habe ich immer mal wieder Versuche gestartet dies zu ändern, leider ohne Erfolg. Da ich wichtigere Aufgabe zu erledigen habe, verwerfe ich es
dann immer wieder. Deswegen hier meine Frage an Leute die sich damit auskennen:
Gibt es hier eine schnelle einfache Variante diese Ausführung zu erreichen, sodass die CheckBox Werte im TableView angepasst werden und die CheckBox updated?

Versuche wie

Code:
   TC.setCellFactory(col -> {

                    CheckBoxTableCell<TableViewCheckBox.Item, Boolean> cell = new CheckBoxTableCell<>(index -> {
                        boolean T = (boolean) Boolean.valueOf(TableView.getItems().get(index)[Col].toString());
                        return new SimpleBooleanProperty(true);
                    });
                    Callback<Integer, ObservableValue<Boolean>> Call = new Callback<Integer, ObservableValue<Boolean>>() {
                        @Override
                        public ObservableValue<Boolean> call(Integer index) {
                            String itemBoolean = (String) TableView.getItems().get(index)[Col];
                            ObservableValue<Boolean> o = new ReadOnlyObjectWrapper<>(Boolean.valueOf(true));
                            Print(itemBoolean);
                            return o;
                        }
                    };

                    cell.setSelectedStateCallback(Call);
                    TC.setCellFactory(CheckBoxTableCell.forTableColumn(Call));
                    return cell;
                });

führten hierbei auch zu keinem Erfolg. Bei dieser Variante reagiert die Cell zwar auf Clicks, itemBoolean ist allerdings immer false und
es lässt sich auch hier kein Haken setzen bei der CB.

Hilfe hierzu wäre sehr nett, da ich aktuell in meinem Programm immer true/false Werte in einer String.class-Column verwende und mich das mittlerweile extrem nervt..
 

looparda

Top Contributor
Ohne deine Modelle zu kennen habe ich es wie folgt für ein minimales Beispiel:
Java:
public class MainWindowController implements Initializable {

    private ObservableList<ViewModel> dataList = FXCollections.observableArrayList();

    @FXML
    private TableView<ViewModel> tvView;

    @FXML
    private TableColumn<ViewModel, Boolean> boolColumn;

    @FXML
    private TableColumn<ViewModel, String> nameColumn;

    @Override
    public void initialize(URL location, ResourceBundle resources) {
        boolColumn.setCellFactory(CheckBoxTableCell.forTableColumn(boolColumn));
        boolColumn.setCellValueFactory(new PropertyValueFactory<>("bool"));
        nameColumn.setCellValueFactory(new PropertyValueFactory<>("name"));

        tvView.setItems(dataList);
        tvView.setEditable(true);

        dataList.add(new ViewModel(false, "b"));
    }

    public static class ViewModel {
        private BooleanProperty bool = new SimpleBooleanProperty(false);
        private StringProperty name = new SimpleStringProperty();

        public ViewModel(final boolean bool, final String name) {
            this.bool.set(bool);
            this.name.set(name);
        }

        public boolean isBool() {
            return bool.get();
        }

        public BooleanProperty boolProperty() {
            return bool;
        }

        public void setBool(final boolean bool) {
            this.bool.set(bool);
        }

        public String getName() {
            return name.get();
        }

        public StringProperty nameProperty() {
            return name;
        }

        public void setName(final String name) {
            this.name.set(name);
        }

    }
}
Java:
<AnchorPane id="AnchorPane" prefHeight="200" prefWidth="320" xmlns="http://javafx.com/javafx/8.0.121" xmlns:fx="http://javafx.com/fxml/1" fx:controller="MainWindowController">
    <children>
      <TableView fx:id="tvView" layoutY="7.0" prefHeight="200.0" prefWidth="200.0">
        <columns>
          <TableColumn fx:id="boolColumn" prefWidth="75.0" text="My Bool" />
          <TableColumn fx:id="nameColumn" prefWidth="75.0" text="My String" />
        </columns>
      </TableView>
    </children>
</AnchorPane>
Ich sehe die Schwierigkeit noch nicht. Wie sieht dein Modell aus? Brauchst du zwingend Click-Listener?
 

J90

Aktives Mitglied
Danke für den Hinweis, ich suche allerdings nach einer einfachen Lösung, welche sich in meinem bestehenden Aufbau eingliedert anstatt alles von Grund auf Neu zu bauen (mangels Zeit).

Da ich hier nicht direkt mit meiner kompletten class ankommen wollte, habe ich sie zunächst weggelassen. Hier nun die wichtigsten Teile:

Code:
     private void Init(int[] Weite, Object[] Class, boolean[] Editable, CellFactory CellFactory, Object[][] Daten, String[] Kopfzeile, boolean Sortable) {
        this.Kopfzeile = Kopfzeile;
        TCs = new TableColumn[Kopfzeile.length];
        for (int i = 0; i < Kopfzeile.length; i++) {

            TCs[i] = new TableColumn<>(Kopfzeile[i]);
            TCs[i].setSortable(Sortable);
            if (Weite != null) {
                TCs[i].setPrefWidth(Weite[i]);
            }
            int j = i;
            Object C = Object.class;
            if (Class != null) {
                C = Class[i];
            }
            TCs[i].setEditable(Editable[i]);
            Object CI = C;

            TCs[i].setCellValueFactory(new Callback<TableColumn.CellDataFeatures<Object[], ?>, ObservableValue<?>>() {
                @Override
                public ObservableValue<?> call(TableColumn.CellDataFeatures<Object[], ?> p) {
                    Object[] x = p.getValue();

                    if (j < x.length && x.length > 0) {

                        if (CI instanceof String) {
                            return new SimpleStringProperty(x[j].toString());
                        } else if (CI instanceof Boolean) {
                            return new SimpleBooleanProperty(Boolean.valueOf(x[j].toString()));
                        } else if (CI instanceof Integer) {
                            return new SimpleIntegerProperty(Integer.valueOf(x[j].toString()));
                        } else if (CI instanceof Double) {
                            return new SimpleDoubleProperty(Double.valueOf(x[j].toString()));
                        }
                        return new SimpleObjectProperty(x[j]);
                    } else {
                        return new SimpleObjectProperty("");
                    }
                }
            });

            //Boolean Spalte
            Set_Cell_Factory(TCs[i], C, Class, i, CellFactory);
            this.TableView.getColumns().add(TCs[i]);
        }


        if (Weite == null) {
            this.TableView.setColumnResizePolicy(TableView.CONSTRAINED_RESIZE_POLICY);
        }

        for (int i = 0; i < Daten.length; i++) {
            this.TableView.getItems().add(Daten[i]);
            this.HM_Daten.add(ArrayToMap(Daten[i]));
        }
        this.Items = Clone();


        /*
        Sortieren
         */
        // enable multi-selection
        this.TableView.getSelectionModel().setCellSelectionEnabled(true);
        this.TableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);

        // enable copy/paste
        TableUtils.installCopyPasteHandler(this.TableView);

        this.TableView.setOnKeyPressed(new TableKeyEventHandler());
    }




    private void Set_Cell_Factory(TableColumn TC, Object Class, Object[] Klassen, int Col, CellFactory CF) {
        if (Klassen != null) {
            if (Class == Boolean.class) {

               TC.setCellFactory(new CheckBoxCellFactory());
                TC.setOnEditCommit(new EventHandler<CellEditEvent<Object[], String>>() {
                    @Override
                    public void handle(CellEditEvent<Object[], String> Edit) {

                        TablePosition<Object[], String> TP = Edit.getTablePosition();
                        Object Wert = Edit.getNewValue();
                        int Row = TP.getRow(), Col = TP.getColumn();
                        Print(Wert);
                        setValueAt(String.valueOf(Wert), Row, Kopfzeile[Col], true); //HashMap
                    }
                });
            } else if (Class == String.class) {
                TC.setCellFactory(TextFieldTableCell.forTableColumn());
                TC.setOnEditCommit(new EventHandler<CellEditEvent<Object[], String>>() {
                    @Override
                    public void handle(CellEditEvent<Object[], String> Edit) {
                        TablePosition<Object[], String> TP = Edit.getTablePosition();
                        Object Wert = Edit.getNewValue();
                        int Row = TP.getRow(), Col = TP.getColumn();
                        Print(Wert);
                        setValueAt(String.valueOf(Wert), Row, Kopfzeile[Col], true); //HashMap
                    }
                });
            }
        } else {
            if (CF == null) {
                //Standardeinstellung
                TC.setCellFactory((row) -> {
                    CellFactory CellFactory = new CellFactory_Allgemein();
                    return CellFactory.TableCell(CellFactory.Schriftfarbe());
                });
            } else {
                //Laden dynamisch aus CellFactory Klasse
                TC.setCellFactory((row) -> {
                    return CF.newInstance().TableCell(CF.Schriftfarbe());
                });
            }
        }

    }

    public class CheckBoxCellFactory implements Callback {

        @Override
        public TableCell call(Object param) {
            CheckBoxTableCell<Boolean, Boolean> checkBoxCell = new CheckBoxTableCell();      
            TableColumn TC = (TableColumn) param;
            return checkBoxCell;
        }
    }
 
      public void setValueAt(String Wert, int Zeile, int Column, boolean Refresh) {
        if (Zeile > -1 && Column > -1) {
            Platform.runLater(() -> {
                ObservableList<Object[]> items = this.TableView.getItems();
                         if (items.size() > Zeile) {
                    Object[] get = items.get(Zeile);
 
                    if (get.length > Column) {
                        get[Column] = Wert;
                        //Sleep(50);

                        if (Refresh) {
                            // TableView.getColumns().get(Column).setVisible(false);
                            //TableView.getColumns().get(Column).setVisible(true);
                            this.TableView.refresh();//CPU intensiv
                        }
                    }
                }
            });
        }
    }

Hierzu wäre ein einfache Lösung hilfreich. Ob das ganze insgesamt sinnvoll ist soll jetzt nicht das Thema sein..
Das Problem liegt wie gesagt darin, dass der Boolean.class-Code Teil nicht so funktioniert wie der String.class Teil drunter
in der Methode Set_Cell_Factory() . Es
erfolgt eben kein Umstellen der CheckBox-Cell (auf aktiv) und der Wert ist immer false, die handle() Methode wird allerdings
ausgeführt.
Falls jemand die gesamte class mit Testcase brauch, kann ich das auch einstellen, habe gehofft jemand erfahrenes sieht wo
hier das Problem liegt
 

dzim

Top Contributor
Nur eins: Stell auf Boolean um und halte dich an die Doku... Mehr wüsste ich im Moment auch nicht.
Hab im Moment leider nicht so viel Zeit um zu experimentieren, also kann ich nur mit Wort, aber nicht Tat, zur Seite stehen.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
izoards JavaFX TableView mit Array Inhalt füllen AWT, Swing, JavaFX & SWT 1
A Mit JavaFX einzelne Zeilen in TableView farbig markieren AWT, Swing, JavaFX & SWT 5
N JavaFX Tableview nach Löschen von Element falscher Index AWT, Swing, JavaFX & SWT 4
S JavaFX - mit Listener Veränderungen in einer TableView abhören AWT, Swing, JavaFX & SWT 3
S Alternative JavaFX TableView AWT, Swing, JavaFX & SWT 1
S JavaFX JavaFX TableView scrollen färbt falsche Zeilen AWT, Swing, JavaFX & SWT 1
B JavaFX JavaFX TableView PropertyValueFactory für Werte aus HashMap AWT, Swing, JavaFX & SWT 2
S JavaFx - ausgewählte Rows in Arraylist (Tableview) AWT, Swing, JavaFX & SWT 4
S JavaFX - Populate TableView AWT, Swing, JavaFX & SWT 2
M JavaFX JavaFX ResultSet in TableView ausgeben AWT, Swing, JavaFX & SWT 2
D JavaFX TableView AWT, Swing, JavaFX & SWT 2
P JavaFX, TableView und Datenbank AWT, Swing, JavaFX & SWT 4
Z JavaFX TableView cellValueFactory für Arrays AWT, Swing, JavaFX & SWT 2
C (JavaFX 8) SQLite Datenbank in einem TableView darstellen AWT, Swing, JavaFX & SWT 2
G JavaFX TableView - Änderungen werden nicht übernommen. AWT, Swing, JavaFX & SWT 3
Juelin Javafx hängt sich auf AWT, Swing, JavaFX & SWT 31
Juelin StringGrid in SceneBuilder und javaFX AWT, Swing, JavaFX & SWT 2
Juelin starten maven javafx programm ohne netbeans AWT, Swing, JavaFX & SWT 38
Juelin javax.swing in javafx AWT, Swing, JavaFX & SWT 1
MiMa JUnit5 im JavaFX Projekt AWT, Swing, JavaFX & SWT 2
Juelin in javafx Event auslösen AWT, Swing, JavaFX & SWT 4
MiMa Fonts, Icons, Bilder in JavaFX Anwendung AWT, Swing, JavaFX & SWT 5
MiMa SwingFXUtils in JavaFX 20 AWT, Swing, JavaFX & SWT 6
MiMa JavaFX Fenstertitel zu klein. AWT, Swing, JavaFX & SWT 1
MiMa JavaFX JAR unter Windows ausführen schlägt fehl? AWT, Swing, JavaFX & SWT 5
I JavaFX JavaFx-Anwendung für die Erstellung einer Windows-Anwendung? AWT, Swing, JavaFX & SWT 6
Hatsi09 Javafx MediaPlayer spielt nicht immer AWT, Swing, JavaFX & SWT 3
Hatsi09 Javafx Neuladen von ImageView und MediaView AWT, Swing, JavaFX & SWT 3
Maxim6394 JavaFX Umlaute in JavaFX GUI AWT, Swing, JavaFX & SWT 12
Maxim6394 JavaFX Scene Builder - Crash bei eigener Komponente AWT, Swing, JavaFX & SWT 2
Ernesto95 JavaFX JavaFX GUI mit sehr vielen Update requests AWT, Swing, JavaFX & SWT 4
Telisti Javafx Image wird nicht richtig integiert AWT, Swing, JavaFX & SWT 8
J Netbeans die JavaFX-Anwendung wird nicht ausgeführt AWT, Swing, JavaFX & SWT 16
MartinNeuerlich Kann mir jemand, der einen Mac mit einem m1 oder m2-Chip hat, eine POM geben mit der Javafx-Fullscreen beim Mac mit m-Chip funktioniert? AWT, Swing, JavaFX & SWT 1
tommybalbor JavaFx Anwendung klappt nicht für macOs Nutzern, wenn ich zwei dependecies bei maven hinzufüge AWT, Swing, JavaFX & SWT 6
JavaSchmecktLecker JavaFX JavaFX Ordner automatisch verlinken AWT, Swing, JavaFX & SWT 2
melaniemueller Taschenrechner JavaFX AWT, Swing, JavaFX & SWT 4
R auto. Importanweisungen für javafx funktioniert in Eclipse nicht mehr AWT, Swing, JavaFX & SWT 4
thor_norsk JavaFX Anwendung stürzt ab AWT, Swing, JavaFX & SWT 4
berserkerdq2 Skalieren sich javafx objekte automatisch auf die Bildschirmgröße AWT, Swing, JavaFX & SWT 6
berserkerdq2 Wie füge ich ein Bild in javafx mit dem Scenebuilder ein, das automatisch mitgezogen wird, wenn das Fenster vergrößert wird oder Vollbildmodus AWT, Swing, JavaFX & SWT 6
B Java Projekt mit JavaFX und jfoenix ausführbar machen AWT, Swing, JavaFX & SWT 46
H JavaFX wie JavaFX Projekt aufsetzen? AWT, Swing, JavaFX & SWT 10
thor_norsk JavaFX - Grafikkarte AWT, Swing, JavaFX & SWT 7
MiHimbert Rückmeldung an den aufrufenden JAVAFX-Dialog AWT, Swing, JavaFX & SWT 1
MiMa JavaFX Runtime components are Missing??? AWT, Swing, JavaFX & SWT 3
J JavaFx PDF in einem Element in einem Fenster anzeigen. AWT, Swing, JavaFX & SWT 11
B JavaFX Sprachumschaltung mit Button auf der HMI AWT, Swing, JavaFX & SWT 6
H JavaFX Fehlende JavaFX Package AWT, Swing, JavaFX & SWT 10
K JavaFX unterschiedliche (mehrere Fenster) in seperater Main Methode AWT, Swing, JavaFX & SWT 26
_user_q Kann man ein 2. JavaFX-Fenster auch beenden (exit) statt schließen (close) lassen? AWT, Swing, JavaFX & SWT 8
G JavaFX Line Chart mit Farbverlauf/Gradient in Linie AWT, Swing, JavaFX & SWT 1
thor_norsk JavaFX, FXML und SceneBuilder AWT, Swing, JavaFX & SWT 6
_user_q Über installDist exportiertes Programm wirft "Unsupported JavaFX configuration" AWT, Swing, JavaFX & SWT 0
CodingBerlin JavaFX Programm läuft nur unter Eclipse AWT, Swing, JavaFX & SWT 1
H Fehler: Zum Ausführen dieser Anwendung benötigte JavaFX-Runtime-Komponenten fehlen AWT, Swing, JavaFX & SWT 44
temi JavaFX "Frames" in JavaFx - passende Komponente? AWT, Swing, JavaFX & SWT 13
G JavaFX Steuerung bzw. Test von externer JavaFX Anwendung (liegt nur als jar vor) AWT, Swing, JavaFX & SWT 9
_user_q [JavaFX] Spinner so einstellen, dass er nicht leer bleiben darf? AWT, Swing, JavaFX & SWT 6
S Javafx getResource-Pfad wird nicht erkannt AWT, Swing, JavaFX & SWT 7
A JavaFX exportierte Jar ohne beim starten die Libs hinzufügen? AWT, Swing, JavaFX & SWT 2
J JavaFX Schiffe versenken mit JavaFX und Scene builder AWT, Swing, JavaFX & SWT 3
Encera ArrayList mit eigenen Objekten in JavaFX sortieren und ausgeben AWT, Swing, JavaFX & SWT 50
L JavaFx Textformatierung mittels Datenbank und Funktion anpassen AWT, Swing, JavaFX & SWT 5
sserio Wie funktioniert ein Controller bei JavaFx? AWT, Swing, JavaFX & SWT 1
sserio Kann man bei JavaFx ein Fenster aufkommen lassen? AWT, Swing, JavaFX & SWT 1
Jose05 JavaFx Fxml: GUI aus einer anderen Klasse starten AWT, Swing, JavaFX & SWT 1
Tassos JavaFX/Problem mit der Maussteuerung in Stackpane AWT, Swing, JavaFX & SWT 7
S Ich bringe Code mit JavaFX unter Apache NetBeans IDE 12.6 nicht zum laufen. AWT, Swing, JavaFX & SWT 14
K Bekomme (u.a) javafx.fxml.LoadException trotz "korrektem" Code AWT, Swing, JavaFX & SWT 8
S JavaFX: voneinander abhängige TextFields AWT, Swing, JavaFX & SWT 33
M Gluon will JavaFX in den Browser stecken AWT, Swing, JavaFX & SWT 0
H javafx application does not exist AWT, Swing, JavaFX & SWT 16
A JavaFX Controller Problem AWT, Swing, JavaFX & SWT 1
M Javafx versuch Bibliothek zu erstellen AWT, Swing, JavaFX & SWT 0
N JavaFX Javafx intelij Projekt zu ausführbaren jar Datei Machen AWT, Swing, JavaFX & SWT 1
K JavaFx, Sound Aufnahme und Thread AWT, Swing, JavaFX & SWT 0
izoards JavaFX TextFlow - Sonderzeichen AWT, Swing, JavaFX & SWT 1
maximstein JavaFX WebView - java.lang.NoSuchMethodError: 'boolean com.sun.prism.ResourceFactory.isDisposed()' AWT, Swing, JavaFX & SWT 4
N JavaFX Unicode zeichnen in javafx Label verwenden AWT, Swing, JavaFX & SWT 2
MiHimbert javaFX openfx (17) datepicker AWT, Swing, JavaFX & SWT 3
melaniemueller JavaFX Taschenrechner mit SceneBuilder AWT, Swing, JavaFX & SWT 12
Jose05 Javafx Label Höhe=Breite AWT, Swing, JavaFX & SWT 1
Jose05 JavaFX: eigene FXML-Datei für einen Button AWT, Swing, JavaFX & SWT 3
izoards JavaFX editierbare Tabelle AWT, Swing, JavaFX & SWT 4
N javafx Position der Bustaben finden label AWT, Swing, JavaFX & SWT 1
D Verschieden Scenen ansprechen mit dem Scene Builder und JavaFX (Eclipse) AWT, Swing, JavaFX & SWT 16
izoards JavaFX Background Task warten auf Knopfdruck AWT, Swing, JavaFX & SWT 4
M Zufallsgenerator bei JavaFx AWT, Swing, JavaFX & SWT 1
N Label Schriftart Ändern javafx AWT, Swing, JavaFX & SWT 2
L JavaFX JavaFX, MVVM und SceneBuilder AWT, Swing, JavaFX & SWT 4
S JavaFx Album AWT, Swing, JavaFX & SWT 137
I JavaFX - Pane wechseln über 2. Controller AWT, Swing, JavaFX & SWT 5
melaniemueller JavaFX Beispiel kann nicht ausgeführt werden AWT, Swing, JavaFX & SWT 4
T FXML Datei in Java Code einbinden: javafx.fxml.LoadException AWT, Swing, JavaFX & SWT 2
J JavaFX - Included FXML - Entfernen feststellen AWT, Swing, JavaFX & SWT 2
J JavaFX JavaFX/ Taskmenu / UML Klassendiagramm AWT, Swing, JavaFX & SWT 2
Davee JavaFX JavaFX Jar ausführbar jedoch nicht alle Stages AWT, Swing, JavaFX & SWT 3
2 JavaFX die ChoiceBox leitet den String nicht weiter oder es komm zu einem NullPointer AWT, Swing, JavaFX & SWT 8
C MouseEvent JavaFX AWT, Swing, JavaFX & SWT 4

Ähnliche Java Themen

Neue Themen


Oben