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.
ANSITerm.cpp@1:e3403c93f864, 2012-09-23 (annotated)
- Committer:
- dansummers
- Date:
- Sun Sep 23 23:28:27 2012 +0000
- Revision:
- 1:e3403c93f864
- Parent:
- 0:863811463610
Added symbols to name each glyph in a boxdraw style.
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
dansummers | 0:863811463610 | 1 | /* |
dansummers | 0:863811463610 | 2 | Copyright (c) 2012 dansummers |
dansummers | 0:863811463610 | 3 | |
dansummers | 0:863811463610 | 4 | Permission is hereby granted, free of charge, to any person obtaining a copy of |
dansummers | 0:863811463610 | 5 | this software and associated documentation files (the "Software"), to deal in |
dansummers | 0:863811463610 | 6 | the Software without restriction, including without limitation the rights to |
dansummers | 0:863811463610 | 7 | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies |
dansummers | 0:863811463610 | 8 | of the Software, and to permit persons to whom the Software is furnished to do |
dansummers | 0:863811463610 | 9 | so, subject to the following conditions: |
dansummers | 0:863811463610 | 10 | |
dansummers | 0:863811463610 | 11 | The above copyright notice and this permission notice shall be included in all |
dansummers | 0:863811463610 | 12 | copies or substantial portions of the Software. |
dansummers | 0:863811463610 | 13 | |
dansummers | 0:863811463610 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
dansummers | 0:863811463610 | 15 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
dansummers | 0:863811463610 | 16 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
dansummers | 0:863811463610 | 17 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
dansummers | 0:863811463610 | 18 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
dansummers | 0:863811463610 | 19 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
dansummers | 0:863811463610 | 20 | SOFTWARE. |
dansummers | 0:863811463610 | 21 | */ |
dansummers | 0:863811463610 | 22 | #include "ANSITerm.h" |
dansummers | 0:863811463610 | 23 | #include "mbed.h" |
dansummers | 0:863811463610 | 24 | |
dansummers | 0:863811463610 | 25 | //Box styles |
dansummers | 0:863811463610 | 26 | /* topleft, topright, bottomleft, bottomright, horiz, vert, uptee, downtee, lefttee, righttee,cross */ |
dansummers | 0:863811463610 | 27 | const char ANSITerm::two_lines[11] = {0xC9,0xBB,0xC8,0xBC,0xCD,0xBA,0xCA,0xCB,0xB9,0xCC,0xCE}; |
dansummers | 0:863811463610 | 28 | const char ANSITerm::one_line[11] = {0xDA,0xBF,0xC0,0xD9,0xC4,0xB3,0xC1,0xC2,0xB4,0xC3,0xC5}; |
dansummers | 0:863811463610 | 29 | const char ANSITerm::simple[11] = {'+','+','+','+','-','|','+','+','+','+','+'}; |
dansummers | 0:863811463610 | 30 | |
dansummers | 0:863811463610 | 31 | ANSITerm::ANSITerm(PinName tx, PinName rx) : Serial(tx, rx) {current_style = 0;} |
dansummers | 0:863811463610 | 32 | |
dansummers | 0:863811463610 | 33 | /* Device Status Report (Returns CSIn;mR, where n,m is the coordinates of the cursor). */ |
dansummers | 0:863811463610 | 34 | void ANSITerm::ansi_dsr(int* x_coord, int* y_coord) |
dansummers | 0:863811463610 | 35 | { |
dansummers | 0:863811463610 | 36 | char _r; bool reading_x = false; bool reading_y = false; |
dansummers | 0:863811463610 | 37 | this->puts("6n"); wait(0.02); |
dansummers | 0:863811463610 | 38 | |
dansummers | 0:863811463610 | 39 | /* detect presence of symbol "CMSIS_OS_RTX", and switch on thread safety if we spot it (mutexes, thread.wait instead of wait, etc) */ |
dansummers | 0:863811463610 | 40 | /* TODO: Make thread-safe at some point. */ |
dansummers | 0:863811463610 | 41 | /*Reply format is "\033[<ycoord>;<xcoord>R" */ |
dansummers | 0:863811463610 | 42 | *x_coord = 0; *y_coord = 0; |
dansummers | 0:863811463610 | 43 | while(this->readable() == 1) |
dansummers | 0:863811463610 | 44 | { |
dansummers | 0:863811463610 | 45 | _r = this->getc(); |
dansummers | 0:863811463610 | 46 | switch(_r) { |
dansummers | 0:863811463610 | 47 | case 033: |
dansummers | 0:863811463610 | 48 | //ignore, start of DSR |
dansummers | 0:863811463610 | 49 | break; |
dansummers | 0:863811463610 | 50 | case '[': |
dansummers | 0:863811463610 | 51 | //next char starts the y-coordinate |
dansummers | 0:863811463610 | 52 | reading_y = true; |
dansummers | 0:863811463610 | 53 | break; |
dansummers | 0:863811463610 | 54 | default: |
dansummers | 0:863811463610 | 55 | //this is probably a numeric: check and, if so, subtract 48 to decode the ASCII to int. |
dansummers | 0:863811463610 | 56 | if((_r>47) && (_r<58)) { |
dansummers | 0:863811463610 | 57 | if(reading_x == true) { |
dansummers | 0:863811463610 | 58 | *x_coord = ((*x_coord)*10) + (_r-48); |
dansummers | 0:863811463610 | 59 | } else if(reading_y == true) { |
dansummers | 0:863811463610 | 60 | *y_coord = ((*y_coord)*10) + (_r-48); |
dansummers | 0:863811463610 | 61 | } |
dansummers | 0:863811463610 | 62 | } else { |
dansummers | 0:863811463610 | 63 | /* Not a numeric, ignore it.*/ |
dansummers | 0:863811463610 | 64 | } |
dansummers | 0:863811463610 | 65 | break; |
dansummers | 0:863811463610 | 66 | case ';': |
dansummers | 0:863811463610 | 67 | //next char starts the x-coordinate |
dansummers | 0:863811463610 | 68 | reading_y = false; reading_x = true; |
dansummers | 0:863811463610 | 69 | break; |
dansummers | 0:863811463610 | 70 | case 'R': |
dansummers | 0:863811463610 | 71 | //end of ANSI-DSR reply |
dansummers | 0:863811463610 | 72 | return; |
dansummers | 0:863811463610 | 73 | } |
dansummers | 0:863811463610 | 74 | } |
dansummers | 0:863811463610 | 75 | } |
dansummers | 0:863811463610 | 76 | |
dansummers | 0:863811463610 | 77 | /* Select Graphic Rendition - Make the output bold, underlined, coloured... all manner of things.*/ |
dansummers | 0:863811463610 | 78 | void ANSITerm::ansi_sgr(bool reset, char text_style, char text_colour, char background_colour) |
dansummers | 0:863811463610 | 79 | { |
dansummers | 0:863811463610 | 80 | bool needs_sep = false; |
dansummers | 0:863811463610 | 81 | if(reset) { this->putc('0'); } |
dansummers | 0:863811463610 | 82 | else |
dansummers | 0:863811463610 | 83 | { |
dansummers | 0:863811463610 | 84 | /* text style (select many) */ |
dansummers | 0:863811463610 | 85 | // These two are mutually exclusive |
dansummers | 0:863811463610 | 86 | if((text_style & SGR_BOLD) == SGR_BOLD) { this->putc('1'); } |
dansummers | 0:863811463610 | 87 | else if ((text_style & SGR_FAINT) == SGR_FAINT){ this->putc('2'); } |
dansummers | 0:863811463610 | 88 | else{ this->puts("22"); } |
dansummers | 0:863811463610 | 89 | this->putc(';'); |
dansummers | 0:863811463610 | 90 | |
dansummers | 0:863811463610 | 91 | if((text_style & SGR_ITALIC) == SGR_ITALIC) { this->putc('3'); } |
dansummers | 0:863811463610 | 92 | else{ this->puts("23"); } |
dansummers | 0:863811463610 | 93 | this->putc(';'); |
dansummers | 0:863811463610 | 94 | |
dansummers | 0:863811463610 | 95 | if((text_style & SGR_UNDERLINE) == SGR_UNDERLINE) { this->putc('4'); } |
dansummers | 0:863811463610 | 96 | else{ this->puts("24"); } |
dansummers | 0:863811463610 | 97 | this->putc(';'); |
dansummers | 0:863811463610 | 98 | |
dansummers | 0:863811463610 | 99 | // These two are mutually exclusive |
dansummers | 0:863811463610 | 100 | if((text_style & SGR_BLINK_SLOW) == SGR_BLINK_SLOW) { this->putc('5'); } |
dansummers | 0:863811463610 | 101 | else if ((text_style & SGR_BLINK_RAPID) == SGR_BLINK_RAPID) { this->putc('6'); } |
dansummers | 0:863811463610 | 102 | else{ this->puts("25"); } |
dansummers | 0:863811463610 | 103 | this->putc(';'); |
dansummers | 0:863811463610 | 104 | |
dansummers | 0:863811463610 | 105 | if((text_style & SGR_IMAGE_NEGATIVE) == SGR_IMAGE_NEGATIVE) { this->putc('7'); } |
dansummers | 0:863811463610 | 106 | else{ this->puts("27"); } |
dansummers | 0:863811463610 | 107 | this->putc(';'); |
dansummers | 0:863811463610 | 108 | |
dansummers | 0:863811463610 | 109 | if((text_style & SGR_CROSSED_OUT) == SGR_CROSSED_OUT) { this->putc('9'); } |
dansummers | 0:863811463610 | 110 | else{ this->puts("29"); } |
dansummers | 0:863811463610 | 111 | needs_sep = true; |
dansummers | 0:863811463610 | 112 | |
dansummers | 0:863811463610 | 113 | /* text colour (select 1) */ |
dansummers | 0:863811463610 | 114 | if(text_colour != 0) |
dansummers | 0:863811463610 | 115 | { |
dansummers | 0:863811463610 | 116 | if(needs_sep) { this->putc(';'); }/* Emit a separator if needed. */ |
dansummers | 0:863811463610 | 117 | if ((text_colour & SGR_BLACK) == SGR_BLACK) { this->puts("30"); } |
dansummers | 0:863811463610 | 118 | else if((text_colour & SGR_RED) == SGR_RED) { this->puts("31"); } |
dansummers | 0:863811463610 | 119 | else if((text_colour & SGR_GREEN) == SGR_GREEN) { this->puts("32"); } |
dansummers | 0:863811463610 | 120 | else if((text_colour & SGR_YELLOW) == SGR_YELLOW) { this->puts("33"); } |
dansummers | 0:863811463610 | 121 | else if((text_colour & SGR_BLUE) == SGR_BLUE) { this->puts("34"); } |
dansummers | 0:863811463610 | 122 | else if((text_colour & SGR_MAGENTA) == SGR_MAGENTA) { this->puts("35"); } |
dansummers | 0:863811463610 | 123 | else if((text_colour & SGR_CYAN) == SGR_CYAN) { this->puts("36"); } |
dansummers | 0:863811463610 | 124 | else if((text_colour & SGR_WHITE) == SGR_WHITE) { this->puts("37"); } |
dansummers | 0:863811463610 | 125 | } |
dansummers | 0:863811463610 | 126 | /* background_colour (select 1) */ |
dansummers | 0:863811463610 | 127 | if(background_colour != 0) |
dansummers | 0:863811463610 | 128 | { |
dansummers | 0:863811463610 | 129 | if(needs_sep) { this->putc(';'); needs_sep = false; } |
dansummers | 0:863811463610 | 130 | if ((background_colour & SGR_BLACK) == SGR_BLACK) { this->puts("40"); } |
dansummers | 0:863811463610 | 131 | else if((background_colour & SGR_RED) == SGR_RED) { this->puts("41"); } |
dansummers | 0:863811463610 | 132 | else if((background_colour & SGR_GREEN) == SGR_GREEN) { this->puts("42"); } |
dansummers | 0:863811463610 | 133 | else if((background_colour & SGR_YELLOW) == SGR_YELLOW) { this->puts("43"); } |
dansummers | 0:863811463610 | 134 | else if((background_colour & SGR_BLUE) == SGR_BLUE) { this->puts("44"); } |
dansummers | 0:863811463610 | 135 | else if((background_colour & SGR_MAGENTA) == SGR_MAGENTA) { this->puts("45"); } |
dansummers | 0:863811463610 | 136 | else if((background_colour & SGR_CYAN) == SGR_CYAN) { this->puts("46"); } |
dansummers | 0:863811463610 | 137 | else if((background_colour & SGR_WHITE) == SGR_WHITE) { this->puts("47"); } |
dansummers | 0:863811463610 | 138 | } |
dansummers | 0:863811463610 | 139 | } |
dansummers | 0:863811463610 | 140 | putc('m'); |
dansummers | 0:863811463610 | 141 | } |
dansummers | 0:863811463610 | 142 | |
dansummers | 0:863811463610 | 143 | /* |
dansummers | 0:863811463610 | 144 | Draw a box with the specified top-left and bottom-right points (these are the coords of the box, not its contents!) |
dansummers | 0:863811463610 | 145 | Style is an array of chars to use to draw the box, [topleft, topright, bottomleft, bottomright, horiz, vert, uptee, downtee, lefttee, righttee,cross]. |
dansummers | 0:863811463610 | 146 | if clear_inner is set, the 'contents' of the box onscreen will be erased. If it's clear, they won't be. |
dansummers | 0:863811463610 | 147 | */ |
dansummers | 0:863811463610 | 148 | void ANSITerm::draw_box(int x1, int y1, int x2, int y2, const char* style, bool clear_inner) |
dansummers | 0:863811463610 | 149 | { |
dansummers | 0:863811463610 | 150 | this->set_cursor_position(x1,y1); |
dansummers | 0:863811463610 | 151 | //draw the top row |
dansummers | 0:863811463610 | 152 | this->putc(style[0]); for(int i = x1; i<(x2-1);i++) this->putc(style[4]); this->putc(style[1]); |
dansummers | 0:863811463610 | 153 | |
dansummers | 0:863811463610 | 154 | for(int i=(y1+1); i<(y2);i++) |
dansummers | 0:863811463610 | 155 | { |
dansummers | 0:863811463610 | 156 | this->set_cursor_position(x1,i); |
dansummers | 0:863811463610 | 157 | this->putc(style[5]);//draw the left-hand side of this row |
dansummers | 0:863811463610 | 158 | this->set_cursor_position(x2,i); |
dansummers | 0:863811463610 | 159 | this->putc(style[5]);//draw the right-hand side of this row |
dansummers | 0:863811463610 | 160 | } |
dansummers | 0:863811463610 | 161 | |
dansummers | 0:863811463610 | 162 | if(clear_inner) |
dansummers | 0:863811463610 | 163 | { |
dansummers | 0:863811463610 | 164 | for(int i = (y1+1);i<(y2);i++) |
dansummers | 0:863811463610 | 165 | { |
dansummers | 0:863811463610 | 166 | this->set_cursor_position((x1+1),i); //position at the start of this row |
dansummers | 0:863811463610 | 167 | for(int j=x1;j<(x2-1);j++) this->putc(' '); //fill it with spaces |
dansummers | 0:863811463610 | 168 | } |
dansummers | 0:863811463610 | 169 | } |
dansummers | 0:863811463610 | 170 | |
dansummers | 0:863811463610 | 171 | this->set_cursor_position(x1,y2); |
dansummers | 0:863811463610 | 172 | //draw the bottom row |
dansummers | 0:863811463610 | 173 | this->putc(style[2]); for(int i = x1; i<(x2-1);i++) this->putc(style[4]); this->putc(style[3]); |
dansummers | 0:863811463610 | 174 | } |