Вызовы из JavaScript в JavaFX не работают из-за VLCJ

Вызовы из JavaScript в JavaFX работают нормально. Но если я добавлю переменную, содержащую реализацию плеера VLCJ, то все перестанет работать.

Main.java:

import javafx.application.Application;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.scene.Scene;
import javafx.scene.layout.StackPane;
import javafx.scene.paint.Color;
import javafx.scene.web.WebEngine;
import javafx.scene.web.WebView;
import javafx.stage.Stage;
import javafx.concurrent.Worker;
import javafx.concurrent.Worker.State;
import netscape.javascript.JSObject;

public class Main extends Application {
    WebView browser = new WebView();
    WebEngine webEngine = browser.getEngine();
    VLC player = new VLC(); // It doesn't work because of it. But if you remove that, then everything works.

    @Override
    public void start(final Stage stage) {
        webEngine.setJavaScriptEnabled(true);
        String HTML_STRING = //
                "<html>"//
                        + "<head> " //
                        + "  <script language='javascript'> " //
                        + "     function callToJavaFX()  { "//
                        + "        myJavaMember.log('text'); " //
                        + "     } " //
                        + "  </script> "//
                        + "</head> "//
                        + "<body> "//
                        + "   <h2>This is Html content</h2> "//
                        + "   <button onclick='callToJavaFX();'>Call To JavaFX</button> "//
                        + "</body> "//
                        + "</html> "//
                ;
        webEngine.loadContent(HTML_STRING);
        Worker<Void> worker = webEngine.getLoadWorker();
        worker.stateProperty().addListener(new ChangeListener<State>() {
            @Override
            public void changed(ObservableValue<? extends State> observable, State oldValue, State newValue) {
                if (newValue == Worker.State.SUCCEEDED) {
                    JSObject jsobj = (JSObject) webEngine.executeScript("window");
                    jsobj.setMember("myJavaMember", new JSBridge());
                }
            }
        });

        StackPane stack_pane = new StackPane(browser);
        stage.setScene(new Scene(stack_pane,700, 400, Color.BLACK));
        stage.show();
    }

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

}

JSBridge.java:

public class JSBridge {
    public void log(String text) {
        System.out.println(text);
    }
}

VLC.java:

import uk.co.caprica.vlcj.factory.MediaPlayerFactory;
import uk.co.caprica.vlcj.player.embedded.EmbeddedMediaPlayer;

public class VLC{

    private final MediaPlayerFactory mediaPlayerFactory;
    private final EmbeddedMediaPlayer mediaPlayer;

    public VLC() {
        this.mediaPlayerFactory = new MediaPlayerFactory();
        this.mediaPlayer = mediaPlayerFactory.mediaPlayers().newEmbeddedMediaPlayer(); // It doesn't work because of it. But if you remove that, then everything works.
    }
}

Мои модули: введите здесь описание изображения

До сих пор мне удалось выяснить только то, что вызовы не работают из-за объявления переменной 'player' в Main.java. >. Но это сработает, если вы измените this.mediaPlayer = mediaPlayerFactory.mediaPlayers(). NewEmbeddedMediaPlayer() в VLC.java; на this.mediaPlayer = null

Если убрать строчку 'VLC player = new VLC();' то все работает. Если его не удалить, то метод 'myJavaMember.log('текст')' работать не будет и будет ошибка: 'TypeError: myJavaMember.log не является функцией (В 'myJavaMember.log(' text') ',' myJavaMember.log ' не определен). Почему?

Подскажите, пожалуйста, как это исправить и в чем может быть причина?

P.S.: я знаю, как реализовать VLCJ. Меня устраивает. Вот только код, который может повторить беспокоящую меня ошибку. Проблема в том, что звонки не работают. Повторяю, как реализовать знакомый мне проигрыватель VLCJ.


person Дмитрий Чубаров    schedule 24.02.2021    source источник
comment
Полные подробные логи пожалуйста   -  person mfkl    schedule 01.03.2021
comment
@mfkl Если вы удалите строку «VLC player = new VLC ();» тогда все работает. Если его не удалить, то метод 'myJavaMember.log('текст')' работать не будет и будет ошибка 'TypeError: myJavaMember.log не является функцией (В 'myJavaMember.log('текст')' , 'myJavaMember.log' не определен). Почему это?   -  person Дмитрий Чубаров    schedule 02.03.2021


Ответы (1)


Решил проблему благодаря этому ответу: https://stackoverflow.com/a/53618875/10946427

Оказывается, мой JSBridge уничтожался сборщиком мусора, и из-за VLCJ он работал быстрее.

Если создать JSBridge переменную внутри класса, то сборщик мусора не уничтожит ее и все будет работать.

person Дмитрий Чубаров    schedule 02.03.2021