/*  console.c -- Ncurses console for GRUB.  */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2003,2005,2007  Free Software Foundation, Inc.
 *
 *  GRUB 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 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB 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 GRUB.  If not, see .
 */
#include 
#if defined(HAVE_NCURSES_CURSES_H)
# include 
#elif defined(HAVE_NCURSES_H)
# include 
#elif defined(HAVE_CURSES_H)
# include 
#endif
/* For compatibility.  */
#ifndef A_NORMAL
# define A_NORMAL	0
#endif /* ! A_NORMAL */
#ifndef A_STANDOUT
# define A_STANDOUT	0
#endif /* ! A_STANDOUT */
#include 
#include 
#include 
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 = 2;
      break;
    case KEY_RIGHT:
      c = 6;
      break;
      
    case KEY_UP:
      c = 16;
      break;
    case KEY_DOWN:
      c = 14;
      break;
    case KEY_IC:
      c = 24;
      break;
    case KEY_DC:
      c = 4;
      break;
    case KEY_BACKSPACE:
      /* XXX: For some reason ncurses on xterm does not return
	 KEY_BACKSPACE.  */
    case 127: 
      c = 8;
      break;
    case KEY_HOME:
      c = 1;
      break;
    case KEY_END:
      c = 5;
      break;
    case KEY_NPAGE:
      c = 3;
      break;
    case KEY_PPAGE:
      c = 7;
      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 ();
}