#include "form.priv.h"
MODULE_ID("$Id: fty_num.c,v 1.22 2005/08/20 18:26:16 tom Exp $")
#if HAVE_LOCALE_H
#include <locale.h>
#endif
#if HAVE_LOCALE_H
#define isDecimalPoint(c) ((c) == ((L && L->decimal_point) ? *(L->decimal_point) : '.'))
#else
#define isDecimalPoint(c) ((c) == '.')
#endif
#if USE_WIDEC_SUPPORT
#define isDigit(c) (iswdigit((wint_t)(c)) || isdigit(UChar(c)))
#else
#define isDigit(c) isdigit(UChar(c))
#endif
#define thisARG numericARG
typedef struct
{
int precision;
double low;
double high;
struct lconv *L;
}
thisARG;
static void *
Make_This_Type(va_list *ap)
{
thisARG *argn = (thisARG *) malloc(sizeof(thisARG));
if (argn)
{
argn->precision = va_arg(*ap, int);
argn->low = va_arg(*ap, double);
argn->high = va_arg(*ap, double);
#if HAVE_LOCALE_H
argn->L = localeconv();
#else
argn->L = NULL;
#endif
}
return (void *)argn;
}
static void *
Copy_This_Type(const void *argp)
{
const thisARG *ap = (const thisARG *)argp;
thisARG *result = (thisARG *) 0;
if (argp)
{
result = (thisARG *) malloc(sizeof(thisARG));
if (result)
*result = *ap;
}
return (void *)result;
}
static void
Free_This_Type(void *argp)
{
if (argp)
free(argp);
}
static bool
Check_This_Field(FIELD *field, const void *argp)
{
const thisARG *argn = (const thisARG *)argp;
double low = argn->low;
double high = argn->high;
int prec = argn->precision;
unsigned char *bp = (unsigned char *)field_buffer(field, 0);
char *s = (char *)bp;
double val = 0.0;
struct lconv *L = argn->L;
char buf[64];
bool result = FALSE;
while (*bp && *bp == ' ')
bp++;
if (*bp)
{
if (*bp == '-' || *bp == '+')
bp++;
#if USE_WIDEC_SUPPORT
if (*bp)
{
bool blank = FALSE;
int state = 0;
int len;
int n;
wchar_t *list = _nc_Widen_String((char *)bp, &len);
if (list != 0)
{
result = TRUE;
for (n = 0; n < len; ++n)
{
if (blank)
{
if (list[n] != ' ')
{
result = FALSE;
break;
}
}
else if (list[n] == ' ')
{
blank = TRUE;
}
else if (isDecimalPoint(list[n]))
{
if (++state > 1)
{
result = FALSE;
break;
}
}
else if (!isDigit(list[n]))
{
result = FALSE;
break;
}
}
free(list);
}
}
#else
while (*bp)
{
if (!isdigit(UChar(*bp)))
break;
bp++;
}
if (isDecimalPoint(*bp))
{
bp++;
while (*bp)
{
if (!isdigit(UChar(*bp)))
break;
bp++;
}
}
while (*bp && *bp == ' ')
bp++;
result = (*bp == '\0');
#endif
if (result)
{
val = atof(s);
if (low < high)
{
if (val < low || val > high)
result = FALSE;
}
if (result)
{
sprintf(buf, "%.*f", (prec > 0 ? prec : 0), val);
set_field_buffer(field, 0, buf);
}
}
}
return (result);
}
static bool
Check_This_Character(int c, const void *argp)
{
const thisARG *argn = (const thisARG *)argp;
struct lconv *L = argn->L;
return ((isDigit(c) ||
c == '+' ||
c == '-' ||
isDecimalPoint(c))
? TRUE
: FALSE);
}
static FIELDTYPE typeTHIS =
{
_HAS_ARGS | _RESIDENT,
1,
(FIELDTYPE *)0,
(FIELDTYPE *)0,
Make_This_Type,
Copy_This_Type,
Free_This_Type,
Check_This_Field,
Check_This_Character,
NULL,
NULL
};
NCURSES_EXPORT_VAR(FIELDTYPE*) TYPE_NUMERIC = &typeTHIS;