0
|
1 /* Device functions for X windows.
|
|
2 Copyright (C) 1994, 1995 Board of Trustees, University of Illinois.
|
|
3 Copyright (C) 1994, 1995 Free Software Foundation, Inc.
|
|
4
|
|
5 This file is part of XEmacs.
|
|
6
|
|
7 XEmacs is free software; you can redistribute it and/or modify it
|
|
8 under the terms of the GNU General Public License as published by the
|
|
9 Free Software Foundation; either version 2, or (at your option) any
|
|
10 later version.
|
|
11
|
|
12 XEmacs is distributed in the hope that it will be useful, but WITHOUT
|
|
13 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
15 for more details.
|
|
16
|
|
17 You should have received a copy of the GNU General Public License
|
|
18 along with XEmacs; see the file COPYING. If not, write to
|
|
19 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
20 Boston, MA 02111-1307, USA. */
|
|
21
|
|
22 /* Synched up with: Not in FSF. */
|
|
23
|
|
24 /* Original authors: Jamie Zawinski and the FSF */
|
|
25 /* Rewritten by Ben Wing and Chuck Thompson. */
|
|
26
|
|
27 #include <config.h>
|
|
28 #include "lisp.h"
|
|
29
|
|
30 #include "console-x.h"
|
|
31 #include "xintrinsicp.h" /* CoreP.h needs this */
|
|
32 #include <X11/CoreP.h> /* Numerous places access the fields of
|
|
33 a core widget directly. We could
|
165
|
34 use XtGetValues(), but ... */
|
0
|
35 #include "xgccache.h"
|
|
36 #include <X11/Shell.h>
|
|
37 #include "xmu.h"
|
|
38 #include "glyphs-x.h"
|
|
39 #include "objects-x.h"
|
|
40
|
|
41 #include "buffer.h"
|
|
42 #include "events.h"
|
|
43 #include "faces.h"
|
|
44 #include "frame.h"
|
|
45 #include "redisplay.h"
|
|
46 #include "sysdep.h"
|
|
47 #include "window.h"
|
|
48
|
|
49 #include "sysfile.h"
|
|
50 #include "systime.h"
|
|
51
|
197
|
52 #ifdef HAVE_OFFIX_DND
|
|
53 #include "offix.h"
|
|
54 #endif
|
|
55
|
0
|
56 Lisp_Object Vdefault_x_device;
|
227
|
57 #if defined(MULE) && (defined(LWLIB_MENUBARS_MOTIF) || defined(HAVE_XIM) || defined (USE_XFONTSET))
|
|
58 Lisp_Object Vx_app_defaults_directory;
|
|
59 #endif
|
0
|
60
|
|
61 /* Qdisplay in general.c */
|
183
|
62 Lisp_Object Qx_error;
|
0
|
63 Lisp_Object Qinit_pre_x_win, Qinit_post_x_win;
|
|
64
|
167
|
65 /* $B@ZJ"(B, n. Japanese ritual suicide. */
|
|
66 int x_seppuku_on_epipe;
|
|
67
|
0
|
68 /* The application class of Emacs. */
|
|
69 Lisp_Object Vx_emacs_application_class;
|
|
70
|
|
71 Lisp_Object Vx_initial_argv_list; /* #### ugh! */
|
|
72
|
|
73 static XrmOptionDescRec emacs_options[] =
|
|
74 {
|
183
|
75 {"-geometry", ".geometry", XrmoptionSepArg, NULL},
|
185
|
76 {"-iconic", ".iconic", XrmoptionNoArg, "yes"},
|
2
|
77
|
187
|
78 {"-internal-border-width", "*EmacsFrame.internalBorderWidth", XrmoptionSepArg, NULL},
|
|
79 {"-ib", "*EmacsFrame.internalBorderWidth", XrmoptionSepArg, NULL},
|
|
80 {"-scrollbar-width", "*EmacsFrame.scrollBarWidth", XrmoptionSepArg, NULL},
|
|
81 {"-scrollbar-height", "*EmacsFrame.scrollBarHeight", XrmoptionSepArg, NULL},
|
0
|
82
|
219
|
83 {"-privatecolormap", ".privateColormap", XrmoptionNoArg, "yes"},
|
|
84 {"-visual", ".EmacsVisual", XrmoptionSepArg, NULL},
|
|
85
|
0
|
86 /* #### Beware! If the type of the shell changes, update this. */
|
187
|
87 {"-T", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
|
88 {"-wn", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
|
89 {"-title", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
183
|
90
|
|
91 {"-iconname", "*TopLevelEmacsShell.iconName", XrmoptionSepArg, NULL},
|
187
|
92 {"-in", "*TopLevelEmacsShell.iconName", XrmoptionSepArg, NULL},
|
|
93 {"-mc", "*pointerColor", XrmoptionSepArg, NULL},
|
|
94 {"-cr", "*cursorColor", XrmoptionSepArg, NULL},
|
|
95 {"-fontset", "*FontSet", XrmoptionSepArg, NULL},
|
0
|
96 };
|
|
97
|
|
98 /* Functions to synchronize mirroring resources and specifiers */
|
|
99 int in_resource_setting;
|
|
100 int in_specifier_change_function;
|
|
101
|
|
102
|
|
103 /************************************************************************/
|
|
104 /* helper functions */
|
|
105 /************************************************************************/
|
|
106
|
219
|
107 /* JH 97/11/25 removed the static declaration because I need it during setup in event-Xt... */
|
253
|
108 struct device * get_device_from_display_1 (Display *dpy);
|
219
|
109 struct device *
|
114
|
110 get_device_from_display_1 (Display *dpy)
|
0
|
111 {
|
|
112 Lisp_Object devcons, concons;
|
|
113
|
|
114 DEVICE_LOOP_NO_BREAK (devcons, concons)
|
|
115 {
|
|
116 struct device *d = XDEVICE (XCAR (devcons));
|
|
117 if (DEVICE_X_P (d) && DEVICE_X_DISPLAY (d) == dpy)
|
|
118 return d;
|
|
119 }
|
|
120
|
114
|
121 return 0;
|
|
122 }
|
|
123
|
|
124 struct device *
|
|
125 get_device_from_display (Display *dpy)
|
|
126 {
|
|
127 struct device *d = get_device_from_display_1 (dpy);
|
0
|
128
|
276
|
129 #if !defined(INFODOCK)
|
|
130 # define FALLBACK_RESOURCE_NAME "xemacs"
|
|
131 # else
|
|
132 # define FALLBACK_RESOURCE_NAME "infodock"
|
|
133 #endif
|
|
134
|
114
|
135 if (!d) {
|
|
136 /* This isn't one of our displays. Let's crash? */
|
|
137 stderr_out
|
|
138 ("\n%s: Fatal X Condition. Asked about display we don't own: \"%s\"\n",
|
|
139 (STRINGP (Vinvocation_name) ?
|
276
|
140 (char *) XSTRING_DATA (Vinvocation_name) : FALLBACK_RESOURCE_NAME),
|
114
|
141 DisplayString (dpy) ? DisplayString (dpy) : "???");
|
|
142 abort();
|
|
143 }
|
|
144
|
276
|
145 #undef FALLBACK_RESOURCE_NAME
|
|
146
|
114
|
147 return d;
|
0
|
148 }
|
|
149
|
|
150 struct device *
|
|
151 decode_x_device (Lisp_Object device)
|
|
152 {
|
|
153 XSETDEVICE (device, decode_device (device));
|
|
154 CHECK_X_DEVICE (device);
|
|
155 return XDEVICE (device);
|
|
156 }
|
|
157
|
272
|
158 static Display *
|
0
|
159 get_x_display (Lisp_Object device)
|
|
160 {
|
|
161 return DEVICE_X_DISPLAY (decode_x_device (device));
|
|
162 }
|
|
163
|
|
164
|
|
165 /************************************************************************/
|
|
166 /* initializing an X connection */
|
|
167 /************************************************************************/
|
|
168
|
|
169 static void
|
|
170 allocate_x_device_struct (struct device *d)
|
|
171 {
|
185
|
172 d->device_data = xnew_and_zero (struct x_device);
|
0
|
173 }
|
|
174
|
|
175 static void
|
|
176 Xatoms_of_device_x (struct device *d)
|
|
177 {
|
149
|
178 Display *D = DEVICE_X_DISPLAY (d);
|
0
|
179
|
149
|
180 DEVICE_XATOM_WM_PROTOCOLS (d) = XInternAtom (D, "WM_PROTOCOLS", False);
|
|
181 DEVICE_XATOM_WM_DELETE_WINDOW(d) = XInternAtom (D, "WM_DELETE_WINDOW",False);
|
|
182 DEVICE_XATOM_WM_SAVE_YOURSELF(d) = XInternAtom (D, "WM_SAVE_YOURSELF",False);
|
|
183 DEVICE_XATOM_WM_TAKE_FOCUS (d) = XInternAtom (D, "WM_TAKE_FOCUS", False);
|
|
184 DEVICE_XATOM_WM_STATE (d) = XInternAtom (D, "WM_STATE", False);
|
0
|
185 }
|
|
186
|
|
187 static void
|
|
188 sanity_check_geometry_resource (Display *dpy)
|
|
189 {
|
|
190 char *app_name, *app_class, *s;
|
|
191 char buf1 [255], buf2 [255];
|
|
192 char *type;
|
|
193 XrmValue value;
|
|
194 XtGetApplicationNameAndClass (dpy, &app_name, &app_class);
|
|
195 strcpy (buf1, app_name);
|
|
196 strcpy (buf2, app_class);
|
|
197 for (s = buf1; *s; s++) if (*s == '.') *s = '_';
|
|
198 strcat (buf1, "._no_._such_._resource_.geometry");
|
|
199 strcat (buf2, "._no_._such_._resource_.Geometry");
|
|
200 if (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value) == True)
|
|
201 {
|
|
202 warn_when_safe (Qgeometry, Qerror,
|
|
203 "\n"
|
|
204 "Apparently \"%s*geometry: %s\" or \"%s*geometry: %s\" was\n"
|
|
205 "specified in the resource database. Specifying \"*geometry\" will make\n"
|
|
206 "XEmacs (and most other X programs) malfunction in obscure ways. (i.e.\n"
|
|
207 "the Xt or Xm libraries will probably crash, which is a very bad thing.)\n"
|
|
208 "You should always use \".geometry\" or \"*EmacsFrame.geometry\" instead.\n",
|
|
209 app_name, (char *) value.addr,
|
|
210 app_class, (char *) value.addr);
|
272
|
211 suppress_early_error_handler_backtrace = 1;
|
0
|
212 error ("Invalid geometry resource");
|
|
213 }
|
|
214 }
|
|
215
|
|
216 static void
|
|
217 x_init_device_class (struct device *d)
|
|
218 {
|
219
|
219 if (DEVICE_X_DEPTH(d) > 2)
|
0
|
220 {
|
219
|
221 switch (DEVICE_X_VISUAL(d)->class)
|
0
|
222 {
|
|
223 case StaticGray:
|
|
224 case GrayScale:
|
|
225 DEVICE_CLASS (d) = Qgrayscale;
|
|
226 break;
|
|
227 default:
|
|
228 DEVICE_CLASS (d) = Qcolor;
|
|
229 }
|
|
230 }
|
|
231 else
|
|
232 DEVICE_CLASS (d) = Qmono;
|
|
233 }
|
|
234
|
276
|
235 /*
|
|
236 * Figure out what application name to use for xemacs
|
|
237 *
|
|
238 * Since we have decomposed XtOpenDisplay into XOpenDisplay and
|
|
239 * XtDisplayInitialize, we no longer get this for free.
|
|
240 *
|
|
241 * If there is a `-name' argument in argv, use that.
|
|
242 * Otherwise use the last component of argv[0].
|
|
243 *
|
|
244 * I have removed the gratuitous use of getenv("RESOURCE_NAME")
|
|
245 * which was in X11R5, but left the matching of any prefix of `-name'.
|
|
246 * Finally, if all else fails, return `xemacs', as it is more
|
|
247 * appropriate (X11R5 returns `main').
|
|
248 */
|
|
249 static char *
|
|
250 compute_x_app_name (int argc, char **argv)
|
259
|
251 {
|
276
|
252 int i;
|
|
253 char *ptr;
|
259
|
254
|
276
|
255 for (i = 1; i < argc - 1; i++)
|
|
256 if (!strncmp(argv[i], "-name", max (2, strlen (argv[1]))))
|
|
257 return argv[i+1];
|
|
258
|
|
259 if (argc > 0 && argv[0] && *argv[0])
|
|
260 return (ptr = strrchr (argv[0], '/')) ? ++ptr : argv[0];
|
|
261
|
|
262 return "xemacs";
|
|
263 }
|
|
264
|
|
265 /*
|
|
266 * This function figures out whether the user has any resources of the
|
|
267 * form "XEmacs.foo" or "XEmacs*foo".
|
|
268 *
|
|
269 * Currently we only consult the display's global resources; to look
|
|
270 * for screen specific resources, we would need to also consult:
|
|
271 * xdefs = XScreenResourceString(ScreenOfDisplay(dpy, scrno));
|
|
272 */
|
|
273 static int
|
|
274 have_xemacs_resources_in_xrdb (Display *dpy)
|
|
275 {
|
|
276 char *xdefs, *key;
|
|
277 int len;
|
259
|
278
|
|
279 key = "XEmacs";
|
276
|
280 len = strlen (key);
|
259
|
281
|
276
|
282 if (!dpy)
|
|
283 return 0;
|
272
|
284
|
276
|
285 xdefs = XResourceManagerString (dpy); /* don't free - owned by X */
|
|
286 while (xdefs && *xdefs)
|
|
287 {
|
|
288 if (strncmp (xdefs, key, len) == 0 &&
|
|
289 (xdefs[len] == '*' || xdefs[len] == '.'))
|
|
290 return 1;
|
|
291
|
|
292 while (*xdefs && *xdefs++ != '\n') /* find start of next entry.. */
|
|
293 ;
|
259
|
294 }
|
|
295
|
276
|
296 return 0;
|
259
|
297 }
|
|
298
|
272
|
299 /* Only the characters [-_A-Za-z0-9] are allowed in the individual
|
|
300 components of a resource. Convert invalid characters to `-' */
|
|
301
|
|
302 static char valid_resource_char_p[256];
|
|
303
|
|
304 static void
|
|
305 validify_resource_component (char *str, size_t len)
|
|
306 {
|
|
307 for (; len; len--, str++)
|
|
308 if (!valid_resource_char_p[(unsigned char) (*str)])
|
|
309 *str = '-';
|
|
310 }
|
|
311
|
|
312 static void
|
|
313 Dynarr_add_validified_lisp_string (char_dynarr *cda, Lisp_Object str)
|
|
314 {
|
|
315 Bytecount len = XSTRING_LENGTH (str);
|
|
316 Dynarr_add_many (cda, (char *) XSTRING_DATA (str), len);
|
|
317 validify_resource_component (Dynarr_atp (cda, Dynarr_length (cda) - len), len);
|
|
318 }
|
|
319
|
0
|
320 static void
|
|
321 x_init_device (struct device *d, Lisp_Object props)
|
|
322 {
|
|
323 Lisp_Object display;
|
|
324 Lisp_Object device;
|
|
325 Display *dpy;
|
219
|
326 Widget app_shell;
|
0
|
327 int argc;
|
|
328 char **argv;
|
272
|
329 CONST char *app_class;
|
|
330 CONST char *app_name;
|
0
|
331 CONST char *disp_name;
|
219
|
332 Arg xargs[6];
|
|
333 Cardinal numargs;
|
|
334 Visual *visual = NULL;
|
|
335 int depth = 8; /* shut up the compiler */
|
|
336 Colormap cmap;
|
|
337 int screen;
|
0
|
338
|
|
339 XSETDEVICE (device, d);
|
|
340 display = DEVICE_CONNECTION (d);
|
|
341
|
|
342 allocate_x_device_struct (d);
|
|
343
|
|
344 make_argc_argv (Vx_initial_argv_list, &argc, &argv);
|
|
345
|
259
|
346 GET_C_STRING_CTEXT_DATA_ALLOCA (display, disp_name);
|
|
347
|
276
|
348 /*
|
|
349 * Break apart the old XtOpenDisplay call into XOpenDisplay and
|
|
350 * XtDisplayInitialize so we can figure out whether there
|
|
351 * are any XEmacs resources in the resource database before
|
|
352 * we intitialize Xt. This is so we can automagically support
|
|
353 * both `Emacs' and `XEmacs' application classes.
|
|
354 */
|
0
|
355 slow_down_interrupts ();
|
276
|
356 /* May not be needed but XtOpenDisplay could not deal with signals here. */
|
|
357 dpy = DEVICE_X_DISPLAY (d) = XOpenDisplay (disp_name);
|
0
|
358 speed_up_interrupts ();
|
|
359
|
|
360 if (dpy == 0)
|
|
361 {
|
272
|
362 suppress_early_error_handler_backtrace = 1;
|
0
|
363 signal_simple_error ("X server not responding\n", display);
|
|
364 }
|
|
365
|
276
|
366 if (STRINGP (Vx_emacs_application_class) &&
|
|
367 XSTRING_LENGTH (Vx_emacs_application_class) > 0)
|
|
368 GET_C_STRING_CTEXT_DATA_ALLOCA (Vx_emacs_application_class, app_class);
|
|
369 else
|
|
370 {
|
|
371 app_class = (NILP (Vx_emacs_application_class) &&
|
|
372 have_xemacs_resources_in_xrdb (dpy))
|
|
373 ? "XEmacs"
|
|
374 : "Emacs";
|
|
375 /* need to update Vx_emacs_application_class: */
|
|
376 Vx_emacs_application_class = build_string (app_class);
|
|
377 }
|
|
378
|
|
379 slow_down_interrupts ();
|
|
380 /* May not be needed but XtOpenDisplay could not deal with signals here.
|
|
381 Yuck. */
|
|
382 XtDisplayInitialize (Xt_app_con, dpy, compute_x_app_name (argc, argv),
|
|
383 app_class, emacs_options,
|
|
384 XtNumber (emacs_options), &argc, argv);
|
|
385 speed_up_interrupts ();
|
|
386
|
219
|
387 screen = DefaultScreen(dpy);
|
74
|
388 if (NILP (Vdefault_x_device))
|
|
389 Vdefault_x_device = device;
|
|
390
|
70
|
391 #ifdef MULE
|
209
|
392 #if defined(LWLIB_MENUBARS_MOTIF) || defined(HAVE_XIM) || defined (USE_XFONTSET)
|
70
|
393 {
|
|
394 /* Read in locale-specific resources from
|
263
|
395 data-directory/app-defaults/$LANG/Emacs.
|
70
|
396 This is in addition to the standard app-defaults files, and
|
|
397 does not override resources defined elsewhere */
|
|
398 CONST char *data_dir;
|
267
|
399 char *path;
|
70
|
400 XrmDatabase db = XtDatabase (dpy); /* ### XtScreenDatabase(dpy) ? */
|
|
401 CONST char *locale = XrmLocaleOfDatabase (db);
|
183
|
402
|
227
|
403 if (STRINGP (Vx_app_defaults_directory) &&
|
|
404 XSTRING_LENGTH (Vx_app_defaults_directory) > 0)
|
|
405 {
|
|
406 GET_C_STRING_FILENAME_DATA_ALLOCA(Vx_app_defaults_directory, data_dir);
|
267
|
407 path = (char *)alloca (strlen (data_dir) + strlen (locale) + 7);
|
263
|
408 sprintf (path, "%s%s/Emacs", data_dir, locale);
|
227
|
409 if (!access (path, R_OK))
|
|
410 XrmCombineFileDatabase (path, &db, False);
|
|
411 }
|
|
412 else if (STRINGP (Vdata_directory) && XSTRING_LENGTH (Vdata_directory) > 0)
|
169
|
413 {
|
|
414 GET_C_STRING_FILENAME_DATA_ALLOCA (Vdata_directory, data_dir);
|
267
|
415 path = (char *)alloca (strlen (data_dir) + 13 + strlen (locale) + 7);
|
263
|
416 sprintf (path, "%sapp-defaults/%s/Emacs", data_dir, locale);
|
169
|
417 if (!access (path, R_OK))
|
|
418 XrmCombineFileDatabase (path, &db, False);
|
|
419 }
|
227
|
420 }
|
209
|
421 #endif /* LWLIB_MENUBARS_MOTIF or HAVE_XIM USE_XFONTSET */
|
149
|
422 #endif /* MULE */
|
6
|
423
|
0
|
424 if (NILP (DEVICE_NAME (d)))
|
|
425 DEVICE_NAME (d) = display;
|
|
426
|
|
427 /* We're going to modify the string in-place, so be a nice XEmacs */
|
|
428 DEVICE_NAME (d) = Fcopy_sequence (DEVICE_NAME (d));
|
|
429 /* colons and periods can't appear in individual elements of resource
|
|
430 strings */
|
219
|
431
|
272
|
432 XtGetApplicationNameAndClass (dpy, (char **) &app_name, (char **) &app_class);
|
219
|
433 /* search for a matching visual if requested by the user, or setup the display default */
|
|
434 numargs = 0;
|
|
435 {
|
267
|
436 char *buf1 = (char *)alloca (strlen (app_name) + 17);
|
|
437 char *buf2 = (char *)alloca (strlen (app_class) + 17);
|
219
|
438 char *type;
|
|
439 XrmValue value;
|
|
440
|
|
441 sprintf (buf1, "%s.emacsVisual", app_name);
|
|
442 sprintf (buf2, "%s.EmacsVisual", app_class);
|
|
443 if (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value) == True)
|
|
444 {
|
|
445 int cnt = 0, vis_class= PseudoColor;
|
|
446 XVisualInfo vinfo;
|
|
447 char *res, *str = (char*)value.addr;
|
|
448
|
|
449 if (strncmp(str, "StaticGray", 10) == 0) cnt = 10, vis_class = StaticGray;
|
|
450 else if (strncmp(str, "StaticColor", 11) == 0) cnt = 11, vis_class = StaticColor;
|
|
451 else if (strncmp(str, "TrueColor", 9) == 0) cnt = 9, vis_class = TrueColor;
|
|
452 else if (strncmp(str, "GrayScale", 9) == 0) cnt = 9, vis_class = GrayScale;
|
|
453 else if (strncmp(str, "PseudoColor", 11) == 0) cnt = 11, vis_class = PseudoColor;
|
|
454 else if (strncmp(str, "DirectColor", 11) == 0) cnt = 11, vis_class = DirectColor;
|
|
455 if (cnt)
|
|
456 {
|
|
457 res = str + cnt;
|
|
458 depth = atoi(res);
|
|
459 if (depth == 0)
|
|
460 {
|
|
461 stderr_out("Invalid Depth specification in %s... ignoring...\n",(char*)str);
|
|
462 }
|
|
463 else
|
|
464 {
|
|
465 if (XMatchVisualInfo(dpy, screen, depth, vis_class, &vinfo))
|
|
466 {
|
|
467 visual = vinfo.visual;
|
|
468 }
|
|
469 else
|
|
470 {
|
|
471 stderr_out("Can't match the requested visual %s... using defaults\n",str);
|
|
472 }
|
|
473 }
|
|
474 }
|
|
475 else
|
|
476 {
|
|
477 stderr_out("Invalid Visual specification in %s... ignoring.\n",(char*)str);
|
|
478 }
|
|
479 }
|
|
480 if (visual == NULL)
|
|
481 {
|
|
482 visual = DefaultVisual(dpy, screen);
|
|
483 depth = DefaultDepth(dpy, screen);
|
|
484 }
|
|
485
|
|
486 /* If we've got the same visual as the default and its PseudoColor, check to see if the user
|
|
487 specified that we need a private colormap */
|
|
488 if (visual == DefaultVisual(dpy, screen))
|
|
489 {
|
|
490 sprintf (buf1, "%s.privateColormap", app_name);
|
|
491 sprintf (buf2, "%s.PrivateColormap", app_class);
|
|
492 if ((visual->class == PseudoColor) &&
|
|
493 (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value) == True))
|
|
494 {
|
|
495 cmap = XCopyColormapAndFree(dpy, DefaultColormap(dpy, screen));
|
|
496 }
|
|
497 else
|
|
498 {
|
|
499 cmap = DefaultColormap(dpy, screen);
|
|
500 }
|
|
501 }
|
|
502 else
|
|
503 {
|
|
504 /* We have to create a matching colormap anyway...
|
|
505 ### think about using standard colormaps (need the Xmu libs?) */
|
|
506 cmap = XCreateColormap(dpy, RootWindow(dpy, screen), visual, AllocNone);
|
|
507 XInstallColormap(dpy, cmap);
|
|
508 }
|
|
509 }
|
|
510 XtSetArg(xargs[numargs],XtNvisual, visual); numargs++;
|
|
511 XtSetArg(xargs[numargs],XtNdepth, depth); numargs++;
|
|
512 XtSetArg(xargs[numargs],XtNcolormap, cmap); numargs++;
|
|
513 DEVICE_X_VISUAL (d) = visual;
|
|
514 DEVICE_X_COLORMAP (d) = cmap;
|
|
515 DEVICE_X_DEPTH (d) = depth;
|
|
516
|
272
|
517 validify_resource_component ((char *) XSTRING_DATA (DEVICE_NAME (d)),
|
|
518 XSTRING_LENGTH (DEVICE_NAME (d)));
|
219
|
519 app_shell = XtAppCreateShell (NULL, app_class,
|
|
520 applicationShellWidgetClass,
|
|
521 dpy, xargs, numargs);
|
0
|
522
|
219
|
523 DEVICE_XT_APP_SHELL (d) = app_shell;
|
70
|
524 #ifdef HAVE_XIM
|
|
525 XIM_init_device(d);
|
|
526 #endif /* HAVE_XIM */
|
0
|
527
|
272
|
528 /* Realize the app_shell so that its window exists for GC creation purposes,
|
219
|
529 and set it to the size of the root window for child placement purposes */
|
|
530 {
|
|
531 Screen *scrn = ScreenOfDisplay(dpy, screen);
|
|
532 int screen_width, screen_height;
|
|
533 screen_width = WidthOfScreen(scrn);
|
|
534 screen_height = HeightOfScreen(scrn);
|
|
535 numargs = 0;
|
|
536 XtSetArg (xargs[numargs], XtNmappedWhenManaged, False); numargs++;
|
|
537 XtSetArg (xargs[numargs], XtNx, 0); numargs++;
|
|
538 XtSetArg (xargs[numargs], XtNy, 0); numargs++;
|
|
539 XtSetArg (xargs[numargs], XtNwidth, screen_width); numargs++;
|
|
540 XtSetArg (xargs[numargs], XtNheight, screen_height); numargs++;
|
|
541 XtSetValues (app_shell, xargs, numargs);
|
|
542 XtRealizeWidget (app_shell);
|
|
543 }
|
179
|
544 #ifdef HAVE_SESSION
|
177
|
545 {
|
219
|
546 int new_argc;
|
|
547 char **new_argv;
|
|
548 make_argc_argv (Vcommand_line_args, &new_argc, &new_argv);
|
|
549 XSetCommand (XtDisplay (app_shell), XtWindow (app_shell), new_argv, new_argc);
|
|
550 free_argc_argv (new_argv);
|
177
|
551 }
|
179
|
552 #endif /* HAVE_SESSION */
|
177
|
553
|
219
|
554
|
197
|
555 #ifdef HAVE_OFFIX_DND
|
219
|
556 DndInitialize ( app_shell );
|
197
|
557 #endif
|
|
558
|
0
|
559 Vx_initial_argv_list = make_arg_list (argc, argv);
|
|
560 free_argc_argv (argv);
|
|
561
|
|
562 DEVICE_X_WM_COMMAND_FRAME (d) = Qnil;
|
|
563
|
|
564 sanity_check_geometry_resource (dpy);
|
|
565
|
|
566 /* In event-Xt.c */
|
|
567 x_init_modifier_mapping (d);
|
|
568
|
|
569 DEVICE_INFD (d) = DEVICE_OUTFD (d) = ConnectionNumber (dpy);
|
|
570 init_baud_rate (d);
|
|
571 init_one_device (d);
|
|
572
|
219
|
573 DEVICE_X_GC_CACHE (d) = make_gc_cache (dpy, XtWindow(app_shell));
|
0
|
574 DEVICE_X_GRAY_PIXMAP (d) = None;
|
|
575 Xatoms_of_device_x (d);
|
|
576 Xatoms_of_xselect (d);
|
|
577 Xatoms_of_objects_x (d);
|
|
578 x_init_device_class (d);
|
|
579
|
2
|
580 /* Run the elisp side of the X device initialization. */
|
0
|
581 call0 (Qinit_pre_x_win);
|
|
582 }
|
|
583
|
|
584 static void
|
|
585 x_finish_init_device (struct device *d, Lisp_Object props)
|
|
586 {
|
|
587 call0 (Qinit_post_x_win);
|
|
588 }
|
|
589
|
|
590 static void
|
|
591 x_mark_device (struct device *d, void (*markobj) (Lisp_Object))
|
|
592 {
|
187
|
593 ((markobj) (DEVICE_X_WM_COMMAND_FRAME (d)));
|
|
594 ((markobj) (DEVICE_X_DATA (d)->x_keysym_map_hashtable));
|
0
|
595 }
|
|
596
|
|
597
|
|
598 /************************************************************************/
|
|
599 /* closing an X connection */
|
|
600 /************************************************************************/
|
|
601
|
|
602 static void
|
|
603 free_x_device_struct (struct device *d)
|
|
604 {
|
|
605 xfree (d->device_data);
|
|
606 }
|
|
607
|
|
608 static void
|
|
609 x_delete_device (struct device *d)
|
|
610 {
|
|
611 Lisp_Object device;
|
|
612 Display *display;
|
|
613 #ifdef FREE_CHECKING
|
|
614 extern void (*__free_hook)();
|
|
615 int checking_free;
|
|
616 #endif
|
|
617
|
|
618 XSETDEVICE (device, d);
|
|
619 display = DEVICE_X_DISPLAY (d);
|
|
620
|
|
621 if (display)
|
|
622 {
|
|
623 #ifdef FREE_CHECKING
|
|
624 checking_free = (__free_hook != 0);
|
183
|
625
|
0
|
626 /* Disable strict free checking, to avoid bug in X library */
|
|
627 if (checking_free)
|
|
628 disable_strict_free_check ();
|
|
629 #endif
|
|
630
|
|
631 free_gc_cache (DEVICE_X_GC_CACHE (d));
|
|
632 if (DEVICE_X_DATA (d)->x_modifier_keymap)
|
|
633 XFreeModifiermap (DEVICE_X_DATA (d)->x_modifier_keymap);
|
|
634 if (DEVICE_X_DATA (d)->x_keysym_map)
|
|
635 XFree ((char *) DEVICE_X_DATA (d)->x_keysym_map);
|
|
636
|
|
637 XtCloseDisplay (display);
|
|
638 DEVICE_X_DISPLAY (d) = 0;
|
|
639 #ifdef FREE_CHECKING
|
|
640 if (checking_free)
|
|
641 enable_strict_free_check ();
|
|
642 #endif
|
|
643 }
|
183
|
644
|
0
|
645 if (EQ (device, Vdefault_x_device))
|
|
646 {
|
|
647 Lisp_Object devcons, concons;
|
|
648 /* #### handle deleting last X device */
|
|
649 Vdefault_x_device = Qnil;
|
|
650 DEVICE_LOOP_NO_BREAK (devcons, concons)
|
|
651 {
|
157
|
652 if (DEVICE_X_P (XDEVICE (XCAR (devcons))) &&
|
|
653 !EQ (device, XCAR (devcons)))
|
0
|
654 {
|
|
655 Vdefault_x_device = XCAR (devcons);
|
|
656 goto double_break;
|
|
657 }
|
|
658 }
|
|
659 }
|
|
660 double_break:
|
|
661 free_x_device_struct (d);
|
|
662 }
|
|
663
|
|
664
|
|
665 /************************************************************************/
|
|
666 /* handle X errors */
|
|
667 /************************************************************************/
|
|
668
|
|
669 CONST char *
|
|
670 x_event_name (int event_type)
|
|
671 {
|
272
|
672 static CONST char *events[] =
|
|
673 {
|
|
674 "0: ERROR!",
|
|
675 "1: REPLY",
|
|
676 "KeyPress",
|
|
677 "KeyRelease",
|
|
678 "ButtonPress",
|
|
679 "ButtonRelease",
|
|
680 "MotionNotify",
|
|
681 "EnterNotify",
|
|
682 "LeaveNotify",
|
|
683 "FocusIn",
|
|
684 "FocusOut",
|
|
685 "KeymapNotify",
|
|
686 "Expose",
|
|
687 "GraphicsExpose",
|
|
688 "NoExpose",
|
|
689 "VisibilityNotify",
|
|
690 "CreateNotify",
|
|
691 "DestroyNotify",
|
|
692 "UnmapNotify",
|
|
693 "MapNotify",
|
|
694 "MapRequest",
|
|
695 "ReparentNotify",
|
|
696 "ConfigureNotify",
|
|
697 "ConfigureRequest",
|
|
698 "GravityNotify",
|
|
699 "ResizeRequest",
|
|
700 "CirculateNotify",
|
|
701 "CirculateRequest",
|
|
702 "PropertyNotify",
|
|
703 "SelectionClear",
|
|
704 "SelectionRequest",
|
|
705 "SelectionNotify",
|
|
706 "ColormapNotify",
|
|
707 "ClientMessage",
|
|
708 "MappingNotify",
|
|
709 "LASTEvent"
|
|
710 };
|
|
711
|
|
712 if (event_type < 0 || event_type >= countof (events))
|
|
713 return NULL;
|
0
|
714 return events [event_type];
|
|
715 }
|
|
716
|
|
717 /* Handling errors.
|
|
718
|
|
719 If an X error occurs which we are not expecting, we have no alternative
|
|
720 but to print it to stderr. It would be nice to stuff it into a pop-up
|
|
721 buffer, or to print it in the minibuffer, but that's not possible, because
|
|
722 one is not allowed to do any I/O on the display connection from an error
|
|
723 handler. The guts of Xlib expect these functions to either return or exit.
|
|
724
|
|
725 However, there are occasions when we might expect an error to reasonably
|
|
726 occur. The interface to this is as follows:
|
|
727
|
|
728 Before calling some X routine which may error, call
|
|
729 expect_x_error (dpy);
|
|
730
|
|
731 Just after calling the X routine, call either:
|
|
732
|
|
733 x_error_occurred_p (dpy);
|
|
734
|
|
735 to ask whether an error happened (and was ignored), or:
|
|
736
|
|
737 signal_if_x_error (dpy, resumable_p);
|
|
738
|
|
739 which will call Fsignal() with args appropriate to the X error, if there
|
|
740 was one. (Resumable_p is whether the debugger should be allowed to
|
|
741 continue from the call to signal.)
|
|
742
|
|
743 You must call one of these two routines immediately after calling the X
|
|
744 routine; think of them as bookends like BLOCK_INPUT and UNBLOCK_INPUT.
|
|
745 */
|
|
746
|
|
747 static int error_expected;
|
|
748 static int error_occurred;
|
|
749 static XErrorEvent last_error;
|
|
750
|
|
751 /* OVERKILL! */
|
|
752
|
|
753 #ifdef EXTERNAL_WIDGET
|
|
754 static Lisp_Object
|
|
755 x_error_handler_do_enqueue (Lisp_Object frame)
|
|
756 {
|
|
757 enqueue_magic_eval_event (io_error_delete_frame, frame);
|
|
758 return Qt;
|
|
759 }
|
|
760
|
|
761 static Lisp_Object
|
|
762 x_error_handler_error (Lisp_Object data, Lisp_Object dummy)
|
|
763 {
|
|
764 return Qnil;
|
|
765 }
|
|
766 #endif /* EXTERNAL_WIDGET */
|
|
767
|
|
768 int
|
|
769 x_error_handler (Display *disp, XErrorEvent *event)
|
|
770 {
|
|
771 if (error_expected)
|
|
772 {
|
|
773 error_expected = 0;
|
|
774 error_occurred = 1;
|
|
775 last_error = *event;
|
|
776 }
|
|
777 else
|
|
778 {
|
|
779 #ifdef EXTERNAL_WIDGET
|
|
780 struct frame *f;
|
|
781 struct device *d = get_device_from_display (disp);
|
|
782
|
|
783 if ((event->error_code == BadWindow ||
|
|
784 event->error_code == BadDrawable)
|
|
785 && ((f = x_any_window_to_frame (d, event->resourceid)) != 0))
|
|
786 {
|
|
787 Lisp_Object frame;
|
|
788
|
|
789 /* one of the windows comprising one of our frames has died.
|
|
790 This occurs particularly with ExternalShell frames when the
|
|
791 client that owns the ExternalShell's window dies.
|
|
792
|
|
793 We cannot do any I/O on the display connection so we need
|
|
794 to enqueue an eval event so that the deletion happens
|
|
795 later.
|
|
796
|
|
797 Furthermore, we need to trap any errors (out-of-memory) that
|
|
798 may occur when Fenqueue_eval_event is called.
|
|
799 */
|
|
800
|
|
801 if (f->being_deleted)
|
|
802 return 0;
|
|
803 XSETFRAME (frame, f);
|
|
804 if (!NILP (condition_case_1 (Qerror, x_error_handler_do_enqueue,
|
|
805 frame, x_error_handler_error, Qnil)))
|
|
806 {
|
|
807 f->being_deleted = 1;
|
|
808 f->visible = 0;
|
|
809 }
|
|
810 return 0;
|
|
811 }
|
|
812 #endif /* EXTERNAL_WIDGET */
|
|
813
|
|
814 stderr_out ("\n%s: ",
|
|
815 (STRINGP (Vinvocation_name)
|
14
|
816 ? (char *) XSTRING_DATA (Vinvocation_name)
|
0
|
817 : "xemacs"));
|
|
818 XmuPrintDefaultErrorMessage (disp, event, stderr);
|
|
819 }
|
|
820 return 0;
|
|
821 }
|
|
822
|
|
823 void
|
|
824 expect_x_error (Display *dpy)
|
|
825 {
|
|
826 assert (!error_expected);
|
|
827 XSync (dpy, 0); /* handle pending errors before setting flag */
|
|
828 error_expected = 1;
|
|
829 error_occurred = 0;
|
|
830 }
|
|
831
|
|
832 int
|
|
833 x_error_occurred_p (Display *dpy)
|
|
834 {
|
|
835 int val;
|
|
836 XSync (dpy, 0); /* handle pending errors before setting flag */
|
|
837 val = error_occurred;
|
|
838 error_expected = 0;
|
|
839 error_occurred = 0;
|
|
840 return val;
|
|
841 }
|
|
842
|
|
843 int
|
|
844 signal_if_x_error (Display *dpy, int resumable_p)
|
|
845 {
|
|
846 char buf[1024];
|
|
847 Lisp_Object data;
|
|
848 if (! x_error_occurred_p (dpy))
|
|
849 return 0;
|
|
850 data = Qnil;
|
|
851 sprintf (buf, "0x%X", (unsigned int) last_error.resourceid);
|
|
852 data = Fcons (build_string (buf), data);
|
|
853 {
|
|
854 char num [32];
|
|
855 sprintf (num, "%d", last_error.request_code);
|
|
856 XGetErrorDatabaseText (last_error.display, "XRequest", num, "",
|
|
857 buf, sizeof (buf));
|
|
858 if (! *buf)
|
|
859 sprintf (buf, "Request-%d", last_error.request_code);
|
|
860 data = Fcons (build_string (buf), data);
|
|
861 }
|
|
862 XGetErrorText (last_error.display, last_error.error_code, buf, sizeof (buf));
|
|
863 data = Fcons (build_string (buf), data);
|
|
864 again:
|
|
865 Fsignal (Qx_error, data);
|
|
866 if (! resumable_p) goto again;
|
|
867 return 1;
|
|
868 }
|
|
869
|
|
870 int
|
|
871 x_IO_error_handler (Display *disp)
|
|
872 {
|
|
873 /* This function can GC */
|
|
874 Lisp_Object dev;
|
114
|
875 struct device *d = get_device_from_display_1 (disp);
|
|
876
|
|
877 if (d)
|
|
878 XSETDEVICE (dev, d);
|
|
879 else
|
|
880 dev = Qnil;
|
0
|
881
|
|
882 if (NILP (find_nonminibuffer_frame_not_on_device (dev)))
|
|
883 {
|
|
884 /* We're going down. */
|
|
885 stderr_out
|
|
886 ("\n%s: Fatal I/O Error %d (%s) on display connection \"%s\"\n",
|
|
887 (STRINGP (Vinvocation_name) ?
|
14
|
888 (char *) XSTRING_DATA (Vinvocation_name) : "xemacs"),
|
0
|
889 errno, strerror (errno), DisplayString (disp));
|
|
890 stderr_out
|
|
891 (" after %lu requests (%lu known processed) with %d events remaining.\n",
|
|
892 NextRequest (disp) - 1, LastKnownRequestProcessed (disp),
|
|
893 QLength (disp));
|
|
894 /* assert (!_Xdebug); */
|
|
895 }
|
|
896 else
|
|
897 {
|
|
898 warn_when_safe
|
|
899 (Qx, Qcritical,
|
|
900 "I/O Error %d (%s) on display connection \"%s\"\n"
|
|
901 " after %lu requests (%lu known processed) with "
|
|
902 "%d events remaining.\n",
|
|
903 errno, strerror (errno), DisplayString (disp),
|
|
904 NextRequest (disp) - 1, LastKnownRequestProcessed (disp),
|
|
905 QLength (disp));
|
|
906 }
|
|
907
|
114
|
908 if (d)
|
|
909 enqueue_magic_eval_event (io_error_delete_device, dev);
|
0
|
910
|
167
|
911 /* CvE, July 16, 1996, XEmacs 19.14 */
|
|
912 /* Test for broken pipe error, which indicates X-server has gone down */
|
|
913 if (errno == EPIPE && x_seppuku_on_epipe)
|
|
914 {
|
|
915 /* Most probably X-server has gone down: Avoid infinite loop by just */
|
|
916 /* exiting */
|
|
917 /* slb: This sounds really, really dangerous to do by default, so */
|
|
918 /* I'm adding a guard to avoid doing this as default behavior */
|
|
919 stderr_out( "\n\nXEmacs exiting on broken pipe (errno %d, %s)\n",
|
|
920 errno, strerror(errno));
|
|
921 exit(errno);
|
|
922 }
|
|
923
|
0
|
924 return 0;
|
|
925 }
|
|
926
|
20
|
927 DEFUN ("x-debug-mode", Fx_debug_mode, 1, 2, 0, /*
|
0
|
928 With a true arg, make the connection to the X server synchronous.
|
|
929 With false, make it asynchronous. Synchronous connections are much slower,
|
|
930 but are useful for debugging. (If you get X errors, make the connection
|
|
931 synchronous, and use a debugger to set a breakpoint on `x_error_handler'.
|
|
932 Your backtrace of the C stack will now be useful. In asynchronous mode,
|
|
933 the stack above `x_error_handler' isn't helpful because of buffering.)
|
|
934 If DEVICE is not specified, the selected device is assumed.
|
|
935
|
|
936 Calling this function is the same as calling the C function `XSynchronize',
|
|
937 or starting the program with the `-sync' command line argument.
|
20
|
938 */
|
|
939 (arg, device))
|
0
|
940 {
|
|
941 struct device *d = decode_x_device (device);
|
|
942
|
|
943 XSynchronize (DEVICE_X_DISPLAY (d), !NILP (arg));
|
|
944
|
|
945 if (!NILP (arg))
|
|
946 message ("X connection is synchronous");
|
|
947 else
|
|
948 message ("X connection is asynchronous");
|
|
949
|
|
950 return arg;
|
|
951 }
|
|
952
|
|
953
|
|
954 /************************************************************************/
|
|
955 /* X resources */
|
|
956 /************************************************************************/
|
|
957
|
|
958 #if 0 /* bah humbug. The whole "widget == resource" stuff is such
|
|
959 a crock of shit that I'm just going to ignore it all. */
|
|
960
|
|
961 /* If widget is NULL, we are retrieving device or global face data. */
|
|
962
|
|
963 static void
|
|
964 construct_name_list (Display *display, Widget widget, char *fake_name,
|
|
965 char *fake_class, char *name, char *class)
|
|
966 {
|
|
967 char *stack [100][2];
|
|
968 Widget this;
|
|
969 int count = 0;
|
|
970 char *name_tail, *class_tail;
|
|
971
|
|
972 if (widget)
|
|
973 {
|
|
974 for (this = widget; this; this = XtParent (this))
|
|
975 {
|
|
976 stack [count][0] = this->core.name;
|
|
977 stack [count][1] = XtClass (this)->core_class.class_name;
|
|
978 count++;
|
|
979 }
|
|
980 count--;
|
|
981 }
|
|
982 else if (fake_name && fake_class)
|
|
983 {
|
|
984 stack [count][0] = fake_name;
|
|
985 stack [count][1] = fake_class;
|
|
986 count++;
|
|
987 }
|
|
988
|
|
989 /* The root widget is an application shell; resource lookups use the
|
|
990 specified application name and application class in preference to
|
|
991 the name/class of that widget (which is argv[0] / "ApplicationShell").
|
|
992 Generally the app name and class will be argv[0] / "Emacs" but
|
|
993 the former can be set via the -name command-line option, and the
|
|
994 latter can be set by changing `x-emacs-application-class' in
|
|
995 lisp/term/x-win.el.
|
|
996 */
|
|
997 XtGetApplicationNameAndClass (display,
|
|
998 &stack [count][0],
|
|
999 &stack [count][1]);
|
|
1000
|
|
1001 name [0] = 0;
|
|
1002 class [0] = 0;
|
|
1003
|
|
1004 name_tail = name;
|
|
1005 class_tail = class;
|
|
1006 for (; count >= 0; count--)
|
|
1007 {
|
|
1008 strcat (name_tail, stack [count][0]);
|
|
1009 for (; *name_tail; name_tail++)
|
|
1010 if (*name_tail == '.') *name_tail = '_';
|
|
1011 strcat (name_tail, ".");
|
|
1012 name_tail++;
|
|
1013
|
|
1014 strcat (class_tail, stack [count][1]);
|
|
1015 for (; *class_tail; class_tail++)
|
|
1016 if (*class_tail == '.') *class_tail = '_';
|
|
1017 strcat (class_tail, ".");
|
|
1018 class_tail++;
|
|
1019 }
|
|
1020 }
|
|
1021
|
183
|
1022 #endif /* 0 */
|
0
|
1023
|
272
|
1024 static char_dynarr *name_char_dynarr;
|
|
1025 static char_dynarr *class_char_dynarr;
|
0
|
1026
|
|
1027 /* Given a locale and device specification from x-get-resource or
|
|
1028 x-get-resource-prefix, return the resource prefix and display to
|
|
1029 fetch the resource on. */
|
|
1030
|
|
1031 static void
|
|
1032 x_get_resource_prefix (Lisp_Object locale, Lisp_Object device,
|
272
|
1033 Display **display_out, char_dynarr *name,
|
|
1034 char_dynarr *class)
|
0
|
1035 {
|
|
1036 if (NILP (locale))
|
|
1037 locale = Qglobal;
|
|
1038 if (NILP (Fvalid_specifier_locale_p (locale)))
|
|
1039 signal_simple_error ("Invalid locale", locale);
|
|
1040 if (WINDOWP (locale))
|
|
1041 /* #### I can't come up with any coherent way of naming windows.
|
|
1042 By relative position? That seems tricky because windows
|
|
1043 can change position, be split, etc. By order of creation?
|
|
1044 That seems less than useful. */
|
|
1045 signal_simple_error ("Windows currently can't be resourced", locale);
|
|
1046
|
|
1047 if (!NILP (device) && !DEVICEP (device))
|
|
1048 CHECK_DEVICE (device);
|
|
1049 if (DEVICEP (device) && !DEVICE_X_P (XDEVICE (device)))
|
|
1050 device = Qnil;
|
|
1051 if (NILP (device))
|
|
1052 {
|
|
1053 device = DFW_DEVICE (locale);
|
|
1054 if (DEVICEP (device) && !DEVICE_X_P (XDEVICE (device)))
|
|
1055 device = Qnil;
|
|
1056 if (NILP (device))
|
|
1057 device = Vdefault_x_device;
|
|
1058 if (NILP (device))
|
|
1059 {
|
|
1060 *display_out = 0;
|
|
1061 return;
|
|
1062 }
|
|
1063 }
|
|
1064
|
|
1065 *display_out = DEVICE_X_DISPLAY (XDEVICE (device));
|
|
1066
|
272
|
1067 {
|
|
1068 char *appname, *appclass;
|
|
1069 int name_len, class_len;
|
|
1070 XtGetApplicationNameAndClass (*display_out, &appname, &appclass);
|
|
1071 name_len = strlen (appname);
|
|
1072 class_len = strlen (appclass);
|
|
1073 Dynarr_add_many (name , appname, name_len);
|
|
1074 Dynarr_add_many (class, appclass, class_len);
|
|
1075 validify_resource_component (Dynarr_atp (name, 0), name_len);
|
|
1076 validify_resource_component (Dynarr_atp (class, 0), class_len);
|
|
1077 }
|
0
|
1078
|
|
1079 if (EQ (locale, Qglobal))
|
|
1080 return;
|
|
1081 if (BUFFERP (locale))
|
|
1082 {
|
272
|
1083 Dynarr_add_literal_string (name, ".buffer.");
|
0
|
1084 /* we know buffer is live; otherwise we got an error above. */
|
272
|
1085 Dynarr_add_validified_lisp_string (name, Fbuffer_name (locale));
|
|
1086 Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsBuffer");
|
0
|
1087 }
|
|
1088 else if (FRAMEP (locale))
|
|
1089 {
|
272
|
1090 Dynarr_add_literal_string (name, ".frame.");
|
0
|
1091 /* we know frame is live; otherwise we got an error above. */
|
272
|
1092 Dynarr_add_validified_lisp_string (name, Fframe_name (locale));
|
|
1093 Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsFrame");
|
0
|
1094 }
|
|
1095 else
|
|
1096 {
|
|
1097 assert (DEVICEP (locale));
|
272
|
1098 Dynarr_add_literal_string (name, ".device.");
|
0
|
1099 /* we know device is live; otherwise we got an error above. */
|
272
|
1100 Dynarr_add_validified_lisp_string (name, Fdevice_name (locale));
|
|
1101 Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsDevice");
|
0
|
1102 }
|
|
1103 return;
|
|
1104 }
|
|
1105
|
20
|
1106 DEFUN ("x-get-resource", Fx_get_resource, 3, 6, 0, /*
|
0
|
1107 Retrieve an X resource from the resource manager.
|
|
1108
|
183
|
1109 The first arg is the name of the resource to retrieve, such as "font".
|
272
|
1110 The second arg is the class of the resource to retrieve, such as "Font".
|
|
1111 The third arg must be one of the symbols 'string, 'integer, 'natnum, or
|
0
|
1112 'boolean, specifying the type of object that the database is searched for.
|
|
1113 The fourth arg is the locale to search for the resources on, and can
|
272
|
1114 currently be a buffer, a frame, a device, or 'global. If omitted, it
|
0
|
1115 defaults to 'global.
|
|
1116 The fifth arg is the device to search for the resources on. (The resource
|
|
1117 database for a particular device is constructed by combining non-device-
|
183
|
1118 specific resources such as any command-line resources specified and any
|
0
|
1119 app-defaults files found [or the fallback resources supplied by XEmacs,
|
|
1120 if no app-defaults file is found] with device-specific resources such as
|
|
1121 those supplied using xrdb.) If omitted, it defaults to the device of
|
|
1122 LOCALE, if a device can be derived (i.e. if LOCALE is a frame or device),
|
|
1123 and otherwise defaults to the value of `default-x-device'.
|
|
1124 The sixth arg NOERROR, if non-nil, means do not signal an error if a
|
|
1125 bogus resource specification was retrieved (e.g. if a non-integer was
|
|
1126 given when an integer was requested). In this case, a warning is issued
|
|
1127 instead.
|
|
1128
|
|
1129 The resource names passed to this function are looked up relative to the
|
|
1130 locale.
|
|
1131
|
|
1132 If you want to search for a subresource, you just need to specify the
|
|
1133 resource levels in NAME and CLASS. For example, NAME could be
|
183
|
1134 "modeline.attributeFont", and CLASS "Face.AttributeFont".
|
0
|
1135
|
|
1136 Specifically,
|
|
1137
|
|
1138 1) If LOCALE is a buffer, a call
|
|
1139
|
183
|
1140 (x-get-resource "foreground" "Foreground" 'string SOME-BUFFER)
|
0
|
1141
|
|
1142 is an interface to a C call something like
|
|
1143
|
183
|
1144 XrmGetResource (db, "xemacs.buffer.BUFFER-NAME.foreground",
|
|
1145 "Emacs.EmacsLocaleType.EmacsBuffer.Foreground",
|
|
1146 "String");
|
0
|
1147
|
|
1148 2) If LOCALE is a frame, a call
|
|
1149
|
183
|
1150 (x-get-resource "foreground" "Foreground" 'string SOME-FRAME)
|
0
|
1151
|
|
1152 is an interface to a C call something like
|
|
1153
|
183
|
1154 XrmGetResource (db, "xemacs.frame.FRAME-NAME.foreground",
|
|
1155 "Emacs.EmacsLocaleType.EmacsFrame.Foreground",
|
|
1156 "String");
|
0
|
1157
|
|
1158 3) If LOCALE is a device, a call
|
|
1159
|
183
|
1160 (x-get-resource "foreground" "Foreground" 'string SOME-DEVICE)
|
0
|
1161
|
|
1162 is an interface to a C call something like
|
|
1163
|
183
|
1164 XrmGetResource (db, "xemacs.device.DEVICE-NAME.foreground",
|
|
1165 "Emacs.EmacsLocaleType.EmacsDevice.Foreground",
|
|
1166 "String");
|
0
|
1167
|
|
1168 4) If LOCALE is 'global, a call
|
|
1169
|
183
|
1170 (x-get-resource "foreground" "Foreground" 'string 'global)
|
0
|
1171
|
|
1172 is an interface to a C call something like
|
|
1173
|
183
|
1174 XrmGetResource (db, "xemacs.foreground",
|
|
1175 "Emacs.Foreground",
|
|
1176 "String");
|
0
|
1177
|
|
1178 Note that for 'global, no prefix is added other than that of the
|
|
1179 application itself; thus, you can use this locale to retrieve
|
|
1180 arbitrary application resources, if you really want to.
|
|
1181
|
|
1182 The returned value of this function is nil if the queried resource is not
|
|
1183 found. If the third arg is `string', a string is returned, and if it is
|
|
1184 `integer', an integer is returned. If the third arg is `boolean', then the
|
|
1185 returned value is the list (t) for true, (nil) for false, and is nil to
|
|
1186 mean ``unspecified.''
|
20
|
1187 */
|
|
1188 (name, class, type, locale, device, no_error))
|
0
|
1189 {
|
272
|
1190 char* name_string, *class_string;
|
0
|
1191 char *raw_result;
|
|
1192 XrmDatabase db;
|
|
1193 Display *display;
|
|
1194 Error_behavior errb = decode_error_behavior_flag (no_error);
|
|
1195
|
|
1196 CHECK_STRING (name);
|
|
1197 CHECK_STRING (class);
|
|
1198 CHECK_SYMBOL (type);
|
|
1199
|
272
|
1200 Dynarr_reset (name_char_dynarr);
|
|
1201 Dynarr_reset (class_char_dynarr);
|
0
|
1202
|
272
|
1203 x_get_resource_prefix (locale, device, &display,
|
|
1204 name_char_dynarr, class_char_dynarr);
|
0
|
1205 if (!display)
|
|
1206 return Qnil;
|
|
1207
|
|
1208 db = XtDatabase (display);
|
|
1209
|
272
|
1210 Dynarr_add (name_char_dynarr, '.');
|
|
1211 Dynarr_add_lisp_string (name_char_dynarr, name);
|
|
1212 Dynarr_add (class_char_dynarr, '.');
|
|
1213 Dynarr_add_lisp_string (class_char_dynarr, class);
|
|
1214 Dynarr_add (name_char_dynarr, '\0');
|
|
1215 Dynarr_add (class_char_dynarr, '\0');
|
|
1216
|
|
1217 name_string = Dynarr_atp (name_char_dynarr, 0);
|
|
1218 class_string = Dynarr_atp (class_char_dynarr, 0);
|
0
|
1219
|
|
1220 {
|
|
1221 XrmValue xrm_value;
|
|
1222 XrmName namelist[100];
|
|
1223 XrmClass classlist[100];
|
|
1224 XrmName *namerest = namelist;
|
|
1225 XrmClass *classrest = classlist;
|
|
1226 XrmRepresentation xrm_type;
|
|
1227 XrmRepresentation string_quark;
|
|
1228 int result;
|
|
1229 XrmStringToNameList (name_string, namelist);
|
|
1230 XrmStringToClassList (class_string, classlist);
|
|
1231 string_quark = XrmStringToQuark ("String");
|
|
1232
|
|
1233 /* ensure that they have the same length */
|
|
1234 while (namerest[0] && classrest[0])
|
|
1235 namerest++, classrest++;
|
|
1236 if (namerest[0] || classrest[0])
|
|
1237 signal_simple_error_2
|
|
1238 ("class list and name list must be the same length", name, class);
|
|
1239 result = XrmQGetResource (db, namelist, classlist, &xrm_type, &xrm_value);
|
|
1240
|
|
1241 if (result != True || xrm_type != string_quark)
|
|
1242 return Qnil;
|
|
1243 raw_result = (char *) xrm_value.addr;
|
|
1244 }
|
|
1245
|
|
1246 if (EQ (type, Qstring))
|
|
1247 return build_string (raw_result);
|
|
1248 else if (EQ (type, Qboolean))
|
|
1249 {
|
183
|
1250 if (!strcasecmp (raw_result, "off") ||
|
0
|
1251 !strcasecmp (raw_result, "false") ||
|
183
|
1252 !strcasecmp (raw_result, "no"))
|
0
|
1253 return Fcons (Qnil, Qnil);
|
272
|
1254 if (!strcasecmp (raw_result, "on") ||
|
|
1255 !strcasecmp (raw_result, "true") ||
|
|
1256 !strcasecmp (raw_result, "yes"))
|
0
|
1257 return Fcons (Qt, Qnil);
|
272
|
1258 return maybe_continuable_error
|
|
1259 (Qresource, errb,
|
|
1260 "can't convert %s: %s to a Boolean", name_string, raw_result);
|
0
|
1261 }
|
|
1262 else if (EQ (type, Qinteger) || EQ (type, Qnatnum))
|
|
1263 {
|
|
1264 int i;
|
|
1265 char c;
|
|
1266 if (1 != sscanf (raw_result, "%d%c", &i, &c))
|
|
1267 return maybe_continuable_error
|
|
1268 (Qresource, errb,
|
272
|
1269 "can't convert %s: %s to an integer", name_string, raw_result);
|
0
|
1270 else if (EQ (type, Qnatnum) && i < 0)
|
|
1271 return maybe_continuable_error
|
|
1272 (Qresource, errb,
|
272
|
1273 "invalid numerical value %d for resource %s", i, name_string);
|
0
|
1274 else
|
|
1275 return make_int (i);
|
|
1276 }
|
|
1277 else
|
272
|
1278 {
|
|
1279 return maybe_signal_continuable_error
|
|
1280 (Qwrong_type_argument,
|
|
1281 list2 (build_translated_string
|
|
1282 ("should be string, integer, natnum or boolean"),
|
|
1283 type),
|
|
1284 Qresource, errb);
|
|
1285 }
|
0
|
1286 }
|
|
1287
|
20
|
1288 DEFUN ("x-get-resource-prefix", Fx_get_resource_prefix, 1, 2, 0, /*
|
0
|
1289 Return the resource prefix for LOCALE on DEVICE.
|
|
1290 The resource prefix is the strings used to prefix resources if
|
|
1291 the LOCALE and DEVICE arguments were passed to `x-get-resource'.
|
|
1292 The returned value is a cons of a name prefix and a class prefix.
|
|
1293 For example, if LOCALE is a frame, the returned value might be
|
183
|
1294 \("xemacs.frame.FRAME-NAME" . "Emacs.EmacsLocaleType.EmacsFrame").
|
0
|
1295 If no valid X device for resourcing can be obtained, this function
|
|
1296 returns nil. (In such a case, `x-get-resource' would always return nil.)
|
20
|
1297 */
|
|
1298 (locale, device))
|
0
|
1299 {
|
|
1300 Display *display;
|
|
1301
|
272
|
1302 Dynarr_reset (name_char_dynarr );
|
|
1303 Dynarr_reset (class_char_dynarr);
|
|
1304
|
|
1305 x_get_resource_prefix (locale, device, &display,
|
|
1306 name_char_dynarr, class_char_dynarr);
|
0
|
1307 if (!display)
|
|
1308 return Qnil;
|
272
|
1309
|
|
1310 return Fcons (make_string ((Bufbyte *) Dynarr_atp (name_char_dynarr, 0),
|
|
1311 Dynarr_length (name_char_dynarr)),
|
|
1312 make_string ((Bufbyte *) Dynarr_atp (class_char_dynarr, 0),
|
|
1313 Dynarr_length (class_char_dynarr)));
|
0
|
1314 }
|
|
1315
|
20
|
1316 DEFUN ("x-put-resource", Fx_put_resource, 1, 2, 0, /*
|
0
|
1317 Add a resource to the resource database for DEVICE.
|
|
1318 RESOURCE-LINE specifies the resource to add and should be a
|
|
1319 standard resource specification.
|
20
|
1320 */
|
|
1321 (resource_line, device))
|
0
|
1322 {
|
|
1323 struct device *d = decode_device (device);
|
|
1324 char *str, *colon_pos;
|
|
1325
|
|
1326 CHECK_STRING (resource_line);
|
14
|
1327 str = (char *) XSTRING_DATA (resource_line);
|
0
|
1328 if (!(colon_pos = strchr (str, ':')) || strchr (str, '\n'))
|
|
1329 invalid:
|
|
1330 signal_simple_error ("Invalid resource line", resource_line);
|
|
1331 if (strspn (str,
|
|
1332 /* Only the following chars are allowed before the colon */
|
|
1333 " \t.*?abcdefghijklmnopqrstuvwxyz"
|
272
|
1334 "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_-")
|
|
1335 != (size_t) (colon_pos - str))
|
0
|
1336 goto invalid;
|
|
1337
|
|
1338 if (DEVICE_X_P (d))
|
|
1339 {
|
|
1340 XrmDatabase db = XtDatabase (DEVICE_X_DISPLAY (d));
|
|
1341 XrmPutLineResource (&db, str);
|
|
1342 }
|
|
1343
|
|
1344 return Qnil;
|
|
1345 }
|
|
1346
|
|
1347
|
|
1348 /************************************************************************/
|
|
1349 /* display information functions */
|
|
1350 /************************************************************************/
|
|
1351
|
20
|
1352 DEFUN ("default-x-device", Fdefault_x_device, 0, 0, 0, /*
|
0
|
1353 Return the default X device for resourcing.
|
|
1354 This is the first-created X device that still exists.
|
20
|
1355 */
|
|
1356 ())
|
0
|
1357 {
|
|
1358 return Vdefault_x_device;
|
|
1359 }
|
|
1360
|
20
|
1361 DEFUN ("x-display-visual-class", Fx_display_visual_class, 0, 1, 0, /*
|
219
|
1362 Return the visual class of the X display `device' is using.
|
|
1363 This can be altered from the default at startup using the XResource "EmacsVisual".
|
0
|
1364 The returned value will be one of the symbols `static-gray', `gray-scale',
|
|
1365 `static-color', `pseudo-color', `true-color', or `direct-color'.
|
20
|
1366 */
|
|
1367 (device))
|
0
|
1368 {
|
219
|
1369 Visual *vis = DEVICE_X_VISUAL (decode_x_device (device));
|
|
1370 switch (vis->class)
|
0
|
1371 {
|
193
|
1372 case StaticGray: return intern ("static-gray");
|
|
1373 case GrayScale: return intern ("gray-scale");
|
|
1374 case StaticColor: return intern ("static-color");
|
|
1375 case PseudoColor: return intern ("pseudo-color");
|
|
1376 case TrueColor: return intern ("true-color");
|
|
1377 case DirectColor: return intern ("direct-color");
|
0
|
1378 default:
|
|
1379 error ("display has an unknown visual class");
|
272
|
1380 return Qnil; /* suppress compiler warning */
|
0
|
1381 }
|
|
1382 }
|
|
1383
|
219
|
1384 DEFUN ("x-display-visual-depth", Fx_display_visual_depth, 0, 1, 0, /*
|
|
1385 Return the bitplane depth of the visual the X display `device' is using.
|
|
1386 */
|
|
1387 (device))
|
|
1388 {
|
|
1389 return make_int (DEVICE_X_DEPTH (decode_x_device (device)));
|
|
1390 }
|
|
1391
|
0
|
1392 static int
|
|
1393 x_device_pixel_width (struct device *d)
|
|
1394 {
|
|
1395 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1396
|
|
1397 return DisplayWidth (dpy, DefaultScreen (dpy));
|
|
1398 }
|
|
1399
|
|
1400 static int
|
|
1401 x_device_pixel_height (struct device *d)
|
|
1402 {
|
|
1403 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1404
|
|
1405 return DisplayHeight (dpy, DefaultScreen (dpy));
|
|
1406 }
|
|
1407
|
|
1408 static int
|
|
1409 x_device_mm_width (struct device *d)
|
|
1410 {
|
|
1411 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1412
|
|
1413 return DisplayWidthMM (dpy, DefaultScreen (dpy));
|
|
1414 }
|
|
1415
|
|
1416 static int
|
|
1417 x_device_mm_height (struct device *d)
|
|
1418 {
|
|
1419 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1420
|
|
1421 return DisplayHeightMM (dpy, DefaultScreen (dpy));
|
|
1422 }
|
|
1423
|
|
1424 static int
|
|
1425 x_device_bitplanes (struct device *d)
|
|
1426 {
|
|
1427 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1428
|
|
1429 return DisplayPlanes (dpy, DefaultScreen (dpy));
|
|
1430 }
|
|
1431
|
|
1432 static int
|
|
1433 x_device_color_cells (struct device *d)
|
|
1434 {
|
|
1435 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1436
|
|
1437 return DisplayCells (dpy, DefaultScreen (dpy));
|
|
1438 }
|
|
1439
|
20
|
1440 DEFUN ("x-server-vendor", Fx_server_vendor, 0, 1, 0, /*
|
0
|
1441 Return the vendor ID string of the X server `device' on.
|
20
|
1442 */
|
|
1443 (device))
|
0
|
1444 {
|
|
1445 Display *dpy = get_x_display (device);
|
|
1446 char *vendor = ServerVendor (dpy);
|
|
1447
|
183
|
1448 return build_string (vendor ? vendor : "");
|
0
|
1449 }
|
|
1450
|
20
|
1451 DEFUN ("x-server-version", Fx_server_version, 0, 1, 0, /*
|
0
|
1452 Return the version numbers of the X server `device' is on.
|
|
1453 The returned value is a list of three integers: the major and minor
|
|
1454 version numbers of the X Protocol in use, and the vendor-specific release
|
|
1455 number. See also `x-server-vendor'.
|
20
|
1456 */
|
|
1457 (device))
|
0
|
1458 {
|
|
1459 Display *dpy = get_x_display (device);
|
|
1460
|
187
|
1461 return list3 (make_int (ProtocolVersion (dpy)),
|
0
|
1462 make_int (ProtocolRevision (dpy)),
|
187
|
1463 make_int (VendorRelease (dpy)));
|
0
|
1464 }
|
|
1465
|
20
|
1466 DEFUN ("x-valid-keysym-name-p", Fx_valid_keysym_name_p, 1, 1, 0, /*
|
0
|
1467 Return true if KEYSYM names a keysym that the X library knows about.
|
|
1468 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1469 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
20
|
1470 */
|
|
1471 (keysym))
|
0
|
1472 {
|
|
1473 CONST char *keysym_ext;
|
183
|
1474
|
0
|
1475 CHECK_STRING (keysym);
|
|
1476 GET_C_STRING_CTEXT_DATA_ALLOCA (keysym, keysym_ext);
|
183
|
1477
|
|
1478 return XStringToKeysym (keysym_ext) ? Qt : Qnil;
|
0
|
1479 }
|
|
1480
|
187
|
1481 DEFUN ("x-keysym-hashtable", Fx_keysym_hashtable, 0, 1, 0, /*
|
|
1482 Return a hashtable which contains a hash key for all keysyms which
|
|
1483 name keys on the keyboard. See `x-keysym-on-keyboard-p'.
|
|
1484 */
|
|
1485 (device))
|
|
1486 {
|
|
1487 struct device *d = decode_device (device);
|
|
1488 if (!DEVICE_X_P (d))
|
|
1489 signal_simple_error ("Not an X device", device);
|
|
1490
|
|
1491 return DEVICE_X_DATA (d)->x_keysym_map_hashtable;
|
|
1492 }
|
|
1493
|
|
1494 DEFUN ("x-keysym-on-keyboard-sans-modifiers-p",
|
|
1495 Fx_keysym_on_keyboard_sans_modifiers_p, 1, 2, 0, /*
|
0
|
1496 Return true if KEYSYM names a key on the keyboard of DEVICE.
|
|
1497 More precisely, return true if pressing a physical key
|
|
1498 on the keyboard of DEVICE without any modifier keys generates KEYSYM.
|
|
1499 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1500 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
253
|
1501 The keysym name can be provided in two forms:
|
|
1502 - if keysym is a string, it must be the name as known to X windows.
|
|
1503 - if keysym is a symbol, it must be the name as known to XEmacs.
|
|
1504 The two names differ in capitalization and underscoring.
|
20
|
1505 */
|
|
1506 (keysym, device))
|
0
|
1507 {
|
187
|
1508 struct device *d = decode_device (device);
|
|
1509 if (!DEVICE_X_P (d))
|
|
1510 signal_simple_error ("Not an X device", device);
|
|
1511
|
|
1512 return (EQ (Qsans_modifiers,
|
|
1513 Fgethash (keysym, DEVICE_X_KEYSYM_MAP_HASHTABLE (d), Qnil)) ?
|
|
1514 Qt : Qnil);
|
|
1515 }
|
|
1516
|
|
1517
|
|
1518 DEFUN ("x-keysym-on-keyboard-p", Fx_keysym_on_keyboard_p, 1, 2, 0, /*
|
|
1519 Return true if KEYSYM names a key on the keyboard of DEVICE.
|
|
1520 More precisely, return true if some keystroke (possibly including modifiers)
|
|
1521 on the keyboard of DEVICE keys generates KEYSYM.
|
|
1522 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1523 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
253
|
1524 The keysym name can be provided in two forms:
|
|
1525 - if keysym is a string, it must be the name as known to X windows.
|
|
1526 - if keysym is a symbol, it must be the name as known to XEmacs.
|
|
1527 The two names differ in capitalization and underscoring.
|
187
|
1528 */
|
|
1529 (keysym, device))
|
|
1530 {
|
|
1531 struct device *d = decode_device (device);
|
0
|
1532 if (!DEVICE_X_P (d))
|
|
1533 signal_simple_error ("Not an X device", device);
|
183
|
1534
|
187
|
1535 return (NILP (Fgethash (keysym, DEVICE_X_KEYSYM_MAP_HASHTABLE (d), Qnil)) ?
|
|
1536 Qnil : Qt);
|
0
|
1537 }
|
|
1538
|
|
1539
|
|
1540 /************************************************************************/
|
|
1541 /* grabs and ungrabs */
|
|
1542 /************************************************************************/
|
|
1543
|
20
|
1544 DEFUN ("x-grab-pointer", Fx_grab_pointer, 0, 3, 0, /*
|
0
|
1545 Grab the pointer and restrict it to its current window.
|
|
1546 If optional DEVICE argument is nil, the default device will be used.
|
|
1547 If optional CURSOR argument is non-nil, change the pointer shape to that
|
|
1548 until `x-ungrab-pointer' is called (it should be an object returned by the
|
|
1549 `make-cursor-glyph' function).
|
|
1550 If the second optional argument IGNORE-KEYBOARD is non-nil, ignore all
|
|
1551 keyboard events during the grab.
|
|
1552 Returns t if the grab is successful, nil otherwise.
|
20
|
1553 */
|
|
1554 (device, cursor, ignore_keyboard))
|
0
|
1555 {
|
|
1556 Window w;
|
|
1557 int pointer_mode, result;
|
|
1558 struct device *d = decode_x_device (device);
|
|
1559
|
|
1560 if (!NILP (cursor))
|
|
1561 {
|
|
1562 CHECK_POINTER_GLYPH (cursor);
|
|
1563 cursor = glyph_image_instance (cursor, device, ERROR_ME, 0);
|
|
1564 }
|
|
1565
|
|
1566 if (!NILP (ignore_keyboard))
|
|
1567 pointer_mode = GrabModeSync;
|
|
1568 else
|
|
1569 pointer_mode = GrabModeAsync;
|
|
1570
|
|
1571 w = XtWindow (FRAME_X_TEXT_WIDGET (device_selected_frame (d)));
|
|
1572
|
|
1573 /* #### Possibly this needs to gcpro the cursor somehow, but it doesn't
|
|
1574 seem to cause a problem if XFreeCursor is called on a cursor in use
|
|
1575 in a grab; I suppose the X server counts the grab as a reference
|
|
1576 and doesn't free it until it exits? */
|
|
1577 result = XGrabPointer (DEVICE_X_DISPLAY (d), w,
|
|
1578 False,
|
193
|
1579 ButtonMotionMask |
|
|
1580 ButtonPressMask |
|
|
1581 ButtonReleaseMask |
|
|
1582 PointerMotionHintMask,
|
0
|
1583 GrabModeAsync, /* Keep pointer events flowing */
|
|
1584 pointer_mode, /* Stall keyboard events */
|
|
1585 w, /* Stay in this window */
|
|
1586 (NILP (cursor) ? 0
|
|
1587 : XIMAGE_INSTANCE_X_CURSOR (cursor)),
|
|
1588 CurrentTime);
|
183
|
1589 return (result == GrabSuccess) ? Qt : Qnil;
|
0
|
1590 }
|
|
1591
|
20
|
1592 DEFUN ("x-ungrab-pointer", Fx_ungrab_pointer, 0, 1, 0, /*
|
0
|
1593 Release a pointer grab made with `x-grab-pointer'.
|
|
1594 If optional first arg DEVICE is nil the default device is used.
|
|
1595 If it is t the pointer will be released on all X devices.
|
20
|
1596 */
|
|
1597 (device))
|
0
|
1598 {
|
|
1599 if (!EQ (device, Qt))
|
|
1600 {
|
|
1601 Display *dpy = get_x_display (device);
|
|
1602 XUngrabPointer (dpy, CurrentTime);
|
|
1603 }
|
|
1604 else
|
|
1605 {
|
|
1606 Lisp_Object devcons, concons;
|
|
1607
|
|
1608 DEVICE_LOOP_NO_BREAK (devcons, concons)
|
|
1609 {
|
|
1610 struct device *d = XDEVICE (XCAR (devcons));
|
|
1611
|
|
1612 if (DEVICE_X_P (d))
|
|
1613 XUngrabPointer (DEVICE_X_DISPLAY (d), CurrentTime);
|
|
1614 }
|
|
1615 }
|
|
1616
|
|
1617 return Qnil;
|
|
1618 }
|
|
1619
|
20
|
1620 DEFUN ("x-grab-keyboard", Fx_grab_keyboard, 0, 1, 0, /*
|
0
|
1621 Grab the keyboard on the given device (defaulting to the selected one).
|
|
1622 So long as the keyboard is grabbed, all keyboard events will be delivered
|
|
1623 to emacs -- it is not possible for other X clients to eavesdrop on them.
|
|
1624 Ungrab the keyboard with `x-ungrab-keyboard' (use an unwind-protect).
|
|
1625 Returns t if the grab was successful; nil otherwise.
|
20
|
1626 */
|
|
1627 (device))
|
0
|
1628 {
|
|
1629 struct device *d = decode_x_device (device);
|
|
1630 Window w = XtWindow (FRAME_X_TEXT_WIDGET (device_selected_frame (d)));
|
|
1631 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1632 Status status;
|
|
1633 XSync (dpy, False);
|
|
1634 status = XGrabKeyboard (dpy, w, True,
|
|
1635 /* I don't really understand sync-vs-async
|
|
1636 grabs, but this is what xterm does. */
|
|
1637 GrabModeAsync, GrabModeAsync,
|
|
1638 /* Use the timestamp of the last user action
|
|
1639 read by emacs proper; xterm uses CurrentTime
|
|
1640 but there's a comment that says "wrong"...
|
|
1641 (Despite the name this is the time of the
|
|
1642 last key or mouse event.) */
|
|
1643 DEVICE_X_MOUSE_TIMESTAMP (d));
|
|
1644 if (status == GrabSuccess)
|
|
1645 {
|
|
1646 /* The XUngrabKeyboard should generate a FocusIn back to this
|
|
1647 window but it doesn't unless we explicitly set focus to the
|
|
1648 window first (which should already have it. The net result
|
|
1649 is that without this call when x-ungrab-keyboard is called
|
|
1650 the selected frame ends up not having focus. */
|
|
1651 XSetInputFocus (dpy, w, RevertToParent, DEVICE_X_MOUSE_TIMESTAMP (d));
|
|
1652 return Qt;
|
|
1653 }
|
|
1654 else
|
|
1655 return Qnil;
|
|
1656 }
|
|
1657
|
20
|
1658 DEFUN ("x-ungrab-keyboard", Fx_ungrab_keyboard, 0, 1, 0, /*
|
0
|
1659 Release a keyboard grab made with `x-grab-keyboard'.
|
20
|
1660 */
|
|
1661 (device))
|
0
|
1662 {
|
|
1663 Display *dpy = get_x_display (device);
|
|
1664 XUngrabKeyboard (dpy, CurrentTime);
|
|
1665 return Qnil;
|
|
1666 }
|
|
1667
|
|
1668
|
|
1669 /************************************************************************/
|
|
1670 /* initialization */
|
|
1671 /************************************************************************/
|
|
1672
|
|
1673 void
|
|
1674 syms_of_device_x (void)
|
|
1675 {
|
20
|
1676 DEFSUBR (Fx_debug_mode);
|
|
1677 DEFSUBR (Fx_get_resource);
|
|
1678 DEFSUBR (Fx_get_resource_prefix);
|
|
1679 DEFSUBR (Fx_put_resource);
|
0
|
1680
|
20
|
1681 DEFSUBR (Fdefault_x_device);
|
|
1682 DEFSUBR (Fx_display_visual_class);
|
219
|
1683 DEFSUBR (Fx_display_visual_depth);
|
20
|
1684 DEFSUBR (Fx_server_vendor);
|
|
1685 DEFSUBR (Fx_server_version);
|
|
1686 DEFSUBR (Fx_valid_keysym_name_p);
|
187
|
1687 DEFSUBR (Fx_keysym_hashtable);
|
20
|
1688 DEFSUBR (Fx_keysym_on_keyboard_p);
|
187
|
1689 DEFSUBR (Fx_keysym_on_keyboard_sans_modifiers_p);
|
0
|
1690
|
20
|
1691 DEFSUBR (Fx_grab_pointer);
|
|
1692 DEFSUBR (Fx_ungrab_pointer);
|
|
1693 DEFSUBR (Fx_grab_keyboard);
|
|
1694 DEFSUBR (Fx_ungrab_keyboard);
|
0
|
1695
|
|
1696 defsymbol (&Qx_error, "x-error");
|
|
1697 defsymbol (&Qinit_pre_x_win, "init-pre-x-win");
|
|
1698 defsymbol (&Qinit_post_x_win, "init-post-x-win");
|
|
1699 }
|
|
1700
|
|
1701 void
|
|
1702 console_type_create_device_x (void)
|
|
1703 {
|
|
1704 CONSOLE_HAS_METHOD (x, init_device);
|
|
1705 CONSOLE_HAS_METHOD (x, finish_init_device);
|
|
1706 CONSOLE_HAS_METHOD (x, mark_device);
|
|
1707 CONSOLE_HAS_METHOD (x, delete_device);
|
|
1708 CONSOLE_HAS_METHOD (x, device_pixel_width);
|
|
1709 CONSOLE_HAS_METHOD (x, device_pixel_height);
|
|
1710 CONSOLE_HAS_METHOD (x, device_mm_width);
|
|
1711 CONSOLE_HAS_METHOD (x, device_mm_height);
|
|
1712 CONSOLE_HAS_METHOD (x, device_bitplanes);
|
|
1713 CONSOLE_HAS_METHOD (x, device_color_cells);
|
272
|
1714
|
|
1715 {
|
|
1716 /* Initialize variables to speed up X resource interactions */
|
|
1717 CONST char *valid_resource_chars =
|
|
1718 "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
|
1719 while (*valid_resource_chars)
|
|
1720 valid_resource_char_p[(unsigned int) (*valid_resource_chars++)] = 1;
|
|
1721
|
|
1722 name_char_dynarr = Dynarr_new (char);
|
|
1723 class_char_dynarr = Dynarr_new (char);
|
|
1724 }
|
0
|
1725 }
|
|
1726
|
|
1727 void
|
|
1728 vars_of_device_x (void)
|
|
1729 {
|
|
1730 DEFVAR_LISP ("x-emacs-application-class", &Vx_emacs_application_class /*
|
|
1731 The X application class of the XEmacs process.
|
|
1732 This controls, among other things, the name of the `app-defaults' file
|
|
1733 that XEmacs will use. For changes to this variable to take effect, they
|
|
1734 must be made before the connection to the X server is initialized, that is,
|
|
1735 this variable may only be changed before emacs is dumped, or by setting it
|
|
1736 in the file lisp/term/x-win.el.
|
259
|
1737
|
|
1738 If this variable is nil before the connection to the X server is first
|
|
1739 initialized (which it is by default), the X resource database will be
|
|
1740 consulted and the value will be set according to whether any resources
|
|
1741 are found for the application class `XEmacs'. If the user has set any
|
|
1742 resources for the XEmacs application class, the XEmacs process will use
|
272
|
1743 the application class `XEmacs'. Otherwise, the XEmacs process will use
|
259
|
1744 the application class `Emacs' which is backwards compatible to previous
|
|
1745 XEmacs versions but may conflict with resources intended for GNU Emacs.
|
0
|
1746 */ );
|
259
|
1747 Vx_emacs_application_class = Qnil;
|
0
|
1748
|
|
1749 DEFVAR_LISP ("x-initial-argv-list", &Vx_initial_argv_list /*
|
|
1750 You don't want to know.
|
|
1751 This is used during startup to communicate the remaining arguments in
|
|
1752 `command-line-args-left' to the C code, which passes the args to
|
|
1753 the X initialization code, which removes some args, and then the
|
|
1754 args are placed back into `x-initial-arg-list' and thence into
|
|
1755 `command-line-args-left'. Perhaps `command-line-args-left' should
|
|
1756 just reside in C.
|
|
1757 */ );
|
|
1758 Vx_initial_argv_list = Qnil;
|
|
1759
|
167
|
1760 DEFVAR_BOOL ("x-seppuku-on-epipe", &x_seppuku_on_epipe /*
|
|
1761 When non-nil terminate XEmacs immediately on SIGPIPE from the X server.
|
|
1762 XEmacs doesn't terminate properly on some systems.
|
|
1763 When this variable is non-nil, XEmacs will commit immediate suicide
|
|
1764 when it gets a sigpipe from the X Server.
|
|
1765 */ );
|
|
1766 x_seppuku_on_epipe = 0;
|
|
1767
|
227
|
1768 #if defined(MULE) && (defined(LWLIB_MENUBARS_MOTIF) || defined(HAVE_XIM) || defined (USE_XFONTSET))
|
|
1769 DEFVAR_LISP ("x-app-defaults-directory", &Vx_app_defaults_directory /*
|
|
1770 Used by the Lisp code to communicate to the low level X initialization
|
|
1771 where the localized init files are.
|
|
1772 */ );
|
|
1773 Vx_app_defaults_directory = Qnil;
|
|
1774 #endif
|
|
1775
|
0
|
1776 Fprovide (Qx);
|
|
1777
|
|
1778 staticpro (&Vdefault_x_device);
|
|
1779 Vdefault_x_device = Qnil;
|
|
1780
|
|
1781 error_expected = 0;
|
|
1782 error_occurred = 0;
|
|
1783
|
|
1784 in_resource_setting = 0;
|
|
1785 in_specifier_change_function = 0;
|
|
1786 }
|