lik*_*sky 4 javafx processbuilder javafx-2 fxml
关于JavaFX的新手问题我无法回答,尽管知道它必须非常简单,并且在我看过的任何地方都找不到任何资源(教程,许多Oracle在线文档,文章,以及已知的JavaFX博主等)
我正在开发一个运行应用程序的命令行(脚本),我已经成功地从脚本中获取输出(通过ProcessBuilder),我可以持续显示,就像在命令行上发生的那样.也就是说,我可以System.out.println(line);整天做,在控制台中显示输出,它只是返回正在运行的'myProcess'返回的输入流的输出,如下所示:
BufferedReader bri = new BufferedReader(new InputStreamReader(myProcess.getInputStream()))
Run Code Online (Sandbox Code Playgroud)
所以我能够看到从脚本返回的所有输出.
我想设置一个JavaFX TextArea或ScrollPane,或者不确定是什么,以显示这个输出文本(它有很多,如几千行)作为脚本中正在发生的'进展',当它发生的时候.我有一个场景,我有按钮并从这个场景获取输入以启动脚本运行,但现在我想显示点击"运行此脚本"按钮的结果,可以这么说.
我假设我需要创建一个TextArea,如此处所述,或者TextBuilder可能对开始制作它很有用.不确定.
我需要一些帮助来设置绑定或自动滚动/自动更新的一部分.
有人可以为我提供一个开始的地方,用JavaFX做到这一点吗?我宁愿不使用Swing.
(我正在使用JavaFX 2.2,JDK 1.7u7,所有最新的东西,是的,这是一个FXML应用程序 - 所以这样做是首选.)
更新:Sergey Grinev的回答在绑定部分非常有帮助.但是,当我要求"如何设置一些帮助"时,这里有一些更详细的信息 - 基本上,我需要将控制权返回到主场景以允许用户取消脚本,或者以其他方式监视这是怎么回事.所以我想"生成"运行该脚本的进程(也就是说,有某种'自由运行进程'),但仍然从中获取输出.(在我最初的问题中,我对此并不十分清楚.)
我在这里使用的技术(见下文)是在进程上执行waitFor,但这当然意味着在脚本执行时对话框/场景被"挂起".我想获得控制权,但是如何将'p'(进程)传递给其他控制器部分(或者,只需启动其他进程传入参数以启动脚本并让它启动脚本)然后通过绑定Sergey Grinev提到自动更新 - 没有"挂"场景/窗口?另外:如果用户请求,我可以"停止"这个其他过程吗?
这是我当前的代码('等待'而脚本 - 运行需要20-40分钟! - 完成;这不是我想要的,我想控制返回给用户):
public class MyController implements Initializable {
@FXML
private void handleRunScript(ActionEvent event) throws IOException {
ProcessBuilder pb = new ProcessBuilder("myscript.sh", "arg1", "arg2", ...);
Process p = pb.start();
try {
BufferedReader bri = new BufferedReader
(new InputStreamReader(p.getInputStream()));
String line;
while ((line = bri.readLine()) != null) {
System.out.println(line);
textAreaRight.setText(line);
}
bri.close();
p.waitFor();
}
catch (Exception err) {
err.printStackTrace();
}
}
@FXML
private void handleCancel(ActionEvent event) {
doSomethingDifferent();
}
}
Run Code Online (Sandbox Code Playgroud)
public class DoTextAreaLog extends Application {
TextArea log = new TextArea();
Process p;
@Override
public void start(Stage stage) {
try {
ProcessBuilder pb = new ProcessBuilder("ping", "stackoverflow.com", "-n", "100");
p = pb.start();
// this thread will read from process without blocking an application
new Thread(new Runnable() {
@Override
public void run() {
try {
//try-with-resources from jdk7, change it back if you use older jdk
try (BufferedReader bri = new BufferedReader(new InputStreamReader(p.getInputStream()))) {
String line;
while ((line = bri.readLine()) != null) {
log(line);
}
}
} catch (IOException ex) {
ex.printStackTrace();
}
}
}).start();
stage.setScene(new Scene(new Group(log), 400, 300));
stage.show();
} catch (IOException ex) {
ex.printStackTrace();
}
}
@Override
public void stop() throws Exception {
super.stop();
// this called on fx app close, you may call it in user action handler
if (p!=null ) {
p.destroy();
}
}
private void log(final String st) {
// we can access fx objects only from fx thread
// so we need to wrap log access into Platform#runLater
Platform.runLater(new Runnable() {
@Override
public void run() {
log.setText(st + "\n" + log.getText());
}
});
}
public static void main(String[] args) {
launch();
}
}
Run Code Online (Sandbox Code Playgroud)