/* Part of XPCE --- The SWI-Prolog GUI toolkit Author: Jan Wielemaker and Anjo Anjewierden E-mail: jan@swi.psy.uva.nl WWW: http://www.swi.psy.uva.nl/projects/xpce/ Copyright (c) 1985-2002, University of Amsterdam 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 COPYRIGHT HOLDERS AND CONTRIBUTORS "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 COPYRIGHT OWNER OR CONTRIBUTORS 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 #include #include "include.h" #include "canvas.h" #ifdef O_XDND #include "xdnd.h" #endif static void event_window(Widget w, XtPointer xsw, XtPointer xevent); static void expose_window(Widget w, XtPointer xsw, XtPointer xregion); static void resize_window(Widget w, XtPointer xsw, XtPointer data); static void destroy_window(Widget w, XtPointer xsw, XtPointer data); /******************************* * WIDGET REFERENCE * *******************************/ static status setWidgetWindow(PceWindow sw, Widget w) { sw->ws_ref = (WsRef) w; if ( !w ) assign(sw, displayed, OFF); succeed; } /******************************** * (UN)CREATE * ********************************/ status ws_created_window(PceWindow sw) { if ( widgetWindow(sw) != NULL ) succeed; fail; } void ws_uncreate_window(PceWindow sw) { Widget w; if ( grabbedWindows ) deleteChain(grabbedWindows, sw); if ( (w=widgetWindow(sw)) ) { XtRemoveAllCallbacks(w, XtNeventCallback); XtRemoveAllCallbacks(w, XtNexposeCallback); XtRemoveAllCallbacks(w, XtNresizeCallback); XtRemoveAllCallbacks(w, XtNdestroyCallback); destroy_window(w, (XtPointer)sw, NULL); /* callback may be delayed */ XtDestroyWidget(w); /* too long */ } } status ws_create_window(PceWindow sw, PceWindow parent) { Widget w; DisplayObj d = getDisplayGraphical((Graphical)sw); /* create the widget */ { Arg args[8]; Cardinal n = 0; int pen = valInt(sw->pen); /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Actually, it appears Xt is ignoring the geometry parameters. Hence, ws_realise_frame() sends ->geometry to all windows. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ XtSetArg(args[n], XtNx, valInt(sw->area->x)); n++; XtSetArg(args[n], XtNy, valInt(sw->area->y)); n++; XtSetArg(args[n], XtNwidth, valInt(sw->area->w) - 2*pen); n++; XtSetArg(args[n], XtNheight, valInt(sw->area->h) - 2*pen); n++; XtSetArg(args[n], XtNborderWidth, pen); n++; XtSetArg(args[n], XtNinput, True); n++; if ( instanceOfObject(sw->background, ClassColour) ) { XtSetArg(args[n], XtNbackground, getPixelColour(sw->background, d)); n++; } else { Pixmap pm = (Pixmap) getXrefObject(sw->background, d); XtSetArg(args[n], XtNbackgroundPixmap, pm); n++; } DEBUG(NAME_create, Cprintf("Calling XtCreateWidget ...")); w = XtCreateWidget(strName(sw->name), canvasWidgetClass, isDefault(parent) ? widgetFrame(sw->frame) : widgetWindow(parent), args, n); DEBUG(NAME_create, Cprintf("Widget = %p\n", w)); } if ( !w ) return errorPce(w, NAME_createFailed); setWidgetWindow(sw, w); XtAddCallback(w, XtNeventCallback, event_window, sw); XtAddCallback(w, XtNexposeCallback, expose_window, sw); XtAddCallback(w, XtNresizeCallback, resize_window, sw); XtAddCallback(w, XtNdestroyCallback, destroy_window, sw); if ( notDefault(parent) ) /* make a sub-window */ { XtManageChild(w); send(sw, NAME_displayed, ON, EAV); } succeed; } void ws_manage_window(PceWindow sw) { XtManageChild(widgetWindow(sw)); } void ws_unmanage_window(PceWindow sw) { XtUnmanageChild(widgetWindow(sw)); } /******************************* * DECORATE SUPPORT * *******************************/ /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Reassocicate the WS window of `from' with `to'. Used by ->decorate. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ void ws_reassociate_ws_window(PceWindow from, PceWindow to) { Widget w = widgetWindow(from); if ( w ) { XtRemoveAllCallbacks(w, XtNeventCallback); XtRemoveAllCallbacks(w, XtNexposeCallback); XtRemoveAllCallbacks(w, XtNresizeCallback); setWidgetWindow(from, NULL); setWidgetWindow(to, w); XtAddCallback(w, XtNeventCallback, event_window, to); XtAddCallback(w, XtNexposeCallback, expose_window, to); XtAddCallback(w, XtNresizeCallback, resize_window, to); } } /******************************* * GEOMETRY * *******************************/ /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Define the size and thickness of outside border of the window. The described area is the *outline* of the window *including* its border. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ void ws_geometry_window(PceWindow sw, int x, int y, int w, int h, int pen) { Widget wid = widgetWindow(sw); w -= 2*pen; h -= 2*pen; if ( w < 1 ) w = 1; /* avoid X-errors */ if ( h < 1 ) h = 1; if ( wid ) { DEBUG(NAME_tile, Cprintf("ws_geometry_window(%s, %d, %d, %d, %d, %d)\n", pp(sw), x, y, w, h, pen)); XtConfigureWidget(wid, x, y, w, h, pen); } } /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Used to get the tile_adjuster windows on top of the other windows in a frame. Sofar not needed in the X11 version. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ void ws_topmost_window(PceWindow sw, BoolObj topmost) { } /******************************** * XT CALLBACK HANDLING * ********************************/ static void x_event_window(PceWindow sw, XEvent *event) { EventObj ev; FrameObj fr = getFrameWindow(sw, OFF); FrameObj bfr; Any receiver = sw; #ifdef O_XDND if ( event->xany.type == MapNotify ) { if ( hasSendMethodObject(sw, NAME_dropFiles) ) setDndAwareFrame(fr); } #endif /*O_XDND*/ if ( fr && (bfr=blockedByModalFrame(fr)) ) { switch( event->xany.type ) { case KeyPress: { receiver = bfr; break; } case ButtonRelease: send(fr, NAME_bell, EAV); case ButtonPress: send(bfr, NAME_expose, EAV); default: return; } } if ( (ev = CtoEvent(sw, event)) ) { addCodeReference(ev); postNamedEvent(ev, receiver, DEFAULT, NAME_postEvent); delCodeReference(ev); freeableObj(ev); RedrawDisplayManager(TheDisplayManager()); /* optional? */ } } static void event_window(Widget w, XtPointer xsw, XtPointer xevent) { PceWindow sw = (PceWindow) xsw; XEvent *event = (XEvent *)xevent; pceMTLock(LOCK_PCE); DEBUG(NAME_event, Cprintf("event_window(): X-event %d on %s\n", event->xany.type, pp(sw))); if ( isFreeingObj(sw) || isFreedObj(sw) || sw->sensitive == OFF ) { pceMTUnlock(LOCK_PCE); return; } ServiceMode(is_service_window(sw), { AnswerMark mark; markAnswerStack(mark); x_event_window(sw, event); rewindAnswerStack(mark, NIL); }); pceMTUnlock(LOCK_PCE); } static void expose_window(Widget w, XtPointer xsw, XtPointer xregion) { XRectangle rect; Area a; PceWindow sw = (PceWindow) xsw; Region region = (Region) xregion; Window win; pceMTLock(LOCK_PCE); DEBUG(NAME_window, Cprintf("Window %ld ---> %s\n", XtWindow(w), pp(sw))); if ( !getMemberHashTable(WindowTable, (Any) (win=XtWindow(w))) ) appendHashTable(WindowTable, (Any) win, sw); XClipBox(region, &rect); ServiceMode(is_service_window(sw), a = tempObject(ClassArea, toInt(rect.x), toInt(rect.y), toInt(rect.width), toInt(rect.height), EAV); redrawWindow(sw, a); considerPreserveObject(a)); pceMTUnlock(LOCK_PCE); } static void resize_window(Widget w, XtPointer xsw, XtPointer data) { PceWindow sw; Area a; Int ow, oh; pceMTLock(LOCK_PCE); sw = (PceWindow) xsw; a = sw->area; ow = a->w; oh = a->h; ServiceMode(is_service_window(sw), qadSendv(sw, NAME_resize, 0, NULL); changedUnionWindow(sw, a->x, a->y, ow, oh)); pceMTUnlock(LOCK_PCE); } static void destroy_window(Widget w, XtPointer xsw, XtPointer data) { PceWindow sw = (PceWindow) xsw; DEBUG(NAME_window, Cprintf("destroy_window(%s)\n", pp(sw))); deleteHashTable(WindowTable, (Any) XtWindow(w)); setWidgetWindow(sw, NULL); } int /* MT: see msw/mswindow.c */ ws_delayed_redraw_window(PceWindow sw) { fail; } Int ws_window_thread(PceWindow sw) { fail; /* for now */ } /******************************** * GRAB POINTER/KEYBOARD * ********************************/ static status do_grab_window(PceWindow sw) { if ( widgetWindow(sw) != NULL ) { int rval; char *msg = NULL; rval = XtGrabPointer(widgetWindow(sw), False, ButtonPressMask|ButtonReleaseMask| EnterWindowMask|LeaveWindowMask| PointerMotionMask|ButtonMotionMask, GrabModeAsync, GrabModeAsync, None, None, CurrentTime); switch(rval) { case GrabNotViewable: msg = "not viewable"; break; case AlreadyGrabbed: msg = "already grabbed"; break; case GrabFrozen: msg = "grab frozen"; break; case GrabInvalidTime: msg = "invalid time"; break; } if ( msg ) return errorPce(sw, NAME_cannotGrabPointer, CtoName(msg)); succeed; } fail; } void ws_grab_pointer_window(PceWindow sw, BoolObj val) { if ( widgetWindow(sw) != NULL ) { if ( val == ON ) { if ( getHeadChain(grabbedWindows) != sw ) { do_grab_window(sw); prependChain(grabbedWindows, sw); } } else { XtUngrabPointer(widgetWindow(sw), CurrentTime); flushWindow(sw); deleteChain(grabbedWindows, sw); if ( notNil(grabbedWindows->head) ) do_grab_window(grabbedWindows->head->value); } } } void ws_grab_keyboard_window(PceWindow sw, BoolObj val) { if ( widgetWindow(sw) != NULL ) { if ( val == ON ) XtGrabKeyboard(widgetWindow(sw), True, /* owner_events */ GrabModeAsync, /* pointer mode */ GrabModeAsync, /* keyboard mode */ CurrentTime); else XtUngrabKeyboard(widgetWindow(sw), CurrentTime); } } void ws_ungrab_all() { if ( grabbedWindows ) { if ( notNil(grabbedWindows->tail) ) { PceWindow sw = grabbedWindows->tail->value; if ( widgetWindow(sw) ) { XtUngrabPointer(widgetWindow(sw), CurrentTime); flushWindow(sw); } } clearChain(grabbedWindows); } } void ws_flash_area_window(PceWindow sw, int x, int y, int w, int h, int msecs) { if ( sw->displayed == ON ) { int ox, oy, dw, dh; int maxrect = 100; compute_window(sw, &ox, &oy, &dw, &dh); ox += valInt(sw->scroll_offset->x); oy += valInt(sw->scroll_offset->y); d_offset(ox, oy); if ( w > maxrect ) { x += (w-maxrect)/2; w = maxrect; } if ( h > maxrect ) { y += (h-maxrect)/2; h = maxrect; } if ( d_window(sw, x, y, w, h, FALSE, TRUE) ) { r_complement(x, y, w, h); d_flush(); msleep(msecs); r_complement(x, y, w, h); d_flush(); d_done(); } } } void ws_flash_window(PceWindow sw, int msecs) { if ( sw->displayed == ON ) { int w = valInt(sw->area->w); int h = valInt(sw->area->h); int x = 0, y = 0; int maxrect = 100; if ( w > maxrect ) { x = (w-maxrect)/2; w = maxrect; } if ( h > maxrect ) { y = (h-maxrect)/2; h = maxrect; } d_offset(0, 0); if ( d_window(sw, x, y, w, h, FALSE, FALSE) ) { r_complement(x, y, w, h); d_flush(); msleep(msecs); r_complement(x, y, w, h); d_flush(); d_done(); } } } /******************************* * POINTER * *******************************/ void ws_move_pointer(PceWindow sw, int x, int y) { DisplayObj d = getDisplayGraphical((Graphical)sw); DisplayWsXref r = d->ws_ref; XWarpPointer(r->display_xref, None, XtWindow(widgetWindow(sw)), 0, 0, 0, 0, x, y); } void ws_window_cursor(PceWindow sw, CursorObj cursor) { DisplayObj d = getDisplayGraphical((Graphical)sw); DisplayWsXref r = d->ws_ref; XDefineCursor(r->display_xref, XtWindow(widgetWindow(sw)), isNil(cursor) ? None : (Cursor) getXrefObject(cursor, d)); } /******************************* * COLOURS * *******************************/ void ws_window_background(PceWindow sw, Any c) { Widget w = widgetWindow(sw); if ( w ) { Arg args[2]; DisplayObj d = getDisplayGraphical((Graphical)sw); int i=0; if ( instanceOfObject(c, ClassColour) ) { XtSetArg(args[i], XtNbackground, getPixelColour(c, d)); i++; XtSetArg(args[i], XtNbackgroundPixmap, XtUnspecifiedPixmap); i++; } else { Pixmap pm = (Pixmap) getXrefObject(c, d); XtSetArg(args[i], XtNbackgroundPixmap, pm); i++; } XtSetValues(w, args, i); } } /******************************* * HIDE/EXPOSE * *******************************/ void ws_raise_window(PceWindow sw) { DisplayObj d = getDisplayGraphical((Graphical)sw); DisplayWsXref r = d->ws_ref; Widget w = widgetWindow(sw); if ( w ) XRaiseWindow(r->display_xref, XtWindow(w)); } void ws_lower_window(PceWindow sw) { DisplayObj d = getDisplayGraphical((Graphical)sw); DisplayWsXref r = d->ws_ref; Widget w = widgetWindow(sw); if ( w ) XLowerWindow(r->display_xref, XtWindow(w)); } /******************************* * INPUT * *******************************/ /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Enable/disable event processing on the window. I guess this must be done by modifying the event-mask. See XCreateWindow() and XChangeWindowAttributes() - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ int ws_enable_window(PceWindow sw, int enable) { Widget w; if ( (w = widgetWindow(sw)) ) { Arg args[1]; XtSetArg(args[0], XtNinput, enable ? True : False); XtSetValues(w, args, 1); succeed; } fail; }