带有ChangeListener的JavaFX TableView,用于多种选择

时间:2016-01-01 17:47:56

标签: javafx tableview selecteditem multipleselection changelistener

我想在JavaFX 8中监听表视图的选择更改。如果我将ChangeListener添加到MultipleSelectionModel的selectedItemProperty(或selectedIndexProperty),则在以下情况下不会触发: 已经选择了多行,现在我选择一个新的单行。如果此行是之前选择的最后一行,则它不会做出反应。这是因为selectedItemProperty只包含一个项目而不包含所有选定项目的列表。但遗憾的是,由于方法getSelectedItems()存在,因此没有选择属性,这很奇怪。我不知道在哪里添加监听器,以使其在任何情况下都能正常工作。

以下是一个示例代码:

import java.util.stream.Collectors;
import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.control.ListView;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.layout.HBox;
import javafx.stage.Stage;

public class MultipleSelectionTest extends Application
{
    private ListView<Integer> list = new ListView<>();
    private TableView<String> table = new TableView<>();
    private TableColumn<String, String> column1 = new TableColumn<>("Numbers");

    @Override
    public void start(Stage primaryStage) throws Exception
    {
        HBox box = new HBox(list, table);
        primaryStage.setScene(new Scene(box, 300, 500));
        initializeTable();
        initializeList();
        primaryStage.show();
    }

    private void initializeTable()
    {
        table.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
        table.getColumns().add(column1);
        column1.setCellValueFactory(s -> new SimpleStringProperty(s.getValue()));
        table.setItems(FXCollections.observableArrayList("One", "Two", "Three", "Four", "Five", "Six", "Seven"));
    }

    private void initializeList()
    {
        table.getSelectionModel().selectedItemProperty().addListener((property, oldValue, newValue) ->
        {
            ObservableList<Integer> convertedItems = FXCollections.observableList(
                table.getSelectionModel().getSelectedItems()
                .stream()
                .map(this::convertNumber)
                .collect(Collectors.toList()));

            list.setItems(convertedItems);
        });
    }

    private int convertNumber(String expression)
    {
        switch (expression)
        {
            case "One": return 1;
            case "Two": return 2;
            case "Three": return 3;
            case "Four": return 4;
            case "Five": return 5;
            case "Six": return 6;
            case "Seven": return 7;
            default: throw new IllegalArgumentException();
        }
    }

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

请注意,此示例中的ListView仅用于视觉目的。

2 个答案:

答案 0 :(得分:3)

您不需要selectedItemsProperty - getSelectedItems返回ObservableList即可添加侦听器。

拥有一个属性意味着列表对象发生了变化,但实际上发生的是列表保持相同的对象,但它的内容发生了变化。

答案 1 :(得分:0)

这是一个更完整的解决方案,这要归功于SillyFly的回应指针。

在我的示例中,假设表已设置为使用任意类Person定义名字,姓氏和电子邮件。

//Set the table to multi selection mode
table.getSelectionModel().setSelectionMode( SelectionMode.MULTIPLE);

//Regiseter the listener on the ObervableList<Person>
table.getSelectionModel().getSelectedItems().addListener(multiSelection);


  /**A listener for list selections, multiple selections in the TableView**/
     ListChangeListener< Person> multiSelection = new ListChangeListener<Person>(){
        @Override
        public void onChanged(  ListChangeListener.Change<? extends Person> changed){
            for( Person p : changed.getList())
                System.out.println(p);
        } 
     };