过滤视图列表的出界异常
OutBounds Exception on Filtered ViewList
我正在开发一个从数据库中获取信息并将其显示在 ListView 中的小型应用程序,然后我 select 列表的元素并将其移动到第二个列表,我必须能够添加和删除列表之间的元素。我还使用 FilteredLists 实现了两个列表以实现搜索功能。我的代码有效,我能够在列表之间切换元素,但是当我尝试将 return 元素从第二个列表移到第一个列表时,或者当我将第一个列表的最后一个元素移动到第二个。
谁能帮我解决一下?
提前致谢。
异常
Exception in thread "JavaFX Application Thread" java.lang.IndexOutOfBoundsException: Index: 3, Size: 3
at java.util.ArrayList.rangeCheck(ArrayList.java:657)
at java.util.ArrayList.get(ArrayList.java:433)
at com.sun.javafx.collections.ObservableListWrapper.get(ObservableListWrapper.java:89)
at javafx.collections.transformation.FilteredList.get(FilteredList.java:172)
at javafx.scene.control.ListCell.updateItem(ListCell.java:459)
at javafx.scene.control.ListCell.lambda$new0(ListCell.java:167)
at javafx.collections.WeakListChangeListener.onChanged(WeakListChangeListener.java:88)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:329)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:73)
at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
at javafx.collections.transformation.FilteredList.sourceChanged(FilteredList.java:147)
at javafx.collections.transformation.TransformationList.lambda$getListener(TransformationList.java:106)
at javafx.collections.WeakListChangeListener.onChanged(WeakListChangeListener.java:88)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:329)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:73)
at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
at com.sun.javafx.collections.ObservableListWrapper.removeAll(ObservableListWrapper.java:185)
at com.kalypso.WCExporter.MainController.removeItemFromList(MainController.java:351)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at sun.reflect.misc.Trampoline.invoke(MethodUtil.java:71)
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at sun.reflect.misc.MethodUtil.invoke(MethodUtil.java:275)
at javafx.fxml.FXMLLoader$MethodHandler.invoke(FXMLLoader.java:1771)
at javafx.fxml.FXMLLoader$ControllerMethodEventHandler.handle(FXMLLoader.java:1657)
at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:86)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:49)
at javafx.event.Event.fireEvent(Event.java:198)
at javafx.scene.Node.fireEvent(Node.java:8411)
at javafx.scene.control.Button.fire(Button.java:185)
at com.sun.javafx.scene.control.behavior.ButtonBehavior.mouseReleased(ButtonBehavior.java:182)
at com.sun.javafx.scene.control.skin.BehaviorSkinBase.handle(BehaviorSkinBase.java:96)
at com.sun.javafx.scene.control.skin.BehaviorSkinBase.handle(BehaviorSkinBase.java:89)
at com.sun.javafx.event.CompositeEventHandler$NormalEventHandlerRecord.handleBubblingEvent(CompositeEventHandler.java:218)
at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:80)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
at javafx.event.Event.fireEvent(Event.java:198)
at javafx.scene.Scene$MouseHandler.process(Scene.java:3757)
at javafx.scene.Scene$MouseHandler.access00(Scene.java:3485)
at javafx.scene.Scene.impl_processMouseEvent(Scene.java:1762)
at javafx.scene.Scene$ScenePeerListener.mouseEvent(Scene.java:2494)
at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:394)
at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:295)
at java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.tk.quantum.GlassViewEventHandler.lambda$handleMouseEvent3(GlassViewEventHandler.java:432)
at com.sun.javafx.tk.quantum.QuantumToolkit.runWithoutRenderLock(QuantumToolkit.java:389)
at com.sun.javafx.tk.quantum.GlassViewEventHandler.handleMouseEvent(GlassViewEventHandler.java:431)
at com.sun.glass.ui.View.handleMouseEvent(View.java:555)
at com.sun.glass.ui.View.notifyMouse(View.java:937)
列表声明
@FXML
private JFXListView leftList;
@FXML
private JFXTextField leftFilter;
@FXML
private JFXListView rightList;
@FXML
private JFXTextField rightFilter;
private ObservableList<String> leftData = FXCollections.observableArrayList();
private ObservableList<String> rightData = FXCollections.observableArrayList();
private FilteredList<String> leftFilteredData;
private FilteredList<String> rightFilteredData;
此方法填充第一个列表
此列表包含将要使用的所有值
private void loadResultList(){
if( session != null && session.isOpen()) {
//Clear list View
leftList.getItems().clear();
//Get entity node
HibernateUtil hibernateUtil = new HibernateUtil();
//Get query results
List resultSet = hibernateUtil.executeSQLQuery(session, selectedEntity.getValue().get("query").asText());
//Wrap resultset into Observable list
resultSet.forEach(result -> leftData.add(((Map)result).get("NAME").toString()));
// 1. Wrap the ObservableList in a FilteredList (initially display all data).
leftFilteredData = new FilteredList<>(leftData, p -> true);
// 2. Set the filter Predicate whenever the filter changes.
wrapListAndAddFiltering(leftFilter, leftFilteredData);
leftList.setItems(leftFilteredData);
leftList.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
} else{
Alert noDBCon = new Alert(Alert.AlertType.ERROR);
noDBCon.setTitle("No Database connection");
noDBCon.setHeaderText("No Database Connection");
noDBCon.setContentText("Select and connect to a DB before doing any transaction.");
noDBCon.showAndWait();
}
}
private void wrapListAndAddFiltering(JFXTextField filterField,FilteredList<String> filteredData) {
filterField.textProperty().addListener((observable, oldValue, newValue) -> {
filteredData.setPredicate(item -> {
String filter = filterField.getText().toLowerCase();
if (newValue == null || newValue.isEmpty()) {
return true;
}
if (item.toLowerCase().contains(filter)) {
return true;
}
return false;
});
});
}
添加和删除元素的方法
@FXML
private void addItemToList(){
//1.- Add elements to right list
//Wrap left elements into Observable list
leftList.getSelectionModel().getSelectedItems().forEach(item -> rightData.add(item.toString()));
//Wrap the ObservableList in a FilteredList (initially display all data).
rightFilteredData = new FilteredList<>(rightData, p -> true);
wrapListAndAddFiltering(rightFilter, rightFilteredData);
rightList.setItems(rightFilteredData);
//2.- Remove elements from left list
leftData.removeAll(leftList.getSelectionModel().getSelectedItems());
leftFilteredData = new FilteredList<>(leftData, p -> true);
wrapListAndAddFiltering( leftFilter, leftFilteredData );
leftList.setItems(leftFilteredData);
}
@FXML
private void removeItemFromList(){
//1.-add elements to left list
rightList.getSelectionModel().getSelectedItems().forEach(item -> leftData.add(item.toString()));
leftFilteredData = new FilteredList<>(leftData, p -> true);
wrapListAndAddFiltering(leftFilter, leftFilteredData);
leftList.setItems(leftFilteredData);
//1.-Remove items from right list
ObservableList<String> temp = FXCollections.observableArrayList();
rightList.getSelectionModel().getSelectedItems().forEach(el -> temp.add(el.toString()));
//rightList.getSelectionModel().getSelectedItems().forEach(rightData::remove);
rightData.removeAll(temp);
//rightData.removeAll();
rightFilteredData = new FilteredList<>(rightData, p -> true);
wrapListAndAddFiltering( rightFilter, rightFilteredData );
rightList.setItems(rightFilteredData);
}
GUI
我创建了一个 MCVE。代码中的注释。它显示了如何设置 FilteredList
以及如何将数据从一个 ListView
移动到另一个。
Main
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
/**
*
* @author blj0011
*/
public class JavaFXApplication239 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();
}
/**
* @param args the command line arguments
*/
public static void main(String[] args)
{
launch(args);
}
}
Controller
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import java.util.ResourceBundle;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.collections.transformation.FilteredList;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.ListView;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TextField;
/**
*
* @author blj0011
*/
public class FXMLDocumentController implements Initializable
{
@FXML
private ListView lvLeft, lvRight;
@FXML
private TextField tfLeft, tfRight;
ObservableList<String> leftData = FXCollections.observableArrayList();
ObservableList<String> rightData = FXCollections.observableArrayList();
FilteredList<String> filteredLeftData, filteredRightData;
@Override
public void initialize(URL url, ResourceBundle rb)
{
leftData.addAll(getFakeDataFromDB());//get data from DB
//rightData.addAll(getFakeDataFromDB());//Used for testing
filteredLeftData = new FilteredList(leftData, s -> true);
filteredRightData = new FilteredList(rightData, s -> true);
//Set filtered Lists
tfLeft.textProperty().addListener((observable, oldValue, newValue) -> {
if (newValue == null || newValue.length() == 0) {
filteredLeftData.setPredicate(null);
}
else {
filteredLeftData.setPredicate(t -> {
return t.toUpperCase().startsWith(newValue.toUpperCase());
});
}
});
tfRight.textProperty().addListener((observable, oldValue, newValue) -> {
if (newValue == null || newValue.length() == 0) {
filteredRightData.setPredicate(null);
}
else {
filteredRightData.setPredicate(t -> {
return t.toUpperCase().startsWith(newValue.toUpperCase());
});
}
});
//Set listview items
lvLeft.setItems(filteredLeftData);
lvRight.setItems(filteredRightData);
//Set selecton model selection mode
lvLeft.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
lvRight.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
}
@FXML
private void handleBtnActionMoveToLeft(ActionEvent actionEvent)
{
List<String> itemsToMove = new ArrayList(lvRight.getSelectionModel().getSelectedItems());//If you don't do this "new ArrayList(..):", then you need to first addAll then removeAll
System.out.println("list size: " + itemsToMove.size());
rightData.removeAll(itemsToMove);
leftData.addAll(itemsToMove);
}
@FXML
private void handleBtnActionMoveToRight(ActionEvent actionEvent)
{
List<String> itemsToMove = new ArrayList(lvLeft.getSelectionModel().getSelectedItems());//If you don't do this "new ArrayList(..):", then you need to first addAll then removeAll
System.out.println("list size: " + itemsToMove.size());
leftData.removeAll(itemsToMove);
rightData.addAll(itemsToMove);
}
List<String> getFakeDataFromDB()
{
List<String> tempList = new ArrayList();
tempList.add("Hello");
tempList.add("Hello World!");
tempList.add("Bye");
tempList.add("Bye World!");
tempList.add("Been");
tempList.add("Bad");
return tempList;
}
}
FXML
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.ListView?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.VBox?>
<HBox maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8.0.141" xmlns:fx="http://javafx.com/fxml/1" fx:controller="javafxapplication239.FXMLDocumentController">
<children>
<ListView fx:id="lvLeft" prefHeight="200.0" prefWidth="200.0" />
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" HBox.hgrow="ALWAYS">
<children>
<Label text="Left" />
<TextField fx:id="tfLeft" />
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" spacing="5.0">
<children>
<Button mnemonicParsing="false" onAction="#handleBtnActionMoveToRight" text="Move To Right" />
<Button mnemonicParsing="false" onAction="#handleBtnActionMoveToLeft" text="Move To Left" />
</children>
<opaqueInsets>
<Insets />
</opaqueInsets>
</VBox>
<Label text="Right" />
<TextField fx:id="tfRight" />
</children>
</VBox>
<ListView fx:id="lvRight" prefHeight="200.0" prefWidth="200.0" />
</children>
</HBox>
我正在开发一个从数据库中获取信息并将其显示在 ListView 中的小型应用程序,然后我 select 列表的元素并将其移动到第二个列表,我必须能够添加和删除列表之间的元素。我还使用 FilteredLists 实现了两个列表以实现搜索功能。我的代码有效,我能够在列表之间切换元素,但是当我尝试将 return 元素从第二个列表移到第一个列表时,或者当我将第一个列表的最后一个元素移动到第二个。 谁能帮我解决一下?
提前致谢。
异常
Exception in thread "JavaFX Application Thread" java.lang.IndexOutOfBoundsException: Index: 3, Size: 3
at java.util.ArrayList.rangeCheck(ArrayList.java:657)
at java.util.ArrayList.get(ArrayList.java:433)
at com.sun.javafx.collections.ObservableListWrapper.get(ObservableListWrapper.java:89)
at javafx.collections.transformation.FilteredList.get(FilteredList.java:172)
at javafx.scene.control.ListCell.updateItem(ListCell.java:459)
at javafx.scene.control.ListCell.lambda$new0(ListCell.java:167)
at javafx.collections.WeakListChangeListener.onChanged(WeakListChangeListener.java:88)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:329)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:73)
at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
at javafx.collections.transformation.FilteredList.sourceChanged(FilteredList.java:147)
at javafx.collections.transformation.TransformationList.lambda$getListener(TransformationList.java:106)
at javafx.collections.WeakListChangeListener.onChanged(WeakListChangeListener.java:88)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:329)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:73)
at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
at com.sun.javafx.collections.ObservableListWrapper.removeAll(ObservableListWrapper.java:185)
at com.kalypso.WCExporter.MainController.removeItemFromList(MainController.java:351)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at sun.reflect.misc.Trampoline.invoke(MethodUtil.java:71)
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at sun.reflect.misc.MethodUtil.invoke(MethodUtil.java:275)
at javafx.fxml.FXMLLoader$MethodHandler.invoke(FXMLLoader.java:1771)
at javafx.fxml.FXMLLoader$ControllerMethodEventHandler.handle(FXMLLoader.java:1657)
at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:86)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:49)
at javafx.event.Event.fireEvent(Event.java:198)
at javafx.scene.Node.fireEvent(Node.java:8411)
at javafx.scene.control.Button.fire(Button.java:185)
at com.sun.javafx.scene.control.behavior.ButtonBehavior.mouseReleased(ButtonBehavior.java:182)
at com.sun.javafx.scene.control.skin.BehaviorSkinBase.handle(BehaviorSkinBase.java:96)
at com.sun.javafx.scene.control.skin.BehaviorSkinBase.handle(BehaviorSkinBase.java:89)
at com.sun.javafx.event.CompositeEventHandler$NormalEventHandlerRecord.handleBubblingEvent(CompositeEventHandler.java:218)
at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:80)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
at javafx.event.Event.fireEvent(Event.java:198)
at javafx.scene.Scene$MouseHandler.process(Scene.java:3757)
at javafx.scene.Scene$MouseHandler.access00(Scene.java:3485)
at javafx.scene.Scene.impl_processMouseEvent(Scene.java:1762)
at javafx.scene.Scene$ScenePeerListener.mouseEvent(Scene.java:2494)
at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:394)
at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:295)
at java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.tk.quantum.GlassViewEventHandler.lambda$handleMouseEvent3(GlassViewEventHandler.java:432)
at com.sun.javafx.tk.quantum.QuantumToolkit.runWithoutRenderLock(QuantumToolkit.java:389)
at com.sun.javafx.tk.quantum.GlassViewEventHandler.handleMouseEvent(GlassViewEventHandler.java:431)
at com.sun.glass.ui.View.handleMouseEvent(View.java:555)
at com.sun.glass.ui.View.notifyMouse(View.java:937)
列表声明
@FXML
private JFXListView leftList;
@FXML
private JFXTextField leftFilter;
@FXML
private JFXListView rightList;
@FXML
private JFXTextField rightFilter;
private ObservableList<String> leftData = FXCollections.observableArrayList();
private ObservableList<String> rightData = FXCollections.observableArrayList();
private FilteredList<String> leftFilteredData;
private FilteredList<String> rightFilteredData;
此方法填充第一个列表
此列表包含将要使用的所有值
private void loadResultList(){
if( session != null && session.isOpen()) {
//Clear list View
leftList.getItems().clear();
//Get entity node
HibernateUtil hibernateUtil = new HibernateUtil();
//Get query results
List resultSet = hibernateUtil.executeSQLQuery(session, selectedEntity.getValue().get("query").asText());
//Wrap resultset into Observable list
resultSet.forEach(result -> leftData.add(((Map)result).get("NAME").toString()));
// 1. Wrap the ObservableList in a FilteredList (initially display all data).
leftFilteredData = new FilteredList<>(leftData, p -> true);
// 2. Set the filter Predicate whenever the filter changes.
wrapListAndAddFiltering(leftFilter, leftFilteredData);
leftList.setItems(leftFilteredData);
leftList.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
} else{
Alert noDBCon = new Alert(Alert.AlertType.ERROR);
noDBCon.setTitle("No Database connection");
noDBCon.setHeaderText("No Database Connection");
noDBCon.setContentText("Select and connect to a DB before doing any transaction.");
noDBCon.showAndWait();
}
}
private void wrapListAndAddFiltering(JFXTextField filterField,FilteredList<String> filteredData) {
filterField.textProperty().addListener((observable, oldValue, newValue) -> {
filteredData.setPredicate(item -> {
String filter = filterField.getText().toLowerCase();
if (newValue == null || newValue.isEmpty()) {
return true;
}
if (item.toLowerCase().contains(filter)) {
return true;
}
return false;
});
});
}
添加和删除元素的方法
@FXML
private void addItemToList(){
//1.- Add elements to right list
//Wrap left elements into Observable list
leftList.getSelectionModel().getSelectedItems().forEach(item -> rightData.add(item.toString()));
//Wrap the ObservableList in a FilteredList (initially display all data).
rightFilteredData = new FilteredList<>(rightData, p -> true);
wrapListAndAddFiltering(rightFilter, rightFilteredData);
rightList.setItems(rightFilteredData);
//2.- Remove elements from left list
leftData.removeAll(leftList.getSelectionModel().getSelectedItems());
leftFilteredData = new FilteredList<>(leftData, p -> true);
wrapListAndAddFiltering( leftFilter, leftFilteredData );
leftList.setItems(leftFilteredData);
}
@FXML
private void removeItemFromList(){
//1.-add elements to left list
rightList.getSelectionModel().getSelectedItems().forEach(item -> leftData.add(item.toString()));
leftFilteredData = new FilteredList<>(leftData, p -> true);
wrapListAndAddFiltering(leftFilter, leftFilteredData);
leftList.setItems(leftFilteredData);
//1.-Remove items from right list
ObservableList<String> temp = FXCollections.observableArrayList();
rightList.getSelectionModel().getSelectedItems().forEach(el -> temp.add(el.toString()));
//rightList.getSelectionModel().getSelectedItems().forEach(rightData::remove);
rightData.removeAll(temp);
//rightData.removeAll();
rightFilteredData = new FilteredList<>(rightData, p -> true);
wrapListAndAddFiltering( rightFilter, rightFilteredData );
rightList.setItems(rightFilteredData);
}
GUI
我创建了一个 MCVE。代码中的注释。它显示了如何设置 FilteredList
以及如何将数据从一个 ListView
移动到另一个。
Main
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
/**
*
* @author blj0011
*/
public class JavaFXApplication239 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();
}
/**
* @param args the command line arguments
*/
public static void main(String[] args)
{
launch(args);
}
}
Controller
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import java.util.ResourceBundle;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.collections.transformation.FilteredList;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.ListView;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TextField;
/**
*
* @author blj0011
*/
public class FXMLDocumentController implements Initializable
{
@FXML
private ListView lvLeft, lvRight;
@FXML
private TextField tfLeft, tfRight;
ObservableList<String> leftData = FXCollections.observableArrayList();
ObservableList<String> rightData = FXCollections.observableArrayList();
FilteredList<String> filteredLeftData, filteredRightData;
@Override
public void initialize(URL url, ResourceBundle rb)
{
leftData.addAll(getFakeDataFromDB());//get data from DB
//rightData.addAll(getFakeDataFromDB());//Used for testing
filteredLeftData = new FilteredList(leftData, s -> true);
filteredRightData = new FilteredList(rightData, s -> true);
//Set filtered Lists
tfLeft.textProperty().addListener((observable, oldValue, newValue) -> {
if (newValue == null || newValue.length() == 0) {
filteredLeftData.setPredicate(null);
}
else {
filteredLeftData.setPredicate(t -> {
return t.toUpperCase().startsWith(newValue.toUpperCase());
});
}
});
tfRight.textProperty().addListener((observable, oldValue, newValue) -> {
if (newValue == null || newValue.length() == 0) {
filteredRightData.setPredicate(null);
}
else {
filteredRightData.setPredicate(t -> {
return t.toUpperCase().startsWith(newValue.toUpperCase());
});
}
});
//Set listview items
lvLeft.setItems(filteredLeftData);
lvRight.setItems(filteredRightData);
//Set selecton model selection mode
lvLeft.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
lvRight.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
}
@FXML
private void handleBtnActionMoveToLeft(ActionEvent actionEvent)
{
List<String> itemsToMove = new ArrayList(lvRight.getSelectionModel().getSelectedItems());//If you don't do this "new ArrayList(..):", then you need to first addAll then removeAll
System.out.println("list size: " + itemsToMove.size());
rightData.removeAll(itemsToMove);
leftData.addAll(itemsToMove);
}
@FXML
private void handleBtnActionMoveToRight(ActionEvent actionEvent)
{
List<String> itemsToMove = new ArrayList(lvLeft.getSelectionModel().getSelectedItems());//If you don't do this "new ArrayList(..):", then you need to first addAll then removeAll
System.out.println("list size: " + itemsToMove.size());
leftData.removeAll(itemsToMove);
rightData.addAll(itemsToMove);
}
List<String> getFakeDataFromDB()
{
List<String> tempList = new ArrayList();
tempList.add("Hello");
tempList.add("Hello World!");
tempList.add("Bye");
tempList.add("Bye World!");
tempList.add("Been");
tempList.add("Bad");
return tempList;
}
}
FXML
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.ListView?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.VBox?>
<HBox maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8.0.141" xmlns:fx="http://javafx.com/fxml/1" fx:controller="javafxapplication239.FXMLDocumentController">
<children>
<ListView fx:id="lvLeft" prefHeight="200.0" prefWidth="200.0" />
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" HBox.hgrow="ALWAYS">
<children>
<Label text="Left" />
<TextField fx:id="tfLeft" />
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" spacing="5.0">
<children>
<Button mnemonicParsing="false" onAction="#handleBtnActionMoveToRight" text="Move To Right" />
<Button mnemonicParsing="false" onAction="#handleBtnActionMoveToLeft" text="Move To Left" />
</children>
<opaqueInsets>
<Insets />
</opaqueInsets>
</VBox>
<Label text="Right" />
<TextField fx:id="tfRight" />
</children>
</VBox>
<ListView fx:id="lvRight" prefHeight="200.0" prefWidth="200.0" />
</children>
</HBox>