не получать все события изменения выбора из ListView при (отмене) выборе нескольких элементов с помощью Ctrl + левая клавиша мыши

используя последнюю версию java-fx, я создал ListView с SelectionMode.MULTIPLE, вставил некоторые данные и прикрепил ChangeListener следующим образом:

myTableView.getSelectionModel().selectedItemProperty().addListener(this);

Нет, когда я выбираю некоторые элементы, уведомляется о каждом изменении выбора. Когда я затем отменяю выбор отдельных элементов с помощью Ctrl-щелчков мыши, события изменения выбора не запускаются, только когда отменяется выбор последнего ранее выбранного элемента, я получаю событие. Что я делаю неправильно?

с уважением Ганс


person Hans Huckebein    schedule 03.07.2015    source источник
comment
Сейчас нет времени проверять, но я думаю, что это ошибка, исправленная в 1.8.0u60.   -  person James_D    schedule 03.07.2015
comment
Ответ NwDx идеален. До сих пор я упускал из виду существование ListChangeListeners и задавался вопросом, как назвать selectedItemProperty. Это довольно подвержено ошибкам, так как есть много примеров кода с прослушивателями изменений для selectedItemProperties, и они, на первый взгляд, работают.   -  person Hans Huckebein    schedule 06.07.2015


Ответы (1)


Проблема

Как selectedItemProperty должен отражать выбор нескольких строк? Это невозможно, потому что он предназначался для использования с SelectionMode.SINGLE.

Решение

Вы можете использовать ListChangeListener в своем TableView или ListView вместо этого. Это рекомендуемый способ в моделях множественного выбора. При этом вы сможете узнать, какие строки были удалены, обновлены, удалены и т. д. для выбора.

Я сделал пример на основе Table Посмотреть учебник от Oracle. Не пугайтесь, он немного великоват, но это заложено в природе списков и таблиц. Интересные разделы я отметил комментариями.

Пример

Вот минимальный, полный и проверяемый пример

import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.geometry.Insets;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.layout.VBox;
import javafx.scene.text.Font;
import javafx.stage.Stage;

public class TableViewDemo extends Application {

  private final TableView<Person> table = new TableView<>();
  private final ObservableList<Person> data
      = FXCollections.<Person>observableArrayList(
          new Person("Jacob", "Smith", "[email protected]"),
          new Person("Isabella", "Johnson", "[email protected]"),
          new Person("Ethan", "Williams", "[email protected]"),
          new Person("Emma", "Jones", "[email protected]"),
          new Person("Michael", "Brown", "[email protected]")
      );

  @Override
  public void start(Stage primaryStage) {
    Scene scene = new Scene(new Group());
    primaryStage.setTitle("Table View Sample");
    primaryStage.setWidth(450);
    primaryStage.setHeight(500);

    final Label label = new Label("Address Book");
    label.setFont(new Font("Arial", 20));

    table.setEditable(true);
    table.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);

    // begin of interesting code
    table.getSelectionModel().getSelectedIndices().addListener(new ListChangeListener<Integer>() {

      @Override
      public void onChanged(ListChangeListener.Change<? extends Integer> c) {
        while (c.next()) {
          System.out.println("#######- Indicies -#######");
          if (c.wasAdded()) {
            c.getAddedSubList().stream().forEach((item) -> {
              System.out.println("Added: " + item);
              System.out.println("--------------------------");
            });
          }
          if (c.wasRemoved()) {
            c.getRemoved().stream().forEach((item) -> {
              System.out.println("Removed: " + item);
              System.out.println("--------------------------");
            });
          }
          //c.wasPermutated();
          //c.wasReplaced();
        }
      }
    });

    table.getSelectionModel().getSelectedItems().addListener(new ListChangeListener<Person>() {

      @Override
      public void onChanged(ListChangeListener.Change<? extends Person> c) {
        while (c.next()) {
          System.out.println("######- Items -########");
          if (c.wasAdded()) {
            c.getAddedSubList().stream().forEach((item) -> {
              System.out.println("Added: " + item);
              System.out.println("--------------------------");
            });
          }
          if (c.wasRemoved()) {
            c.getRemoved().stream().forEach((item) -> {
              System.out.println("Removed: " + item);
              System.out.println("--------------------------");
            });
          }
          //c.wasPermutated();
          //c.wasReplaced();
        }
      }
    });

    // end of interesting code
    TableColumn<Person, String> firstNameCol = new TableColumn<>("First Name");
    firstNameCol.setMinWidth(100);
    firstNameCol.setCellValueFactory(
        new PropertyValueFactory<>("firstName"));

    TableColumn<Person, String> lastNameCol = new TableColumn<>("Last Name");
    lastNameCol.setMinWidth(100);
    lastNameCol.setCellValueFactory(
        new PropertyValueFactory<>("lastName"));

    TableColumn<Person, String> emailCol = new TableColumn<>("Email");
    emailCol.setMinWidth(200);
    emailCol.setCellValueFactory(
        new PropertyValueFactory<>("email"));

    table.setItems(data);
    // I know this could be done by invoking addAll, but Xlint will give warning
    table.getColumns().add(firstNameCol);
    table.getColumns().add(lastNameCol);
    table.getColumns().add(emailCol);

    final VBox vbox = new VBox();
    vbox.setSpacing(5);
    vbox.setPadding(new Insets(10, 0, 0, 10));
    vbox.getChildren().addAll(label, table);

    ((Group) scene.getRoot()).getChildren().addAll(vbox);

    primaryStage.setScene(scene);
    primaryStage.show();
  }

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

  public static class Person {

    private final SimpleStringProperty firstName;
    private final SimpleStringProperty lastName;
    private final SimpleStringProperty email;

    private Person(String fName, String lName, String email) {
      this.firstName = new SimpleStringProperty(fName);
      this.lastName = new SimpleStringProperty(lName);
      this.email = new SimpleStringProperty(email);
    }

    public String getFirstName() {
      return firstName.get();
    }

    public void setFirstName(String fName) {
      firstName.set(fName);
    }

    public String getLastName() {
      return lastName.get();
    }

    public void setLastName(String fName) {
      lastName.set(fName);
    }

    public String getEmail() {
      return email.get();
    }

    public void setEmail(String fName) {
      email.set(fName);
    }

    @Override
    public String toString() {
      return "Person{" + "first name=" + firstName.get()
             + ", last name=" + lastName.get()
             + ", email=" + email.get() + '}';
    }
  }
}

Пример вывода в консоль

######- Items -########
Added: Person{first name=Michael, last name=Brown, [email protected]}
--------------------------
Removed: Person{first name=Isabella, last name=Johnson, [email protected]}
--------------------------
Removed: Person{first name=Ethan, last name=Williams, [email protected]}
--------------------------
Removed: Person{first name=Emma, last name=Jones, [email protected]}
--------------------------
Removed: Person{first name=Jacob, last name=Smith, [email protected]}
--------------------------
#######- Indicies -#######
Added: 4
--------------------------
Removed: 1
--------------------------
Removed: 2
--------------------------
Removed: 3
--------------------------
Removed: 0
--------------------------
person aw-think    schedule 03.07.2015
comment
Спасибо за ваш точный ответ, который я смог легко использовать! - person Hans Huckebein; 06.07.2015