gambas c0345b99ab Raise an error if point or rectangle constructor take an incorrect number of arguments.
[GB.GEOM]
* BUG: Raise an error if the rectangle constructor does not take zero or four arguments.
* BUG: Raise an error if the point constructor does not take zero or two arguments.
2021-11-02 12:20:07 +01:00

263 lines
31 KiB
C

/***************************************************************************
cpoint_temp.h
(c) 2000-2017 Benoît Minisini <g4mba5@gmail.com>
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, 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., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
***************************************************************************/
#define IMPLEMENT_POINT_CLASS(__struct, __name, __gtype, __ctype, __sign, __return, __this, __rstruct, __rname) \
\
__struct * __struct##_create(__ctype x, __ctype y) \
{ \
__struct *p = GB.New(GB.FindClass(#__name), NULL, NULL); \
p->x = x; \
p->y = y; \
return p; \
} \
\
static inline __struct *__struct##_make(__struct *a, const __ctype x, const __ctype y) \
{ \
if (a->ob.ref <= 1) \
{ \
a->x = x; \
a->y = y; \
return a; \
} \
else \
return __struct##_create(x, y); \
} \
\
static __struct *_add_##__name(__struct *a, __struct *b, bool invert) \
{ \
return __struct##_make(a, a->x + b->x, a->y + b->y); \
} \
\
static __struct *_sub_##__name(__struct *a, __struct *b, bool invert) \
{ \
return __struct##_make(a, a->x - b->x, a->y - b->y); \
} \
\
static __struct *_mulf_##__name(__struct *a, double f, bool invert) \
{ \
return __struct##_make(a, a->x * f, a->y * f); \
} \
\
static __struct *_mulo_##__name(__struct *a, void *b, bool invert) \
{ \
return NULL; \
} \
\
static __struct *_divf_##__name(__struct *a, double f, bool invert) \
{ \
if (invert) \
return NULL; \
if (f == 0.0) \
return NULL; \
\
return __struct##_make(a, a->x / f, a->y / f); \
} \
\
static __struct *_divo_##__name(__struct *a, void *b, bool invert) \
{ \
return NULL; \
} \
\
static int _equal_##__name(__struct *a, __struct *b, bool invert) \
{ \
return a->x == b->x && a->y == b->y; \
} \
\
static double _fabs_##__name(__struct *a) \
{ \
return hypot(a->x, a->y); \
} \
\
static __struct *_neg_##__name(__struct *a) \
{ \
return __struct##_make(a, -a->x, -a->y); \
} \
\
static GB_OPERATOR_DESC _operator_##__name = \
{ \
.equal = (void *)_equal_##__name, \
.add = (void *)_add_##__name, \
.sub = (void *)_sub_##__name, \
.mulf = (void *)_mulf_##__name, \
.mulo = (void *)_mulo_##__name, \
.divf = (void *)_divf_##__name, \
.divo = (void *)_divo_##__name, \
.fabs = (void *)_fabs_##__name, \
.neg = (void *)_neg_##__name, \
}; \
\
char *__struct##_to_string(void *a, bool local) \
{ \
char *result = NULL; \
char *str; \
int len; \
\
__ctype x = ((__struct *)a)->x; \
__ctype y = ((__struct *)a)->y; \
\
result = GB.AddChar(result, '['); \
\
GB.NumberToString(local, x, NULL, &str, &len); \
result = GB.AddString(result, str, len); \
\
result = GB.AddChar(result, local ? ' ' : ','); \
\
GB.NumberToString(local, y, NULL, &str, &len); \
result = GB.AddString(result, str, len); \
\
result = GB.AddChar(result, ']'); \
\
return result; \
} \
\
static bool _convert_##__name(void *a, GB_TYPE type, GB_VALUE *conv) \
{ \
if (a) \
{ \
double norm = _fabs_##__name(a); \
\
switch (type) \
{ \
case GB_T_FLOAT: \
conv->_float.value = norm; \
return FALSE; \
\
case GB_T_SINGLE: \
conv->_single.value = norm; \
return FALSE; \
\
case GB_T_INTEGER: \
case GB_T_SHORT: \
case GB_T_BYTE: \
conv->_integer.value = norm; \
return FALSE; \
\
case GB_T_LONG: \
conv->_long.value = norm; \
return FALSE; \
\
case GB_T_STRING: \
case GB_T_CSTRING: \
conv->_string.value.addr = __struct##_to_string(a, type == GB_T_CSTRING); \
conv->_string.value.start = 0; \
conv->_string.value.len = GB.StringLength(conv->_string.value.addr); \
return FALSE; \
\
default: \
if (type == GB.FindClass("Point")) \
{ \
conv->_object.value = CPOINT_create(((__struct *)a)->x, ((__struct *)a)->y); \
return FALSE; \
} \
if (type == GB.FindClass("PointF")) \
{ \
conv->_object.value = CPOINTF_create(((__struct *)a)->x, ((__struct *)a)->y); \
return FALSE; \
} \
else \
return TRUE; \
} \
} \
else \
return TRUE; \
} \
\
\
BEGIN_METHOD(__name##_new, __gtype x; __gtype y; __gtype w; __gtype h) \
\
if (!MISSING(x) && !MISSING(y)) \
{ \
__this->x = VARG(x); \
__this->y = VARG(y); \
} \
else if (!MISSING(x) || !MISSING(y)) \
{ \
GB.Error("Not enough arguments"); \
} \
\
END_METHOD \
\
BEGIN_METHOD(__name##_call, __gtype x; __gtype y) \
\
GB.ReturnObject(__struct##_create(VARGOPT(x, 0), VARGOPT(y, 0))); \
\
END_METHOD \
\
BEGIN_PROPERTY(__name##_X) \
\
if (READ_PROPERTY) \
__return(__this->x); \
else \
__this->x = VPROP(__gtype); \
\
END_PROPERTY \
\
BEGIN_PROPERTY(__name##_Y) \
\
if (READ_PROPERTY) \
__return(__this->y); \
else \
__this->y = VPROP(__gtype); \
\
END_PROPERTY \
\
BEGIN_METHOD_VOID(__name##_Copy) \
\
GB.ReturnObject(__struct##_create(__this->x, __this->y)); \
\
END_METHOD \
\
BEGIN_METHOD(__name##_InRect, GB_OBJECT rect) \
\
__rstruct *rect = VARG(rect); \
\
if (GB.CheckObject(rect)) \
return; \
\
GB.ReturnBoolean( \
(__this->x >= rect->x) && (__this->x < (rect->x + rect->w)) \
&& (__this->y >= rect->y) && (__this->y < (rect->y + rect->h)) \
); \
\
END_METHOD \
\
GB_DESC __name##Desc[] = \
{ \
GB_DECLARE(#__name, sizeof(__struct)), \
\
GB_METHOD("_new", NULL, __name##_new, "[(X)" __sign "(Y)" __sign), \
GB_STATIC_METHOD("_call", #__name, __name##_call, "[(X)" __sign "(Y)" __sign), \
\
GB_PROPERTY("X", __sign, __name##_X), \
GB_PROPERTY("Y", __sign, __name##_Y), \
\
GB_METHOD("Copy", #__name, __name##_Copy, NULL), \
GB_METHOD("InRect", "b", __name##_InRect, "(Rectangle)" #__rname ";"), \
\
GB_INTERFACE("_operator", &_operator_##__name), \
GB_INTERFACE("_convert", &_convert_##__name), \
\
GB_END_DECLARE \
};