У меня есть окно графического интерфейса с JavaFX
TextArea
, где я хотел бы выводить весь вывод логгера (информация, отладка, ошибка и т. Д.) В реальном времени (аналогично выводу консоли из IDE при запускепроект).
Я посмотрел на этот ответ , и я попытался адаптировать код, но он не работает, у меня такое ощущение, что что-то не инициализируется или не вызываетсяили logback.xml
не указывает на правильный класс.
Слишком сложно включить весь код здесь, поэтому у меня есть демонстрационный проект на GitHub , пожалуйста, клонируйте и позвольте мнезнаете, что вы думаете?
Регистратор: logback
, slf4j
, log4j
Используемая среда разработки: IntelliJ 2018.1
package Demo_Logger;
import javafx.application.Application;
import javafx.application.Platform;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.OutputStream;
import static Demo_Logger.controller.ControllerMain.CONTROLLER_MAIN;
public class Main extends Application {
private static final Logger logger = LoggerFactory.getLogger(Main.class);
public static void main(String[] args) {
launch(args);
}
@Override
public void start(Stage primaryStage) throws Exception {
Parent parent = FXMLLoader.load(getClass().getResource("/view/main.fxml"));
primaryStage.setTitle("Demo logger");
primaryStage.setWidth(400);
primaryStage.setHeight(400);
primaryStage.setScene(new Scene(parent));
primaryStage.show();
Platform.runLater(this::setTextArea);
Platform.runLater(this::logToTexArea);
System.out.println("Main -> init"); //OK called
}
private void setTextArea() {
OutputStream outputStream = new TextAreaOutputStream(CONTROLLER_MAIN.textArea);
MyStaticOutputStreamAppender.setStaticOutputStream(outputStream);
}
private void logToTexArea() {
logger.debug("Debug example");
logger.error("Error example");
CONTROLLER_MAIN.textArea.setText("ADD LOGGER OUTPUT HERE");
}
}
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.TextArea?>
<?import javafx.scene.layout.BorderPane?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.VBox?>
<BorderPane xmlns="http://javafx.com/javafx/8.0.121" xmlns:fx="http://javafx.com/fxml/1"
fx:controller="Demo_Logger.controller.ControllerMain">
<center>
<HBox alignment="CENTER">
<VBox alignment="CENTER" minWidth="100.0" spacing="10.0">
<Button fx:id="btnLogInfo" mnemonicParsing="false" onAction="#btnLogInfo" text="Log Info" />
<Button fx:id="btnLogError" mnemonicParsing="false" onAction="#btnLogError" text="Log Error" />
</VBox>
<TextArea fx:id="textArea" promptText="logger has to output here" />
</HBox>
</center>
</BorderPane>
package Demo_Logger.controller;
import Demo_Logger.Main;
import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.TextArea;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class ControllerMain {
public static ControllerMain CONTROLLER_MAIN;
private static final Logger logger = LoggerFactory.getLogger(Main.class);
@FXML
public TextArea textArea;
@FXML
public Button btnLogInfo, btnLogError;
public ControllerMain() {
CONTROLLER_MAIN = this;
}
@FXML
public void btnLogInfo() {
logger.info("Button info click");
}
@FXML
public void btnLogError() {
logger.error("Button error click");
}
}
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<layout class="ch.qos.logback.classic.PatternLayout">
<Pattern>
<!-- THE PATTERN IS NOT WORKING - INVESTIGATE -->
%-50(%d{HH:mm:ss} [%level{}] %class{0}.%method{0}\(\)) - %message{}%n{}
</Pattern>
</layout>
</appender>
<!--<appender name="MyCustomAppender" class=".src.main.java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class="src.main.java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class=".main.java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class="main.java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class=".java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class="java.Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class=".Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class="Demo_Logger.MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class=".MyStaticOutputStreamAppender">-->
<!--<appender name="MyCustomAppender" class="MyStaticOutputStreamAppender">-->
<appender name="MyCustomAppender" class="Demo_Logger.MyStaticOutputStreamAppender">
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
<level>ALL</level>
</filter>
<encoder>
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{5} - %msg%n</pattern>
</encoder>
</appender>
<root>
<appender-ref ref="STDOUT" />
<appender-ref ref="MyCustomAppender" />
</root>
</configuration>
package Demo_Logger;
import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.OutputStream;
public class DelegatingOutputStream extends FilterOutputStream {
/**
* Creates a delegating outputstream with a NO-OP delegate
*/
public DelegatingOutputStream(OutputStream out) {
super(new OutputStream() {
@Override
public void write(int b) throws IOException {
}
});
System.out.println("DelegatingOutputStream -> init"); //OK called
}
void setOutputStream(OutputStream outputStream) {
this.out = outputStream;
System.out.println("DelegatingOutputStream -> setOutputStream"); //OK called
}
}
package Demo_Logger;
import javafx.scene.control.TextArea;
import java.io.IOException;
import java.io.OutputStream;
public class TextAreaOutputStream extends OutputStream {
private TextArea textArea;
public TextAreaOutputStream(TextArea textArea) {
this.textArea = textArea;
System.out.println("TextAreaOutputStream -> init"); //OK called
}
@Override
public void write(int b) throws IOException {
textArea.appendText(String.valueOf((char) b));
System.out.println("TextAreaOutputStream -> write");
}
}
package Demo_Logger;
import ch.qos.logback.core.OutputStreamAppender;
import java.io.OutputStream;
public class MyStaticOutputStreamAppender<E> extends OutputStreamAppender<E> {
private static final DelegatingOutputStream DELEGATING_OUTPUT_STREAM = new DelegatingOutputStream(null);
@Override
public void start() {
setOutputStream(DELEGATING_OUTPUT_STREAM);
System.out.println("MyStaticOutputStreamAppender -> start");
super.start();
}
static void setStaticOutputStream(OutputStream outputStream) {
DELEGATING_OUTPUT_STREAM.setOutputStream(outputStream);
System.out.println("MyStaticOutputStreamAppender -> setStaticOutputStream"); //OK called
}
}