1
0
mirror of https://github.com/mitb-archive/filebot synced 2025-01-12 14:28:29 -05:00
filebot/source/net/filebot/cli/GroovyPad.java
2016-11-25 23:59:26 +08:00

322 lines
8.9 KiB
Java

package net.filebot.cli;
import static net.filebot.Logging.*;
import static net.filebot.util.ui.SwingUI.*;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dialog.ModalExclusionType;
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.PrintStream;
import java.io.UnsupportedEncodingException;
import java.util.HashMap;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;
import javax.script.Bindings;
import javax.script.ScriptException;
import javax.script.SimpleBindings;
import javax.swing.Action;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JSplitPane;
import javax.swing.JToolBar;
import javax.swing.KeyStroke;
import javax.swing.SwingUtilities;
import javax.swing.SwingWorker;
import javax.swing.text.JTextComponent;
import org.fife.ui.rsyntaxtextarea.FileLocation;
import org.fife.ui.rsyntaxtextarea.SyntaxConstants;
import org.fife.ui.rsyntaxtextarea.TextEditorPane;
import org.fife.ui.rtextarea.RTextScrollPane;
import net.filebot.ApplicationFolder;
import net.filebot.ResourceManager;
import net.filebot.Settings;
import net.filebot.util.TeePrintStream;
public class GroovyPad extends JFrame {
public static final String DEFAULT_SCRIPT = "runScript 'sysinfo'";
public GroovyPad() throws IOException {
super("Groovy Pad");
JSplitPane splitPane = new JSplitPane(JSplitPane.VERTICAL_SPLIT, true, createEditor(), createOutputLog());
splitPane.setResizeWeight(0.7);
JComponent c = (JComponent) getContentPane();
c.setLayout(new BorderLayout(0, 0));
c.add(splitPane, BorderLayout.CENTER);
JToolBar tools = new JToolBar("Run", JToolBar.HORIZONTAL);
tools.setFloatable(true);
tools.add(run);
tools.add(cancel);
tools.addSeparator();
tools.add(newAction(DEFAULT_SCRIPT, ResourceManager.getIcon("status.info"), evt -> runScript(DEFAULT_SCRIPT)));
c.add(tools, BorderLayout.NORTH);
run.setEnabled(true);
cancel.setEnabled(false);
installAction(c, KeyStroke.getKeyStroke(KeyEvent.VK_F5, 0), run);
installAction(c, KeyStroke.getKeyStroke(KeyEvent.VK_R, KeyEvent.CTRL_DOWN_MASK), run);
addWindowListener(new WindowAdapter() {
@Override
public void windowClosed(WindowEvent evt) {
cancel.actionPerformed(null);
console.unhook();
try {
editor.save();
} catch (IOException e) {
// ignore
}
}
});
console = new MessageConsole(output);
console.hook();
shell = createScriptShell();
editor.requestFocusInWindow();
setModalExclusionType(ModalExclusionType.TOOLKIT_EXCLUDE);
setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
setLocationByPlatform(true);
setSize(800, 600);
}
protected MessageConsole console;
protected TextEditorPane editor;
protected TextEditorPane output;
protected JComponent createEditor() {
editor = new TextEditorPane(TextEditorPane.INSERT_MODE, false);
editor.setSyntaxEditingStyle(SyntaxConstants.SYNTAX_STYLE_GROOVY);
editor.setAutoscrolls(false);
editor.setAnimateBracketMatching(false);
editor.setAntiAliasingEnabled(true);
editor.setAutoIndentEnabled(true);
editor.setBracketMatchingEnabled(true);
editor.setCloseCurlyBraces(true);
editor.setClearWhitespaceLinesEnabled(true);
editor.setCodeFoldingEnabled(true);
editor.setHighlightSecondaryLanguages(false);
editor.setRoundedSelectionEdges(false);
editor.setTabsEmulated(false);
try {
// use this default value so people can easily submit bug reports with fn:sysinfo logs
File pad = ApplicationFolder.AppData.resolve("pad.groovy");
if (!pad.exists()) {
ScriptShellMethods.saveAs(DEFAULT_SCRIPT, pad);
}
// restore on open
editor.load(FileLocation.create(pad), "UTF-8");
} catch (Exception e) {
debug.log(Level.WARNING, e, e::toString);
}
return new RTextScrollPane(editor, true);
}
protected JComponent createOutputLog() throws IOException {
output = new TextEditorPane(TextEditorPane.INSERT_MODE, false);
output.setEditable(false);
output.setReadOnly(true);
output.setAutoscrolls(true);
output.setBackground(new Color(255, 255, 218));
output.setSyntaxEditingStyle(SyntaxConstants.SYNTAX_STYLE_NONE);
output.setAnimateBracketMatching(false);
output.setAntiAliasingEnabled(true);
output.setAutoIndentEnabled(false);
output.setBracketMatchingEnabled(false);
output.setCloseCurlyBraces(false);
output.setClearWhitespaceLinesEnabled(false);
output.setCodeFoldingEnabled(false);
output.setHighlightCurrentLine(false);
output.setHighlightSecondaryLanguages(false);
output.setRoundedSelectionEdges(false);
output.setTabsEmulated(false);
return new RTextScrollPane(output, true);
}
protected final ScriptShell shell;
protected ScriptShell createScriptShell() {
try {
return new ScriptShell(s -> ScriptSource.GITHUB_STABLE.getScriptProvider(s).getScript(s), new CmdlineOperations(), new HashMap<String, Object>());
} catch (Exception e) {
throw new RuntimeException(e);
}
}
protected final Action run = newAction("Run", ResourceManager.getIcon("script.go"), this::runScript);
protected final Action cancel = newAction("Cancel", ResourceManager.getIcon("script.cancel"), this::cancelScript);
private Runner currentRunner = null;
public void runScript(ActionEvent evt) {
try {
editor.save();
} catch (IOException e) {
debug.log(Level.WARNING, e, e::toString);
}
runScript(editor.getText());
}
public void runScript(String script) {
if (currentRunner == null || currentRunner.isDone()) {
currentRunner = new Runner(script) {
@Override
protected void done() {
run.setEnabled(true);
cancel.setEnabled(false);
}
};
run.setEnabled(false);
cancel.setEnabled(true);
output.setText(null);
currentRunner.execute();
}
}
@SuppressWarnings("deprecation")
protected void cancelScript(ActionEvent evt) {
if (currentRunner != null && !currentRunner.isDone()) {
currentRunner.cancel(true);
currentRunner.getExecutionThread().stop();
try {
currentRunner.get(2, TimeUnit.SECONDS);
} catch (Exception e) {
debug.log(Level.WARNING, e, e::getMessage);
}
}
}
protected class Runner extends SwingWorker<Object, Object> {
private Thread executionThread;
private Object result;
public Runner(final String script) {
executionThread = new Thread("GroovyPadRunner") {
@Override
public void run() {
try {
Bindings bindings = new SimpleBindings();
bindings.put(ScriptShell.SHELL_ARGS_BINDING_NAME, Settings.getApplicationArguments());
bindings.put(ScriptShell.ARGV_BINDING_NAME, Settings.getApplicationArguments().getFiles(false));
result = shell.evaluate(script, bindings);
// print result and make sure to flush Groovy output
SimpleBindings resultBindings = new SimpleBindings();
resultBindings.put("result", result);
if (result != null) {
shell.evaluate("print('Result: '); println(result);", resultBindings);
} else {
shell.evaluate("println();", resultBindings);
}
} catch (ScriptException e) {
while (e.getCause() instanceof ScriptException) {
e = (ScriptException) e.getCause();
}
e.printStackTrace();
} catch (Throwable e) {
e.printStackTrace();
}
};
};
executionThread.setDaemon(false);
executionThread.setPriority(Thread.MIN_PRIORITY);
}
@Override
protected Object doInBackground() throws Exception {
executionThread.start();
executionThread.join();
return result;
}
public Thread getExecutionThread() {
return executionThread;
}
};
public static class MessageConsole {
private final PrintStream system_out = System.out;
private final PrintStream system_err = System.err;
private JTextComponent textComponent;
public MessageConsole(JTextComponent textComponent) {
this.textComponent = textComponent;
}
public void hook() {
try {
System.setOut(new TeePrintStream(new ConsoleOutputStream(), true, "UTF-8", system_out));
System.setErr(new TeePrintStream(new ConsoleOutputStream(), true, "UTF-8", system_err));
} catch (UnsupportedEncodingException e) {
debug.log(Level.WARNING, e, e::getMessage);
}
}
public void unhook() {
System.setOut(system_out);
System.setErr(system_err);
}
private class ConsoleOutputStream extends ByteArrayOutputStream {
@Override
public void flush() {
try {
String message = this.toString("UTF-8");
reset();
commit(message);
} catch (Exception e) {
// can't happen
}
}
private void commit(final String line) {
SwingUtilities.invokeLater(() -> {
try {
int offset = textComponent.getDocument().getLength();
textComponent.getDocument().insertString(offset, line, null);
textComponent.setCaretPosition(textComponent.getDocument().getLength());
} catch (Exception e) {
// ignore
}
});
}
}
}
}