Skip to content

Latest commit

 

History

History
131 lines (101 loc) · 6.5 KB

Tutorial02.md

File metadata and controls

131 lines (101 loc) · 6.5 KB

Tutorial 2

In this second tutorial, we'll expand on how to use the Terminal interface to provide more advanced functionality.

    DefaultTerminalFactory defaultTerminalFactory = new DefaultTerminalFactory();
    Terminal terminal = null;
    try {
        terminal = defaultTerminalFactory.createTerminal();

Most terminals and terminal emulators supports what's known as "private mode" which is a separate buffer for the text content that does not support any scrolling. This is frequently used by text editors such as nano and vi in order to give a "fullscreen" view. When exiting from private mode, the previous content is usually restored, including the scrollback history. Emulators that don't support this properly might at least clear the screen after exiting.

        terminal.enterPrivateMode();

You can use the enterPrivateMode() to activate private mode, but you'll need to remember to also exit private mode afterwards so that you don't leave the terminal in a weird state when the application exists. The usual close() at the end will do this automatically, but you can also manually call exitPrivateMode() and finally Lanterna will register a shutdown hook that tries to restore the terminal (including exiting private mode, if necessary) as well.

The terminal content should already be cleared after switching to private mode, but in case it's not, the clear method should make all content set to default background color with no characters and the input cursor in the top-left corner.

        terminal.clearScreen();

It's possible to tell the terminal to hide the text input cursor

        terminal.setCursorVisible(false);

Instead of manually writing one character at a time like we did in the previous tutorial, an easier way is to use a TextGraphic object. You'll see more of this object later on, it's reused on the Screen and TextGUI layers too.

        final TextGraphics textGraphics = terminal.newTextGraphics();

The TextGraphics object keeps its own state of the current colors, separate from the terminal. You can use the foreground and background set methods to specify it and it will take effect on all operations until further modified.

        textGraphics.setForegroundColor(TextColor.ANSI.WHITE);
        textGraphics.setBackgroundColor(TextColor.ANSI.BLACK);

putString(..) exists in a couple of different flavors but it generally works by taking a string and outputting it somewhere in terminal window. Notice that it doesn't take the current position of the text cursor when doing this.

        textGraphics.putString(2, 1, "Lanterna Tutorial 2 - Press ESC to exit", SGR.BOLD);
        textGraphics.setForegroundColor(TextColor.ANSI.DEFAULT);
        textGraphics.setBackgroundColor(TextColor.ANSI.DEFAULT);
        textGraphics.putString(5, 3, "Terminal Size: ", SGR.BOLD);
        textGraphics.putString(5 + "Terminal Size: ".length(), 3, terminal.getTerminalSize().toString());

You still need to flush for changes to become visible

        terminal.flush();

You can attach a resize listener to your Terminal object, which will invoke a callback method (usually on a separate thread) when it is informed of the terminal emulator window changing size. Notice that maybe not all implementations supports this. The UnixTerminal, for example, relies on the WINCH signal being sent to the java process, which might not make it though if you remote shell isn't forwarding the signal properly.

        terminal.addResizeListener(new TerminalResizeListener() {
            @Override
            public void onResized(Terminal terminal, TerminalSize newSize) {
                // Be careful here though, this is likely running on a separate thread. Lanterna is threadsafe in 
                // a best-effort way so while it shouldn't blow up if you call terminal methods on multiple threads, 
                // it might have unexpected behavior if you don't do any external synchronization
                textGraphics.drawLine(5, 3, newSize.getColumns() - 1, 3, ' ');
                textGraphics.putString(5, 3, "Terminal Size: ", SGR.BOLD);
                textGraphics.putString(5 + "Terminal Size: ".length(), 3, newSize.toString());
                try {
                    terminal.flush();
                }
                catch(IOException e) {
                    // Not much we can do here
                    throw new RuntimeException(e);
                }
            }
        });

        textGraphics.putString(5, 4, "Last Keystroke: ", SGR.BOLD);
        textGraphics.putString(5 + "Last Keystroke: ".length(), 4, "<Pending>");
        terminal.flush();

Now let's try reading some input. There are two methods for this, pollInput() and readInput(). One is blocking (readInput) and one isn't (pollInput), returning null if there was nothing to read.

        KeyStroke keyStroke = terminal.readInput();

The KeyStroke class has a couple of different methods for getting details on the particular input that was read. Notice that some keys, like CTRL and ALT, cannot be individually distinguished as the standard input stream doesn't report these as individual keys. Generally special keys are categorized with a special KeyType, while regular alphanumeric and symbol keys are all under KeyType.Character. Notice that tab and enter are not considered KeyType.Character but special types (KeyType.Tab and KeyType.Enter respectively)

        while(keyStroke.getKeyType() != KeyType.Escape) {
            textGraphics.drawLine(5, 4, terminal.getTerminalSize().getColumns() - 1, 4, ' ');
            textGraphics.putString(5, 4, "Last Keystroke: ", SGR.BOLD);
            textGraphics.putString(5 + "Last Keystroke: ".length(), 4, keyStroke.toString());
            terminal.flush();
            keyStroke = terminal.readInput();
        }

    }
    catch(IOException e) {
        e.printStackTrace();
    }
    finally {
        if(terminal != null) {
            try {

The close() call here will exit private mode

                terminal.close();
            }
            catch(IOException e) {
                e.printStackTrace();
            }
        }
    }

The full code to this tutorial is available in the test section of the source code

Move on to tutorial 3