Search code examples
javajavafxtreeviewtreeviewitem

JavaFX TreeView ChangeListener old value is always null


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. enter image description here

But when I`m selecting another item nothing happens. enter image description here

And after switching back to my object(which I want to represent) app creates second instance of it(cause previous wasnt deleted).enter image description here

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.


Solution

  • 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.