2006-10-02 05:22:43 +08:00
/** \file screen.h High level library for handling the terminal screen
The screen library allows the interactive reader to write its
output to screen efficiently by keeping an inetrnal representation
2008-01-14 00:47:47 +08:00
of the current screen contents and trying to find a reasonably
2006-10-02 05:22:43 +08:00
efficient way for transforming that to the desired screen content .
2008-01-14 00:47:47 +08:00
The current implementation is less smart than ncurses allows
and can not for example move blocks of text around to handle text
insertion .
*/
2006-10-02 00:02:58 +08:00
# ifndef FISH_SCREEN_H
# define FISH_SCREEN_H
2011-12-28 10:41:38 +08:00
# include <vector>
/**
A class representing a single line of a screen .
*/
2012-07-16 01:45:18 +08:00
struct line_t
2011-12-28 10:41:38 +08:00
{
2012-07-16 01:45:18 +08:00
std : : vector < wchar_t > text ;
std : : vector < int > colors ;
2012-10-01 18:29:18 +08:00
bool is_soft_wrapped ;
line_t ( ) : text ( ) , colors ( ) , is_soft_wrapped ( false )
{
}
2011-12-28 10:41:38 +08:00
2012-07-16 01:45:18 +08:00
void clear ( void )
{
text . clear ( ) ;
colors . clear ( ) ;
2011-12-28 10:41:38 +08:00
}
2012-07-16 01:45:18 +08:00
void append ( wchar_t txt , int color )
{
text . push_back ( txt ) ;
colors . push_back ( color ) ;
}
size_t size ( void ) const
{
return text . size ( ) ;
2011-12-28 10:41:38 +08:00
}
2012-07-16 01:45:18 +08:00
wchar_t char_at ( size_t idx ) const
{
return text . at ( idx ) ;
2011-12-28 10:41:38 +08:00
}
2012-07-16 01:45:18 +08:00
int color_at ( size_t idx ) const
{
return colors . at ( idx ) ;
2011-12-28 10:41:38 +08:00
}
2012-07-16 01:45:18 +08:00
2011-12-28 10:41:38 +08:00
} ;
/**
A class representing screen contents .
*/
class screen_data_t
{
std : : vector < line_t > line_datas ;
public :
2012-08-05 04:54:20 +08:00
struct cursor_t {
int x ;
int y ;
2012-10-01 18:29:18 +08:00
cursor_t ( ) : x ( 0 ) , y ( 0 ) { }
cursor_t ( int a , int b ) : x ( a ) , y ( b ) { }
2012-08-05 04:54:20 +08:00
} cursor ;
2011-12-28 10:41:38 +08:00
line_t & add_line ( void ) {
line_datas . resize ( line_datas . size ( ) + 1 ) ;
return line_datas . back ( ) ;
}
void resize ( size_t size ) {
line_datas . resize ( size ) ;
}
2012-07-16 01:45:18 +08:00
2011-12-28 10:41:38 +08:00
line_t & create_line ( size_t idx ) {
if ( idx > = line_datas . size ( ) ) {
line_datas . resize ( idx + 1 ) ;
}
return line_datas . at ( idx ) ;
}
line_t & line ( size_t idx ) {
return line_datas . at ( idx ) ;
}
size_t line_count ( void ) {
return line_datas . size ( ) ;
}
} ;
2006-10-02 05:22:43 +08:00
/**
2012-03-25 18:00:38 +08:00
The class representing the current and desired screen contents .
2006-10-02 05:22:43 +08:00
*/
2012-03-25 18:00:38 +08:00
class screen_t
2006-10-02 00:02:58 +08:00
{
2012-03-25 18:00:38 +08:00
public :
2011-12-28 10:41:38 +08:00
2012-10-01 18:29:18 +08:00
/** Constructor */
screen_t ( ) ;
2006-10-05 05:45:02 +08:00
2012-10-01 18:29:18 +08:00
/**
The internal representation of the desired screen contents .
*/
screen_data_t desired ;
/**
The internal representation of the actual screen contents .
*/
screen_data_t actual ;
/**
A string containing the prompt which was last printed to
the screen .
*/
wcstring actual_prompt ;
/**
The actual width of the screen at the time of the last screen
write .
*/
int actual_width ;
/** If we support soft wrapping, we can output to this location without any cursor motion. */
screen_data_t : : cursor_t soft_wrap_location ;
2006-10-05 07:33:12 +08:00
2012-10-01 18:29:18 +08:00
/**
2006-10-09 21:26:42 +08:00
This flag is set to true when there is reason to suspect that
the parts of the screen lines where the actual content is not
filled in may be non - empty . This means that a clr_eol command
has to be sent to the terminal at the end of each line .
*/
2012-10-03 08:30:07 +08:00
bool need_clear ;
/** If we need to clear, this is how many lines the actual screen had, before we reset it. This is used when resizing the window larger: if the cursor jumps to the line above, we need to remember to clear the subsequent lines. */
size_t actual_lines_before_reset ;
2011-12-27 11:18:46 +08:00
2006-10-05 07:33:12 +08:00
/**
These status buffers are used to check if any output has occurred
other than from fish ' s main loop , in which case we need to redraw .
*/
struct stat prev_buff_1 , prev_buff_2 , post_buff_1 , post_buff_2 ;
2012-03-25 18:00:38 +08:00
} ;
2006-10-02 00:02:58 +08:00
2006-10-02 05:22:43 +08:00
/**
This is the main function for the screen putput library . It is used
to define the desired contents of the screen . The screen command
will use it ' s knowlege of the current contents of the screen in
order to render the desired output using as few terminal commands
as possible .
2012-07-13 03:51:47 +08:00
\ param s the screen on which to write
\ param prompt the prompt to prepend to the command line
\ param commandline the command line
\ param explicit_len the number of characters of the " explicit " ( non - autosuggestion ) portion of the command line
\ param colors the colors to use for the comand line
\ param indent the indent to use for the command line
\ param cursor_pos where the cursor is
2006-10-02 05:22:43 +08:00
*/
2011-12-27 11:18:46 +08:00
void s_write ( screen_t * s ,
2011-12-29 04:36:47 +08:00
const wchar_t * prompt ,
const wchar_t * commandline ,
2012-08-02 07:32:52 +08:00
size_t explicit_len ,
2011-12-29 04:36:47 +08:00
const int * colors ,
const int * indent ,
2012-08-05 04:47:56 +08:00
size_t cursor_pos ) ;
2006-10-02 05:22:43 +08:00
2011-12-27 11:18:46 +08:00
/**
2007-10-01 06:53:54 +08:00
This function resets the screen buffers internal knowledge about
the contents of the screen . Use this function when some other
function than s_write has written to the screen .
\ param s the screen to reset
2011-12-27 11:18:46 +08:00
\ param reset_cursor whether the line on which the curor has changed should be assumed to have changed . If \ c reset_cursor is set to 0 , the library will attempt to make sure that the screen area does not seem to move up or down on repaint .
2007-10-01 06:53:54 +08:00
If reset_cursor is incorreclt set to 0 , this may result in screen
contents being erased . If it is incorrectly set to one , it may
result in one or more lines of garbage on screen on the next
repaint . If this happens during a loop , such as an interactive
resizing , there will be one line of garbage for every repaint ,
which will quicly fill the screen .
2006-10-02 05:22:43 +08:00
*/
2012-03-25 18:00:38 +08:00
void s_reset ( screen_t * s , bool reset_cursor ) ;
2006-10-02 00:02:58 +08:00
# endif