#ifdef M_RCSID
#ifndef lint
static char rcsID[] = "$Header: /rd/src/libc/xcurses/rcs/wgetn_ws.c 1.3 1995/10/02 15:07:23 ant Exp $";
#endif
#endif
#include <private.h>
#include <limits.h>
#include <stdlib.h>
#include <m_wio.h>
static wint_t fld_key;
static int fld_echo;
static int fld_index;
static int fld_bytes;
static int fld_mb;
static wint_t *fld_buffer;
static int fld_maxlength;
static WINDOW *fld_window;
static int fld_row, fld_col;
STATIC int fld_done(void);
STATIC int fld_erase(void);
STATIC int fld_kill(void);
STATIC int fld_insert(void);
typedef struct t_key_entry {
int type;
wint_t code;
int (*func)(void);
} t_key_entry;
#define ERASE_KEY 0
#define KILL_KEY 1
#define EOF_KEY 2
#define EOL_KEY 3
static t_key_entry key_table[] = {
{ OK, 0, fld_erase },
{ OK, 0, fld_kill },
{ OK, 0, fld_done },
{ OK, 0, fld_done },
{ OK, '\n', fld_done },
{ OK, WEOF, fld_done },
{ KEY_CODE_YES, KEY_LEFT, fld_erase },
{ KEY_CODE_YES, KEY_BACKSPACE, fld_erase },
{ ERR, 0, fld_insert }
};
int
__m_wgetn_wstr(w, s, n, mb_flag)
WINDOW *w;
void *s;
int n, mb_flag;
{
int type;
wchar_t wc;
t_key_entry *k;
struct termios saved;
#ifdef M_CURSES_TRACE
__m_trace("__m_wgetn_wstr(%p, %p, %d, %d)", w, s, n, mb_flag);
#endif
fld_window = w;
fld_index = 0;
fld_bytes = 0;
fld_mb = mb_flag;
fld_maxlength = n < 0 ? LINE_MAX : n;
if (fld_mb && fld_maxlength < MB_CUR_MAX)
return __m_return_code("wgetn_wstr", ERR);
if (mb_flag) {
fld_buffer = (wint_t *) calloc(
fld_maxlength+1, sizeof *fld_buffer
);
if (fld_buffer == (wint_t *) 0)
return __m_return_code("wgetn_wstr", ERR);
} else {
fld_buffer = (wint_t *) s;
}
(void) __m_tty_wc(VEOL, &wc);
key_table[EOL_KEY].code = (wint_t) wc;
(void) __m_tty_wc(VEOF, &wc);
key_table[EOF_KEY].code = (wint_t) wc;
(void) __m_tty_wc(VKILL, &wc);
key_table[KILL_KEY].code = (wint_t) wc;
(void) __m_tty_wc(VERASE, &wc);
key_table[ERASE_KEY].code = (wint_t) wc;
getyx(fld_window, fld_row, fld_col);
fld_echo = __m_set_echo(0);
saved = cur_term->_prog;
(void) cbreak();
for (;;) {
type = wget_wch(fld_window, &fld_key);
for (k = key_table; k->type != ERR; ++k)
if (k->type == type && k->code == fld_key)
break;
if (k->func != (int (*)(void)) 0 && !(*k->func)()) {
fld_buffer[fld_index] = '\0';
break;
}
}
(void) __m_set_echo(fld_echo);
(void) __m_tty_set(&saved);
if (mb_flag) {
(void) wistombs((char *) s, fld_buffer, fld_maxlength+1);
free(fld_buffer);
}
return __m_return_code("__m_wgetn_wstr", OK);
}
STATIC int
wint_len(wc)
wint_t wc;
{
int len;
char mb[MB_LEN_MAX];
if (wc == WEOF)
return 0;
len = wctomb(mb, (wchar_t) wc);
return len < 0 ? 0 : len;
}
STATIC int
fld_done()
{
return 0;
}
STATIC int
fld_erase()
{
int x, y, width;
if (fld_index <= 0)
return 1;
width = wcwidth(fld_buffer[--fld_index]);
fld_bytes -= wint_len(fld_buffer[fld_index]);
fld_buffer[fld_index] = '\0';
getyx(fld_window, y, x);
if (0 < x) {
x -= width;
} else if (0 < y) {
--y;
x = fld_window->_maxx - 1;
while (fld_buffer[fld_index] != fld_window->_line[y][x]._wc[0])
--x;
x = __m_cc_first(fld_window, y, x);
}
(void) __m_cc_erase(fld_window, y, x, y, x);
fld_window->_cury = y;
fld_window->_curx = x;
return 1;
}
STATIC int
fld_kill()
{
int y, x;
getyx(fld_window, y, x);
(void) __m_cc_erase(fld_window, fld_row, fld_col, y, x);
fld_window->_cury = fld_row;
fld_window->_curx = fld_col;
fld_index = fld_bytes = 0;
fld_buffer[0] = '\0';
return 1;
}
STATIC int
fld_insert()
{
cchar_t cc;
t_wide_io *wio;
if (fld_maxlength < fld_index)
return 0;
wio = (t_wide_io *) __m_screen->_in;
if (fld_mb && fld_maxlength < fld_bytes + wio->_size) {
if (fld_bytes < fld_maxlength) {
(void) beep();
return 1;
}
return 0;
}
if (0 <= fld_key) {
fld_buffer[fld_index++] = fld_key;
fld_bytes += wio->_size;
if (fld_echo) {
(void) __m_wc_cc(fld_key, &cc);
(void) wadd_wch(fld_window, &cc);
}
} else {
(void) beep();
}
return 1;
}
int
wgetnstr(w, s, n)
WINDOW *w;
char *s;
int n;
{
int code;
#ifdef M_CURSES_TRACE
__m_trace("wgetnstr(%p, %p, %d)", w, s, n);
#endif
code = __m_wgetn_wstr(w, (void *) s, n, 1);
return __m_return_code("wgetnstr", code);
}
int
wgetn_wstr(w, s, n)
WINDOW *w;
wint_t *s;
int n;
{
int code;
#ifdef M_CURSES_TRACE
__m_trace("wgetn_wstr(%p, %p, %d)", w, s, n);
#endif
code = __m_wgetn_wstr(w, (void *) s, n, 0);
return __m_return_code("wgetn_wstr", code);
}