java - 如何解决我的 JavaFX TableView 中的这个视觉故障?

标签 java javafx tableview javafx-8

左列:用于选择或取消选择行的复选框,默认选中。右列:一个字符串,表示直到并包括该行的所选行的数量。因此,取消选中一行中的复选框会更改下方行中的值。

错误:向下滚动到表格底部。取消选中邀请代码为 74 的行中的复选框。再次选中它。最后三个邀请码应再次显示为 73、74 和 75。但通常情况下,它们会显示 73、73、74 或 73、74、74。

错误并不总是发生!如果没有出现,用表格的滚动条上下滚动一点,重复上面的过程就可以出现。

似乎这个错误只是视觉上的——我让它把 ObservableList 的内容转储到控制台,它显示了正确的值。除了这个视觉故障,我的应用程序运行正常。单击窗口中的任何其他控件(例如表格的滚动条)会将表格中的邀请代码翻转为正确的值。切换到我桌面上的另一个工作区并返回,它也会显示正确的值。

Small image ,在左侧显示 ObservableList 的控制台转储,在右侧显示有问题的表。

问题 非常合乎逻辑: 我怎样才能消除这个错误!?

编辑:按照 Kleopatra 的建议丢弃了更多代码。谢谢!

MCV:

FXMLDocumentController.java

package invcodebug;

import java.net.URL;
import java.util.ResourceBundle;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.CheckBoxTableCell;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.util.Callback;

public class FXMLDocumentController implements Initializable {
    @FXML private TableView<Person> personTable;
    @FXML private TableColumn<Person, Boolean> invitedCol;
    @FXML private TableColumn<Person, String> inviteCodeCol;
    private final ObservableList<Person> persons
            = FXCollections.observableArrayList();

    @Override
    public void initialize(URL location, ResourceBundle resources) {
        initPersonTable();
        populatePersons();
    }

    private void initPersonTable() {
        invitedCol.setCellValueFactory(new PropertyValueFactory<>("invited"));
        inviteCodeCol.setCellValueFactory(new PropertyValueFactory<>("inviteCode"));

        invitedCol.setCellFactory(CheckBoxTableCell.forTableColumn(new Callback<Integer, ObservableValue<Boolean>>() {
            @Override
            public ObservableValue<Boolean> call(Integer param) {
                doInvCode();

// SHOWS: underlying ObservableList has correct values
System.out.println("--------------------------");
for (Person p : persons) {
    System.out.println(p.isInvited() + " " + p.getInviteCode()
    );
}
                return persons.get(param).invitedProperty();
            }
        }));

        personTable.setItems(persons);
    }

    private void doInvCode() {
        int invCounter = 1;
        for (Person p : persons) {
            if (p.isInvited()) {
                p.setInviteCode(((Integer) invCounter).toString());
                invCounter++;
            } else p.setInviteCode("");
        }
    }

    private void populatePersons() {
        for (int i = 0; i < 75; i++) {
            persons.add(new Person(true, ""));
        }
    }
}

Person.java

package invcodebug;

import javafx.beans.property.SimpleBooleanProperty;
import javafx.beans.property.SimpleStringProperty;

public class Person {
    private final SimpleBooleanProperty  invited;
    private final SimpleStringProperty   inviteCode;

    public Person(boolean invited, String inviteCode) {
        this.invited = new SimpleBooleanProperty(invited);
        this.inviteCode = new SimpleStringProperty(inviteCode);
    }

    public boolean isInvited() {
        return invited.get();
    }
    public SimpleBooleanProperty invitedProperty() {
        return invited;
    }

    public String getInviteCode(){
        return inviteCode.get();
    }
    public void setInviteCode(String invCode) {
        this.inviteCode.set(invCode);
    }
    public SimpleStringProperty inviteCodeProperty() {
        return inviteCode;
    }
}

FXML文档.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TableColumn?>
<?import javafx.scene.control.TableView?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.AnchorPane?>

<AnchorPane id="AnchorPane" prefHeight="464.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8.0.65" xmlns:fx="http://javafx.com/fxml/1" fx:controller="invcodebug.FXMLDocumentController">
    <children>
        <TableView fx:id="personTable" editable="true" layoutX="26.0" layoutY="28.0" prefHeight="347.0" prefWidth="572.0" AnchorPane.leftAnchor="14.0" AnchorPane.rightAnchor="14.0" AnchorPane.topAnchor="20.0">
            <columns>
                <TableColumn fx:id="invitedCol" prefWidth="27.0" sortable="false" />
                <TableColumn fx:id="inviteCodeCol" editable="false" prefWidth="110.0" resizable="false" sortable="false" text="Invite Code" />
            </columns>
        </TableView>
    </children>
</AnchorPane>

InvCodeBug.java

package invcodebug;

import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class InvCodeBug extends Application {

    @Override
    public void start(Stage stage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("FXMLDocument.fxml"));
        Scene scene = new Scene(root);
        stage.setScene(scene);
        stage.show();
    }

    public static void main(String[] args) {
        launch(args);
    }
}

最佳答案

可能不是最技术性的答案,但是通过在 doInvCode() 方法的末尾使用 personTable.requestFocus();,表格在视觉上被刷新并且看起来解决问题。

关于java - 如何解决我的 JavaFX TableView 中的这个视觉故障?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36674184/

相关文章:

java - 无法解析符号。无法将位置数据添加到图像的 EXIF。

java - 我可以将 pipedoutputstream 用于多线程吗?

java - 如何将javafx条形图刻度线设置为条形图的中间?

iOS/自动布局/ Storyboard : Pin view not responsive for table header

ios - 将 TableViewController 中的 TableView 嵌入到另一个 View 中

swift - 无法在 tableview Swift 4 中查看 Firebase 数据

java - 为什么反射不想加载我的测试类?

java - 使用 Java 检测鼠标点击屏幕上的任意位置

maven - JavaFX Native Bundle 初始屏幕

javaFX-什么是 getBoundsInLocal() 、getBoundsInParent() 方法?