JavaFX 8 WebEngine:如何在java中将console.log()从javascript导入System.out?

时间:2015-02-24 03:26:05

标签: webview javafx-8 javafx-webengine

我使用JavaFX和JavaFX WebEngine中的javascript引擎来开发应用程序。我想从javascript获取反馈以进行调试。 WebEngine中的控制台输出会发生什么?有什么办法可以访问它,或者在java中重定向到System.out吗?

3 个答案:

答案 0 :(得分:23)

以下代码将console.log()重定向到JavaBridge.log()

import netscape.javascript.JSObject;

[...]

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

// Maintain a strong reference to prevent garbage collection:
// https://bugs.openjdk.java.net/browse/JDK-8154127
private final JavaBridge bridge = new JavaBridge();

[...]

webEngine.getLoadWorker().stateProperty().addListener((observable, oldValue, newValue) ->
{
    JSObject window = (JSObject) webEngine.executeScript("window");
    window.setMember("java", bridge);
    webEngine.executeScript("console.log = function(message)\n" +
        "{\n" +
        "    java.log(message);\n" +
        "};");
});

答案 1 :(得分:8)

您只需添加消息监听器即可查看输出中发生的情况。您不必为每个加载的页面注入js桥,重新定义console.log等函数

WebConsoleListener.setDefaultListener((webView, message, lineNumber, sourceId) -> {
    System.out.println(message + "[at " + lineNumber + "]");
});

答案 2 :(得分:5)

我喜欢走另一个方向。我们使用log4j,所以我创建了一个javascript包装器,如下所示:

module.exports = {

    levels:[ "ALL", "TRACE", "DEBUG", "INFO", "WARN", "ERROR", "OFF"],

    level:"INFO",

    error:function(msg){
      if(this.isErrorEnabled()){
        console.error(msg)
      }
    },
    warn:function(msg){
      if(this.isWarnEnabled()){
        console.warn(msg)
      }
    },
    info:function(msg){
      if(this.isInfoEnabled()){
        console.log("INFO: "+msg)
      }
    },
    debug:function(msg){
      if(this.isDebugEnabled()){
        console.log("DEBUG: "+msg)
      }
    },
    trace:function(msg){
      if(this.isTraceEnabled()){
        console.log("TRACE: "+msg)
      }
    },

    isErrorEnabled:function(){
      return this.isEnabled("ERROR");
    },
    isWarnEnabled:function(){
      return this.isEnabled("WARN");
    },
    isInfoEnabled:function(){
      return this.isEnabled("INFO");
    },
    isDebugEnabled:function(){
      return this.isEnabled("DEBUG");
    },
    isTraceEnabled:function(){
      return this.isEnabled("TRACE");
    },
    isEnabled:function(statementLevel){
      return this.levels.indexOf(this.level)<=this.levels.indexOf(statementLevel);
    }
  }

然后在javascript的开头我检查日志是否存在并设置它:

if(window.log == undefined){
  window.log = require("./utils/log4j-wrapper")
  window.log.level = "INFO"
}

如果您在加载网址之前直接在引擎上设置Log4j记录器,就像这样:

WebEngine webEngine = webView.getEngine()
JSObject win = (JSObject) webEngine.executeScript("window")
win.setMember("log", log)  //log being the java log4j logger

这样,如果我直接在浏览器中打开或者从JavaFX程序中的WebView运行,我就可以登录。并且具有与javascript中的日志记录级别相匹配的额外好处,这些级别与WebView控制器的包相匹配。只是更大的javascript视图的替代方案。