ncursesw-morphos/form/fty_num.c

265 lines
6.2 KiB
C
Raw Normal View History

1997-05-15 12:00:00 +08:00
/*
* THIS CODE IS SPECIFICALLY EXEMPTED FROM THE NCURSES PACKAGE COPYRIGHT.
* You may freely copy it for use as a template for your own field types.
* If you develop a field type that might be of general use, please send
* it back to the ncurses maintainers for inclusion in the next version.
*/
/***************************************************************************
* *
2002-10-13 11:35:53 +08:00
* Author : Juergen Pfeifer *
1997-05-15 12:00:00 +08:00
* *
***************************************************************************/
#include "form.priv.h"
2005-10-10 02:41:57 +08:00
MODULE_ID("$Id: fty_num.c,v 1.22 2005/08/20 18:26:16 tom Exp $")
1997-05-15 12:00:00 +08:00
#if HAVE_LOCALE_H
#include <locale.h>
#endif
2005-10-10 02:41:57 +08:00
#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;
1997-05-15 12:00:00 +08:00
/*---------------------------------------------------------------------------
2005-10-10 02:41:57 +08:00
| Facility : libnform
| Function : static void *Make_This_Type(va_list * ap)
|
1997-05-15 12:00:00 +08:00
| Description : Allocate structure for numeric type argument.
|
| Return Values : Pointer to argument structure or NULL on error
+--------------------------------------------------------------------------*/
2005-10-10 02:41:57 +08:00
static void *
Make_This_Type(va_list *ap)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
thisARG *argn = (thisARG *) malloc(sizeof(thisARG));
1997-05-15 12:00:00 +08:00
if (argn)
{
2005-10-10 02:41:57 +08:00
argn->precision = va_arg(*ap, int);
argn->low = va_arg(*ap, double);
argn->high = va_arg(*ap, double);
1997-05-15 12:00:00 +08:00
#if HAVE_LOCALE_H
2005-10-10 02:41:57 +08:00
argn->L = localeconv();
1997-05-15 12:00:00 +08:00
#else
2005-10-10 02:41:57 +08:00
argn->L = NULL;
1997-05-15 12:00:00 +08:00
#endif
}
return (void *)argn;
}
/*---------------------------------------------------------------------------
2005-10-10 02:41:57 +08:00
| Facility : libnform
| Function : static void *Copy_This_Type(const void * argp)
|
| Description : Copy structure for numeric type argument.
1997-05-15 12:00:00 +08:00
|
| Return Values : Pointer to argument structure or NULL on error.
+--------------------------------------------------------------------------*/
2005-10-10 02:41:57 +08:00
static void *
Copy_This_Type(const void *argp)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
const thisARG *ap = (const thisARG *)argp;
thisARG *result = (thisARG *) 0;
1997-05-15 12:00:00 +08:00
if (argp)
{
2005-10-10 02:41:57 +08:00
result = (thisARG *) malloc(sizeof(thisARG));
1998-03-01 12:21:12 +08:00
if (result)
2005-10-10 02:41:57 +08:00
*result = *ap;
1997-05-15 12:00:00 +08:00
}
1998-03-01 12:21:12 +08:00
return (void *)result;
1997-05-15 12:00:00 +08:00
}
/*---------------------------------------------------------------------------
2005-10-10 02:41:57 +08:00
| Facility : libnform
| Function : static void Free_This_Type(void * argp)
|
1997-05-15 12:00:00 +08:00
| Description : Free structure for numeric type argument.
|
| Return Values : -
+--------------------------------------------------------------------------*/
2005-10-10 02:41:57 +08:00
static void
Free_This_Type(void *argp)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
if (argp)
1997-05-15 12:00:00 +08:00
free(argp);
}
/*---------------------------------------------------------------------------
2005-10-10 02:41:57 +08:00
| Facility : libnform
| Function : static bool Check_This_Field(FIELD * field,
| const void * argp)
|
1997-05-15 12:00:00 +08:00
| Description : Validate buffer content to be a valid numeric value
|
| Return Values : TRUE - field is valid
| FALSE - field is invalid
+--------------------------------------------------------------------------*/
2005-10-10 02:41:57 +08:00
static bool
Check_This_Field(FIELD *field, const void *argp)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
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;
1997-05-15 12:00:00 +08:00
char buf[64];
2005-10-10 02:41:57 +08:00
bool result = FALSE;
1997-05-15 12:00:00 +08:00
2005-10-10 02:41:57 +08:00
while (*bp && *bp == ' ')
bp++;
1997-05-15 12:00:00 +08:00
if (*bp)
{
2005-10-10 02:41:57 +08:00
if (*bp == '-' || *bp == '+')
1997-05-15 12:00:00 +08:00
bp++;
2005-10-10 02:41:57 +08:00
#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)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
if (!isdigit(UChar(*bp)))
break;
1997-05-15 12:00:00 +08:00
bp++;
}
2005-10-10 02:41:57 +08:00
if (isDecimalPoint(*bp))
1997-05-15 12:00:00 +08:00
{
bp++;
2005-10-10 02:41:57 +08:00
while (*bp)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
if (!isdigit(UChar(*bp)))
break;
1997-05-15 12:00:00 +08:00
bp++;
}
}
2005-10-10 02:41:57 +08:00
while (*bp && *bp == ' ')
bp++;
result = (*bp == '\0');
#endif
if (result)
1997-05-15 12:00:00 +08:00
{
val = atof(s);
2005-10-10 02:41:57 +08:00
if (low < high)
{
if (val < low || val > high)
result = FALSE;
}
if (result)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
sprintf(buf, "%.*f", (prec > 0 ? prec : 0), val);
set_field_buffer(field, 0, buf);
1997-05-15 12:00:00 +08:00
}
}
}
2005-10-10 02:41:57 +08:00
return (result);
1997-05-15 12:00:00 +08:00
}
/*---------------------------------------------------------------------------
2005-10-10 02:41:57 +08:00
| Facility : libnform
| Function : static bool Check_This_Character(
1997-05-15 12:00:00 +08:00
| int c,
| const void * argp)
2005-10-10 02:41:57 +08:00
|
1997-05-15 12:00:00 +08:00
| Description : Check a character for the numeric type.
|
| Return Values : TRUE - character is valid
| FALSE - character is invalid
+--------------------------------------------------------------------------*/
2005-10-10 02:41:57 +08:00
static bool
Check_This_Character(int c, const void *argp)
1997-05-15 12:00:00 +08:00
{
2005-10-10 02:41:57 +08:00
const thisARG *argn = (const thisARG *)argp;
struct lconv *L = argn->L;
1997-05-15 12:00:00 +08:00
2005-10-10 02:41:57 +08:00
return ((isDigit(c) ||
c == '+' ||
c == '-' ||
isDecimalPoint(c))
? TRUE
: FALSE);
1997-05-15 12:00:00 +08:00
}
2005-10-10 02:41:57 +08:00
static FIELDTYPE typeTHIS =
{
1997-05-15 12:00:00 +08:00
_HAS_ARGS | _RESIDENT,
2005-10-10 02:41:57 +08:00
1, /* this is mutable, so we can't be const */
1997-05-15 12:00:00 +08:00
(FIELDTYPE *)0,
(FIELDTYPE *)0,
2005-10-10 02:41:57 +08:00
Make_This_Type,
Copy_This_Type,
Free_This_Type,
Check_This_Field,
Check_This_Character,
1997-05-15 12:00:00 +08:00
NULL,
NULL
};
2005-10-10 02:41:57 +08:00
NCURSES_EXPORT_VAR(FIELDTYPE*) TYPE_NUMERIC = &typeTHIS;
1997-05-15 12:00:00 +08:00
/* fty_num.c ends here */