Inspired by Simon Ford's "Terminal" library, this is a clean-room reimplementation that supports a larger set of the ANSI escape sequences and includes a few handy drawing routines. Useful for making console UIs for your projects. The box-drawing stuff requires your terminal to be set to codepage 850.
ANSITerm is a class extending the mbed Serial class, and is designed to wrap a serial connection.
In addition to the standard Serial functions, ANSITerm exposes a set of functions for moving the cursor around the screen, setting and getting the current cursor position, and setting text display colours and styles.
It's primarily a simple utility class, providing a set of simply-named functions that wrap the escape sequences, rather than requiring the programmer to remember the sequence of characters to perform a particular task.
Diff: ANSITerm.cpp
- Revision:
- 0:863811463610
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/ANSITerm.cpp Tue Sep 18 18:58:19 2012 +0000 @@ -0,0 +1,174 @@ +/* +Copyright (c) 2012 dansummers + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + */ +#include "ANSITerm.h" +#include "mbed.h" + +//Box styles +/* topleft, topright, bottomleft, bottomright, horiz, vert, uptee, downtee, lefttee, righttee,cross */ +const char ANSITerm::two_lines[11] = {0xC9,0xBB,0xC8,0xBC,0xCD,0xBA,0xCA,0xCB,0xB9,0xCC,0xCE}; +const char ANSITerm::one_line[11] = {0xDA,0xBF,0xC0,0xD9,0xC4,0xB3,0xC1,0xC2,0xB4,0xC3,0xC5}; +const char ANSITerm::simple[11] = {'+','+','+','+','-','|','+','+','+','+','+'}; + +ANSITerm::ANSITerm(PinName tx, PinName rx) : Serial(tx, rx) {current_style = 0;} + +/* Device Status Report (Returns CSIn;mR, where n,m is the coordinates of the cursor). */ +void ANSITerm::ansi_dsr(int* x_coord, int* y_coord) +{ + char _r; bool reading_x = false; bool reading_y = false; + this->puts("6n"); wait(0.02); + + /* detect presence of symbol "CMSIS_OS_RTX", and switch on thread safety if we spot it (mutexes, thread.wait instead of wait, etc) */ + /* TODO: Make thread-safe at some point. */ + /*Reply format is "\033[<ycoord>;<xcoord>R" */ + *x_coord = 0; *y_coord = 0; + while(this->readable() == 1) + { + _r = this->getc(); + switch(_r) { + case 033: + //ignore, start of DSR + break; + case '[': + //next char starts the y-coordinate + reading_y = true; + break; + default: + //this is probably a numeric: check and, if so, subtract 48 to decode the ASCII to int. + if((_r>47) && (_r<58)) { + if(reading_x == true) { + *x_coord = ((*x_coord)*10) + (_r-48); + } else if(reading_y == true) { + *y_coord = ((*y_coord)*10) + (_r-48); + } + } else { + /* Not a numeric, ignore it.*/ + } + break; + case ';': + //next char starts the x-coordinate + reading_y = false; reading_x = true; + break; + case 'R': + //end of ANSI-DSR reply + return; + } + } +} + +/* Select Graphic Rendition - Make the output bold, underlined, coloured... all manner of things.*/ +void ANSITerm::ansi_sgr(bool reset, char text_style, char text_colour, char background_colour) +{ + bool needs_sep = false; + if(reset) { this->putc('0'); } + else + { + /* text style (select many) */ + // These two are mutually exclusive + if((text_style & SGR_BOLD) == SGR_BOLD) { this->putc('1'); } + else if ((text_style & SGR_FAINT) == SGR_FAINT){ this->putc('2'); } + else{ this->puts("22"); } + this->putc(';'); + + if((text_style & SGR_ITALIC) == SGR_ITALIC) { this->putc('3'); } + else{ this->puts("23"); } + this->putc(';'); + + if((text_style & SGR_UNDERLINE) == SGR_UNDERLINE) { this->putc('4'); } + else{ this->puts("24"); } + this->putc(';'); + + // These two are mutually exclusive + if((text_style & SGR_BLINK_SLOW) == SGR_BLINK_SLOW) { this->putc('5'); } + else if ((text_style & SGR_BLINK_RAPID) == SGR_BLINK_RAPID) { this->putc('6'); } + else{ this->puts("25"); } + this->putc(';'); + + if((text_style & SGR_IMAGE_NEGATIVE) == SGR_IMAGE_NEGATIVE) { this->putc('7'); } + else{ this->puts("27"); } + this->putc(';'); + + if((text_style & SGR_CROSSED_OUT) == SGR_CROSSED_OUT) { this->putc('9'); } + else{ this->puts("29"); } + needs_sep = true; + + /* text colour (select 1) */ + if(text_colour != 0) + { + if(needs_sep) { this->putc(';'); }/* Emit a separator if needed. */ + if ((text_colour & SGR_BLACK) == SGR_BLACK) { this->puts("30"); } + else if((text_colour & SGR_RED) == SGR_RED) { this->puts("31"); } + else if((text_colour & SGR_GREEN) == SGR_GREEN) { this->puts("32"); } + else if((text_colour & SGR_YELLOW) == SGR_YELLOW) { this->puts("33"); } + else if((text_colour & SGR_BLUE) == SGR_BLUE) { this->puts("34"); } + else if((text_colour & SGR_MAGENTA) == SGR_MAGENTA) { this->puts("35"); } + else if((text_colour & SGR_CYAN) == SGR_CYAN) { this->puts("36"); } + else if((text_colour & SGR_WHITE) == SGR_WHITE) { this->puts("37"); } + } + /* background_colour (select 1) */ + if(background_colour != 0) + { + if(needs_sep) { this->putc(';'); needs_sep = false; } + if ((background_colour & SGR_BLACK) == SGR_BLACK) { this->puts("40"); } + else if((background_colour & SGR_RED) == SGR_RED) { this->puts("41"); } + else if((background_colour & SGR_GREEN) == SGR_GREEN) { this->puts("42"); } + else if((background_colour & SGR_YELLOW) == SGR_YELLOW) { this->puts("43"); } + else if((background_colour & SGR_BLUE) == SGR_BLUE) { this->puts("44"); } + else if((background_colour & SGR_MAGENTA) == SGR_MAGENTA) { this->puts("45"); } + else if((background_colour & SGR_CYAN) == SGR_CYAN) { this->puts("46"); } + else if((background_colour & SGR_WHITE) == SGR_WHITE) { this->puts("47"); } + } + } + putc('m'); +} + +/* + Draw a box with the specified top-left and bottom-right points (these are the coords of the box, not its contents!) + Style is an array of chars to use to draw the box, [topleft, topright, bottomleft, bottomright, horiz, vert, uptee, downtee, lefttee, righttee,cross]. + if clear_inner is set, the 'contents' of the box onscreen will be erased. If it's clear, they won't be. + */ +void ANSITerm::draw_box(int x1, int y1, int x2, int y2, const char* style, bool clear_inner) +{ + this->set_cursor_position(x1,y1); + //draw the top row + this->putc(style[0]); for(int i = x1; i<(x2-1);i++) this->putc(style[4]); this->putc(style[1]); + + for(int i=(y1+1); i<(y2);i++) + { + this->set_cursor_position(x1,i); + this->putc(style[5]);//draw the left-hand side of this row + this->set_cursor_position(x2,i); + this->putc(style[5]);//draw the right-hand side of this row + } + + if(clear_inner) + { + for(int i = (y1+1);i<(y2);i++) + { + this->set_cursor_position((x1+1),i); //position at the start of this row + for(int j=x1;j<(x2-1);j++) this->putc(' '); //fill it with spaces + } + } + + this->set_cursor_position(x1,y2); + //draw the bottom row + this->putc(style[2]); for(int i = x1; i<(x2-1);i++) this->putc(style[4]); this->putc(style[3]); +} \ No newline at end of file