blob: df0be84a070a2c0dc797bd6991abba5231c8ed49 [file] [log] [blame] [edit]
// Copyright 2018 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef SRC_LIB_LINE_INPUT_LINE_INPUT_H_
#define SRC_LIB_LINE_INPUT_LINE_INPUT_H_
#include <deque>
#include <string>
#include <vector>
#include "lib/fit/function.h"
#if !defined(__Fuchsia__)
struct termios;
#endif
namespace line_input {
struct SpecialCharacters {
static constexpr char kKeyControlA = 1;
static constexpr char kKeyControlB = 2;
static constexpr char kKeyControlC = 3;
static constexpr char kKeyControlD = 4;
static constexpr char kKeyControlE = 5;
static constexpr char kKeyControlF = 6;
static constexpr char kKeyControlH = 8;
static constexpr char kKeyTab = 9;
static constexpr char kKeyNewline = 10;
static constexpr char kKeyControlK = 11;
static constexpr char kKeyFormFeed = 12;
static constexpr char kKeyEnter = 13;
static constexpr char kKeyControlN = 14;
static constexpr char kKeyControlP = 16;
static constexpr char kKeyControlR = 18;
static constexpr char kKeyControlT = 20;
static constexpr char kKeyControlU = 21;
static constexpr char kKeyControlW = 23;
static constexpr char kKeyEsc = 27;
static constexpr char kKeyBackspace = 127;
// Escape sequences for terminal output.
static const char* kTermBeginningOfLine;
static const char* kTermClearToEnd;
static const char* kTermCursorToColFormat; // printf format.
};
// Abtract base class for line input.
//
// This class implements a push model for input of characters, allowing it to be used in
// asynchronous contexts.
//
// The model is you create a LineInput class outside of the input loop. It encapsulates the history
// state and remembers the prompt. When you want to read a line:
//
// 1. Call Show().
// 2. Push data to it via OnInput().
// 3. On an accept callback:
// 3a. Handle the input.
// 3b. Add line to history if desired.
// 4. Repeat until done.
// 5. Call Hide() to put the terminal back.
//
// If your application has data that it needs to print asynchronously, just:
// 1. Call Hide().
// 2. Print the stuff you want.
// 3. Call Show().
class LineInput {
public:
virtual ~LineInput() = default;
// Called with the user input when the user acceps a line.
using AcceptCallback = fit::function<void(const std::string&)>;
// Called when the current line changes. This is not called for <Enter> which doesn't change
// anything but will call the AcceptCallback.
using ChangeCallback = fit::function<void(const std::string&)>;
// Given some typing, returns a prioritized list of completions.
using AutocompleteCallback = fit::function<std::vector<std::string>(const std::string&)>;
// Callback that indicates Control-C or EOF (Control-D) was typed.
using CancelCallback = fit::function<void()>;
using EofCallback = fit::function<void()>;
// Setup -----------------------------------------------------------------------------------------
// Provides the callback for tab completion.
virtual void SetAutocompleteCallback(AutocompleteCallback cb) = 0;
// Provides the callback for when the current line changes.
virtual void SetChangeCallback(ChangeCallback cb) = 0;
// Provides the callback for handling Control-C. If unset, "^C" will be echoed and the line
// will be cleared.
virtual void SetCancelCallback(CancelCallback cb) = 0;
// Provides the callback for handling EOF. If unset EOF will be ignored.
virtual void SetEofCallback(EofCallback cb) = 0;
// Sets the maximum width of a line. Beyond this the input will scroll. Setitng to 0 will disable
// horizontal scrolling.
virtual void SetMaxCols(size_t max) = 0;
// Querying --------------------------------------------------------------------------------------
// Returns the current input text.
virtual const std::string& GetLine() const = 0;
// Returns the current history. The most resent input is at the begin().
virtual const std::deque<std::string>& GetHistory() const = 0;
// State -----------------------------------------------------------------------------------------
// Provides one character of input to the editor. Callbacks for autocomplete or line done will be
// issued from within this function.
virtual void OnInput(char c) = 0;
// Adds the given line to history. If the history is longer than max_history_, the oldest thing
// will be deleted.
//
// AddToHistory should be called on startup before the initial Show() call, or from within the
// accept callback (typically you would add the current line at this point).
virtual void AddToHistory(const std::string& line) = 0;
// The input can be hidden and re-shown. Hiding it will erase the current line and put the cursor
// at the beginning of the line, but not change any internal state. Showing it again will repaint
// the line at the new cursor position. This allows other output to be printed to the screen
// without interfering with the input.
//
// Hiding or showing when it's already in that state will do nothing. There is not a reference
// count on the hide calls.
//
// OnInput() should not be called while hidden.
//
// Tip: When the application is done (the user types "quit" or whatever), call Hide() from within
// the AcceptCallback or EofCallback. This will ensure the prompt isn't repainted when the
// callback is complete only to be rehidden on exit (which will cause flickering).
virtual void Hide() = 0;
virtual void Show() = 0;
};
// Implementation of LineInput that implements the editing state. Output is still abstract to
// allow for output to different places.
class LineInputEditor : public LineInput {
public:
explicit LineInputEditor(AcceptCallback accept_cb, const std::string& prompt);
virtual ~LineInputEditor();
// LineInput implementation.
void SetAutocompleteCallback(AutocompleteCallback cb) override;
void SetChangeCallback(ChangeCallback cb) override;
void SetCancelCallback(CancelCallback cb) override;
void SetEofCallback(EofCallback cb) override;
void SetMaxCols(size_t max) override;
const std::string& GetLine() const override;
const std::deque<std::string>& GetHistory() const override;
void OnInput(char c) override;
void AddToHistory(const std::string& line) override;
void Hide() override;
void Show() override;
size_t pos() const { return pos_; }
bool in_reverse_history_mode() const { return reverse_history_mode_; }
size_t reverse_history_index() const { return reverse_history_index_; }
// Exposed for testing purposes.
std::string GetReverseHistoryPrompt() const;
std::string GetReverseHistorySuggestion() const;
protected:
// Enables or disables console raw mode if applicable.
virtual void EnsureRawMode() {}
virtual void EnsureNoRawMode() {}
// Abstract output function, overridden by a derived class to output to screen.
virtual void Write(const std::string& data) = 0;
// Helper to return the current line of text.
std::string& cur_line() { return history_[history_index_]; }
// Useful for testing.
void set_pos(size_t pos) { pos_ = pos; }
const std::string& prompt() const { return prompt_; }
void set_prompt(std::string prompt) { prompt_ = std::move(prompt); }
private:
void HandleEscapedInput(char c);
void HandleBackspace();
void HandleDelete();
// FormFeed is the name of Ctrl-L in ASCII world.
void HandleFormFeed();
void HandleEnter();
void HandleTab();
// NegAck is the name of Ctrl-U in ASCII world.
void HandleNegAck();
// EndOfTransimission is the name for Ctrl-W in ASCII world.
void HandleEndOfTransimission();
// EndOfFile means Ctrl-D with an empty input line.
void HandleEndOfFile();
// ReverseHistory means Ctrl-R.
void HandleReverseHistory(char c);
void StartReverseHistoryMode();
void EndReverseHistoryMode(bool accept_suggestion);
void SearchNextReverseHistory(bool restart);
void Insert(char c);
void MoveLeft();
void MoveRight();
void MoveUp();
void MoveDown();
void MoveHome();
void MoveEnd();
void TransposeLastTwoCharacters();
void CancelCommand();
void DeleteToEnd();
void CancelCompletion();
void AcceptCompletion();
// Issues a line change notification and repaints the current line.
void LineChanged();
// Repaints the current line without issuing a line change notification.
void RepaintLine();
void ResetLineState();
AcceptCallback accept_callback_;
ChangeCallback change_callback_; // Possibly null.
std::string prompt_;
size_t max_cols_ = 0;
AutocompleteCallback autocomplete_callback_; // Possibly null.
CancelCallback cancel_callback_; // Possibly null;.
EofCallback eof_callback_; // Possibly null;.
// Indicates whether the line is currently visible (as controlled by Show()/Hide()).
bool visible_ = false;
// The history is basically the line stack going back in time as indices increase. The currently
// viewed line is at [history_index_] and this is where editing happens. When you start a new text
// entry, a new history item is added and you delete it.
//
// This is simple but can be a bit confusing if you go back, edit, and then press enter. The
// history item itself will be edited, and the same edited version will be added again as the most
// recent history entry.
//
// This is weird because the editing has actually changed history. A more complex model might be
// to maintain a virtual shadow copy of history that you edit, and this shadow copy is replaced
// with the actual history whenever you start editing a new line.
std::deque<std::string> history_; // front() is newest.
size_t history_index_ = 0; // Offset from history_.front().
const size_t max_history_ = 256;
bool completion_mode_ = false;
std::vector<std::string> completions_;
size_t completion_index_ = 0;
// Tracks the current line's state before suggesting completions so we can
// put them back if necessary. Only valid when completion_mode_ = true.
std::string line_before_completion_;
size_t pos_before_completion_;
// When an escape is read, we enter "escaped input" mode which interprets the
// next few characters input as an escape sequence.
bool reading_escaped_input_ = false;
std::string escape_sequence_;
bool reverse_history_mode_ = false;
std::string reverse_history_input_;
// Index within history the reverse search suggestion current is. 0 means not found, as that is
// pointing to the current line, which we don't want to do a history search in.
size_t reverse_history_index_ = 0;
size_t pos_ = 0; // Current editing position.
};
// Implementation of LineInput that prints to stdout. The caller is still responsible for providing
// input asynchronously. The initial width of the output will be automatically derived from the
// terminal associated wit, as that is pointing to the current line, which we don't want to do a
// history search in. stdout (if any).
class LineInputStdout : public LineInputEditor {
public:
LineInputStdout(AcceptCallback accept_cb, const std::string& prompt);
~LineInputStdout() override;
protected:
// LineInputEditor implementation.
void EnsureRawMode() override;
void EnsureNoRawMode() override;
void Write(const std::string& str) override;
private:
#if !defined(__Fuchsia__)
// The terminal is converted into raw mode when the prompt is visible and accepting input. Then
// it's switched back. This block tracks that information. Use unique_ptr to avoid bringing all
// terminal headers into this header.
bool raw_mode_enabled_ = false;
std::unique_ptr<termios> raw_termios_;
std::unique_ptr<termios> original_termios_;
#endif
};
} // namespace line_input
#endif // SRC_LIB_LINE_INPUT_LINE_INPUT_H_