/** ASCII Table Utility by Bruce Hill A simple utility that uses ncurses to display the ASCII table and lets you manually output ASCII characters one byte at a time. Use arrow keys or h/j/k/l to move, 'enter' to add a character to the output, '/' to search, and 'q' or escape to quit and print the output. Licensed under the MIT License (see LICENSE file for details) */ #include #include #include #include #include #include #include typedef enum {BROWSE_MODE, INSERT_MODE, SEARCH_MODE} Mode; #define DELETE_KEY 127 #define ESCAPE_KEY 27 // The descriptive names of the ASCII characters const char *NAMES[128][2] = { {"NUL", "null"}, // 0 {"SOH", "start of heading"}, // 1 {"STX", "start of text"}, // 2 {"ETX", "end of text"}, // 3 {"EOT", "end of transmission"}, // 4 {"ENQ", "enquiry"}, // 5 {"ACK", "acknowledge"}, // 6 {"BEL", "bell"}, // 7 {"BS", "backspace"}, // 8 {"TAB", "horizontal tab"}, // 9 {"LF", "NL line feed, new line"}, // 10 {"VT", "vertical tab"}, // 11 {"FF", "NP form feed, new page"}, // 12 {"CR", "carriage return"}, // 13 {"SO", "shift out"}, // 14 {"SI", "shift in"}, // 15 {"DLE", "data link escape"}, // 16 {"DC1", "device control 1"}, // 17 {"DC2", "device control 2"}, // 18 {"DC3", "device control 3"}, // 19 {"DC4", "device control 4"}, // 20 {"NAK", "negative acknowledge"}, // 21 {"SYN", "synchonous idle"}, // 22 {"ETB", "end of transmission block"}, // 23 {"CAN", "cancel"}, // 24 {"EM", "end of medium"}, // 25 {"SUB", "substitute"}, // 26 {"ESC", "escape"}, // 27 {"FS", "file separator"}, // 28 {"GS", "group separator"}, // 29 {"RS", "record separator"}, // 30 {"US", "unit separator"}, // 31 {" ", "space"}, // 32 [127] = {"DEL", "delete"} }; static int new_colorpair(int fg, int bg) { static int next_colornum = 1; init_pair(next_colornum, fg, bg); return COLOR_PAIR(next_colornum++); } int DECIMAL_COLORS, HEX_COLORS, CHAR_COLORS, DESCRIPTION_SPECIAL_COLORS, DESCRIPTION_LETTER_COLORS, DESCRIPTION_NUMBER_COLORS, DESCRIPTION_SYMBOL_COLORS, OUTPUT_CHAR_COLORS, OUTPUT_ESCAPE_COLORS, OUTPUT_LABEL_COLORS, BROWSE_MODE_COLORS, INSERT_MODE_COLORS, SEARCH_MODE_COLORS; const int COL_WIDTH = 42; int W = 0; int H = 0; static inline int min(int a, int b) { return a < b ? a : b; } static inline int max(int a, int b) { return a >= b ? a : b; } // Shift the cursor static inline void shift(int dy, int dx) { int x, y; getyx(stdscr, y, x); move(y+dy, x+dx); } // Same as printw, except it doesn't do text wrapping, and it sets attributes #define printwattrs(_attrs, ...) \ { char _buf[W+1];\ chtype _bufch[W+1];\ sprintf(_buf, __VA_ARGS__);\ for (int i = 0; i == 0 || _buf[i-1]; i++) _bufch[i] = _buf[i] | _attrs;\ addchstr(_bufch);\ shift(0, strlen(_buf));} // Draw the information for one character static inline void draw_entry(int i, int scrollx, int attrs) { int x = (i / (H-2)) * COL_WIDTH - scrollx; if (x > W || x < 0) return; int y = (i % (H-2)) + 1; move(y, x); printwattrs(DECIMAL_COLORS | attrs, "%3d ", i); printwattrs(HEX_COLORS | attrs, " 0x%02X ", i); if (NAMES[i][0]) { printwattrs(CHAR_COLORS | attrs, " %-3s ", NAMES[i][0]); } else { printwattrs(CHAR_COLORS | attrs, " %c ", i); } if (NAMES[i][1]) { printwattrs(DESCRIPTION_SPECIAL_COLORS | attrs, " %-25s", NAMES[i][1]); } else if (('a' <= i && i <= 'z') || ('A' <= i && i <= 'Z')) { printwattrs(DESCRIPTION_LETTER_COLORS | attrs, " the letter %c ", i); } else if ('0' <= i && i <= '9') { printwattrs(DESCRIPTION_NUMBER_COLORS | attrs, " the number %c ", i); } else { printwattrs(DESCRIPTION_SYMBOL_COLORS | A_BOLD | attrs, " the %c symbol ", i); } } // Redraw the parts of the ASCII table necessary for the selected item static void redraw(int selected) { static int last_selected = -1, last_scrollx = -1, last_W = -1, last_H = -1; int selectedx = (selected / (H-2)) * COL_WIDTH; int scrollx = (selectedx + COL_WIDTH < W) ? 0 : (((selectedx + COL_WIDTH) - W)/COL_WIDTH+1)*COL_WIDTH; if (last_selected == -1 || last_scrollx != scrollx || last_W != W || last_H != H) { // Redraw all the entries erase(); for (int i = 0; i < 128; i++) { draw_entry(i, scrollx, i == selected ? A_REVERSE : 0); } last_scrollx = scrollx; last_H = H; last_W = W; } else if (selected != last_selected) { // Otherwise, we can lazily just redraw 2 entries draw_entry(selected, scrollx, A_REVERSE); draw_entry(last_selected, scrollx, 0); } last_selected = selected; } // Find the ascii character starting at `start`, going in `direction`, matching // `searchbuff`, and return `fallback` if nothing matches. static int find_ascii_match(int start, int fallback, int direction, char *searchbuf) { if (searchbuf[0] == '\0') return fallback; char tmp[COL_WIDTH+6]; for (int i = 0; i <= 127; i++) { int c = (start + i*direction) % 128; // relative to starting point if (searchbuf[0] == c && searchbuf[1] == '\0') return c; // Only do full match for search strings longer than 1 if (searchbuf[1]) { if (NAMES[c][1]) { sprintf(tmp, "%d 0x%02X 0x%02x %s %s", c, c, c, NAMES[c][0] ? NAMES[c][0] : "", NAMES[c][1]); } else if (('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z')) { sprintf(tmp, "%d 0x%02X 0x%02x %s the letter %c", c, c, c, NAMES[c][0] ? NAMES[c][0] : "", c); } else if ('0' <= c && c <= '9') { sprintf(tmp, "%d 0x%02X 0x%02x %s the number %c", c, c, c, NAMES[c][0] ? NAMES[c][0] : "", c); } else { sprintf(tmp, "%d 0x%02X 0x%02x %s the %c character", c, c, c, NAMES[c][0] ? NAMES[c][0] : "", c); } if (strstr(tmp, searchbuf) != NULL) return c; } } return fallback; } // To prevent wasted work, wait until 200ms has elapsed with no additional // resizes before doing anything, then set the correct W/H values. static void handle_resize() { timeout(0); do { // Wait 200ms usleep(2e5); // Peek if there's another resize event int k = getch(); if (k == KEY_RESIZE) continue; if (k > 0) ungetch(k); } while (0); // Back to blocking input timeout(-1); W = getmaxx(stdscr); H = getmaxy(stdscr); } int main(int argc, char *argv[]) { if (argc > 1 && (strcmp(argv[1], "--help") == 0 || strcmp(argv[1], "-h") == 0)) { printf("ASCII table viewer/editor, Usage: ascii [N...]\n"); return 0; } // Instead of initscr(), set up the terminal this way to play nicely with // the unix pipeline: { char* term_type = getenv("TERM"); if (term_type == NULL || *term_type == '\0') { term_type = "unknown"; } FILE* term_in = fopen("/dev/tty", "r"); if (term_in == NULL) { perror("fopen(/dev/tty)"); exit(EXIT_FAILURE); } SCREEN *screen = newterm(term_type, stderr, term_in); set_term(screen); } W = getmaxx(stdscr); H = getmaxy(stdscr); noecho(); set_escdelay(5); keypad(stdscr, 1); curs_set(2); start_color(); DECIMAL_COLORS = new_colorpair(COLOR_YELLOW, COLOR_BLACK) | A_BOLD; HEX_COLORS = new_colorpair(COLOR_GREEN, COLOR_BLACK) | A_BOLD; CHAR_COLORS = new_colorpair(COLOR_MAGENTA, COLOR_BLACK) | A_BOLD; DESCRIPTION_SPECIAL_COLORS = new_colorpair(COLOR_RED, COLOR_BLACK) | A_BOLD; DESCRIPTION_LETTER_COLORS = new_colorpair(COLOR_BLUE, COLOR_BLACK) | A_BOLD; DESCRIPTION_NUMBER_COLORS = new_colorpair(COLOR_CYAN, COLOR_BLACK) | A_BOLD; DESCRIPTION_SYMBOL_COLORS = new_colorpair(COLOR_WHITE, COLOR_BLACK) | A_BOLD; OUTPUT_CHAR_COLORS = new_colorpair(COLOR_BLACK, COLOR_WHITE) | A_BOLD; OUTPUT_ESCAPE_COLORS = new_colorpair(COLOR_RED, COLOR_WHITE) | A_BOLD; OUTPUT_LABEL_COLORS = new_colorpair(COLOR_BLACK, COLOR_YELLOW); BROWSE_MODE_COLORS = new_colorpair(COLOR_BLACK, COLOR_YELLOW); INSERT_MODE_COLORS = new_colorpair(COLOR_WHITE, COLOR_RED) | A_BOLD; SEARCH_MODE_COLORS = new_colorpair(COLOR_BLACK, COLOR_BLUE); size_t buf_chunk = 1024; size_t buf_size = buf_chunk, buf_i = 0; char *outbuf = calloc(buf_size, sizeof(char)); { // Read stdin if anything was piped in struct pollfd desc; desc.fd = STDIN_FILENO; desc.events = POLLIN; int ret = poll(&desc, 1, 50); if (ret > 0) { size_t consumed; while ((consumed = fread(&outbuf[buf_i], 1, buf_chunk, stdin))) { buf_i += consumed; if (consumed == buf_chunk) { buf_chunk *= 2; buf_size += buf_chunk; outbuf = realloc(outbuf, buf_size); } } } } char searchbuf[1024] = {'\0'}; size_t searchlen = 0; // If any command line args, treat them as bytes to append to // the output buffer for (int i = 1; i < argc; i++) { long c = strtol(argv[i], NULL, 0); outbuf[buf_i++] = (char)c; if (buf_i >= buf_size) { buf_size *= 2; outbuf = realloc(outbuf, buf_size); } } Mode mode = BROWSE_MODE; int prev_selected = 0, selected = 0; while (1) { redraw(selected); // Title bar move(0,0); char *title; int title_colors; switch (mode) { case INSERT_MODE: title = "ASCII TABLE (typing, 'esc' to browse)"; title_colors = INSERT_MODE_COLORS; break; case SEARCH_MODE: title = "ASCII TABLE (searching, 'esc' to browse)"; title_colors = SEARCH_MODE_COLORS; break; case BROWSE_MODE: title = "ASCII TABLE (browsing, 'esc' to exit)"; title_colors = BROWSE_MODE_COLORS; break; } int spacer = W/2-strlen(title)/2; printwattrs(title_colors, "%*s%s%*s", spacer, "", title, spacer+1, ""); if (mode == SEARCH_MODE) { attron(SEARCH_MODE_COLORS); mvprintw(H-1, 0, " Search:"); attroff(SEARCH_MODE_COLORS); addch(' '); printwattrs(OUTPUT_CHAR_COLORS, "%s", searchbuf); } else { // Bar at the bottom of the screen showing the output text: // left-truncate output so the end of it fits on screen: char *visible = &outbuf[buf_i]; for (int space = W-strlen(" Output: ")-1-5; space > 0 && visible > outbuf; visible--) { int chlen = (' ' <= *visible && *visible <= '~') ? 1 : 4; // printable range if (chlen > space) break; else space -= chlen; } attron(mode == INSERT_MODE ? INSERT_MODE_COLORS : OUTPUT_LABEL_COLORS); mvprintw(H-1, 0, " Output:"); attroff(mode == INSERT_MODE ? INSERT_MODE_COLORS : OUTPUT_LABEL_COLORS); addch(' '); for (char *c = visible; c < &outbuf[buf_i]; c++) { if (' ' <= *c && *c <= '~') { // printable range printwattrs(OUTPUT_CHAR_COLORS, "%c", *c); } else { printwattrs(OUTPUT_ESCAPE_COLORS, "\\x%02X", *c); } } } int y, x; getyx(stdscr, y, x); // Clear to end of line for (int x2 = x; x2 < W; x2++) mvaddch(y, x2, ' '); move(y, x); refresh(); int key; next_input: key = getch(); switch (mode) { case INSERT_MODE: switch (key) { case ESCAPE_KEY: mode = BROWSE_MODE; break; case DELETE_KEY: if (buf_i > 0) buf_i--; break; case KEY_RESIZE: handle_resize(); break; default: if (0 <= key && key < 127) { outbuf[buf_i++] = (char)key; if (buf_i >= buf_size) { buf_size *= 2; outbuf = realloc(outbuf, buf_size); } break; } else goto next_input; // skip redraw } break; case SEARCH_MODE: switch (key) { case ESCAPE_KEY: // Escape selected = prev_selected; searchbuf[0] = '\0'; searchlen = 0; mode = BROWSE_MODE; break; case KEY_ENTER: case '\n': mode = BROWSE_MODE; break; case DELETE_KEY: if (searchlen > 0) { searchbuf[searchlen--] = '\0'; selected = find_ascii_match(prev_selected, prev_selected, 1, searchbuf); } break; case KEY_RESIZE: handle_resize(); break; default: if (0 <= key && key < 127) { if (searchlen < sizeof(searchbuf)-1) { searchbuf[searchlen++] = (char)key; searchbuf[searchlen] = '\0'; } selected = find_ascii_match(prev_selected, prev_selected, 1, searchbuf); break; } else goto next_input; // skip redraw } break; case BROWSE_MODE: switch (key) { case 'j': case KEY_DOWN: selected++; break; case 'J': selected += 10; break; case 'k': case KEY_UP: selected--; break; case 'K': selected -= 10; break; case 'l': case 'L': case KEY_RIGHT: case KEY_SRIGHT: selected += H-2; break; case 'h': case 'H': case KEY_LEFT: case KEY_SLEFT: selected -= H-2; break; case 'i': case 'a': mode = INSERT_MODE; break; case '/': mode = SEARCH_MODE; prev_selected = selected; searchbuf[0] = '\0'; searchlen = 0; break; case 'n': selected = find_ascii_match(selected+1, selected, 1, searchbuf); break; case 'N': selected = find_ascii_match(selected-1, selected, -1, searchbuf); break; case KEY_RESIZE: handle_resize(); break; case KEY_ENTER: case '\n': outbuf[buf_i++] = (char)selected; if (buf_i >= buf_size) { buf_size *= 2; outbuf = realloc(outbuf, buf_size); } break; case KEY_BACKSPACE: case KEY_DC: case DELETE_KEY: if (buf_i > 0) buf_i--; break; case 'q': case 'Q': case KEY_CANCEL: case KEY_CLOSE: case KEY_EXIT: case ESCAPE_KEY: endwin(); write(STDOUT_FILENO, outbuf, buf_i); return 0; default: goto next_input; // skip redraw } break; } if (selected < 0) selected = 0; if (selected > 127) selected = 127; } endwin(); return 0; }