Mercurial > hg > xemacs-beta
diff src/ExternalClient-Xlib.c @ 0:376386a54a3c r19-14
Import from CVS: tag r19-14
author | cvs |
---|---|
date | Mon, 13 Aug 2007 08:45:50 +0200 |
parents | |
children | 8de911beca70 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/ExternalClient-Xlib.c Mon Aug 13 08:45:50 2007 +0200 @@ -0,0 +1,147 @@ +/* External client, raw Xlib version. + Copyright (C) 1993, 1994 Sun Microsystems, Inc. + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Library General Public +License as published by the Free Software Foundation; either +version 2 of the License, or (at your option) any later version. + +This library 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 +Library General Public License for more details. + +You should have received a copy of the GNU Library General Public +License along with this library; if not, write to +the Free Software Foundation, Inc., 59 Temple Place - Suite 330, +Boston, MA 02111-1307, USA. */ + +/* Synched up with: Not in FSF. */ + +/* Written by Ben Wing, February 1994. */ + +#include <X11/Xlib.h> +#include <X11/Xresource.h> +#include <X11/Xutil.h> +#include "extw-Xlib.h" + +/* this is not a perfect solution, but otherwise we have to include all + of the Xt junk */ + +#define XtGeometryNo 1 + +#if (XlibSpecificationRelease < 5) +# define XPointer char * +#endif + +static int context_inited; +static XContext focus_context; + +/* does the specified window have the focus, given that the pointer just + entered (or left) the window (according to enter_p)? This question + does not have an obvious answer in X. (Basically, X sucks.) */ + +static int +window_has_focus_p (Display *display, Window win, int enter_p) +{ + Window focuswin; + int dummy; + + XGetInputFocus(display, &focuswin, &dummy); + if (focuswin == PointerRoot) + return enter_p; + if (focuswin == win) + return True; + if (!enter_p) + return False; + do + { + Status st; + Window root_win, parent_win; + Window *child_win; + int nchild; + + st = XQueryTree(display, win, &root_win, &parent_win, &child_win, + &nchild); + if (!st) + return False; + XFree((XPointer)child_win); + if (parent_win == focuswin) + return True; + if (parent_win == root_win) + return False; + win = parent_win; + } + while (1); +} + + +/* External entry points when using XLib directly */ + +void ExternalClientInitialize (Display *display, Window win); +void +ExternalClientInitialize (Display *display, Window win) +{ + extw_initialize_atoms(display); + extw_which_side = extw_client_send; + if (!context_inited) + { + focus_context = XUniqueContext(); + context_inited = 1; + } + XSaveContext(display, win, focus_context, 0); + XSelectInput(display, win, EnterWindowMask | LeaveWindowMask | + FocusChangeMask); +} + +void ExternalClientEventHandler (Display *display, Window win, XEvent *event); +void +ExternalClientEventHandler (Display *display, Window win, XEvent *event) +{ + if (win != event->xany.window) + return; + + if (event->type == ClientMessage && + event->xclient.message_type == a_EXTW_NOTIFY && + event->xclient.data.l[0] == extw_shell_send) + switch (event->xclient.data.l[1]) { + case extw_notify_gm: + /* for the moment, just refuse geometry requests. */ + extw_send_notify_3(display, win, extw_notify_gm, XtGeometryNo, 0, 0); + break; + + case extw_notify_init: + extw_send_notify_3(display, win, extw_notify_init, EXTW_TYPE_XLIB, 0, 0); + break; + + case extw_notify_end: + XClearArea(display, win, 0, 0, 0, 0, True); + break; + } + else + { + int focus_status; + XPointer current_focus; + + if (event->type == FocusIn) + focus_status = 1; + else if (event->type == FocusOut) + focus_status = 0; + else if (event->type == EnterNotify && + event->xcrossing.detail != NotifyInferior) + focus_status = window_has_focus_p(display, win, 1); + else if (event->type == LeaveNotify && + event->xcrossing.detail != NotifyInferior) + focus_status = window_has_focus_p(display, win, 0); + else + return; + XFindContext(display, win, focus_context, ¤t_focus); + if (focus_status != (int) current_focus) + { + XSaveContext(display, win, focus_context, (XPointer) focus_status); + extw_send_notify_3(display, win, focus_status ? + extw_notify_focus_in : extw_notify_focus_out, + 0, 0, 0); + } + } +}