2001-05-08 10:21:00 -04:00
|
|
|
#include "xlib.h"
|
|
|
|
|
|
|
|
Generic_Predicate (Color)
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static Color_Equal (x, y) s48_value x, y; {
|
2001-05-08 10:21:00 -04:00
|
|
|
register XColor *p = &COLOR(x)->c, *q = &COLOR(y)->c;
|
|
|
|
return p->red == q->red && p->green == q->green && p->blue == q->blue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Generic_Print (Color, "#[color %lu]", POINTER(x))
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
s48_value Make_Color (r, g, b) unsigned int r, g, b; {
|
|
|
|
s48_value c;
|
2001-05-08 10:21:00 -04:00
|
|
|
|
|
|
|
c = Find_Object (T_Color, (GENERIC)0, Match_X_Obj, r, g, b);
|
2001-05-14 09:48:37 -04:00
|
|
|
if (S48_NULL_P (c)) {
|
2001-05-08 10:21:00 -04:00
|
|
|
c = Alloc_Object (sizeof (struct S_Color), T_Color, 0);
|
2001-05-14 09:48:37 -04:00
|
|
|
COLOR(c)->tag = S48_NULL;
|
2001-05-08 10:21:00 -04:00
|
|
|
COLOR(c)->c.red = r;
|
|
|
|
COLOR(c)->c.green = g;
|
|
|
|
COLOR(c)->c.blue = b;
|
|
|
|
Register_Object (c, (GENERIC)0, (PFO)0, 0);
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
XColor *Get_Color (c) s48_value c; {
|
2001-05-08 10:21:00 -04:00
|
|
|
Check_Type (c, T_Color);
|
|
|
|
return &COLOR(c)->c;
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static unsigned short Get_RGB_Value (x) s48_value x; {
|
2001-05-08 10:21:00 -04:00
|
|
|
double d;
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
d = s48_extract_double (x);
|
2001-05-08 10:21:00 -04:00
|
|
|
if (d < 0.0 || d > 1.0)
|
|
|
|
Primitive_Error ("bad RGB value: ~s", x);
|
|
|
|
return (unsigned short)(d * 65535);
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static s48_value P_Make_Color (r, g, b) s48_value r, g, b; {
|
2001-05-08 10:21:00 -04:00
|
|
|
return Make_Color (Get_RGB_Value (r), Get_RGB_Value (g), Get_RGB_Value (b));
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static s48_value P_Color_Rgb_Values (c) s48_value c; {
|
|
|
|
s48_value ret, t, x;
|
|
|
|
S48_DECLARE_GC_PROTECT(3);
|
2001-05-08 10:21:00 -04:00
|
|
|
|
|
|
|
Check_Type (c, T_Color);
|
2001-05-14 09:48:37 -04:00
|
|
|
ret = t = S48_NULL;
|
|
|
|
S48_GC_PROTECT_3 (c, ret, t);
|
|
|
|
t = ret = P_Make_List (s48_enter_integer (3), S48_NULL);
|
|
|
|
S48_GC_UNPROTECT;
|
2001-05-08 10:21:00 -04:00
|
|
|
x = Make_Reduced_Flonum ((double)COLOR(c)->c.red / 65535.0);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_CAR (t) = x; t = S48_CDR (t);
|
2001-05-08 10:21:00 -04:00
|
|
|
x = Make_Reduced_Flonum ((double)COLOR(c)->c.green / 65535.0);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_CAR (t) = x; t = S48_CDR (t);
|
2001-05-08 10:21:00 -04:00
|
|
|
x = Make_Reduced_Flonum ((double)COLOR(c)->c.blue / 65535.0);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_CAR (t) = x;
|
2001-05-08 10:21:00 -04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static s48_value P_Query_Color (cmap, p) s48_value cmap, p; {
|
2001-05-08 10:21:00 -04:00
|
|
|
XColor c;
|
|
|
|
Colormap cm = Get_Colormap (cmap);
|
|
|
|
|
|
|
|
c.pixel = Get_Pixel (p);
|
|
|
|
Disable_Interrupts;
|
|
|
|
XQueryColor (COLORMAP(cmap)->dpy, cm, &c);
|
|
|
|
Enable_Interrupts;
|
|
|
|
return Make_Color (c.red, c.green, c.blue);
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static s48_value P_Query_Colors (cmap, v) s48_value cmap, v; {
|
2001-05-08 10:21:00 -04:00
|
|
|
Colormap cm = Get_Colormap (cmap);
|
|
|
|
register i, n;
|
2001-05-14 09:48:37 -04:00
|
|
|
s48_value ret;
|
2001-05-08 10:21:00 -04:00
|
|
|
register XColor *p;
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_DECLARE_GC_PROTECT(1);
|
2001-05-08 10:21:00 -04:00
|
|
|
Alloca_Begin;
|
|
|
|
|
|
|
|
Check_Type (v, T_Vector);
|
2001-05-14 09:48:37 -04:00
|
|
|
n = S48_VECTOR_LENGTH(v);
|
2001-05-08 10:21:00 -04:00
|
|
|
Alloca (p, XColor*, n * sizeof (XColor));
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
p[i].pixel = Get_Pixel (VECTOR(v)->data[i]);
|
|
|
|
Disable_Interrupts;
|
|
|
|
XQueryColors (COLORMAP(cmap)->dpy, cm, p, n);
|
|
|
|
Enable_Interrupts;
|
2001-05-14 09:48:37 -04:00
|
|
|
ret = s48_make_vector (n, S48_NULL);
|
|
|
|
S48_GC_PROTECT_1 (ret);
|
2001-05-08 10:21:00 -04:00
|
|
|
for (i = 0; i < n; i++, p++) {
|
2001-05-14 09:48:37 -04:00
|
|
|
s48_value x;
|
2001-05-08 10:21:00 -04:00
|
|
|
|
|
|
|
x = Make_Color (p->red, p->green, p->blue);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_VECTOR_SET(ret, i, x;)
|
2001-05-08 10:21:00 -04:00
|
|
|
}
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_GC_UNPROTECT;
|
2001-05-08 10:21:00 -04:00
|
|
|
Alloca_End;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2001-05-14 09:48:37 -04:00
|
|
|
static s48_value P_Lookup_Color (cmap, name) s48_value cmap, name; {
|
2001-05-08 10:21:00 -04:00
|
|
|
XColor visual, exact;
|
|
|
|
Colormap cm = Get_Colormap (cmap);
|
2001-05-14 09:48:37 -04:00
|
|
|
s48_value ret, x;
|
|
|
|
S48_DECLARE_GC_PROTECT(1);
|
2001-05-08 10:21:00 -04:00
|
|
|
|
|
|
|
if (!XLookupColor (COLORMAP(cmap)->dpy, cm, Get_Strsym (name),
|
|
|
|
&visual, &exact))
|
|
|
|
Primitive_Error ("no such color: ~s", name);
|
2001-05-14 09:48:37 -04:00
|
|
|
ret = s48_cons (S48_NULL, S48_NULL);
|
|
|
|
S48_GC_PROTECT_1 (ret);
|
2001-05-08 10:21:00 -04:00
|
|
|
x = Make_Color (visual.red, visual.green, visual.blue);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_CAR (ret) = x;
|
2001-05-08 10:21:00 -04:00
|
|
|
x = Make_Color (exact.red, exact.green, exact.blue);
|
2001-05-14 09:48:37 -04:00
|
|
|
S48_CDR (ret) = x;
|
|
|
|
S48_GC_UNPROTECT;
|
2001-05-08 10:21:00 -04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
elk_init_xlib_color () {
|
|
|
|
Generic_Define (Color, "color", "color?");
|
|
|
|
Define_Primitive (P_Make_Color, "make-color", 3, 3, EVAL);
|
|
|
|
Define_Primitive (P_Color_Rgb_Values, "color-rgb-values", 1, 1, EVAL);
|
|
|
|
Define_Primitive (P_Query_Color, "query-color", 2, 2, EVAL);
|
|
|
|
Define_Primitive (P_Query_Colors, "query-colors", 2, 2, EVAL);
|
|
|
|
Define_Primitive (P_Lookup_Color, "lookup-color", 2, 2, EVAL);
|
|
|
|
}
|