I have a TreeView of Objects. I want to add a ChangeListener that will draw my object on right pane and remove old one if present. But I can not do that, because of non obvious reasons my old value is always a null. Here is my Listener:
treeView.getSelectionModel().selectedItemProperty().addListener((ObservableValue<? extends TreeItem<Asset>> ov, TreeItem<Asset> t, TreeItem<Asset> t1) -> {
t.getValue().derepresent();
t1.getValue().represent();
});
This code makes my app to "represent" object well.
But when I`m selecting another item nothing happens.
And after switching back to my object(which I want to represent) app creates second instance of it(cause previous wasn
t deleted).
So I`m asking, how can I solve that?
EDIT: Here is useful code from my Controller class:
public class Controller implements Initializable {
@FXML
private VBox infoVBox;
@FXML
private TreeView<Asset> treeView;
private Stage stage;
@Override
public void initialize(URL location, ResourceBundle resources) {
}
public void selectRootOnClick(ActionEvent actionEvent) {
try {
final DirectoryChooser directoryChooser = new DirectoryChooser();
final File selectedDirectory = directoryChooser.showDialog(stage);
TreeItem<Asset> treeItem = new TreeItem<>(new Asset(selectedDirectory.getAbsolutePath(), infoVBox));
createTree(treeItem);
treeItem.getChildren().sort(Comparator.comparing(t -> t.getValue().getAssetPath().toLowerCase()));
treeView.setRoot(treeItem);
treeView.setCellFactory(new Callback<>() {
public TreeCell<Asset> call(TreeView<Asset> tv) {
return new TreeCell<>() {
@Override
protected void updateItem(Asset item, boolean empty) {
super.updateItem(item, empty);
setText((empty || item == null) ? "" : item.getAssetPath());
}
};
}
});
treeView.getSelectionModel().selectedItemProperty().addListener((ObservableValue<? extends TreeItem<Asset>> ov, TreeItem<Asset> t, TreeItem<Asset> t1) -> {
if (t != null) t.getValue().derepresent();
t1.getValue().represent();
});
} catch (IOException e) {
e.printStackTrace();
}
}
public void createTree(TreeItem<Asset> rootItem) throws IOException {
try (DirectoryStream<Path> directoryStream = Files.newDirectoryStream(Paths.get(rootItem.getValue().getAssetPath()))) {
for (Path path : directoryStream) {
TreeItem<Asset> newItem;
if (!Files.isDirectory(path)) {
String extension = "";
int i = path.toString().lastIndexOf('.');
if (i > 0) extension = path.toString().substring(i + 1);
switch (extension) {
case "mp3":
newItem = new TreeItem<>(new Audio(path.toString(), infoVBox));
rootItem.getChildren().add(newItem);
break;
default:
newItem = new TreeItem<>(new Asset(path.toString(), infoVBox));
rootItem.getChildren().add(newItem);
break;
}
} else {
newItem = new TreeItem<>(new Asset(path.toString(), infoVBox));
newItem.setExpanded(true);
rootItem.getChildren().add(newItem);
createTree(newItem);
}
}
}
}
}
I have an hierarchy of classes. In this hierarchy Audio class is inheriting Asset class. Here is Asset class:
public class Asset {
protected final String assetPath;
public String getAssetPath() {
return assetPath;
}
protected VBox parent;
public Asset() {
assetPath = "";
parent = null;
}
public Asset(String path, VBox node) {
assetPath = path;
parent = node;
}
public void represent() {}
public void derepresent() {}
}
And my Audio Class:
public final class Audio extends Asset {
private Media media;
private MediaPlayer mediaPlayer;
private MediaControl mediaControl;
public Audio() {
super();
}
public Audio(String path, VBox parent) {
super(path, parent);
}
@Override
public void represent() {
media = new Media("file:///"+assetPath);
mediaPlayer = new MediaPlayer(media);
mediaControl = new MediaControl(mediaPlayer);
parent.getChildren().add(mediaControl);
}
@Override
public void derepresent() {
mediaControl = null;
mediaPlayer = null;
media = null;
parent.getChildren().remove(mediaControl);
}
}
Also I have MediaControl but I suppose it`s implementation is unnecessary for current thread.
You set mediaControl
too early to null
, so parent.getChildren().remove(mediaControl)
is basically just parent.getChildren().remove(null)
which always throws a NullPointerException
.
Change derepresent()
to:
@Override
public void derepresent() {
// you can drop this null-check when you're sure that mediaControl is set
if(mediaControl != null) {
parent.getChildren().remove(mediaControl);
}
mediaControl = null;
mediaPlayer = null;
media = null;
}
I also recommend to remove the the constructor Audio()
because you probably get a NullPointerException
in derepresent()
and represent
when parent
is still null
.