2022-12-15 13:21:26 +00:00
|
|
|
package bin;
|
|
|
|
|
2023-06-07 18:10:51 +00:00
|
|
|
import kernel.log.Log;
|
2023-05-23 12:36:51 +00:00
|
|
|
import kernel.binstore.BinStore;
|
2023-05-22 20:25:37 +00:00
|
|
|
import kernel.ps.ProcessHandle;
|
2023-04-13 22:19:49 +00:00
|
|
|
import kernel.ps.Process;
|
|
|
|
import kernel.ps.ProcessManager;
|
2022-12-19 20:06:23 +00:00
|
|
|
import lib.Color;
|
2022-12-15 13:21:26 +00:00
|
|
|
import kernel.ui.WindowContext;
|
|
|
|
|
2023-01-28 02:55:32 +00:00
|
|
|
using tink.CoreApi;
|
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
class Terminal implements Process {
|
2022-12-15 13:21:26 +00:00
|
|
|
private var context:WindowContext;
|
|
|
|
private var input:String = "";
|
2022-12-19 16:35:52 +00:00
|
|
|
private var backlog:Array<String> = [];
|
2023-05-22 20:25:37 +00:00
|
|
|
private var handle:ProcessHandle;
|
|
|
|
private var requestRender: () -> Void;
|
2023-06-07 18:10:51 +00:00
|
|
|
private var runningPID:PID = -1;
|
2022-12-15 13:21:26 +00:00
|
|
|
|
|
|
|
public function new() {}
|
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
public function run(handle: ProcessHandle): Void {
|
|
|
|
this.handle = handle;
|
|
|
|
|
|
|
|
var statelessContext = handle.createStatelessWindowContext();
|
|
|
|
|
|
|
|
this.context = statelessContext.ctx;
|
|
|
|
this.requestRender = statelessContext.requestRender;
|
|
|
|
|
|
|
|
statelessContext.setRenderFunc(this.render);
|
2022-12-15 13:21:26 +00:00
|
|
|
|
2023-06-07 18:10:51 +00:00
|
|
|
handle.addCallbackLink(this.context.onChar.handle(char -> {
|
|
|
|
if (this.runningPID > 0) return;
|
2022-12-15 13:21:26 +00:00
|
|
|
this.input += char;
|
2023-05-22 20:25:37 +00:00
|
|
|
this.requestRender();
|
2023-06-07 18:10:51 +00:00
|
|
|
}));
|
|
|
|
|
|
|
|
handle.addCallbackLink(this.context.onKey.handle(e -> {
|
|
|
|
switch (e.keyCode) {
|
|
|
|
case 259: // Backspace
|
|
|
|
if (this.runningPID > 0) return;
|
|
|
|
this.input = this.input.substr(0, this.input.length - 1);
|
|
|
|
this.requestRender();
|
|
|
|
case 257: // Enter
|
|
|
|
if (this.runningPID > 0) return;
|
|
|
|
this.backlog.push("> " + this.input);
|
|
|
|
var command = this.input;
|
|
|
|
this.input = "";
|
|
|
|
this.requestRender();
|
|
|
|
this.invokeCommand(command);
|
|
|
|
case 269: // END
|
|
|
|
this.stopCurrentPS();
|
2022-12-15 13:21:26 +00:00
|
|
|
}
|
2023-06-07 18:10:51 +00:00
|
|
|
}));
|
2022-12-15 13:21:26 +00:00
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
this.requestRender();
|
|
|
|
}
|
2023-01-28 02:55:32 +00:00
|
|
|
|
2023-06-07 18:10:51 +00:00
|
|
|
private function stopCurrentPS() {
|
|
|
|
if (this.runningPID < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcessManager.kill(this.runningPID);
|
|
|
|
}
|
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
private function render() {
|
|
|
|
redrawBacklog();
|
|
|
|
redrawInput();
|
2022-12-15 13:21:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private function redrawBacklog() {
|
|
|
|
var size = this.context.getSize();
|
|
|
|
var linesAvailable = size.y - 1;
|
|
|
|
|
|
|
|
var start:Int = this.backlog.length - linesAvailable;
|
|
|
|
|
2022-12-19 16:35:52 +00:00
|
|
|
for (i in 0...linesAvailable) {
|
2022-12-15 13:21:26 +00:00
|
|
|
var line = this.backlog[start + i];
|
|
|
|
|
|
|
|
this.context.setCursorPos(0, i);
|
|
|
|
this.context.clearLine();
|
|
|
|
|
|
|
|
if (line != null) {
|
|
|
|
this.context.write(line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.moveCursorToInput();
|
|
|
|
}
|
|
|
|
|
|
|
|
private function redrawInput() {
|
|
|
|
var size = this.context.getSize();
|
|
|
|
|
|
|
|
this.context.setCursorPos(0, size.y - 1);
|
|
|
|
this.context.clearLine();
|
|
|
|
|
|
|
|
this.context.setTextColor(Color.Blue);
|
|
|
|
this.context.write("> ");
|
|
|
|
|
|
|
|
this.context.setTextColor(Color.White);
|
|
|
|
this.context.write(this.input);
|
|
|
|
|
|
|
|
this.context.setCursorBlink(true);
|
|
|
|
}
|
|
|
|
|
2022-12-19 16:35:52 +00:00
|
|
|
private function invokeCommand(command:String):Void {
|
2022-12-15 13:21:26 +00:00
|
|
|
var args = this.parseArgs(command);
|
|
|
|
|
|
|
|
if (args.length == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var commandName = args[0];
|
|
|
|
|
|
|
|
// Handle built-in commands
|
|
|
|
switch (commandName) {
|
|
|
|
case "clear":
|
|
|
|
this.clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var commandArgs:Array<String> = args.slice(1);
|
|
|
|
var hadInput = false;
|
2023-04-13 22:19:49 +00:00
|
|
|
|
|
|
|
var ps = getProgByName(commandName);
|
|
|
|
if (ps == null) {
|
|
|
|
this.backlog.push("Unknown command: " + commandName);
|
|
|
|
this.redrawBacklog();
|
|
|
|
this.redrawInput();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-06-07 18:10:51 +00:00
|
|
|
this.runningPID = ProcessManager.run(ps,{
|
2023-04-13 22:19:49 +00:00
|
|
|
args: commandArgs,
|
2022-12-15 13:21:26 +00:00
|
|
|
onWrite: (s:String) -> {
|
2023-05-29 20:10:25 +00:00
|
|
|
if (s == "") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-12-15 13:21:26 +00:00
|
|
|
if (!hadInput) {
|
2023-05-29 20:10:25 +00:00
|
|
|
// Add a new line, so that the input is not on the same line as the command
|
2022-12-15 13:21:26 +00:00
|
|
|
this.backlog.push("");
|
|
|
|
hadInput = true;
|
|
|
|
}
|
2023-05-29 20:10:25 +00:00
|
|
|
|
|
|
|
for (line in s.split("\n")) {
|
|
|
|
if (line == ""){
|
|
|
|
this.backlog.push("");
|
|
|
|
} else {
|
|
|
|
this.backlog[this.backlog.length - 1] += s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
this.requestRender();
|
2022-12-15 13:21:26 +00:00
|
|
|
},
|
2023-04-13 22:19:49 +00:00
|
|
|
onExit: (success:Bool) -> {
|
2023-06-07 18:10:51 +00:00
|
|
|
this.runningPID = -1;
|
2023-04-13 22:19:49 +00:00
|
|
|
if (this.backlog[this.backlog.length - 1] == "") {
|
|
|
|
this.backlog.pop();
|
|
|
|
}
|
|
|
|
|
2023-05-22 20:25:37 +00:00
|
|
|
this.requestRender();
|
2022-12-15 13:21:26 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.context.setCursorBlink(false);
|
|
|
|
}
|
|
|
|
|
2022-12-19 16:35:52 +00:00
|
|
|
private function parseArgs(command:String):Array<String> {
|
2022-12-15 13:21:26 +00:00
|
|
|
// TODO: tim and quote handling
|
|
|
|
return command.split(" ");
|
|
|
|
}
|
|
|
|
|
|
|
|
private function clear() {
|
|
|
|
this.backlog = [];
|
|
|
|
this.redrawBacklog();
|
|
|
|
}
|
|
|
|
|
2023-04-13 22:19:49 +00:00
|
|
|
private function getProgByName(name:String):Process {
|
2023-05-23 12:36:51 +00:00
|
|
|
var bin = BinStore.instance.getBinByAlias(name);
|
|
|
|
if (bin == null) {
|
|
|
|
return null;
|
2022-12-15 13:21:26 +00:00
|
|
|
}
|
2023-05-23 12:36:51 +00:00
|
|
|
|
|
|
|
return Type.createInstance(bin.c,[]);
|
2022-12-15 13:21:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private function moveCursorToInput() {
|
|
|
|
var size = this.context.getSize();
|
|
|
|
this.context.setCursorPos(this.input.length + 2, size.y - 1);
|
|
|
|
}
|
|
|
|
}
|