880 lines
24 KiB
C
880 lines
24 KiB
C
/*
|
|
* $Id$
|
|
*
|
|
Copyright (c) 2014-2019 Chung, Hyung-Hwan. All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
1. Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in the
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
|
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "_x11.h"
|
|
#include <moo-utl.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
|
|
|
|
typedef struct x11_modctx_t x11_modctx_t;
|
|
struct x11_modctx_t
|
|
{
|
|
moo_oop_class_t x11_class;
|
|
};
|
|
|
|
/* TODO: bchars_to_xchar2bstr??? */
|
|
static XChar2b* uchars_to_xchar2bstr (moo_t* moo, const moo_uch_t* inptr, moo_oow_t inlen, moo_oow_t* outlen)
|
|
{
|
|
moo_uch_t uch;
|
|
const moo_uch_t* endptr;
|
|
XChar2b* outbuf, * outptr;
|
|
|
|
outbuf = moo_allocmem(moo, (inlen + 1) * MOO_SIZEOF(*outptr));
|
|
if (!outbuf) return MOO_NULL;
|
|
|
|
outptr = outbuf;
|
|
endptr = inptr + inlen;
|
|
while (inptr < endptr)
|
|
{
|
|
uch = *inptr++;
|
|
|
|
#if (MOO_SIZEOF_UCH_T > 2)
|
|
if (uch > 0xFFFF) uch = 0xFFFD; /* unicode replacement character */
|
|
#endif
|
|
|
|
outptr->byte1 = (uch >> 8) & 0xFF;
|
|
outptr->byte2 = uch & 0xFF;
|
|
outptr++;
|
|
}
|
|
|
|
outptr->byte1 = 0;
|
|
outptr->byte2 = 0;
|
|
|
|
if (outlen) *outlen = outptr - outbuf;
|
|
return outbuf;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------ */
|
|
static moo_pfrc_t pf_open_display (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
x11_trailer_t* tr;
|
|
|
|
Display* disp = MOO_NULL;
|
|
XEvent* event = MOO_NULL;
|
|
char* dispname = MOO_NULL;
|
|
moo_ooi_t connno;
|
|
|
|
// TODO: CHECK if the receiver is an X11 object
|
|
|
|
if (nargs >= 1)
|
|
{
|
|
moo_oop_t np;
|
|
|
|
np = MOO_STACK_GETARG(moo, nargs, 0);
|
|
if (np != moo->_nil)
|
|
{
|
|
moo_oow_t bl;
|
|
|
|
if (!MOO_OBJ_IS_CHAR_POINTER(np))
|
|
{
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
goto oops;
|
|
}
|
|
|
|
bl = MOO_OBJ_GET_SIZE(np);
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
dispname = moo_dupootobcstr(moo, MOO_OBJ_GET_CHAR_SLOT(np), &bl);
|
|
if (!dispname)
|
|
{
|
|
MOO_DEBUG2 (moo, "<x11.connect> Cannot convert display name %.*js\n", MOO_OBJ_GET_SIZE(np), MOO_OBJ_GET_CHAR_SLOT(np));
|
|
MOO_STACK_SETRETTOERRNUM (moo, nargs);
|
|
goto oops;
|
|
}
|
|
#else
|
|
dispname = MOO_OBJ_GET_CHAR_SLOT(np);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
event = moo_allocmem(moo, MOO_SIZEOF(*event));
|
|
if (!event)
|
|
{
|
|
MOO_STACK_SETRETTOERRNUM (moo, nargs);
|
|
goto oops;
|
|
}
|
|
|
|
disp = XOpenDisplay(dispname);
|
|
if (!disp)
|
|
{
|
|
MOO_DEBUG1 (moo, "<x11.open_display> Cannot connect to X11 server %hs\n", XDisplayName(dispname));
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ESYSERR);
|
|
goto oops;
|
|
}
|
|
|
|
if (!MOO_IN_SMPTR_RANGE(disp))
|
|
{
|
|
MOO_DEBUG1 (moo, "<x11.open_display> Display pointer to %hs not in small pointer range\n", XDisplayName(dispname));
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ERANGE);
|
|
goto oops;
|
|
}
|
|
|
|
connno = ConnectionNumber(disp);
|
|
if (!MOO_IN_SMOOI_RANGE(connno))
|
|
{
|
|
MOO_DEBUG1 (moo, "<x11.open_display> Connection number to %hs out of small integer range\n", XDisplayName(dispname));
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ERANGE);
|
|
goto oops;
|
|
}
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
tr = (x11_trailer_t*)moo_getobjtrailer (moo, (moo_oop_t)x11, MOO_NULL);
|
|
tr->event = event;
|
|
tr->connection_number = connno;
|
|
tr->wm_delete_window = XInternAtom (disp, "WM_DELETE_WINDOW", False);
|
|
|
|
MOO_ASSERT (moo, MOO_IN_SMPTR_RANGE(disp));
|
|
x11->display = MOO_SMPTR_TO_OOP(disp);
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
if (dispname) moo_freemem (moo, dispname);
|
|
#endif
|
|
return MOO_PF_SUCCESS;
|
|
|
|
oops:
|
|
if (disp) XCloseDisplay (disp);
|
|
if (event) moo_freemem (moo, event);
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
if (dispname) moo_freemem (moo, dispname);
|
|
#endif
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_close_display (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
x11_trailer_t* tr;
|
|
|
|
// TODO: CHECK if the receiver is an X11 object
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
if (x11->display != moo->_nil)
|
|
{
|
|
MOO_ASSERT (moo, MOO_OOP_IS_SMPTR(x11->display));
|
|
XCloseDisplay (MOO_OOP_TO_SMPTR(x11->display));
|
|
x11->display = moo->_nil;
|
|
}
|
|
|
|
|
|
tr = moo_getobjtrailer (moo, MOO_STACK_GETRCV(moo,nargs), MOO_NULL);
|
|
if (tr->event)
|
|
{
|
|
moo_freemem (moo, tr->event);
|
|
tr->event = MOO_NULL;
|
|
}
|
|
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_get_fd (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
x11_trailer_t* tr;
|
|
// TODO: CHECK if the receiver is an X11 object
|
|
tr = moo_getobjtrailer (moo, MOO_STACK_GETRCV(moo,nargs), MOO_NULL);
|
|
MOO_STACK_SETRET(moo, nargs, MOO_SMOOI_TO_OOP(tr->connection_number));
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_get_llevent (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
x11_trailer_t* tr;
|
|
|
|
Display* disp;
|
|
XEvent* event;
|
|
|
|
// TODO: CHECK if the receiver is an X11 object
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
//MOO_ASSERT (moo, MOO_CLASSOF(moo,x11) == modctx->x11_class);
|
|
tr = moo_getobjtrailer(moo, (moo_oop_t)x11, MOO_NULL);
|
|
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
event = tr->event;
|
|
if (XPending(disp))
|
|
{
|
|
oop_x11_llevent_t e;
|
|
|
|
XNextEvent (disp, event);
|
|
|
|
e = (oop_x11_llevent_t)MOO_STACK_GETARG(moo, nargs, 0);
|
|
/* TOOD: check if e is an instance of X11.LLEvent */
|
|
e->type = MOO_SMOOI_TO_OOP(event->type);
|
|
e->window = MOO_SMOOI_TO_OOP(0);
|
|
|
|
/* if the following is going to trigger GC directly or indirectly,
|
|
* e must be proteced with moo_pushvolat().
|
|
* also x11, tr must be refetched from the stack. */
|
|
|
|
switch (event->type)
|
|
{
|
|
case ClientMessage:
|
|
if (event->xclient.data.l[0] == tr->wm_delete_window)
|
|
{
|
|
e->type = MOO_SMOOI_TO_OOP(65537); /* match SHELL_CLOSE in X11.LLEventType */
|
|
e->window = MOO_SMOOI_TO_OOP(event->xclient.window);
|
|
/* WINDOW CLSOE EVENT */
|
|
}
|
|
|
|
break;
|
|
|
|
case Expose:
|
|
{
|
|
XRectangle rect;
|
|
|
|
rect.x = event->xexpose.x;
|
|
rect.y = event->xexpose.y;
|
|
rect.width = event->xexpose.width;
|
|
rect.height = event->xexpose.height;
|
|
if (XCheckWindowEvent(disp, event->xexpose.window, ExposureMask, event))
|
|
{
|
|
Region reg;
|
|
|
|
/* merge all expose events in the event queue */
|
|
reg = XCreateRegion();
|
|
XUnionRectWithRegion (&rect, reg, reg);
|
|
|
|
do
|
|
{
|
|
rect.x = event->xexpose.x;
|
|
rect.y = event->xexpose.y;
|
|
rect.width = event->xexpose.width;
|
|
rect.height = event->xexpose.height;
|
|
XUnionRectWithRegion (&rect, reg, reg);
|
|
}
|
|
while (XCheckWindowEvent(disp, event->xexpose.window, ExposureMask, event));
|
|
|
|
XClipBox (reg, &rect);
|
|
XDestroyRegion (reg);
|
|
}
|
|
|
|
e->window = MOO_SMOOI_TO_OOP(event->xexpose.window);
|
|
e->x = MOO_SMOOI_TO_OOP(rect.x);
|
|
e->y = MOO_SMOOI_TO_OOP(rect.y);
|
|
e->width = MOO_SMOOI_TO_OOP(rect.width);
|
|
e->height = MOO_SMOOI_TO_OOP(rect.height);
|
|
break;
|
|
}
|
|
|
|
case ButtonPress:
|
|
case ButtonRelease:
|
|
{
|
|
e->window = MOO_SMOOI_TO_OOP(event->xbutton.window);
|
|
e->x = MOO_SMOOI_TO_OOP(event->xbutton.x);
|
|
e->y = MOO_SMOOI_TO_OOP(event->xbutton.y);
|
|
break;
|
|
}
|
|
}
|
|
|
|
MOO_STACK_SETRET (moo, nargs, (moo_oop_t)e);
|
|
}
|
|
else
|
|
{
|
|
/* nil if there is no event */
|
|
MOO_DEBUG0 (moo, "NO PENDING EVENT....\n");
|
|
MOO_STACK_SETRET (moo, nargs, moo->_nil);
|
|
}
|
|
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_create_window (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
Display* disp;
|
|
Window wind; /* Window -> XID, unsigned long */
|
|
int scrn;
|
|
Window parent;
|
|
XSetWindowAttributes attrs;
|
|
|
|
oop_x11_t x11;
|
|
x11_trailer_t* tr;
|
|
moo_oop_t a0, a1, a2, a3, a4, a5, a6;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
|
|
a0 = MOO_STACK_GETARG(moo, nargs, 0); /* parent window - Integer or nil (Window) */
|
|
a1 = MOO_STACK_GETARG(moo, nargs, 1); /* x - SmallInteger */
|
|
a2 = MOO_STACK_GETARG(moo, nargs, 2); /* y - SmallInteger */
|
|
a3 = MOO_STACK_GETARG(moo, nargs, 3); /* width - SmallInteger */
|
|
a4 = MOO_STACK_GETARG(moo, nargs, 4); /* height - SmallInteger */
|
|
a5 = MOO_STACK_GETARG(moo, nargs, 5); /* fgcolor - SmallInteger */
|
|
a6 = MOO_STACK_GETARG(moo, nargs, 6); /* bgcolor - SmallInteger */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a1) || !MOO_OOP_IS_SMOOI(a2) || !MOO_OOP_IS_SMOOI(a3) ||
|
|
!MOO_OOP_IS_SMOOI(a4) || !MOO_OOP_IS_SMOOI(a5) || !MOO_OOP_IS_SMOOI(a6))
|
|
{
|
|
einval:
|
|
MOO_DEBUG0 (moo, "<x11.create_window> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
tr = moo_getobjtrailer (moo, (moo_oop_t)x11, MOO_NULL);
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
|
|
/* ScreenCount (disp); -> the number of screens available in this display server */
|
|
|
|
if (a0 == moo->_nil)
|
|
{
|
|
scrn = DefaultScreen (disp);
|
|
parent = RootWindow (disp, scrn);
|
|
}
|
|
else if (!MOO_OOP_IS_SMOOI(a0)) goto einval;
|
|
else
|
|
{
|
|
XWindowAttributes wa;
|
|
|
|
parent = MOO_OOP_TO_SMOOI(a0);
|
|
XGetWindowAttributes (disp, parent, &wa);
|
|
scrn = XScreenNumberOfScreen(wa.screen);
|
|
}
|
|
|
|
attrs.event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | ExposureMask/* | StructureNotifyMask*/; /* TODO: accept it as a parameter??? */
|
|
attrs.border_pixel = BlackPixel (disp, scrn); /* TODO: use a6 */
|
|
attrs.background_pixel = WhitePixel (disp, scrn);/* TODO: use a7 */
|
|
|
|
wind = XCreateWindow (
|
|
disp,
|
|
parent,
|
|
MOO_OOP_TO_SMOOI(a1), /* x */
|
|
MOO_OOP_TO_SMOOI(a2), /* y */
|
|
MOO_OOP_TO_SMOOI(a3), /* width */
|
|
MOO_OOP_TO_SMOOI(a4), /* height */
|
|
0, /* border width */
|
|
CopyFromParent, /* depth */
|
|
InputOutput, /* class */
|
|
CopyFromParent, /* visual */
|
|
CWEventMask | CWBackPixel | CWBorderPixel, &attrs);
|
|
if (!wind)
|
|
{
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ESYSERR);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
if (parent == RootWindow(disp, scrn))
|
|
{
|
|
XSetWMProtocols (disp, wind, &tr->wm_delete_window, 1);
|
|
}
|
|
|
|
/*if (!MOO_IN_SMOOI_RANGE ((moo_ooi_t)wind))*/
|
|
if (wind > MOO_SMOOI_MAX)
|
|
{
|
|
XDestroyWindow (disp, wind);
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ERANGE);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
XMapWindow (disp, wind);
|
|
XFlush (disp);
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMOOI_TO_OOP(wind));
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_destroy_window (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
moo_oop_t a0;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
a0 = MOO_STACK_GETARG(moo, nargs, 0); /* window - Integer (Window) */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a0))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.destroy_window> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
}
|
|
else
|
|
{
|
|
Display* disp;
|
|
Window wind;
|
|
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
wind = MOO_OOP_TO_SMOOI(a0);
|
|
|
|
XUnmapWindow (disp, wind);
|
|
XDestroyWindow (disp, wind);
|
|
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
}
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_create_gc (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
Display* disp;
|
|
Window wind;
|
|
GC gc;
|
|
XWindowAttributes wa;
|
|
|
|
oop_x11_t x11;
|
|
moo_oop_t a0;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
|
|
a0 = MOO_STACK_GETARG(moo, nargs, 0); /* Window - SmallInteger */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a0))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.create_gc> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
wind = MOO_OOP_TO_SMOOI(a0);
|
|
|
|
gc = XCreateGC (disp, wind, 0, MOO_NULL);
|
|
if (!MOO_IN_SMPTR_RANGE(gc))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.create_gc> GC pointer not in small pointer range\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ERANGE);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
/* XGetWindowAttributes (disp, wind, &wa);
|
|
XCopyGC (disp, DefaultGC(disp, XScreenNumberOfScreen(wa.screen)), GCForeground | GCBackground | GCLineWidth | GCLineStyle, gc);*/
|
|
|
|
MOO_STACK_SETRET (moo, nargs, MOO_SMPTR_TO_OOP(gc));
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_destroy_gc (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
oop_x11_gc_t gc;
|
|
|
|
Display* disp;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
gc = (oop_x11_gc_t)MOO_STACK_GETARG(moo, nargs, 0); /* GC object */
|
|
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
if (MOO_OOP_IS_SMPTR(gc->font_ptr))
|
|
{
|
|
XFreeFont (disp, MOO_OOP_TO_SMPTR(gc->font_ptr));
|
|
gc->font_ptr = moo->_nil;
|
|
}
|
|
|
|
if (MOO_OOP_IS_SMPTR(gc->font_set))
|
|
{
|
|
XFreeFontSet (disp, MOO_OOP_TO_SMPTR(gc->font_set));
|
|
gc->font_set = moo->_nil;
|
|
MOO_DEBUG0 (moo, "Freed Font Set\n");
|
|
}
|
|
|
|
if (MOO_OOP_IS_SMPTR(gc->gc_handle))
|
|
{
|
|
XFreeGC (disp, MOO_OOP_TO_SMPTR(gc->gc_handle));
|
|
gc->gc_handle= moo->_nil;
|
|
}
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_apply_gc (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
oop_x11_gc_t a0;
|
|
|
|
Display* disp;
|
|
GC gc;
|
|
unsigned long int mask = 0;
|
|
XGCValues v;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
a0 = (oop_x11_gc_t)MOO_STACK_GETARG(moo, nargs, 0);
|
|
/* TODO check if a0 is an instance of X11.GC */
|
|
|
|
if (!MOO_OOP_IS_SMPTR(a0->gc_handle))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.apply_gc> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
gc = MOO_OOP_TO_SMPTR(a0->gc_handle);
|
|
|
|
if (MOO_OBJ_IS_CHAR_POINTER(a0->font_name) && MOO_OBJ_GET_SIZE(a0->font_name) > 0)
|
|
{
|
|
XFontSet fs;
|
|
char **missing_charsets;
|
|
int num_missing_charsets = 0;
|
|
char *default_string;
|
|
|
|
/* TODO: don't create this again and again */
|
|
/* TODO: use font name */
|
|
fs = XCreateFontSet (disp, "-adobe-*-medium-r-normal-*-14-*-*-*-*-*-*-*,-baekmuk-*-medium-r-normal-*-14-*-*-*-*-*-*-*,-*-*-medium-r-normal-*-*-*-*-*-*-*-*-*",
|
|
&missing_charsets, &num_missing_charsets, &default_string);
|
|
if (num_missing_charsets)
|
|
{
|
|
int i;
|
|
|
|
MOO_DEBUG0 (moo, "The following charsets are missing:\n");
|
|
for(i = 0; i < num_missing_charsets; i++)
|
|
MOO_DEBUG1 (moo, "\t%s\n", missing_charsets[i]);
|
|
MOO_DEBUG1 (moo, "The string %s will be used in place of any characters from those set\n", default_string);
|
|
XFreeStringList(missing_charsets);
|
|
}
|
|
|
|
if (fs)
|
|
{
|
|
/* TODO: error handling. rollback upon failure... etc */
|
|
MOO_ASSERT (moo, MOO_IN_SMPTR_RANGE(fs));
|
|
if (MOO_OOP_IS_SMPTR(a0->font_set))
|
|
{
|
|
MOO_DEBUG0 (moo, "Freed Font Set ..\n");
|
|
XFreeFontSet (disp, MOO_OOP_TO_SMPTR(a0->font_set));
|
|
}
|
|
a0->font_set = MOO_SMPTR_TO_OOP (fs);
|
|
MOO_DEBUG0 (moo, "XCreateFontSet ok....\n");
|
|
}
|
|
else
|
|
{
|
|
XFontStruct* font;
|
|
|
|
/* TODO: .... use font_name */
|
|
font = XLoadQueryFont (disp, "-misc-fixed-medium-r-normal-ko-18-120-100-100-c-180-iso10646-1");
|
|
if (font)
|
|
{
|
|
if (!MOO_IN_SMPTR_RANGE(font))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.apply_gc> Font pointer not in small pointer range\n");
|
|
XFreeFont (disp, font);
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ERANGE);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
XSetFont (disp, gc, font->fid);
|
|
if (MOO_OOP_IS_SMPTR(a0->font_ptr)) XFreeFont (disp, MOO_OOP_TO_SMPTR(a0->font_ptr));
|
|
a0->font_ptr = MOO_SMPTR_TO_OOP(font);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
MOO_DEBUG2 (moo, "<x11.apply_gc> Cannot load font - %.*js\n", MOO_OBJ_GET_SIZE(a0->font_name), MOO_OBJ_GET_CHAR_SLOT(a0->font_name));
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_ESYSERR);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* TODO: accept mask as an option parameter. then only apply fields that matches this given mask */
|
|
if (MOO_OOP_IS_SMOOI(a0->foreground))
|
|
{
|
|
mask |= GCForeground;
|
|
v.foreground = MOO_OOP_TO_SMOOI(a0->foreground);
|
|
}
|
|
if (MOO_OOP_IS_SMOOI(a0->background))
|
|
{
|
|
mask |= GCBackground;
|
|
v.background = MOO_OOP_TO_SMOOI(a0->background);
|
|
}
|
|
if (MOO_OOP_IS_SMOOI(a0->line_width))
|
|
{
|
|
mask |= GCLineWidth;
|
|
v.line_width = MOO_OOP_TO_SMOOI(a0->line_width);
|
|
}
|
|
if (MOO_OOP_IS_SMOOI(a0->line_style))
|
|
{
|
|
mask |= GCLineStyle;
|
|
v.line_style = MOO_OOP_TO_SMOOI(a0->line_style);
|
|
}
|
|
if (MOO_OOP_IS_SMOOI(a0->fill_style))
|
|
{
|
|
mask |= GCFillStyle;
|
|
v.fill_style = MOO_OOP_TO_SMOOI(a0->fill_style);
|
|
}
|
|
XChangeGC (disp, gc, mask, &v);
|
|
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
static moo_pfrc_t pf_draw_rectangle (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
Display* disp;
|
|
moo_oop_t a0, a1, a2, a3, a4, a5;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
|
|
a0 = MOO_STACK_GETARG(moo, nargs, 0); /* Window - SmallInteger */
|
|
a1 = MOO_STACK_GETARG(moo, nargs, 1); /* GC - SMPTR */
|
|
a2 = MOO_STACK_GETARG(moo, nargs, 2); /* x - SmallInteger */
|
|
a3 = MOO_STACK_GETARG(moo, nargs, 3); /* y - SmallInteger */
|
|
a4 = MOO_STACK_GETARG(moo, nargs, 4); /* width - SmallInteger */
|
|
a5 = MOO_STACK_GETARG(moo, nargs, 5); /* height - SmallInteger */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a0) || !MOO_OOP_IS_SMPTR(a1) ||
|
|
!MOO_OOP_IS_SMOOI(a2) || !MOO_OOP_IS_SMOOI(a3) ||
|
|
!MOO_OOP_IS_SMOOI(a4) || !MOO_OOP_IS_SMOOI(a5))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.draw_rectangle> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
XDrawRectangle (disp, (Window)MOO_OOP_TO_SMOOI(a0), (GC)MOO_OOP_TO_SMPTR(a1),
|
|
MOO_OOP_TO_SMOOI(a2), MOO_OOP_TO_SMOOI(a3),
|
|
MOO_OOP_TO_SMOOI(a4), MOO_OOP_TO_SMOOI(a5));
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
|
|
static moo_pfrc_t pf_fill_rectangle (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
Display* disp;
|
|
moo_oop_t a0, a1, a2, a3, a4, a5;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
|
|
a0 = MOO_STACK_GETARG(moo, nargs, 0); /* Window - SmallInteger */
|
|
a1 = MOO_STACK_GETARG(moo, nargs, 1); /* GC - SMPTR */
|
|
a2 = MOO_STACK_GETARG(moo, nargs, 2); /* x - SmallInteger */
|
|
a3 = MOO_STACK_GETARG(moo, nargs, 3); /* y - SmallInteger */
|
|
a4 = MOO_STACK_GETARG(moo, nargs, 4); /* width - SmallInteger */
|
|
a5 = MOO_STACK_GETARG(moo, nargs, 5); /* height - SmallInteger */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a0) || !MOO_OOP_IS_SMPTR(a1) ||
|
|
!MOO_OOP_IS_SMOOI(a2) || !MOO_OOP_IS_SMOOI(a3) ||
|
|
!MOO_OOP_IS_SMOOI(a4) || !MOO_OOP_IS_SMOOI(a5))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.fill_rectangle> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
XFillRectangle (disp, (Window)MOO_OOP_TO_SMOOI(a0), (GC)MOO_OOP_TO_SMPTR(a1),
|
|
MOO_OOP_TO_SMOOI(a2), MOO_OOP_TO_SMOOI(a3),
|
|
MOO_OOP_TO_SMOOI(a4), MOO_OOP_TO_SMOOI(a5));
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
|
|
static moo_pfrc_t pf_draw_string (moo_t* moo, moo_mod_t* mod, moo_ooi_t nargs)
|
|
{
|
|
oop_x11_t x11;
|
|
oop_x11_gc_t gc;
|
|
moo_oop_t a1, a2, a3;
|
|
|
|
Display* disp;
|
|
|
|
x11 = (oop_x11_t)MOO_STACK_GETRCV(moo, nargs);
|
|
disp = MOO_OOP_TO_SMPTR(x11->display);
|
|
|
|
gc = (oop_x11_gc_t)MOO_STACK_GETARG(moo, nargs, 0); /* GC object */
|
|
a1 = MOO_STACK_GETARG(moo, nargs, 1); /* x - SmallInteger */
|
|
a2 = MOO_STACK_GETARG(moo, nargs, 2); /* y - SmallInteger */
|
|
a3 = MOO_STACK_GETARG(moo, nargs, 3); /* string */
|
|
|
|
/* TODO: check if gc is an instance of X11.GC */
|
|
|
|
if (!MOO_OOP_IS_SMOOI(a1) || !MOO_OOP_IS_SMOOI(a2) ||
|
|
!MOO_OBJ_IS_CHAR_POINTER(a3))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.draw_string> Invalid parameters\n");
|
|
MOO_STACK_SETRETTOERROR (moo, nargs, MOO_EINVAL);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
if (MOO_OOP_IS_SMPTR(gc->font_set))
|
|
{
|
|
moo_oow_t bcslen;
|
|
moo_bch_t* bb;
|
|
int ascent = 10;
|
|
XRectangle r;
|
|
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
moo_oow_t oocslen;
|
|
|
|
oocslen = MOO_OBJ_GET_SIZE(a3);
|
|
if (moo_convootobchars(moo, MOO_OBJ_GET_CHAR_SLOT(a3), &oocslen, MOO_NULL, &bcslen) <= -1 ||
|
|
!(bb = moo_allocmem(moo, MOO_SIZEOF(moo_bch_t) * bcslen)))
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.draw_string> Error in converting a string\n");
|
|
MOO_STACK_SETRETTOERRNUM (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
moo_convootobchars (moo, MOO_OBJ_GET_CHAR_SLOT(a3), &oocslen, bb, &bcslen);
|
|
#else
|
|
bb = MOO_OBJ_GET_CHAR_SLOT(a3);
|
|
bcslen = MOO_OBJ_GET_SIZE(a3);
|
|
#endif
|
|
|
|
XmbTextExtents(MOO_OOP_TO_SMPTR(gc->font_set), bb, bcslen, MOO_NULL, &r);
|
|
ascent = r.height;
|
|
|
|
/* what about Xutf8DrawString? */
|
|
XmbDrawString (disp, (Window)MOO_OOP_TO_SMOOI(((oop_x11_widget_t)gc->widget)->window_handle),
|
|
MOO_OOP_TO_SMPTR(gc->font_set), MOO_OOP_TO_SMPTR(gc->gc_handle),
|
|
MOO_OOP_TO_SMOOI(a1), MOO_OOP_TO_SMOOI(a2) + ascent, bb, bcslen);
|
|
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
moo_freemem (moo, bb);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
#if defined(MOO_OOCH_IS_UCH)
|
|
XChar2b* stptr;
|
|
moo_oow_t stlen;
|
|
int ascent = 0;
|
|
|
|
/* TODO: draw string chunk by chunk to avoid memory allocation in uchars_to_xchars2bstr */
|
|
stptr = uchars_to_xchar2bstr(moo, MOO_OBJ_GET_CHAR_SLOT(a3), MOO_OBJ_GET_SIZE(a3), &stlen);
|
|
if (!stptr)
|
|
{
|
|
MOO_DEBUG0 (moo, "<x11.draw_string> Error in converting a string\n");
|
|
MOO_STACK_SETRETTOERRNUM (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
if (MOO_OOP_IS_SMPTR(gc->font_ptr))
|
|
{
|
|
int direction, descent;
|
|
XCharStruct overall;
|
|
XTextExtents16 (MOO_OOP_TO_SMPTR(gc->font_ptr), stptr, stlen, &direction, &ascent, &descent, &overall);
|
|
}
|
|
|
|
XDrawString16 (disp, (Window)MOO_OOP_TO_SMOOI(((oop_x11_widget_t)gc->widget)->window_handle), MOO_OOP_TO_SMPTR(gc->gc_handle),
|
|
MOO_OOP_TO_SMOOI(a1), MOO_OOP_TO_SMOOI(a2) + ascent, stptr, stlen);
|
|
|
|
moo_freemem (moo, stptr);
|
|
#else
|
|
int ascent = 0;
|
|
|
|
if (MOO_OOP_IS_SMPTR(gc->font_ptr))
|
|
{
|
|
int direction, descent;
|
|
XCharStruct overall;
|
|
XTextExtents16 (MOO_OOP_TO_SMPTR(gc->font_ptr), MOO_OBJ_GET_CHAR_SLOT(a3), MOO_OBJ_GET_SIZE(a3), &direction, &ascent, &descent, &overall);
|
|
}
|
|
|
|
XDrawString (disp, (Window)MOO_OOP_TO_SMOOI(((oop_x11_widget_t)gc->widget)->window_handle), MOO_OOP_TO_SMPTR(gc->gc_handle),
|
|
MOO_OOP_TO_SMOOI(a1), MOO_OOP_TO_SMOOI(a2) + ascent, MOO_OBJ_GET_CHAR_SLOT(a3), MOO_OBJ_GET_SIZE(a3));
|
|
#endif
|
|
}
|
|
|
|
MOO_STACK_SETRETTORCV (moo, nargs);
|
|
return MOO_PF_SUCCESS;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------ */
|
|
|
|
static moo_pfinfo_t x11_pfinfo[] =
|
|
{
|
|
{ MI, "apply_gc", { pf_apply_gc, 1, 1 } },
|
|
{ MI, "close_display", { pf_close_display, 0, 0 } },
|
|
{ MI, "create_gc", { pf_create_gc, 1, 1 } },
|
|
{ MI, "create_window", { pf_create_window, 7, 7 } },
|
|
{ MI, "destroy_gc", { pf_destroy_gc, 1, 1 } },
|
|
{ MI, "destroy_window", { pf_destroy_window, 1, 1 } },
|
|
|
|
{ MI, "draw_rectangle", { pf_draw_rectangle, 6, 6 } },
|
|
{ MI, "draw_string", { pf_draw_string, 4, 4 } },
|
|
{ MI, "fill_rectangle", { pf_fill_rectangle, 6, 6 } },
|
|
{ MI, "get_fd", { pf_get_fd, 0, 0 } },
|
|
{ MI, "get_llevent", { pf_get_llevent, 1, 1 } },
|
|
{ MI, "open_display", { pf_open_display, 0, 1 } }
|
|
};
|
|
|
|
static int x11_import (moo_t* moo, moo_mod_t* mod, moo_oop_class_t _class)
|
|
{
|
|
if (moo_setclasstrsize(moo, _class, MOO_SIZEOF(x11_trailer_t), MOO_NULL) <= -1) return -1;
|
|
return 0;
|
|
}
|
|
|
|
static moo_pfbase_t* x11_querypf (moo_t* moo, moo_mod_t* mod, const moo_ooch_t* name, moo_oow_t namelen)
|
|
{
|
|
return moo_findpfbase(moo, x11_pfinfo, MOO_COUNTOF(x11_pfinfo), name, namelen);
|
|
}
|
|
|
|
static void x11_unload (moo_t* moo, moo_mod_t* mod)
|
|
{
|
|
/* TODO: anything else? close all open dll handles? For that, pf_open must store the value it returns to mod->ctx or somewhere..*/
|
|
moo_freemem (moo, mod->ctx);
|
|
}
|
|
|
|
static void gc_mod_x11 (moo_t* moo, moo_mod_t* mod)
|
|
{
|
|
x11_modctx_t* ctx = mod->ctx;
|
|
|
|
MOO_ASSERT (moo, ctx != MOO_NULL);
|
|
ctx->x11_class = (moo_oop_class_t)moo_updateoopforgc(moo, (moo_oop_t)ctx->x11_class);
|
|
}
|
|
|
|
int moo_mod_x11 (moo_t* moo, moo_mod_t* mod)
|
|
{
|
|
if (mod->hints & MOO_MOD_LOAD_FOR_IMPORT)
|
|
{
|
|
mod->gc = MOO_NULL;
|
|
mod->ctx = MOO_NULL;
|
|
}
|
|
else
|
|
{
|
|
x11_modctx_t* ctx;
|
|
moo_oop_t tmp;
|
|
|
|
static moo_ooch_t name_x11[] = { 'X','1','1','\0' };
|
|
|
|
ctx = moo_callocmem (moo, MOO_SIZEOF(*ctx));
|
|
if (!ctx) return -1;
|
|
|
|
tmp = moo_findclass(moo, moo->sysdic, name_x11);
|
|
if (!tmp)
|
|
{
|
|
/* Not a single X11.XXX has been defined. */
|
|
MOO_DEBUG0 (moo, "X11 class not found\n");
|
|
moo_freemem (moo, ctx);
|
|
return -1;
|
|
}
|
|
MOO_STORE_OOP (moo, (moo_oop_t*)&ctx->x11_class, tmp);
|
|
|
|
mod->gc = gc_mod_x11;
|
|
mod->ctx = ctx;
|
|
}
|
|
|
|
mod->import = x11_import;
|
|
mod->querypf = x11_querypf;
|
|
mod->querypv = MOO_NULL;
|
|
mod->unload = x11_unload;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------ */
|