// **************************************************************************
/** @class Shell
@brief Implementation of a console based user shell with an input and output window
Shell is based on the ReadlineWindow class. It creates two windows
(panels) using ncurses which are used for input and output. The output
window panel is displayed on top of the input panel, but can be hidden
for example by pressing F1.
The idea is that continous messages like logging messages do not interfere
with the input area, although one still has both diplayed in the same
console window.
To get a list of the command and functions supported by Shell
type 'h' or 'help' at a command line prompt.
The usage is quite simple. Instantiate an object of Shell with the
programname as an argument. For its meaning see the base class
documentation Readline::Readline(). The created input- and output-
stream can be accessed through GetStreamIn() and GetStreamOut()
whihc are both ostreams, one redirected to the input window and the
other one redirected to the output window. Especially, GetStreamIn()
should not be used while the Readline prompt is in progress, but can for
example be used to display errors about what was entered.
The recommanded way of usage is:
\code
static Shell shell("myprog"); // readline will read the myprog.his history file
while (1)
{
string txt = shell.Prompt("prompt> ");
if (txt=="quit)
break;
// ... do something ...
shell.AddHistory(txt);
}
// On destruction redline will write the current history to the file
// By declaring the Shell static the correct terminal is restored even
// the program got killed (not if killed with SIGABRT)
\endcode
If for some reason the terminal is not correctly restored type (maybe blindly)
reset in your console. This should restor everything back to normal.
@section References
- GNU Readline
- GNU Ncurses
@todo
- Introduce the possibility to scroll in both windows
- Add redisplay functionality for both panels if the console was resized
*/
// **************************************************************************
#include "Shell.h"
#include
#include
#include // SIGWINCH
#include // waitpid
#include // ioctl
#include
#define FORK_ALLOWED
using namespace std;
Shell *Shell::This = 0;
// --------------------------------------------------------------------------
//
//! This initializes the Ncurses environment. Since the ncurses environment
//! is global only one instance of this class is allowed.
//!
//! The first 8 color pairs (COLOR_PAIR) are set to the first 8 color
//! with default background.
//!
//! The shells windows and panels are created. And their pointers are
//! propagated to the two associated WindowLog streams.
//!
//! Also some key bindings are initialized.
//
Shell::Shell(const char *prgname) : ReadlineWindow(prgname),
fPanelHeight(13), fIsVisible(1), fLine(0)
{
if (stdscr!=0)
{
endwin();
cout << "ERROR - Only one instance of class Shell is allowed." << endl;
exit(-1);
}
This = this;
// ---------------------- Setup ncurses -------------------------
initscr(); // Start curses mode
cbreak(); // Line buffering disabled, Pass on
noecho(); // Switch off echo mode
nonl(); // Associate return with CR
intrflush(stdscr, FALSE);
keypad(stdscr, FALSE); // Switch off keymapping for function keys
start_color(); // Initialize ncurses colors
use_default_colors(); // Assign terminal default colors to -1
//assume_default_colors(-1, -1); // standard terminal colors assigned to pair 0
// Setup colors
for (int i=1; i<8; i++)
init_pair(i, i, -1); // -1: def background
signal(SIGWINCH, HandleResizeImp); // Attach HandleResize to SIGWINCH signal
// ---------------------- Setup pansl --------------------------
// Create the necessary windows
WINDOW *wins[4];
CreateWindows(wins);
// Initialize the panels
fPanelIn = new_panel(wins[0]);
fPanelFrame = new_panel(wins[1]);
fPanelOut = new_panel(wins[2]);
win.SetWindow(wins[0]);
wout.SetWindow(wins[2]);
// Get the panels into the right order for startup
ShowHide(1);
// Setup Readline
SetWindow(wins[0]);
SetColorPrompt(COLOR_PAIR(COLOR_BLUE));
// ------------------- Setup key bindings -----------------------
BindKeySequence("\033OP", rl_proc_F1);
BindKeySequence("\033[1;5B", rl_scroll_top);
BindKeySequence("\033[1;5A", rl_scroll_top);
BindKeySequence("\033[1;3A", rl_scroll_bot);
BindKeySequence("\033[1;3B", rl_scroll_bot);
BindKeySequence("\033[5;3~", rl_top_inc);
BindKeySequence("\033[6;3~", rl_top_dec);
BindKeySequence("\033+", rl_top_resize);
BindKeySequence("\033-", rl_top_resize);
/*
rl_bind_keyseq("\033\t", rl_complete); // Meta-Tab
rl_bind_keyseq("\033[1~", home); // Home (console)
rl_bind_keyseq("\033[H", home); // Home (x)
rl_bind_keyseq("\033[4~", end); // End (console)
rl_bind_keyseq("\033[F", end); // End (x)
rl_bind_keyseq("\033[A", up); // Up
rl_bind_keyseq("\033[B", down); // Down
rl_bind_keyseq("\033[[B", accept); // F2 (console)
rl_bind_keyseq("\033OQ", accept); // F2 (x)
rl_bind_keyseq("\033[21~", cancel); // F10
*/
// Ctrl+dn: \033[1;5B
// Ctrl+up: \033[1;5A
// Alt+up: \033[1;3A
// Alt+dn: \033[1;3B
// Alt+pg up: \033[5;3~
// Alt+pg dn: \033[6;3~
}
// --------------------------------------------------------------------------
//
//! Ends the ncurses environment by calling endwin().
//
Shell::~Shell()
{
// Maybe not needed because the window is more or less valid until the
// object is destructed anyway.
//win.SetWindow(0);
//wout.SetWindow(0);
//SetWindow(0);
endwin();
cout << "The end." << endl;
}
// --------------------------------------------------------------------------
//
//! This function gets the windows into the expected order which is:
//!
//! @param v
//! - \b 0: Do not show the output panel
//! - \b 1: Show the output panel
//! - \b -1: Toggle the visibility of the output panel
//! - \b -2: Just update the panels, do not change their visibility
//
void Shell::ShowHide(int v)
{
if (v>-2)
fIsVisible = v==-1 ? !fIsVisible : v;
if (fIsVisible)
{
show_panel(fPanelIn);
show_panel(fPanelFrame);
show_panel(fPanelOut);
}
else
{
show_panel(fPanelIn);
hide_panel(fPanelFrame);
hide_panel(fPanelOut);
}
update_panels();
doupdate();
}
// --------------------------------------------------------------------------
//
//! Creates the windows to be used as panels and draws a frame around one
//!
//! @param w
//! pointers to the three windows which have been created are returned
//!
//! @param all
//! If all is false do not (re-)create the bottom or input-window
//
void Shell::CreateWindows(WINDOW *w[3], int all)
{
int maxx, maxy;
getmaxyx(stdscr, maxy, maxx);
int separator = maxy-fPanelHeight;
WINDOW *new_in = all ? newwin(maxy, maxx, 0, 0) : 0;
WINDOW *new_frame = newwin(separator-1, maxx, 0, 0);
WINDOW *new_out = newwin(separator-1-2, maxx-2, 1, 1);
box(new_frame, 0,0);
wmove(new_frame, 0, 1);
waddch(new_frame, ACS_RTEE);
wprintw(new_frame, " F1 ");
waddch(new_frame, ACS_LTEE);
scrollok(new_out, true);
leaveok (new_out, true);
if (new_in)
{
scrollok(new_in, true); // Allow scrolling
leaveok (new_in, false); // Move the cursor with the output
wmove(new_in, maxy-1, 0);
}
w[0] = new_in;
w[1] = new_frame;
w[2] = new_out;
}
// --------------------------------------------------------------------------
//
//! Key binding for F1. Toggles upper panel by calling ShowHide(-1)
//
int Shell::rl_proc_F1(int /*cnt*/, int /*key*/)
{
This->ShowHide(-1); // toggle
return 0;
}
int Shell::rl_scroll_top(int, int key)
{
This->win << "Scroll " << key << endl;
return 0;
}
int Shell::rl_scroll_bot(int, int key)
{
This->win << "Scroll " << key << endl;
return 0;
}
int Shell::rl_top_inc(int, int key)
{
This->win << "Increase " << key << endl;
return 0;
}
int Shell::rl_top_dec(int, int key)
{
This->win << "Increase " << key << endl;
return 0;
}
int Shell::rl_top_resize(int, int key)
{
This->Resize(key=='+' ? This->fPanelHeight-1 : This->fPanelHeight+1);
return 0;
}
// --------------------------------------------------------------------------
//
//! Signal handler for SIGWINCH, calls HandleResize
//
void Shell::HandleResizeImp(int)
{
This->HandleResize();
}
// --------------------------------------------------------------------------
//
//! Signal handler for SIGWINCH. It resizes the terminal and all panels
//! according to the new terminal size and redisplay the backlog buffer
//! in both windows
//!
//! @todo
//! Maybe there are more efficient ways than to display the whole buffers
//
void Shell::HandleResize()
{
// Get the new terminal size
struct winsize w;
ioctl(0, TIOCGWINSZ, &w);
// propagate it to the terminal
resize_term(w.ws_row, w.ws_col);
// Store the pointer to the old windows
WINDOW *w_in = panel_window(fPanelIn);
WINDOW *w_frame = panel_window(fPanelFrame);
WINDOW *w_out = panel_window(fPanelOut);
// Create new windows
WINDOW *wins[3];
CreateWindows(wins);
// Redirect the streams and the readline output to the new windows
win.SetWindow(wins[0]);
wout.SetWindow(wins[2]);
SetWindow(wins[0]);
// Replace windows in the panels
replace_panel(fPanelIn, wins[0]);
replace_panel(fPanelFrame, wins[1]);
replace_panel(fPanelOut, wins[2]);
// delete the old obsolete windows
delwin(w_in);
delwin(w_out);
delwin(w_frame);
// FIXME: NEEDED also in Redisplay panel
//Redisplay();
// Redisplay their contents
win.Display();
wout.Display();
}
// --------------------------------------------------------------------------
//
//! This resized the top panel or output panel as requested by the argument.
//! The argument is the number of lines which are kept free for the input
//! panel below the top panel
//!
//! @returns
//! always true
//
bool Shell::Resize(int h)
{
// Get curretn terminal size
int lines, cols;
getmaxyx(stdscr, lines, cols);
// Check if we are in a valid range
if (h<1 || h>lines-5)
return false;
// Set new height for panel to be kept free
fPanelHeight = h;
// Store the pointers of the old windows associated with the panels
// which should be resized
WINDOW *w_frame = panel_window(fPanelFrame);
WINDOW *w_out = panel_window(fPanelOut);
// Create new windows
WINDOW *wins[3];
CreateWindows(wins, false);
// Redirect the output stream to the new window
wout.SetWindow(wins[2]);
// Replace the windows associated with the panels
replace_panel(fPanelFrame, wins[1]);
replace_panel(fPanelOut, wins[2]);
// delete the ols windows
delwin(w_out);
delwin(w_frame);
// FIXME: NEEDED also in Redisplay panel
//Redisplay();
// Redisplay the contents
wout.Display();
return true;
}
// --------------------------------------------------------------------------
//
//! This wrapt the given readline function into a redirection to a file,
//! which contents is then displayed afterwards in the input panel.
//!
//! For convinience the given title is printed in bold before the list.
//!
//! This allows to show some readline output in the panel.
//!
//! @param function
//! Takes a function of type bool(*)() as argument
//!
//! @param title
//! A title streamed in bold before the output starts
//!
//! @returns
//! The return value of the function
//
bool Shell::RedirectionWrapper(bool (*function)(), const char *title)
{
FILE *save = SetStreamOut(tmpfile());
const bool rc = function();
FILE *file = SetStreamOut(save);
const bool empty = ftell(file)==0;
rewind(file);
win << endl;
win << kBold << title << endl;
if (empty)
{
win << " " << endl;
fclose(file);
return rc;
}
while (1)
{
const int c = getc(file);
if (feof(file))
break;
win << (char)c;
}
win << endl;
fclose(file);
return rc;
}
// --------------------------------------------------------------------------
//
//! Can be overwritten to display user defined commands when the command
//! c is issued.
//!
//! @returns
//! always true
//
bool Shell::PrintCommands()
{
win << endl;
win << " " << kUnderline << " Commands:" << endl;
win << " No application specific commands defined." << endl;
win << endl;
return true;
}
// --------------------------------------------------------------------------
//
//! Displays the available ncurses attributes, like color.
//!
//! @returns
//! always true
//
bool Shell::PrintAttributes()
{
//ostream &win = wout;
win << endl;
win << " Attributes:" << endl;
win << " " << kReset << "kReset" << endl;
win << " " << kNormal << "kNormal" << endl;
win << " " << kHighlight << "kHighlight" << endl;
win << " " << kReverse << "kReverse" << endl;
win << " " << kUnderline << "kUnderline" << endl;
win << " " << kBlink << "kBlink" << endl;
win << " " << kDim << "kDim" << endl;
win << " " << kBold << "kBold" << endl;
win << " " << kProtect << "kProtect" << endl;
win << " " << kInvisible << "kInvisible" << endl;
win << " " << kAltCharset << "kAltCharset" << kReset << " (kAltCharset)" << endl;
win << endl;
win << " Colors:" << endl;
win << " " << kDefault << "kDefault " << kBold << "+ kBold" << endl;
win << " " << kRed << "kRed " << kBold << "+ kBold" << endl;
win << " " << kGreen << "kGreen " << kBold << "+ kBold" << endl;
win << " " << kYellow << "kYellow " << kBold << "+ kBold" << endl;
win << " " << kBlue << "kBlue " << kBold << "+ kBold" << endl;
win << " " << kMagenta << "kMagenta " << kBold << "+ kBold" << endl;
win << " " << kCyan << "kCyan " << kBold << "+ kBold" << endl;
win << " " << kWhite << "kWhite " << kBold << "+ kBold" << endl;
win << " " << endl;
return true;
}
// --------------------------------------------------------------------------
//
//! Displays the keybindings available due to the Shell class
//!
//! @returns
//! always true
//!
//! @todo
//! Update output
//
bool Shell::PrintKeyBindings()
{
win << endl;
win << " " << kUnderline << "Key bindings:" << endl;
win << kBold << " Page-up " << kReset << "Search backward in history" << endl;
win << kBold << " Page-dn " << kReset << "Search forward in history" << endl;
win << kBold << " Ctrl-left " << kReset << "One word backward" << endl;
win << kBold << " Ctrl-right " << kReset << "One word forward" << endl;
win << kBold << " Ctrl-y " << kReset << "Delete line" << endl;
win << kBold << " Alt-end/Ctrl-k " << kReset << "Delete until the end of the line" << endl;
win << kBold << " F1 " << kReset << "Toggle visibility of upper panel" << endl;
win << endl;
win << " Default key-bindings are identical with your bash." << endl;
win << endl;
return true;
}
// --------------------------------------------------------------------------
//
//! Print a general help text which also includes the commands pre-defined
//! by the Shell class.
//!
//! @returns
//! always true
//!
//! @todo
//! Get it up-to-date
//
bool Shell::PrintGeneralHelp()
{
win << endl;
win << " " << kUnderline << "General help:" << endl;
win << kBold << " h,help " << kReset << "Print this help message" << endl;
win << kBold << " clear " << kReset << "Clear history buffer" << endl;
win << kBold << " lh,history " << kReset << "Dump the history buffer to the screen" << endl;
win << kBold << " v,variable " << kReset << "Dump readline variables" << endl;
win << kBold << " f,function " << kReset << "Dump readline functions" << endl;
win << kBold << " m,funmap " << kReset << "Dump readline funmap" << endl;
win << kBold << " c,command " << kReset << "Dump available commands" << endl;
win << kBold << " k,keylist " << kReset << "Dump key bindings" << endl;
win << kBold << " a,attrs " << kReset << "Dump available stream attributes" << endl;
win << kBold << " .q,quit " << kReset << "Quit" << endl;
win << endl;
win << " The command history is automatically loaded and saves to" << endl;
win << " and from " << GetName() << endl;
win << endl;
return true;
}
// --------------------------------------------------------------------------
//
//! Processes the command provided by the Shell-class.
//!
//! @returns
//! whether a command was successfully processed or could not be found
//
bool Shell::Process(const string &str)
{
// Implement readline commands:
// rl set (rl_variable_bind(..))
// rl_read_init_file(filename)
// int rl_add_defun (const char *name, rl_command_func_t *function, int key)
// ----------- Readline -----------
if (str=="lh" || str=="history")
return RedirectionWrapper(DumpHistory, "History:");
if (str=="v" || str=="variable")
return RedirectionWrapper(DumpVariables, "Variables:");
if (str=="f" || str=="function")
return RedirectionWrapper(DumpFunctions, "Functions:");
if (str=="m" || str=="funmap")
return RedirectionWrapper(DumpFunmap, "Funmap:");
if (Readline::Process(str))
return true;
// ------------ ReadlineWindow -------------
if (str=="a" || str=="attrs")
return PrintAttributes();
// ----------- ReadlineNcurses -----------
if (string(str)=="hide")
{
ShowHide(0);
return true;
}
if (string(str)=="show")
{
ShowHide(1);
return true;
}
if (str.substr(0, 7)=="height ")
{
int h;
sscanf(str.c_str()+7, "%d", &h);
return Resize(h);
}
if (str=="d")
{
wout.Display();
return true;
}
return false;
}
// --------------------------------------------------------------------------
//
//! Overwrites Shutdown. It's main purpose is to re-output
//! the prompt and the buffer using the WindowLog stream so that it is
//! buffered in its backlog.
//!
//! @param buf
//! A pointer to the buffer returned by readline
//!
void Shell::Shutdown(const char *buf)
{
ReadlineWindow::Shutdown(buf);
// Now move the cursor to the start of the prompt
RewindCursor();
// Output the text ourself to get it into the backlog
// buffer of win. We cannot use GetBuffer() because rl_end
// is not updated finally.
win << kBlue << GetPrompt() << kReset << buf << endl;
}