/* console.c -- Ncurses console for GRUB. */ /* * GRUB -- GRand Unified Bootloader * Copyright (C) 2003,2005 Free Software Foundation, Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include <config.h> #if defined(HAVE_NCURSES_CURSES_H) # include <ncurses/curses.h> #elif defined(HAVE_NCURSES_H) # include <ncurses.h> #elif defined(HAVE_CURSES_H) # include <curses.h> #endif /* For compatibility. */ #ifndef A_NORMAL # define A_NORMAL 0 #endif /* ! A_NORMAL */ #ifndef A_STANDOUT # define A_STANDOUT 0 #endif /* ! A_STANDOUT */ #include <grub/machine/console.h> #include <grub/term.h> #include <grub/types.h> static int grub_console_attr = A_NORMAL; static void grub_ncurses_putchar (grub_uint32_t c) { /* Better than nothing. */ switch (c) { case GRUB_TERM_DISP_LEFT: c = '<'; break; case GRUB_TERM_DISP_UP: c = '^'; break; case GRUB_TERM_DISP_RIGHT: c = '>'; break; case GRUB_TERM_DISP_DOWN: c = 'v'; break; case GRUB_TERM_DISP_HLINE: c = '-'; break; case GRUB_TERM_DISP_VLINE: c = '|'; break; case GRUB_TERM_DISP_UL: case GRUB_TERM_DISP_UR: case GRUB_TERM_DISP_LL: case GRUB_TERM_DISP_LR: c = '+'; break; default: /* ncurses does not support Unicode. */ if (c > 0x7f) c = '?'; break; } addch (c | grub_console_attr); } static grub_ssize_t grub_ncurses_getcharwidth (grub_uint32_t code __attribute__ ((unused))) { return 1; } static void grub_ncurses_setcolorstate (grub_term_color_state state) { switch (state) { case GRUB_TERM_COLOR_STANDARD: grub_console_attr = A_NORMAL; break; case GRUB_TERM_COLOR_NORMAL: grub_console_attr = A_NORMAL; break; case GRUB_TERM_COLOR_HIGHLIGHT: grub_console_attr = A_STANDOUT; break; default: break; } } /* XXX: This function is never called. */ static void grub_ncurses_setcolor (grub_uint8_t normal_color, grub_uint8_t highlight_color) { color_set (normal_color << 8 | highlight_color, 0); } static int saved_char = ERR; static int grub_ncurses_checkkey (void) { int c; /* Check for SAVED_CHAR. This should not be true, because this means checkkey is called twice continuously. */ if (saved_char != ERR) return saved_char; wtimeout (stdscr, 100); c = getch (); /* If C is not ERR, then put it back in the input queue. */ if (c != ERR) { saved_char = c; return c; } return -1; } static int grub_ncurses_getkey (void) { int c; /* If checkkey has already got a character, then return it. */ if (saved_char != ERR) { c = saved_char; saved_char = ERR; } else { wtimeout (stdscr, -1); c = getch (); } switch (c) { case KEY_LEFT: c = GRUB_CONSOLE_KEY_LEFT; break; case KEY_RIGHT: c = GRUB_CONSOLE_KEY_RIGHT; break; case KEY_UP: c = GRUB_CONSOLE_KEY_UP; break; case KEY_DOWN: c = GRUB_CONSOLE_KEY_DOWN; break; case KEY_IC: c = GRUB_CONSOLE_KEY_IC; break; case KEY_DC: c = GRUB_CONSOLE_KEY_DC; break; case KEY_BACKSPACE: /* XXX: For some reason ncurses on xterm does not return KEY_BACKSPACE. */ case 127: c = GRUB_CONSOLE_KEY_BACKSPACE; break; case KEY_HOME: c = GRUB_CONSOLE_KEY_HOME; break; case KEY_END: c = GRUB_CONSOLE_KEY_END; break; case KEY_NPAGE: c = GRUB_CONSOLE_KEY_NPAGE; break; case KEY_PPAGE: c = GRUB_CONSOLE_KEY_PPAGE; break; } return c; } static grub_uint16_t grub_ncurses_getxy (void) { int x; int y; getyx (stdscr, y, x); return (x << 8) | y; } static grub_uint16_t grub_ncurses_getwh (void) { int x; int y; getmaxyx (stdscr, y, x); return (x << 8) | y; } static void grub_ncurses_gotoxy (grub_uint8_t x, grub_uint8_t y) { move (y, x); } static void grub_ncurses_cls (void) { clear (); refresh (); } static void grub_ncurses_setcursor (int on) { curs_set (on ? 1 : 0); } static void grub_ncurses_refresh (void) { refresh (); } static grub_err_t grub_ncurses_init (void) { initscr (); raw (); noecho (); scrollok (stdscr, TRUE); nonl (); intrflush (stdscr, FALSE); keypad (stdscr, TRUE); start_color (); return 0; } static grub_err_t grub_ncurses_fini (void) { endwin (); return 0; } static struct grub_term grub_ncurses_term = { .name = "console", .init = grub_ncurses_init, .fini = grub_ncurses_fini, .putchar = grub_ncurses_putchar, .getcharwidth = grub_ncurses_getcharwidth, .checkkey = grub_ncurses_checkkey, .getkey = grub_ncurses_getkey, .getxy = grub_ncurses_getxy, .getwh = grub_ncurses_getwh, .gotoxy = grub_ncurses_gotoxy, .cls = grub_ncurses_cls, .setcolorstate = grub_ncurses_setcolorstate, .setcolor = grub_ncurses_setcolor, .setcursor = grub_ncurses_setcursor, .refresh = grub_ncurses_refresh, .flags = 0, .next = 0 }; void grub_console_init (void) { grub_term_register (&grub_ncurses_term); grub_term_set_current (&grub_ncurses_term); } void grub_console_fini (void) { grub_ncurses_fini (); }