我正在创建一个页面,该页面将在 Java 的 WebView 中打开,或者在外部浏览器中手动打开。如果页面是从 Java 加载的——我需要它来执行特定的回调,因为 Java 被用作一种后端,但是如果页面是手动加载的,它应该从其资源中加载不同的数据。所以我正在尝试注册一个可以在文档加载时调用的 JS“JavaInit”对象。HTML 看起来有点像这样:
<html>
<head>
<script type='text/javascript' src="http://code.jquery.com/jquery-1.11.2.min.js"></script>
<script type="text/javascript">
DefaultInit = {
init: function(x) {
return "Default init: " + x;
}
};
$(function() {
var text;
if (window["JavaInit"] && JavaInit.init)
text = JavaInit.init("passed text");
else
text = DefaultInit.init("passed text");
$("#title").html(text);
});
</script>
</head>
<body>
<h1 id="title"></h1>
</body>
</html>
我在这里阅读了关于在页面 onload 事件之前设置 JS 成员的问题:javaFX webview window.onload is fire before loadworker succeeds,所以我的 Java 代码如下所示:
public class Test extends Application {
public static void main(String[] args) {
Application.launch(args);
}
@Override
public void start(Stage primaryStage) throws Exception {
WebView web = new WebView();
Scene scene = new Scene(web, 400, 300);
primaryStage.setScene(scene);
primaryStage.show();
initWeb(web);
}
private static void initWeb(WebView web) {
WebEngine eng = web.getEngine();
((JSObject)eng.executeScript("window")).setMember("JavaInit", new JavaInit());
URL url = Test.class.getResource("/test.html");
eng.load(url.toString());
}
public static class JavaInit {
public String init(Object o) {
return "Java init: " + o;
}
}
}
它完全按照要求工作,但只有一次,在任何人调用“重新加载页面”之前。重新加载后引擎似乎重新创建了“窗口”对象,所有注册成员都消失了。
可以将 #setMember 放置在“状态属性”侦听器中:
private static void initWeb(WebView web) {
WebEngine eng = web.getEngine();
eng.getLoadWorker().stateProperty().addListener((val, oldState, newState) -> {
if (newState == State.SUCCEEDED)
((JSObject)eng.executeScript("window")).setMember("JavaInit", new JavaInit());
});
URL url = Test.class.getResource("/test.html");
eng.load(url.toString());
}
但是只有在 window.onload 被触发后才会调用它,并且默认的 init 会一直被调用。这可以通过以最小延迟调用初始化代码来修复:
$(function() {
setTimeout(function() {
var text;
if (window["JavaInit"] && JavaInit.init)
text = JavaInit.init("passed text");
else
text = DefaultInit.init("passed text");
$("#title").html(text);
});
});
然后控制从 JS 处理回 Java(如果存在),然后返回到注册函数,并且程序每次都可以正常工作。
但这似乎有点不对,为了希望Java成功注册所有成员而创建init延迟。那么有人知道实现这种功能的任何“更合适”的方法吗?是否有可能在生产中需要更多的时间,并且 JS 可以在成员注册之前调用延迟函数?或者是否有任何保证 WebEngine 将持续处理 Java 事件和 JS 执行?
谢谢!