428
|
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.
|
2367
|
4 Copyright (C) 2001, 2002, 2004 Ben Wing.
|
428
|
5
|
|
6 This file is part of XEmacs.
|
|
7
|
|
8 XEmacs is free software; you can redistribute it and/or modify it
|
|
9 under the terms of the GNU General Public License as published by the
|
|
10 Free Software Foundation; either version 2, or (at your option) any
|
|
11 later version.
|
|
12
|
|
13 XEmacs is distributed in the hope that it will be useful, but WITHOUT
|
|
14 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
16 for more details.
|
|
17
|
|
18 You should have received a copy of the GNU General Public License
|
|
19 along with XEmacs; see the file COPYING. If not, write to
|
|
20 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
21 Boston, MA 02111-1307, USA. */
|
|
22
|
|
23 /* Synched up with: Not in FSF. */
|
|
24
|
442
|
25 /* 7-8-00 !!#### This file needs definite Mule review. */
|
|
26
|
428
|
27 /* Original authors: Jamie Zawinski and the FSF */
|
|
28 /* Rewritten by Ben Wing and Chuck Thompson. */
|
|
29
|
|
30 #include <config.h>
|
|
31 #include "lisp.h"
|
|
32
|
|
33 #include "buffer.h"
|
872
|
34 #include "device-impl.h"
|
428
|
35 #include "elhash.h"
|
|
36 #include "events.h"
|
|
37 #include "faces.h"
|
3707
|
38 #include "file-coding.h"
|
872
|
39 #include "frame-impl.h"
|
2684
|
40 #include "process.h" /* for egetenv */
|
428
|
41 #include "redisplay.h"
|
|
42 #include "sysdep.h"
|
|
43 #include "window.h"
|
|
44
|
872
|
45 #include "console-x-impl.h"
|
800
|
46 #include "glyphs-x.h"
|
|
47 #include "objects-x.h"
|
|
48
|
428
|
49 #include "sysfile.h"
|
|
50 #include "systime.h"
|
|
51
|
800
|
52 #include "xintrinsicp.h" /* CoreP.h needs this */
|
|
53 #include <X11/CoreP.h> /* Numerous places access the fields of
|
|
54 a core widget directly. We could
|
|
55 use XtGetValues(), but ... */
|
|
56 #include "xgccache.h"
|
|
57 #include <X11/Shell.h>
|
|
58 #include "xmu.h"
|
|
59
|
442
|
60 #if defined(HAVE_SHLIB) && defined(LWLIB_USES_ATHENA) && !defined(HAVE_ATHENA_3D)
|
440
|
61 #include "sysdll.h"
|
442
|
62 #endif /* HAVE_SHLIB and LWLIB_USES_ATHENA and not HAVE_ATHENA_3D */
|
440
|
63
|
428
|
64 #ifdef HAVE_OFFIX_DND
|
|
65 #include "offix.h"
|
|
66 #endif
|
|
67
|
771
|
68 #ifdef MULE
|
428
|
69 Lisp_Object Vx_app_defaults_directory;
|
771
|
70 Lisp_Object Qget_coding_system_from_locale;
|
428
|
71 #endif
|
|
72
|
|
73 /* Qdisplay in general.c */
|
|
74 Lisp_Object Qx_error;
|
|
75 Lisp_Object Qinit_pre_x_win, Qinit_post_x_win;
|
|
76
|
|
77 /* The application class of Emacs. */
|
|
78 Lisp_Object Vx_emacs_application_class;
|
|
79
|
|
80 Lisp_Object Vx_initial_argv_list; /* #### ugh! */
|
|
81
|
|
82 static XrmOptionDescRec emacs_options[] =
|
|
83 {
|
|
84 {"-geometry", ".geometry", XrmoptionSepArg, NULL},
|
|
85 {"-iconic", ".iconic", XrmoptionNoArg, "yes"},
|
|
86
|
|
87 {"-internal-border-width", "*EmacsFrame.internalBorderWidth", XrmoptionSepArg, NULL},
|
|
88 {"-ib", "*EmacsFrame.internalBorderWidth", XrmoptionSepArg, NULL},
|
|
89 {"-scrollbar-width", "*EmacsFrame.scrollBarWidth", XrmoptionSepArg, NULL},
|
|
90 {"-scrollbar-height", "*EmacsFrame.scrollBarHeight", XrmoptionSepArg, NULL},
|
|
91
|
|
92 {"-privatecolormap", ".privateColormap", XrmoptionNoArg, "yes"},
|
|
93 {"-visual", ".EmacsVisual", XrmoptionSepArg, NULL},
|
|
94
|
|
95 /* #### Beware! If the type of the shell changes, update this. */
|
|
96 {"-T", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
|
97 {"-wn", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
|
98 {"-title", "*TopLevelEmacsShell.title", XrmoptionSepArg, NULL},
|
|
99
|
|
100 {"-iconname", "*TopLevelEmacsShell.iconName", XrmoptionSepArg, NULL},
|
|
101 {"-in", "*TopLevelEmacsShell.iconName", XrmoptionSepArg, NULL},
|
|
102 {"-mc", "*pointerColor", XrmoptionSepArg, NULL},
|
|
103 {"-cr", "*cursorColor", XrmoptionSepArg, NULL},
|
|
104 {"-fontset", "*FontSet", XrmoptionSepArg, NULL},
|
|
105 };
|
|
106
|
1204
|
107 static const struct memory_description x_device_data_description_1 [] = {
|
|
108 { XD_LISP_OBJECT, offsetof (struct x_device, x_keysym_map_hash_table) },
|
|
109 { XD_LISP_OBJECT, offsetof (struct x_device, WM_COMMAND_frame) },
|
|
110 { XD_END }
|
|
111 };
|
|
112
|
3092
|
113 #ifdef NEW_GC
|
|
114 DEFINE_LRECORD_IMPLEMENTATION ("x-device", x_device,
|
|
115 1, /*dumpable-flag*/
|
|
116 0, 0, 0, 0, 0,
|
|
117 x_device_data_description_1,
|
|
118 Lisp_X_Device);
|
|
119 #else /* not NEW_GC */
|
1204
|
120 extern const struct sized_memory_description x_device_data_description;
|
|
121
|
|
122 const struct sized_memory_description x_device_data_description = {
|
|
123 sizeof (struct x_device), x_device_data_description_1
|
|
124 };
|
3092
|
125 #endif /* not NEW_GC */
|
1204
|
126
|
428
|
127 /* Functions to synchronize mirroring resources and specifiers */
|
|
128 int in_resource_setting;
|
|
129
|
|
130 /************************************************************************/
|
|
131 /* helper functions */
|
|
132 /************************************************************************/
|
|
133
|
|
134 /* JH 97/11/25 removed the static declaration because I need it during setup in event-Xt... */
|
|
135 struct device * get_device_from_display_1 (Display *dpy);
|
|
136 struct device *
|
|
137 get_device_from_display_1 (Display *dpy)
|
|
138 {
|
|
139 Lisp_Object devcons, concons;
|
|
140
|
|
141 DEVICE_LOOP_NO_BREAK (devcons, concons)
|
|
142 {
|
|
143 struct device *d = XDEVICE (XCAR (devcons));
|
|
144 if (DEVICE_X_P (d) && DEVICE_X_DISPLAY (d) == dpy)
|
|
145 return d;
|
|
146 }
|
|
147
|
|
148 return 0;
|
|
149 }
|
|
150
|
|
151 struct device *
|
|
152 get_device_from_display (Display *dpy)
|
|
153 {
|
|
154 struct device *d = get_device_from_display_1 (dpy);
|
|
155
|
|
156 #if !defined(INFODOCK)
|
|
157 # define FALLBACK_RESOURCE_NAME "xemacs"
|
|
158 # else
|
|
159 # define FALLBACK_RESOURCE_NAME "infodock"
|
|
160 #endif
|
|
161
|
853
|
162 if (!d)
|
|
163 {
|
|
164 /* This isn't one of our displays. Let's crash? */
|
|
165 stderr_out
|
|
166 ("\n%s: Fatal X Condition. Asked about display we don't own: \"%s\"\n",
|
|
167 (STRINGP (Vinvocation_name) ?
|
|
168 (char *) XSTRING_DATA (Vinvocation_name) : FALLBACK_RESOURCE_NAME),
|
|
169 DisplayString (dpy) ? DisplayString (dpy) : "???");
|
2500
|
170 ABORT();
|
853
|
171 }
|
428
|
172
|
|
173 #undef FALLBACK_RESOURCE_NAME
|
|
174
|
|
175 return d;
|
|
176 }
|
|
177
|
|
178 struct device *
|
|
179 decode_x_device (Lisp_Object device)
|
|
180 {
|
793
|
181 device = wrap_device (decode_device (device));
|
428
|
182 CHECK_X_DEVICE (device);
|
|
183 return XDEVICE (device);
|
|
184 }
|
|
185
|
|
186 static Display *
|
|
187 get_x_display (Lisp_Object device)
|
|
188 {
|
|
189 return DEVICE_X_DISPLAY (decode_x_device (device));
|
|
190 }
|
|
191
|
771
|
192 static Lisp_Object
|
2333
|
193 coding_system_of_xrm_database (XrmDatabase USED_IF_MULE (db))
|
771
|
194 {
|
|
195 #ifdef MULE
|
3707
|
196 const Extbyte *locale;
|
|
197 Lisp_Object localestr;
|
|
198 static XrmDatabase last_xrm_db;
|
|
199
|
|
200 /* This will always be zero, nil or an actual coding system object, so no
|
|
201 need to worry about GCPROing it--it'll be protected from garbage
|
|
202 collection by means of Vcoding_system_hash_table in file-coding.c. */
|
|
203 static Lisp_Object last_coding_system;
|
|
204
|
|
205 if (db == last_xrm_db)
|
|
206 {
|
|
207 return last_coding_system;
|
|
208 }
|
|
209
|
|
210 last_xrm_db = db;
|
|
211
|
|
212 locale = XrmLocaleOfDatabase (db);
|
|
213 localestr = build_ext_string (locale, Qbinary);
|
|
214 last_coding_system = call1 (Qget_coding_system_from_locale, localestr);
|
|
215
|
|
216 return last_coding_system;
|
771
|
217 #else
|
|
218 return Qbinary;
|
|
219 #endif
|
|
220 }
|
|
221
|
428
|
222
|
|
223 /************************************************************************/
|
|
224 /* initializing an X connection */
|
|
225 /************************************************************************/
|
|
226
|
756
|
227 static struct device *device_being_initialized = NULL;
|
|
228
|
428
|
229 static void
|
|
230 allocate_x_device_struct (struct device *d)
|
|
231 {
|
3092
|
232 #ifdef NEW_GC
|
|
233 d->device_data = alloc_lrecord_type (struct x_device, &lrecord_x_device);
|
|
234 #else /* not NEW_GC */
|
428
|
235 d->device_data = xnew_and_zero (struct x_device);
|
3092
|
236 #endif /* not NEW_GC */
|
428
|
237 }
|
|
238
|
|
239 static void
|
|
240 Xatoms_of_device_x (struct device *d)
|
|
241 {
|
|
242 Display *D = DEVICE_X_DISPLAY (d);
|
|
243
|
|
244 DEVICE_XATOM_WM_PROTOCOLS (d) = XInternAtom (D, "WM_PROTOCOLS", False);
|
|
245 DEVICE_XATOM_WM_DELETE_WINDOW(d) = XInternAtom (D, "WM_DELETE_WINDOW",False);
|
|
246 DEVICE_XATOM_WM_SAVE_YOURSELF(d) = XInternAtom (D, "WM_SAVE_YOURSELF",False);
|
|
247 DEVICE_XATOM_WM_TAKE_FOCUS (d) = XInternAtom (D, "WM_TAKE_FOCUS", False);
|
|
248 DEVICE_XATOM_WM_STATE (d) = XInternAtom (D, "WM_STATE", False);
|
|
249 }
|
|
250
|
|
251 static void
|
|
252 sanity_check_geometry_resource (Display *dpy)
|
|
253 {
|
771
|
254 Extbyte *app_name, *app_class, *s;
|
|
255 Extbyte buf1 [255], buf2 [255];
|
|
256 Extbyte *type;
|
428
|
257 XrmValue value;
|
|
258 XtGetApplicationNameAndClass (dpy, &app_name, &app_class);
|
|
259 strcpy (buf1, app_name);
|
|
260 strcpy (buf2, app_class);
|
|
261 for (s = buf1; *s; s++) if (*s == '.') *s = '_';
|
|
262 strcat (buf1, "._no_._such_._resource_.geometry");
|
|
263 strcat (buf2, "._no_._such_._resource_.Geometry");
|
|
264 if (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value) == True)
|
|
265 {
|
867
|
266 Ibyte *app_name_int, *app_class_int, *value_addr_int;
|
771
|
267 Lisp_Object codesys = coding_system_of_xrm_database (XtDatabase (dpy));
|
|
268 EXTERNAL_TO_C_STRING (app_name, app_name_int, codesys);
|
|
269 EXTERNAL_TO_C_STRING (app_class, app_class_int, codesys);
|
|
270 EXTERNAL_TO_C_STRING (value.addr, value_addr_int, codesys);
|
|
271
|
428
|
272 warn_when_safe (Qgeometry, Qerror,
|
|
273 "\n"
|
|
274 "Apparently \"%s*geometry: %s\" or \"%s*geometry: %s\" was\n"
|
|
275 "specified in the resource database. Specifying \"*geometry\" will make\n"
|
|
276 "XEmacs (and most other X programs) malfunction in obscure ways. (i.e.\n"
|
|
277 "the Xt or Xm libraries will probably crash, which is a very bad thing.)\n"
|
|
278 "You should always use \".geometry\" or \"*EmacsFrame.geometry\" instead.\n",
|
771
|
279 app_name_int, value_addr_int,
|
|
280 app_class_int, value_addr_int);
|
428
|
281 suppress_early_error_handler_backtrace = 1;
|
563
|
282 syntax_error ("Invalid geometry resource", Qunbound);
|
428
|
283 }
|
|
284 }
|
|
285
|
|
286 static void
|
|
287 x_init_device_class (struct device *d)
|
|
288 {
|
|
289 if (DEVICE_X_DEPTH(d) > 2)
|
|
290 {
|
1204
|
291 switch (DEVICE_X_VISUAL(d)->X_CLASSFIELD)
|
428
|
292 {
|
|
293 case StaticGray:
|
|
294 case GrayScale:
|
|
295 DEVICE_CLASS (d) = Qgrayscale;
|
|
296 break;
|
|
297 default:
|
|
298 DEVICE_CLASS (d) = Qcolor;
|
|
299 }
|
|
300 }
|
|
301 else
|
|
302 DEVICE_CLASS (d) = Qmono;
|
|
303 }
|
|
304
|
|
305 /*
|
|
306 * Figure out what application name to use for xemacs
|
|
307 *
|
|
308 * Since we have decomposed XtOpenDisplay into XOpenDisplay and
|
|
309 * XtDisplayInitialize, we no longer get this for free.
|
|
310 *
|
|
311 * If there is a `-name' argument in argv, use that.
|
|
312 * Otherwise use the last component of argv[0].
|
|
313 *
|
|
314 * I have removed the gratuitous use of getenv("RESOURCE_NAME")
|
|
315 * which was in X11R5, but left the matching of any prefix of `-name'.
|
|
316 * Finally, if all else fails, return `xemacs', as it is more
|
|
317 * appropriate (X11R5 returns `main').
|
|
318 */
|
442
|
319 static Extbyte *
|
|
320 compute_x_app_name (int argc, Extbyte **argv)
|
428
|
321 {
|
|
322 int i;
|
442
|
323 Extbyte *ptr;
|
428
|
324
|
|
325 for (i = 1; i < argc - 1; i++)
|
|
326 if (!strncmp(argv[i], "-name", max (2, strlen (argv[1]))))
|
|
327 return argv[i+1];
|
|
328
|
|
329 if (argc > 0 && argv[0] && *argv[0])
|
|
330 return (ptr = strrchr (argv[0], '/')) ? ++ptr : argv[0];
|
|
331
|
|
332 return "xemacs";
|
|
333 }
|
|
334
|
|
335 /*
|
|
336 * This function figures out whether the user has any resources of the
|
|
337 * form "XEmacs.foo" or "XEmacs*foo".
|
|
338 *
|
|
339 * Currently we only consult the display's global resources; to look
|
|
340 * for screen specific resources, we would need to also consult:
|
|
341 * xdefs = XScreenResourceString(ScreenOfDisplay(dpy, scrno));
|
|
342 */
|
|
343 static int
|
|
344 have_xemacs_resources_in_xrdb (Display *dpy)
|
|
345 {
|
|
346 char *xdefs, *key;
|
|
347 int len;
|
|
348
|
|
349 #ifdef INFODOCK
|
|
350 key = "InfoDock";
|
|
351 #else
|
|
352 key = "XEmacs";
|
|
353 #endif
|
|
354 len = strlen (key);
|
|
355
|
|
356 if (!dpy)
|
|
357 return 0;
|
|
358
|
|
359 xdefs = XResourceManagerString (dpy); /* don't free - owned by X */
|
|
360 while (xdefs && *xdefs)
|
|
361 {
|
|
362 if (strncmp (xdefs, key, len) == 0 &&
|
|
363 (xdefs[len] == '*' || xdefs[len] == '.'))
|
|
364 return 1;
|
|
365
|
|
366 while (*xdefs && *xdefs++ != '\n') /* find start of next entry.. */
|
|
367 ;
|
|
368 }
|
|
369
|
|
370 return 0;
|
|
371 }
|
|
372
|
|
373 /* Only the characters [-_A-Za-z0-9] are allowed in the individual
|
|
374 components of a resource. Convert invalid characters to `-' */
|
|
375
|
|
376 static char valid_resource_char_p[256];
|
|
377
|
|
378 static void
|
771
|
379 validify_resource_component (Extbyte *str, Bytecount len)
|
428
|
380 {
|
|
381 for (; len; len--, str++)
|
|
382 if (!valid_resource_char_p[(unsigned char) (*str)])
|
|
383 *str = '-';
|
|
384 }
|
|
385
|
|
386 static void
|
771
|
387 Dynarr_add_validified_lisp_string (Extbyte_dynarr *cda, Lisp_Object str)
|
428
|
388 {
|
771
|
389 Bytecount len;
|
|
390 Extbyte *data;
|
|
391
|
|
392 TO_EXTERNAL_FORMAT (LISP_STRING, str, ALLOCA, (data, len), Qbinary);
|
|
393 Dynarr_add_many (cda, data, len);
|
|
394 validify_resource_component (Dynarr_atp (cda, Dynarr_length (cda) - len),
|
|
395 len);
|
428
|
396 }
|
|
397
|
|
398 #if 0
|
|
399 /* compare visual info for qsorting */
|
|
400 static int
|
|
401 x_comp_visual_info (const void *elem1, const void *elem2)
|
|
402 {
|
|
403 XVisualInfo *left, *right;
|
|
404
|
|
405 left = (XVisualInfo *)elem1;
|
|
406 right = (XVisualInfo *)elem2;
|
|
407
|
|
408 if ( left == NULL )
|
|
409 return -1;
|
|
410 if ( right == NULL )
|
|
411 return 1;
|
|
412
|
771
|
413 if ( left->depth > right->depth )
|
428
|
414 return 1;
|
771
|
415 else if ( left->depth == right->depth )
|
|
416 {
|
|
417 if ( left->colormap_size > right->colormap_size )
|
|
418 return 1;
|
1204
|
419 if ( left->X_CLASSFIELD > right->X_CLASSFIELD )
|
771
|
420 return 1;
|
1204
|
421 else if ( left->X_CLASSFIELD < right->X_CLASSFIELD )
|
771
|
422 return -1;
|
|
423 else
|
|
424 return 0;
|
|
425 }
|
|
426 else
|
428
|
427 return -1;
|
|
428 }
|
|
429 #endif /* if 0 */
|
|
430
|
|
431 #define XXX_IMAGE_LIBRARY_IS_SOMEWHAT_BROKEN
|
|
432 static Visual *
|
|
433 x_try_best_visual_class (Screen *screen, int scrnum, int visual_class)
|
|
434 {
|
|
435 Display *dpy = DisplayOfScreen (screen);
|
|
436 XVisualInfo vi_in;
|
|
437 XVisualInfo *vi_out = NULL;
|
|
438 int out_count;
|
|
439
|
1204
|
440 vi_in.X_CLASSFIELD = visual_class;
|
428
|
441 vi_in.screen = scrnum;
|
|
442 vi_out = XGetVisualInfo (dpy, (VisualClassMask | VisualScreenMask),
|
|
443 &vi_in, &out_count);
|
|
444 if ( vi_out )
|
|
445 {
|
|
446 int i, best;
|
|
447 Visual *visual;
|
|
448 for (i = 0, best = 0; i < out_count; i++)
|
|
449 /* It's better if it's deeper, or if it's the same depth with
|
|
450 more cells (does that ever happen? Well, it could...)
|
|
451 NOTE: don't allow pseudo color to get larger than 8! */
|
|
452 if (((vi_out [i].depth > vi_out [best].depth) ||
|
|
453 ((vi_out [i].depth == vi_out [best].depth) &&
|
|
454 (vi_out [i].colormap_size > vi_out [best].colormap_size)))
|
|
455 #ifdef XXX_IMAGE_LIBRARY_IS_SOMEWHAT_BROKEN
|
|
456 /* For now, the image library doesn't like PseudoColor visuals
|
|
457 of depths other than 1 or 8. Depths greater than 8 only occur
|
|
458 on machines which have TrueColor anyway, so probably we'll end
|
|
459 up using that (it is the one that `Best' would pick) but if a
|
|
460 PseudoColor visual is explicitly specified, pick the 8 bit one.
|
|
461 */
|
|
462 && (visual_class != PseudoColor ||
|
|
463 vi_out [i].depth == 1 ||
|
|
464 vi_out [i].depth == 8)
|
|
465 #endif
|
|
466
|
|
467 /* SGI has 30-bit deep visuals. Ignore them.
|
|
468 (We only have 24-bit data anyway.)
|
|
469 */
|
|
470 && (vi_out [i].depth <= 24)
|
|
471 )
|
|
472 best = i;
|
|
473 visual = vi_out[best].visual;
|
|
474 XFree ((char *) vi_out);
|
|
475 return visual;
|
|
476 }
|
|
477 else
|
|
478 return 0;
|
|
479 }
|
|
480
|
|
481 static int
|
|
482 x_get_visual_depth (Display *dpy, Visual *visual)
|
|
483 {
|
|
484 XVisualInfo vi_in;
|
|
485 XVisualInfo *vi_out;
|
|
486 int out_count, d;
|
|
487
|
|
488 vi_in.visualid = XVisualIDFromVisual (visual);
|
|
489 vi_out = XGetVisualInfo (dpy, /*VisualScreenMask|*/VisualIDMask,
|
|
490 &vi_in, &out_count);
|
2500
|
491 if (! vi_out) ABORT ();
|
428
|
492 d = vi_out [0].depth;
|
|
493 XFree ((char *) vi_out);
|
|
494 return d;
|
|
495 }
|
|
496
|
|
497 static Visual *
|
|
498 x_try_best_visual (Display *dpy, int scrnum)
|
|
499 {
|
|
500 Visual *visual = NULL;
|
|
501 Screen *screen = ScreenOfDisplay (dpy, scrnum);
|
|
502 if ((visual = x_try_best_visual_class (screen, scrnum, TrueColor))
|
|
503 && x_get_visual_depth (dpy, visual) >= 16 )
|
|
504 return visual;
|
|
505 if ((visual = x_try_best_visual_class (screen, scrnum, PseudoColor)))
|
|
506 return visual;
|
|
507 if ((visual = x_try_best_visual_class (screen, scrnum, TrueColor)))
|
|
508 return visual;
|
|
509 #ifdef DIRECTCOLOR_WORKS
|
|
510 if ((visual = x_try_best_visual_class (screen, scrnum, DirectColor)))
|
|
511 return visual;
|
|
512 #endif
|
|
513
|
|
514 visual = DefaultVisualOfScreen (screen);
|
|
515 if ( x_get_visual_depth (dpy, visual) >= 8 )
|
|
516 return visual;
|
|
517
|
|
518 if ((visual = x_try_best_visual_class (screen, scrnum, StaticGray)))
|
|
519 return visual;
|
|
520 if ((visual = x_try_best_visual_class (screen, scrnum, GrayScale)))
|
|
521 return visual;
|
|
522 return DefaultVisualOfScreen (screen);
|
|
523 }
|
|
524
|
|
525
|
|
526 static void
|
2286
|
527 x_init_device (struct device *d, Lisp_Object UNUSED (props))
|
428
|
528 {
|
2367
|
529 /* !!#### */
|
428
|
530 Lisp_Object display;
|
|
531 Display *dpy;
|
|
532 Widget app_shell;
|
|
533 int argc;
|
442
|
534 Extbyte **argv;
|
|
535 const char *app_class;
|
|
536 const char *app_name;
|
|
537 const char *disp_name;
|
428
|
538 Visual *visual = NULL;
|
|
539 int depth = 8; /* shut up the compiler */
|
|
540 Colormap cmap;
|
|
541 int screen;
|
|
542 /* */
|
|
543 int best_visual_found = 0;
|
|
544
|
442
|
545 #if defined(HAVE_SHLIB) && defined(LWLIB_USES_ATHENA) && !defined(HAVE_ATHENA_3D)
|
440
|
546 /*
|
|
547 * In order to avoid the lossage with flat Athena widgets dynamically
|
|
548 * linking to one of the ThreeD variants, using the dynamic symbol helpers
|
|
549 * to look for symbols that shouldn't be there and refusing to run if they
|
|
550 * are seems a less toxic idea than having XEmacs crash when we try and
|
|
551 * use a subclass of a widget that has changed size.
|
|
552 *
|
|
553 * It's ugly, I know, and not going to work everywhere. It seems better to
|
|
554 * do our damnedest to try and tell the user what to expect rather than
|
|
555 * simply blow up though.
|
|
556 *
|
|
557 * All the ThreeD variants I have access to define the following function
|
|
558 * symbols in the shared library. The flat Xaw library does not define them:
|
|
559 *
|
|
560 * Xaw3dComputeBottomShadowRGB
|
|
561 * Xaw3dComputeTopShadowRGB
|
|
562 *
|
|
563 * So far only Linux has shown this problem. This seems to be portable to
|
|
564 * all the distributions (certainly all the ones I checked - Debian and
|
|
565 * Redhat)
|
|
566 *
|
|
567 * This will only work, sadly, with dlopen() -- the other dynamic linkers
|
|
568 * are simply not capable of doing what is needed. :/
|
|
569 */
|
|
570
|
|
571 {
|
|
572 /* Get a dll handle to the main process. */
|
1706
|
573 dll_handle xaw_dll_handle = dll_open (Qnil);
|
440
|
574
|
|
575 /* Did that fail? If so, continue without error.
|
|
576 * We could die here but, well, that's unfriendly and all -- plus I feel
|
|
577 * better about some crashing somewhere rather than preventing a perfectly
|
|
578 * good configuration working just because dll_open failed.
|
|
579 */
|
|
580 if (xaw_dll_handle != NULL)
|
|
581 {
|
|
582 /* Look for the Xaw3d function */
|
|
583 dll_func xaw_function_handle =
|
|
584 dll_function (xaw_dll_handle, "Xaw3dComputeTopShadowRGB");
|
|
585
|
|
586 /* If we found it, warn the user in big, nasty, unfriendly letters */
|
|
587 if (xaw_function_handle != NULL)
|
|
588 {
|
793
|
589 warn_when_safe (Qdevice, Qcritical, "\n"
|
440
|
590 "It seems that XEmacs is built dynamically linked to the flat Athena widget\n"
|
|
591 "library but it finds a 3D Athena variant with the same name at runtime.\n"
|
|
592 "\n"
|
|
593 "This WILL cause your XEmacs process to dump core at some point.\n"
|
|
594 "You should not continue to use this binary without resolving this issue.\n"
|
|
595 "\n"
|
|
596 "This can be solved with the xaw-wrappers package under Debian\n"
|
|
597 "(register XEmacs as incompatible with all 3d widget sets, see\n"
|
|
598 "update-xaw-wrappers(8) and .../doc/xaw-wrappers/README.packagers). It\n"
|
|
599 "can be verified by checking the runtime path in /etc/ld.so.conf and by\n"
|
|
600 "using `ldd /path/to/xemacs' under other Linux distributions. One\n"
|
|
601 "solution is to use LD_PRELOAD or LD_LIBRARY_PATH to force ld.so to\n"
|
|
602 "load the flat Athena widget library instead of the aliased 3D widget\n"
|
|
603 "library (see ld.so(8) for use of these environment variables).\n\n"
|
|
604 );
|
|
605
|
|
606 }
|
|
607
|
|
608 /* Otherwise release the handle to the library
|
|
609 * No error catch here; I can't think of a way to recover anyhow.
|
|
610 */
|
|
611 dll_close (xaw_dll_handle);
|
|
612 }
|
|
613 }
|
442
|
614 #endif /* HAVE_SHLIB and LWLIB_USES_ATHENA and not HAVE_ATHENA_3D */
|
440
|
615
|
428
|
616 display = DEVICE_CONNECTION (d);
|
|
617
|
|
618 allocate_x_device_struct (d);
|
|
619
|
|
620 make_argc_argv (Vx_initial_argv_list, &argc, &argv);
|
|
621
|
442
|
622 LISP_STRING_TO_EXTERNAL (display, disp_name, Qctext);
|
428
|
623
|
|
624 /*
|
|
625 * Break apart the old XtOpenDisplay call into XOpenDisplay and
|
|
626 * XtDisplayInitialize so we can figure out whether there
|
|
627 * are any XEmacs resources in the resource database before
|
|
628 * we initialize Xt. This is so we can automagically support
|
|
629 * both `Emacs' and `XEmacs' application classes.
|
|
630 */
|
|
631 slow_down_interrupts ();
|
|
632 /* May not be needed but XtOpenDisplay could not deal with signals here. */
|
756
|
633 device_being_initialized = d;
|
428
|
634 dpy = DEVICE_X_DISPLAY (d) = XOpenDisplay (disp_name);
|
756
|
635 device_being_initialized = NULL;
|
428
|
636 speed_up_interrupts ();
|
|
637
|
|
638 if (dpy == 0)
|
|
639 {
|
|
640 suppress_early_error_handler_backtrace = 1;
|
563
|
641 gui_error ("X server not responding\n", display);
|
428
|
642 }
|
|
643
|
|
644 if (STRINGP (Vx_emacs_application_class) &&
|
|
645 XSTRING_LENGTH (Vx_emacs_application_class) > 0)
|
442
|
646 LISP_STRING_TO_EXTERNAL (Vx_emacs_application_class, app_class, Qctext);
|
428
|
647 else
|
|
648 {
|
2681
|
649 if (egetenv ("USE_EMACS_AS_DEFAULT_APPLICATION_CLASS"))
|
|
650 {
|
|
651 app_class = (NILP (Vx_emacs_application_class) &&
|
|
652 have_xemacs_resources_in_xrdb (dpy))
|
428
|
653 #ifdef INFODOCK
|
2681
|
654 ? "InfoDock"
|
428
|
655 #else
|
2681
|
656 ? "XEmacs"
|
428
|
657 #endif
|
2681
|
658 : "Emacs";
|
|
659 }
|
|
660 else
|
|
661 {
|
|
662 app_class = "XEmacs";
|
|
663 }
|
|
664
|
428
|
665 /* need to update Vx_emacs_application_class: */
|
|
666 Vx_emacs_application_class = build_string (app_class);
|
|
667 }
|
|
668
|
|
669 slow_down_interrupts ();
|
|
670 /* May not be needed but XtOpenDisplay could not deal with signals here.
|
|
671 Yuck. */
|
|
672 XtDisplayInitialize (Xt_app_con, dpy, compute_x_app_name (argc, argv),
|
|
673 app_class, emacs_options,
|
442
|
674 XtNumber (emacs_options), &argc, (char **) argv);
|
428
|
675 speed_up_interrupts ();
|
|
676
|
|
677 screen = DefaultScreen (dpy);
|
|
678
|
|
679 #ifdef MULE
|
|
680 {
|
|
681 /* Read in locale-specific resources from
|
|
682 data-directory/app-defaults/$LANG/Emacs.
|
|
683 This is in addition to the standard app-defaults files, and
|
|
684 does not override resources defined elsewhere */
|
771
|
685 const Extbyte *data_dir;
|
|
686 Extbyte *path;
|
3644
|
687 Extbyte *format;
|
428
|
688 XrmDatabase db = XtDatabase (dpy); /* #### XtScreenDatabase(dpy) ? */
|
3644
|
689 Extbyte *locale = xstrdup (XrmLocaleOfDatabase (db));
|
|
690 Extbyte *locale_end;
|
428
|
691
|
|
692 if (STRINGP (Vx_app_defaults_directory) &&
|
|
693 XSTRING_LENGTH (Vx_app_defaults_directory) > 0)
|
|
694 {
|
771
|
695 LISP_STRING_TO_EXTERNAL (Vx_app_defaults_directory, data_dir,
|
|
696 Qfile_name);
|
2367
|
697 path = alloca_extbytes (strlen (data_dir) + strlen (locale) + 7);
|
3644
|
698 format = "%s%s/Emacs";
|
428
|
699 }
|
|
700 else if (STRINGP (Vdata_directory) && XSTRING_LENGTH (Vdata_directory) > 0)
|
|
701 {
|
442
|
702 LISP_STRING_TO_EXTERNAL (Vdata_directory, data_dir, Qfile_name);
|
2367
|
703 path = alloca_extbytes (strlen (data_dir) + 13 + strlen (locale) + 7);
|
3644
|
704 format = "%sapp-defaults/%s/Emacs";
|
428
|
705 }
|
3644
|
706
|
|
707 /*
|
|
708 * The general form for $LANG is <language>_<country>.<encoding>. Try
|
|
709 * that form, <language>_<country> and <language> and load for first
|
|
710 * app-defaults file found.
|
|
711 */
|
|
712
|
|
713 sprintf (path, format, data_dir, locale);
|
|
714 if (!access (path, R_OK))
|
|
715 XrmCombineFileDatabase (path, &db, False);
|
|
716
|
|
717 if ((locale_end = strchr(locale, '.'))) {
|
|
718 *locale_end = '\0';
|
|
719 sprintf (path, format, data_dir, locale);
|
|
720
|
|
721 if (!access (path, R_OK))
|
|
722 XrmCombineFileDatabase (path, &db, False);
|
|
723 }
|
|
724
|
|
725 if ((locale_end = strchr(locale, '_'))) {
|
|
726 *locale_end = '\0';
|
|
727 sprintf (path, format, data_dir, locale);
|
|
728
|
|
729 if (!access (path, R_OK))
|
|
730 XrmCombineFileDatabase (path, &db, False);
|
|
731 }
|
|
732
|
|
733 xfree (locale, Extbyte*);
|
428
|
734 }
|
|
735 #endif /* MULE */
|
|
736
|
|
737 if (NILP (DEVICE_NAME (d)))
|
|
738 DEVICE_NAME (d) = display;
|
|
739
|
|
740 /* We're going to modify the string in-place, so be a nice XEmacs */
|
|
741 DEVICE_NAME (d) = Fcopy_sequence (DEVICE_NAME (d));
|
|
742 /* colons and periods can't appear in individual elements of resource
|
|
743 strings */
|
|
744
|
|
745 XtGetApplicationNameAndClass (dpy, (char **) &app_name, (char **) &app_class);
|
|
746 /* search for a matching visual if requested by the user, or setup the display default */
|
|
747 {
|
|
748 int resource_name_length = max (sizeof (".emacsVisual"),
|
|
749 sizeof (".privateColormap"));
|
|
750 char *buf1 = alloca_array (char, strlen (app_name) + resource_name_length);
|
|
751 char *buf2 = alloca_array (char, strlen (app_class) + resource_name_length);
|
|
752 char *type;
|
|
753 XrmValue value;
|
|
754
|
|
755 sprintf (buf1, "%s.emacsVisual", app_name);
|
|
756 sprintf (buf2, "%s.EmacsVisual", app_class);
|
|
757 if (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value) == True)
|
|
758 {
|
|
759 int cnt = 0;
|
|
760 int vis_class = PseudoColor;
|
|
761 XVisualInfo vinfo;
|
|
762 char *str = (char*) value.addr;
|
|
763
|
|
764 #define CHECK_VIS_CLASS(visual_class) \
|
|
765 else if (memcmp (str, #visual_class, sizeof (#visual_class) - 1) == 0) \
|
|
766 cnt = sizeof (#visual_class) - 1, vis_class = visual_class
|
|
767
|
|
768 if (1)
|
|
769 ;
|
|
770 CHECK_VIS_CLASS (StaticGray);
|
|
771 CHECK_VIS_CLASS (StaticColor);
|
|
772 CHECK_VIS_CLASS (TrueColor);
|
|
773 CHECK_VIS_CLASS (GrayScale);
|
|
774 CHECK_VIS_CLASS (PseudoColor);
|
|
775 CHECK_VIS_CLASS (DirectColor);
|
|
776
|
|
777 if (cnt)
|
|
778 {
|
|
779 depth = atoi (str + cnt);
|
|
780 if (depth == 0)
|
|
781 {
|
771
|
782 stderr_out ("Invalid Depth specification in %s... "
|
|
783 "ignoring...\n", str);
|
428
|
784 }
|
|
785 else
|
|
786 {
|
|
787 if (XMatchVisualInfo (dpy, screen, depth, vis_class, &vinfo))
|
|
788 {
|
|
789 visual = vinfo.visual;
|
|
790 }
|
|
791 else
|
|
792 {
|
771
|
793 stderr_out ("Can't match the requested visual %s... "
|
|
794 "using defaults\n", str);
|
428
|
795 }
|
|
796 }
|
|
797 }
|
|
798 else
|
|
799 {
|
771
|
800 stderr_out ("Invalid Visual specification in %s... "
|
|
801 "ignoring.\n", str);
|
428
|
802 }
|
|
803 }
|
|
804 if (visual == NULL)
|
|
805 {
|
|
806 /*
|
|
807 visual = DefaultVisual(dpy, screen);
|
|
808 depth = DefaultDepth(dpy, screen);
|
|
809 */
|
|
810 visual = x_try_best_visual (dpy, screen);
|
|
811 depth = x_get_visual_depth (dpy, visual);
|
|
812 best_visual_found = (visual != DefaultVisual (dpy, screen));
|
|
813 }
|
|
814
|
|
815 /* If we've got the same visual as the default and it's PseudoColor,
|
|
816 check to see if the user specified that we need a private colormap */
|
|
817 if (visual == DefaultVisual (dpy, screen))
|
|
818 {
|
|
819 sprintf (buf1, "%s.privateColormap", app_name);
|
|
820 sprintf (buf2, "%s.PrivateColormap", app_class);
|
1204
|
821 if ((visual->X_CLASSFIELD == PseudoColor) &&
|
771
|
822 (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value)
|
|
823 == True))
|
|
824 cmap = XCopyColormapAndFree (dpy, DefaultColormap (dpy, screen));
|
428
|
825 else
|
771
|
826 cmap = DefaultColormap (dpy, screen);
|
428
|
827 }
|
|
828 else
|
|
829 {
|
|
830 if ( best_visual_found )
|
771
|
831 cmap = XCreateColormap (dpy, RootWindow (dpy, screen), visual,
|
|
832 AllocNone);
|
428
|
833 else
|
|
834 {
|
771
|
835 /* We have to create a matching colormap anyway... ####
|
|
836 think about using standard colormaps (need the Xmu
|
|
837 libs?) */
|
|
838 cmap = XCreateColormap (dpy, RootWindow (dpy, screen), visual,
|
|
839 AllocNone);
|
|
840 XInstallColormap (dpy, cmap);
|
428
|
841 }
|
|
842 }
|
|
843 }
|
|
844
|
|
845 DEVICE_X_VISUAL (d) = visual;
|
|
846 DEVICE_X_COLORMAP (d) = cmap;
|
|
847 DEVICE_X_DEPTH (d) = depth;
|
|
848 validify_resource_component ((char *) XSTRING_DATA (DEVICE_NAME (d)),
|
|
849 XSTRING_LENGTH (DEVICE_NAME (d)));
|
|
850
|
2007
|
851 /* #### If we're going to implement X session management, this would
|
|
852 be the place. Make sure it doesn't conflict with GNOME. */
|
428
|
853 {
|
|
854 Arg al[3];
|
|
855 XtSetArg (al[0], XtNvisual, visual);
|
|
856 XtSetArg (al[1], XtNdepth, depth);
|
|
857 XtSetArg (al[2], XtNcolormap, cmap);
|
|
858
|
|
859 app_shell = XtAppCreateShell (NULL, app_class,
|
|
860 applicationShellWidgetClass,
|
|
861 dpy, al, countof (al));
|
|
862 }
|
|
863
|
|
864 DEVICE_XT_APP_SHELL (d) = app_shell;
|
|
865
|
|
866 #ifdef HAVE_XIM
|
|
867 XIM_init_device(d);
|
|
868 #endif /* HAVE_XIM */
|
|
869
|
|
870 /* Realize the app_shell so that its window exists for GC creation purposes,
|
|
871 and set it to the size of the root window for child placement purposes */
|
|
872 {
|
|
873 Arg al[5];
|
|
874 XtSetArg (al[0], XtNmappedWhenManaged, False);
|
|
875 XtSetArg (al[1], XtNx, 0);
|
|
876 XtSetArg (al[2], XtNy, 0);
|
|
877 XtSetArg (al[3], XtNwidth, WidthOfScreen (ScreenOfDisplay (dpy, screen)));
|
|
878 XtSetArg (al[4], XtNheight, HeightOfScreen (ScreenOfDisplay (dpy, screen)));
|
|
879 XtSetValues (app_shell, al, countof (al));
|
|
880 XtRealizeWidget (app_shell);
|
|
881 }
|
|
882
|
|
883 #ifdef HAVE_WMCOMMAND
|
|
884 {
|
|
885 int new_argc;
|
442
|
886 Extbyte **new_argv;
|
428
|
887 make_argc_argv (Vcommand_line_args, &new_argc, &new_argv);
|
442
|
888 XSetCommand (XtDisplay (app_shell), XtWindow (app_shell),
|
|
889 (char **) new_argv, new_argc);
|
428
|
890 free_argc_argv (new_argv);
|
|
891 }
|
|
892 #endif /* HAVE_WMCOMMAND */
|
|
893
|
|
894
|
|
895 #ifdef HAVE_OFFIX_DND
|
771
|
896 DndInitialize (app_shell);
|
428
|
897 #endif
|
|
898
|
|
899 Vx_initial_argv_list = make_arg_list (argc, argv);
|
|
900 free_argc_argv (argv);
|
|
901
|
|
902 DEVICE_X_WM_COMMAND_FRAME (d) = Qnil;
|
|
903
|
|
904 sanity_check_geometry_resource (dpy);
|
|
905
|
|
906 /* In event-Xt.c */
|
|
907 x_init_modifier_mapping (d);
|
|
908
|
|
909 DEVICE_INFD (d) = DEVICE_OUTFD (d) = ConnectionNumber (dpy);
|
|
910 init_baud_rate (d);
|
|
911 init_one_device (d);
|
|
912
|
771
|
913 DEVICE_X_GC_CACHE (d) = make_gc_cache (dpy, XtWindow (app_shell));
|
428
|
914 DEVICE_X_GRAY_PIXMAP (d) = None;
|
|
915 Xatoms_of_device_x (d);
|
440
|
916 Xatoms_of_select_x (d);
|
428
|
917 Xatoms_of_objects_x (d);
|
|
918 x_init_device_class (d);
|
|
919
|
|
920 /* Run the elisp side of the X device initialization. */
|
|
921 call0 (Qinit_pre_x_win);
|
|
922 }
|
|
923
|
|
924 static void
|
2286
|
925 x_finish_init_device (struct device *UNUSED (d), Lisp_Object UNUSED (props))
|
428
|
926 {
|
|
927 call0 (Qinit_post_x_win);
|
|
928 }
|
|
929
|
|
930 static void
|
|
931 x_mark_device (struct device *d)
|
|
932 {
|
|
933 mark_object (DEVICE_X_WM_COMMAND_FRAME (d));
|
|
934 mark_object (DEVICE_X_DATA (d)->x_keysym_map_hash_table);
|
|
935 }
|
|
936
|
|
937
|
|
938 /************************************************************************/
|
|
939 /* closing an X connection */
|
|
940 /************************************************************************/
|
|
941
|
|
942 static void
|
|
943 free_x_device_struct (struct device *d)
|
|
944 {
|
3092
|
945 #ifdef NEW_GC
|
|
946 mc_free (d->device_data);
|
|
947 #else /* not NEW_GC */
|
1726
|
948 xfree (d->device_data, void *);
|
3092
|
949 #endif /* not NEW_GC */
|
428
|
950 }
|
|
951
|
|
952 static void
|
|
953 x_delete_device (struct device *d)
|
|
954 {
|
|
955 Display *display;
|
|
956 #ifdef FREE_CHECKING
|
|
957 extern void (*__free_hook) (void *);
|
|
958 int checking_free;
|
|
959 #endif
|
|
960
|
|
961 display = DEVICE_X_DISPLAY (d);
|
|
962
|
|
963 if (display)
|
|
964 {
|
|
965 #ifdef FREE_CHECKING
|
|
966 checking_free = (__free_hook != 0);
|
|
967
|
|
968 /* Disable strict free checking, to avoid bug in X library */
|
|
969 if (checking_free)
|
|
970 disable_strict_free_check ();
|
|
971 #endif
|
|
972
|
|
973 free_gc_cache (DEVICE_X_GC_CACHE (d));
|
|
974 if (DEVICE_X_DATA (d)->x_modifier_keymap)
|
|
975 XFreeModifiermap (DEVICE_X_DATA (d)->x_modifier_keymap);
|
|
976 if (DEVICE_X_DATA (d)->x_keysym_map)
|
|
977 XFree ((char *) DEVICE_X_DATA (d)->x_keysym_map);
|
|
978
|
|
979 if (DEVICE_XT_APP_SHELL (d))
|
|
980 {
|
|
981 XtDestroyWidget (DEVICE_XT_APP_SHELL (d));
|
|
982 DEVICE_XT_APP_SHELL (d) = NULL;
|
|
983 }
|
|
984
|
|
985 XtCloseDisplay (display);
|
|
986 DEVICE_X_DISPLAY (d) = 0;
|
|
987 #ifdef FREE_CHECKING
|
|
988 if (checking_free)
|
|
989 enable_strict_free_check ();
|
|
990 #endif
|
|
991 }
|
|
992
|
|
993 free_x_device_struct (d);
|
|
994 }
|
|
995
|
|
996
|
|
997 /************************************************************************/
|
|
998 /* handle X errors */
|
|
999 /************************************************************************/
|
|
1000
|
442
|
1001 const char *
|
428
|
1002 x_event_name (int event_type)
|
|
1003 {
|
442
|
1004 static const char *events[] =
|
428
|
1005 {
|
|
1006 "0: ERROR!",
|
|
1007 "1: REPLY",
|
|
1008 "KeyPress",
|
|
1009 "KeyRelease",
|
|
1010 "ButtonPress",
|
|
1011 "ButtonRelease",
|
|
1012 "MotionNotify",
|
|
1013 "EnterNotify",
|
|
1014 "LeaveNotify",
|
|
1015 "FocusIn",
|
|
1016 "FocusOut",
|
|
1017 "KeymapNotify",
|
|
1018 "Expose",
|
|
1019 "GraphicsExpose",
|
|
1020 "NoExpose",
|
|
1021 "VisibilityNotify",
|
|
1022 "CreateNotify",
|
|
1023 "DestroyNotify",
|
|
1024 "UnmapNotify",
|
|
1025 "MapNotify",
|
|
1026 "MapRequest",
|
|
1027 "ReparentNotify",
|
|
1028 "ConfigureNotify",
|
|
1029 "ConfigureRequest",
|
|
1030 "GravityNotify",
|
|
1031 "ResizeRequest",
|
|
1032 "CirculateNotify",
|
|
1033 "CirculateRequest",
|
|
1034 "PropertyNotify",
|
|
1035 "SelectionClear",
|
|
1036 "SelectionRequest",
|
|
1037 "SelectionNotify",
|
|
1038 "ColormapNotify",
|
|
1039 "ClientMessage",
|
|
1040 "MappingNotify",
|
|
1041 "LASTEvent"
|
|
1042 };
|
|
1043
|
|
1044 if (event_type < 0 || event_type >= countof (events))
|
|
1045 return NULL;
|
|
1046 return events [event_type];
|
|
1047 }
|
|
1048
|
|
1049 /* Handling errors.
|
|
1050
|
|
1051 If an X error occurs which we are not expecting, we have no alternative
|
|
1052 but to print it to stderr. It would be nice to stuff it into a pop-up
|
|
1053 buffer, or to print it in the minibuffer, but that's not possible, because
|
|
1054 one is not allowed to do any I/O on the display connection from an error
|
|
1055 handler. The guts of Xlib expect these functions to either return or exit.
|
|
1056
|
|
1057 However, there are occasions when we might expect an error to reasonably
|
|
1058 occur. The interface to this is as follows:
|
|
1059
|
|
1060 Before calling some X routine which may error, call
|
|
1061 expect_x_error (dpy);
|
|
1062
|
|
1063 Just after calling the X routine, call either:
|
|
1064
|
|
1065 x_error_occurred_p (dpy);
|
|
1066
|
|
1067 to ask whether an error happened (and was ignored), or:
|
|
1068
|
|
1069 signal_if_x_error (dpy, resumable_p);
|
|
1070
|
|
1071 which will call Fsignal() with args appropriate to the X error, if there
|
|
1072 was one. (Resumable_p is whether the debugger should be allowed to
|
|
1073 continue from the call to signal.)
|
|
1074
|
|
1075 You must call one of these two routines immediately after calling the X
|
|
1076 routine; think of them as bookends like BLOCK_INPUT and UNBLOCK_INPUT.
|
|
1077 */
|
|
1078
|
|
1079 static int error_expected;
|
|
1080 static int error_occurred;
|
|
1081 static XErrorEvent last_error;
|
|
1082
|
|
1083 /* OVERKILL! */
|
|
1084
|
|
1085 #ifdef EXTERNAL_WIDGET
|
|
1086 static Lisp_Object
|
|
1087 x_error_handler_do_enqueue (Lisp_Object frame)
|
|
1088 {
|
|
1089 enqueue_magic_eval_event (io_error_delete_frame, frame);
|
|
1090 return Qt;
|
|
1091 }
|
|
1092
|
|
1093 static Lisp_Object
|
2333
|
1094 x_error_handler_error (Lisp_Object UNUSED (data), Lisp_Object UNUSED (dummy))
|
428
|
1095 {
|
|
1096 return Qnil;
|
|
1097 }
|
|
1098 #endif /* EXTERNAL_WIDGET */
|
|
1099
|
|
1100 int
|
|
1101 x_error_handler (Display *disp, XErrorEvent *event)
|
|
1102 {
|
|
1103 if (error_expected)
|
|
1104 {
|
|
1105 error_expected = 0;
|
|
1106 error_occurred = 1;
|
|
1107 last_error = *event;
|
|
1108 }
|
|
1109 else
|
|
1110 {
|
853
|
1111 int depth;
|
|
1112
|
428
|
1113 #ifdef EXTERNAL_WIDGET
|
|
1114 struct frame *f;
|
|
1115 struct device *d = get_device_from_display (disp);
|
|
1116
|
|
1117 if ((event->error_code == BadWindow ||
|
|
1118 event->error_code == BadDrawable)
|
|
1119 && ((f = x_any_window_to_frame (d, event->resourceid)) != 0))
|
|
1120 {
|
|
1121 Lisp_Object frame;
|
|
1122
|
|
1123 /* one of the windows comprising one of our frames has died.
|
|
1124 This occurs particularly with ExternalShell frames when the
|
|
1125 client that owns the ExternalShell's window dies.
|
|
1126
|
|
1127 We cannot do any I/O on the display connection so we need
|
|
1128 to enqueue an eval event so that the deletion happens
|
|
1129 later.
|
|
1130
|
|
1131 Furthermore, we need to trap any errors (out-of-memory) that
|
|
1132 may occur when Fenqueue_eval_event is called.
|
|
1133 */
|
|
1134
|
|
1135 if (f->being_deleted)
|
|
1136 return 0;
|
793
|
1137 frame = wrap_frame (f);
|
428
|
1138 if (!NILP (condition_case_1 (Qerror, x_error_handler_do_enqueue,
|
|
1139 frame, x_error_handler_error, Qnil)))
|
|
1140 {
|
|
1141 f->being_deleted = 1;
|
|
1142 f->visible = 0;
|
|
1143 }
|
|
1144 return 0;
|
|
1145 }
|
|
1146 #endif /* EXTERNAL_WIDGET */
|
|
1147
|
853
|
1148 /* #### this should issue a warning instead of outputting to stderr */
|
|
1149 depth = begin_dont_check_for_quit ();
|
2007
|
1150 #if 0
|
|
1151 /* This ends up calling X, which isn't allowed in an X error handler
|
|
1152 */
|
428
|
1153 stderr_out ("\n%s: ",
|
|
1154 (STRINGP (Vinvocation_name)
|
|
1155 ? (char *) XSTRING_DATA (Vinvocation_name)
|
|
1156 : "xemacs"));
|
2007
|
1157 #endif
|
428
|
1158 XmuPrintDefaultErrorMessage (disp, event, stderr);
|
853
|
1159 unbind_to (depth);
|
428
|
1160 }
|
|
1161 return 0;
|
|
1162 }
|
|
1163
|
|
1164 void
|
|
1165 expect_x_error (Display *dpy)
|
|
1166 {
|
|
1167 assert (!error_expected);
|
|
1168 XSync (dpy, 0); /* handle pending errors before setting flag */
|
|
1169 error_expected = 1;
|
|
1170 error_occurred = 0;
|
|
1171 }
|
|
1172
|
|
1173 int
|
|
1174 x_error_occurred_p (Display *dpy)
|
|
1175 {
|
|
1176 int val;
|
|
1177 XSync (dpy, 0); /* handle pending errors before setting flag */
|
|
1178 val = error_occurred;
|
|
1179 error_expected = 0;
|
|
1180 error_occurred = 0;
|
|
1181 return val;
|
|
1182 }
|
|
1183
|
|
1184 int
|
|
1185 signal_if_x_error (Display *dpy, int resumable_p)
|
|
1186 {
|
771
|
1187 Extbyte buf[1024];
|
867
|
1188 Ibyte num[100];
|
428
|
1189 Lisp_Object data;
|
|
1190 if (! x_error_occurred_p (dpy))
|
|
1191 return 0;
|
|
1192 data = Qnil;
|
771
|
1193 qxesprintf (num, "0x%X", (unsigned int) last_error.resourceid);
|
|
1194 data = Fcons (build_intstring (num), data);
|
|
1195 qxesprintf (num, "%d", last_error.request_code);
|
|
1196 XGetErrorDatabaseText (last_error.display, "XRequest", (char *) num, "",
|
|
1197 buf, sizeof (buf));
|
|
1198 if (*buf)
|
|
1199 data = Fcons (build_ext_string (buf, Qnative), data);
|
|
1200 else
|
|
1201 {
|
|
1202 qxesprintf (num, "Request-%d", last_error.request_code);
|
|
1203 data = Fcons (build_intstring (num), data);
|
|
1204 }
|
428
|
1205 XGetErrorText (last_error.display, last_error.error_code, buf, sizeof (buf));
|
771
|
1206 data = Fcons (build_ext_string (buf, Qnative), data);
|
428
|
1207 again:
|
|
1208 Fsignal (Qx_error, data);
|
|
1209 if (! resumable_p) goto again;
|
|
1210 return 1;
|
|
1211 }
|
|
1212
|
|
1213 int
|
|
1214 x_IO_error_handler (Display *disp)
|
|
1215 {
|
|
1216 /* This function can GC */
|
|
1217 Lisp_Object dev;
|
|
1218 struct device *d = get_device_from_display_1 (disp);
|
|
1219
|
756
|
1220 if (!d)
|
|
1221 d = device_being_initialized;
|
|
1222
|
428
|
1223 assert (d != NULL);
|
793
|
1224 dev = wrap_device (d);
|
428
|
1225
|
|
1226 if (NILP (find_nonminibuffer_frame_not_on_device (dev)))
|
|
1227 {
|
853
|
1228 int depth = begin_dont_check_for_quit ();
|
428
|
1229 /* We're going down. */
|
867
|
1230 Ibyte *errmess;
|
771
|
1231 GET_STRERROR (errmess, errno);
|
|
1232 stderr_out ("\n%s: Fatal I/O Error %d (%s) on display "
|
|
1233 "connection \"%s\"\n",
|
|
1234 (STRINGP (Vinvocation_name) ?
|
|
1235 (char *) XSTRING_DATA (Vinvocation_name) : "xemacs"),
|
|
1236 errno, errmess, DisplayString (disp));
|
|
1237 stderr_out (" after %lu requests (%lu known processed) with %d "
|
|
1238 "events remaining.\n",
|
|
1239 NextRequest (disp) - 1, LastKnownRequestProcessed (disp),
|
|
1240 QLength (disp));
|
428
|
1241 /* assert (!_Xdebug); */
|
853
|
1242 unbind_to (depth);
|
428
|
1243 }
|
|
1244 else
|
|
1245 {
|
867
|
1246 Ibyte *errmess;
|
771
|
1247 GET_STRERROR (errmess, errno);
|
428
|
1248 warn_when_safe
|
|
1249 (Qx, Qcritical,
|
|
1250 "I/O Error %d (%s) on display connection\n"
|
2116
|
1251 " \"%s\" after %lu requests (%lu known processed)\n"
|
428
|
1252 " with %d events remaining.\n"
|
|
1253 " Throwing to top level.\n",
|
771
|
1254 errno, errmess, DisplayString (disp),
|
428
|
1255 NextRequest (disp) - 1, LastKnownRequestProcessed (disp),
|
|
1256 QLength (disp));
|
|
1257 }
|
|
1258
|
|
1259 /* According to X specs, we should not return from this function, or
|
|
1260 Xlib might just decide to exit(). So we mark the offending
|
|
1261 console for deletion and throw to top level. */
|
|
1262 if (d)
|
3466
|
1263 {
|
|
1264 enqueue_magic_eval_event (io_error_delete_device, dev);
|
|
1265 DEVICE_X_BEING_DELETED (d) = 1;
|
|
1266 }
|
428
|
1267 Fthrow (Qtop_level, Qnil);
|
|
1268
|
2268
|
1269 RETURN_NOT_REACHED (0);
|
428
|
1270 }
|
|
1271
|
|
1272 DEFUN ("x-debug-mode", Fx_debug_mode, 1, 2, 0, /*
|
|
1273 With a true arg, make the connection to the X server synchronous.
|
|
1274 With false, make it asynchronous. Synchronous connections are much slower,
|
|
1275 but are useful for debugging. (If you get X errors, make the connection
|
|
1276 synchronous, and use a debugger to set a breakpoint on `x_error_handler'.
|
|
1277 Your backtrace of the C stack will now be useful. In asynchronous mode,
|
|
1278 the stack above `x_error_handler' isn't helpful because of buffering.)
|
|
1279 If DEVICE is not specified, the selected device is assumed.
|
|
1280
|
|
1281 Calling this function is the same as calling the C function `XSynchronize',
|
|
1282 or starting the program with the `-sync' command line argument.
|
|
1283 */
|
|
1284 (arg, device))
|
|
1285 {
|
|
1286 struct device *d = decode_x_device (device);
|
|
1287
|
|
1288 XSynchronize (DEVICE_X_DISPLAY (d), !NILP (arg));
|
|
1289
|
|
1290 if (!NILP (arg))
|
|
1291 message ("X connection is synchronous");
|
|
1292 else
|
|
1293 message ("X connection is asynchronous");
|
|
1294
|
|
1295 return arg;
|
|
1296 }
|
|
1297
|
|
1298
|
|
1299 /************************************************************************/
|
|
1300 /* X resources */
|
|
1301 /************************************************************************/
|
|
1302
|
|
1303 #if 0 /* bah humbug. The whole "widget == resource" stuff is such
|
|
1304 a crock of shit that I'm just going to ignore it all. */
|
|
1305
|
|
1306 /* If widget is NULL, we are retrieving device or global face data. */
|
|
1307
|
|
1308 static void
|
|
1309 construct_name_list (Display *display, Widget widget, char *fake_name,
|
1204
|
1310 char *fake_class, char *name, char *class_)
|
428
|
1311 {
|
|
1312 char *stack [100][2];
|
2552
|
1313 Widget this_widget;
|
428
|
1314 int count = 0;
|
|
1315 char *name_tail, *class_tail;
|
|
1316
|
|
1317 if (widget)
|
|
1318 {
|
2552
|
1319 for (this_widget = widget; this_widget;
|
|
1320 this_widget = XtParent (this_widget))
|
428
|
1321 {
|
2552
|
1322 stack [count][0] = this_widget->core.name;
|
|
1323 stack [count][1] = XtClass (this_widget)->core_class.class_name;
|
428
|
1324 count++;
|
|
1325 }
|
|
1326 count--;
|
|
1327 }
|
|
1328 else if (fake_name && fake_class)
|
|
1329 {
|
|
1330 stack [count][0] = fake_name;
|
|
1331 stack [count][1] = fake_class;
|
|
1332 count++;
|
|
1333 }
|
|
1334
|
|
1335 /* The root widget is an application shell; resource lookups use the
|
|
1336 specified application name and application class in preference to
|
|
1337 the name/class of that widget (which is argv[0] / "ApplicationShell").
|
|
1338 Generally the app name and class will be argv[0] / "Emacs" but
|
|
1339 the former can be set via the -name command-line option, and the
|
|
1340 latter can be set by changing `x-emacs-application-class' in
|
|
1341 lisp/term/x-win.el.
|
|
1342 */
|
|
1343 XtGetApplicationNameAndClass (display,
|
|
1344 &stack [count][0],
|
|
1345 &stack [count][1]);
|
|
1346
|
|
1347 name [0] = 0;
|
1204
|
1348 class_ [0] = 0;
|
428
|
1349
|
|
1350 name_tail = name;
|
1204
|
1351 class_tail = class_;
|
428
|
1352 for (; count >= 0; count--)
|
|
1353 {
|
|
1354 strcat (name_tail, stack [count][0]);
|
|
1355 for (; *name_tail; name_tail++)
|
|
1356 if (*name_tail == '.') *name_tail = '_';
|
|
1357 strcat (name_tail, ".");
|
|
1358 name_tail++;
|
|
1359
|
|
1360 strcat (class_tail, stack [count][1]);
|
|
1361 for (; *class_tail; class_tail++)
|
|
1362 if (*class_tail == '.') *class_tail = '_';
|
|
1363 strcat (class_tail, ".");
|
|
1364 class_tail++;
|
|
1365 }
|
|
1366 }
|
|
1367
|
|
1368 #endif /* 0 */
|
|
1369
|
771
|
1370 static Extbyte_dynarr *name_Extbyte_dynarr;
|
|
1371 static Extbyte_dynarr *class_Extbyte_dynarr;
|
428
|
1372
|
|
1373 /* Given a locale and device specification from x-get-resource or
|
|
1374 x-get-resource-prefix, return the resource prefix and display to
|
|
1375 fetch the resource on. */
|
|
1376
|
|
1377 static void
|
|
1378 x_get_resource_prefix (Lisp_Object locale, Lisp_Object device,
|
771
|
1379 Display **display_out, Extbyte_dynarr *name,
|
1204
|
1380 Extbyte_dynarr *class_)
|
428
|
1381 {
|
|
1382 if (NILP (locale))
|
|
1383 locale = Qglobal;
|
|
1384 if (NILP (Fvalid_specifier_locale_p (locale)))
|
563
|
1385 invalid_argument ("Invalid locale", locale);
|
428
|
1386 if (WINDOWP (locale))
|
|
1387 /* #### I can't come up with any coherent way of naming windows.
|
|
1388 By relative position? That seems tricky because windows
|
|
1389 can change position, be split, etc. By order of creation?
|
|
1390 That seems less than useful. */
|
563
|
1391 signal_error (Qunimplemented,
|
|
1392 "Windows currently can't be resourced", locale);
|
428
|
1393
|
|
1394 if (!NILP (device) && !DEVICEP (device))
|
|
1395 CHECK_DEVICE (device);
|
|
1396 if (DEVICEP (device) && !DEVICE_X_P (XDEVICE (device)))
|
|
1397 device = Qnil;
|
|
1398 if (NILP (device))
|
|
1399 {
|
|
1400 device = DFW_DEVICE (locale);
|
|
1401 if (DEVICEP (device) && !DEVICE_X_P (XDEVICE (device)))
|
|
1402 device = Qnil;
|
|
1403 if (NILP (device))
|
872
|
1404 device = get_default_device (Qx);
|
428
|
1405 if (NILP (device))
|
|
1406 {
|
|
1407 *display_out = 0;
|
|
1408 return;
|
|
1409 }
|
|
1410 }
|
|
1411
|
|
1412 *display_out = DEVICE_X_DISPLAY (XDEVICE (device));
|
|
1413
|
|
1414 {
|
771
|
1415 Extbyte *appname, *appclass;
|
428
|
1416 int name_len, class_len;
|
|
1417 XtGetApplicationNameAndClass (*display_out, &appname, &appclass);
|
|
1418 name_len = strlen (appname);
|
|
1419 class_len = strlen (appclass);
|
771
|
1420 Dynarr_add_many (name, appname, name_len);
|
1204
|
1421 Dynarr_add_many (class_, appclass, class_len);
|
428
|
1422 validify_resource_component (Dynarr_atp (name, 0), name_len);
|
1204
|
1423 validify_resource_component (Dynarr_atp (class_, 0), class_len);
|
428
|
1424 }
|
|
1425
|
|
1426 if (EQ (locale, Qglobal))
|
|
1427 return;
|
|
1428 if (BUFFERP (locale))
|
|
1429 {
|
|
1430 Dynarr_add_literal_string (name, ".buffer.");
|
|
1431 /* we know buffer is live; otherwise we got an error above. */
|
|
1432 Dynarr_add_validified_lisp_string (name, Fbuffer_name (locale));
|
1204
|
1433 Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsBuffer");
|
428
|
1434 }
|
|
1435 else if (FRAMEP (locale))
|
|
1436 {
|
|
1437 Dynarr_add_literal_string (name, ".frame.");
|
|
1438 /* we know frame is live; otherwise we got an error above. */
|
|
1439 Dynarr_add_validified_lisp_string (name, Fframe_name (locale));
|
1204
|
1440 Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsFrame");
|
428
|
1441 }
|
|
1442 else
|
|
1443 {
|
|
1444 assert (DEVICEP (locale));
|
|
1445 Dynarr_add_literal_string (name, ".device.");
|
|
1446 /* we know device is live; otherwise we got an error above. */
|
|
1447 Dynarr_add_validified_lisp_string (name, Fdevice_name (locale));
|
1204
|
1448 Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsDevice");
|
428
|
1449 }
|
|
1450 return;
|
|
1451 }
|
|
1452
|
|
1453 DEFUN ("x-get-resource", Fx_get_resource, 3, 6, 0, /*
|
|
1454 Retrieve an X resource from the resource manager.
|
|
1455
|
|
1456 The first arg is the name of the resource to retrieve, such as "font".
|
|
1457 The second arg is the class of the resource to retrieve, such as "Font".
|
3025
|
1458 The third arg must be one of the symbols `string', `integer', `natnum', or
|
|
1459 `boolean', specifying the type of object that the database is searched for.
|
428
|
1460 The fourth arg is the locale to search for the resources on, and can
|
3025
|
1461 currently be a buffer, a frame, a device, or `global'. If omitted, it
|
|
1462 defaults to `global'.
|
428
|
1463 The fifth arg is the device to search for the resources on. (The resource
|
|
1464 database for a particular device is constructed by combining non-device-
|
|
1465 specific resources such as any command-line resources specified and any
|
|
1466 app-defaults files found [or the fallback resources supplied by XEmacs,
|
|
1467 if no app-defaults file is found] with device-specific resources such as
|
|
1468 those supplied using xrdb.) If omitted, it defaults to the device of
|
|
1469 LOCALE, if a device can be derived (i.e. if LOCALE is a frame or device),
|
|
1470 and otherwise defaults to the value of `default-x-device'.
|
|
1471 The sixth arg NOERROR, if non-nil, means do not signal an error if a
|
|
1472 bogus resource specification was retrieved (e.g. if a non-integer was
|
|
1473 given when an integer was requested). In this case, a warning is issued
|
442
|
1474 instead, unless NOERROR is t, in which case no warning is issued.
|
428
|
1475
|
|
1476 The resource names passed to this function are looked up relative to the
|
|
1477 locale.
|
|
1478
|
|
1479 If you want to search for a subresource, you just need to specify the
|
|
1480 resource levels in NAME and CLASS. For example, NAME could be
|
|
1481 "modeline.attributeFont", and CLASS "Face.AttributeFont".
|
|
1482
|
|
1483 Specifically,
|
|
1484
|
|
1485 1) If LOCALE is a buffer, a call
|
|
1486
|
|
1487 (x-get-resource "foreground" "Foreground" 'string SOME-BUFFER)
|
|
1488
|
|
1489 is an interface to a C call something like
|
|
1490
|
|
1491 XrmGetResource (db, "xemacs.buffer.BUFFER-NAME.foreground",
|
|
1492 "Emacs.EmacsLocaleType.EmacsBuffer.Foreground",
|
|
1493 "String");
|
|
1494
|
|
1495 2) If LOCALE is a frame, a call
|
|
1496
|
|
1497 (x-get-resource "foreground" "Foreground" 'string SOME-FRAME)
|
|
1498
|
|
1499 is an interface to a C call something like
|
|
1500
|
|
1501 XrmGetResource (db, "xemacs.frame.FRAME-NAME.foreground",
|
|
1502 "Emacs.EmacsLocaleType.EmacsFrame.Foreground",
|
|
1503 "String");
|
|
1504
|
|
1505 3) If LOCALE is a device, a call
|
|
1506
|
|
1507 (x-get-resource "foreground" "Foreground" 'string SOME-DEVICE)
|
|
1508
|
|
1509 is an interface to a C call something like
|
|
1510
|
|
1511 XrmGetResource (db, "xemacs.device.DEVICE-NAME.foreground",
|
|
1512 "Emacs.EmacsLocaleType.EmacsDevice.Foreground",
|
|
1513 "String");
|
|
1514
|
3025
|
1515 4) If LOCALE is `global', a call
|
428
|
1516
|
|
1517 (x-get-resource "foreground" "Foreground" 'string 'global)
|
|
1518
|
|
1519 is an interface to a C call something like
|
|
1520
|
|
1521 XrmGetResource (db, "xemacs.foreground",
|
|
1522 "Emacs.Foreground",
|
|
1523 "String");
|
|
1524
|
3025
|
1525 Note that for `global', no prefix is added other than that of the
|
428
|
1526 application itself; thus, you can use this locale to retrieve
|
|
1527 arbitrary application resources, if you really want to.
|
|
1528
|
|
1529 The returned value of this function is nil if the queried resource is not
|
|
1530 found. If the third arg is `string', a string is returned, and if it is
|
|
1531 `integer', an integer is returned. If the third arg is `boolean', then the
|
|
1532 returned value is the list (t) for true, (nil) for false, and is nil to
|
430
|
1533 mean ``unspecified''.
|
428
|
1534 */
|
1204
|
1535 (name, class_, type, locale, device, noerror))
|
428
|
1536 {
|
771
|
1537 Extbyte *name_string, *class_string;
|
|
1538 Extbyte *raw_result;
|
428
|
1539 XrmDatabase db;
|
|
1540 Display *display;
|
578
|
1541 Error_Behavior errb = decode_error_behavior_flag (noerror);
|
771
|
1542 Lisp_Object codesys;
|
428
|
1543
|
|
1544 CHECK_STRING (name);
|
1204
|
1545 CHECK_STRING (class_);
|
428
|
1546 CHECK_SYMBOL (type);
|
|
1547
|
771
|
1548 Dynarr_reset (name_Extbyte_dynarr);
|
|
1549 Dynarr_reset (class_Extbyte_dynarr);
|
428
|
1550
|
|
1551 x_get_resource_prefix (locale, device, &display,
|
771
|
1552 name_Extbyte_dynarr, class_Extbyte_dynarr);
|
428
|
1553 if (!display)
|
|
1554 return Qnil;
|
|
1555
|
|
1556 db = XtDatabase (display);
|
771
|
1557 codesys = coding_system_of_xrm_database (db);
|
|
1558 Dynarr_add (name_Extbyte_dynarr, '.');
|
|
1559 Dynarr_add_lisp_string (name_Extbyte_dynarr, name, Qbinary);
|
|
1560 Dynarr_add (class_Extbyte_dynarr, '.');
|
1204
|
1561 Dynarr_add_lisp_string (class_Extbyte_dynarr, class_, Qbinary);
|
771
|
1562 Dynarr_add (name_Extbyte_dynarr, '\0');
|
|
1563 Dynarr_add (class_Extbyte_dynarr, '\0');
|
428
|
1564
|
771
|
1565 name_string = Dynarr_atp (name_Extbyte_dynarr, 0);
|
|
1566 class_string = Dynarr_atp (class_Extbyte_dynarr, 0);
|
428
|
1567
|
|
1568 {
|
|
1569 XrmValue xrm_value;
|
|
1570 XrmName namelist[100];
|
|
1571 XrmClass classlist[100];
|
|
1572 XrmName *namerest = namelist;
|
|
1573 XrmClass *classrest = classlist;
|
|
1574 XrmRepresentation xrm_type;
|
|
1575 XrmRepresentation string_quark;
|
|
1576 int result;
|
|
1577 XrmStringToNameList (name_string, namelist);
|
|
1578 XrmStringToClassList (class_string, classlist);
|
|
1579 string_quark = XrmStringToQuark ("String");
|
|
1580
|
|
1581 /* ensure that they have the same length */
|
|
1582 while (namerest[0] && classrest[0])
|
|
1583 namerest++, classrest++;
|
|
1584 if (namerest[0] || classrest[0])
|
563
|
1585 {
|
|
1586 maybe_signal_error_2
|
|
1587 (Qstructure_formation_error,
|
1204
|
1588 "class list and name list must be the same length", name, class_,
|
563
|
1589 Qresource, errb);
|
|
1590 return Qnil;
|
|
1591 }
|
428
|
1592 result = XrmQGetResource (db, namelist, classlist, &xrm_type, &xrm_value);
|
|
1593
|
|
1594 if (result != True || xrm_type != string_quark)
|
|
1595 return Qnil;
|
771
|
1596 raw_result = (Extbyte *) xrm_value.addr;
|
428
|
1597 }
|
|
1598
|
|
1599 if (EQ (type, Qstring))
|
771
|
1600 return build_ext_string (raw_result, codesys);
|
428
|
1601 else if (EQ (type, Qboolean))
|
|
1602 {
|
771
|
1603 if (!strcasecmp (raw_result, "off") ||
|
|
1604 !strcasecmp (raw_result, "false") ||
|
|
1605 !strcasecmp (raw_result, "no"))
|
428
|
1606 return Fcons (Qnil, Qnil);
|
771
|
1607 if (!strcasecmp (raw_result, "on") ||
|
|
1608 !strcasecmp (raw_result, "true") ||
|
|
1609 !strcasecmp (raw_result, "yes"))
|
428
|
1610 return Fcons (Qt, Qnil);
|
563
|
1611 return maybe_signal_continuable_error_2
|
|
1612 (Qinvalid_operation, "Can't convert to a Boolean",
|
771
|
1613 build_ext_string (name_string, Qbinary),
|
|
1614 build_ext_string (raw_result, codesys), Qresource,
|
563
|
1615 errb);
|
428
|
1616 }
|
|
1617 else if (EQ (type, Qinteger) || EQ (type, Qnatnum))
|
|
1618 {
|
|
1619 int i;
|
|
1620 char c;
|
|
1621 if (1 != sscanf (raw_result, "%d%c", &i, &c))
|
563
|
1622 return maybe_signal_continuable_error_2
|
|
1623 (Qinvalid_operation, "Can't convert to an integer",
|
771
|
1624 build_ext_string (name_string, Qbinary),
|
|
1625 build_ext_string (raw_result, codesys), Qresource,
|
563
|
1626 errb);
|
428
|
1627 else if (EQ (type, Qnatnum) && i < 0)
|
563
|
1628 return maybe_signal_continuable_error_2
|
|
1629 (Qinvalid_argument, "Invalid numerical value for resource",
|
771
|
1630 make_int (i), build_ext_string (name_string, Qbinary),
|
|
1631 Qresource, errb);
|
428
|
1632 else
|
|
1633 return make_int (i);
|
|
1634 }
|
|
1635 else
|
|
1636 {
|
|
1637 return maybe_signal_continuable_error
|
563
|
1638 (Qwrong_type_argument, "Should be string, integer, natnum or boolean",
|
|
1639 type, Qresource, errb);
|
428
|
1640 }
|
|
1641 }
|
|
1642
|
|
1643 DEFUN ("x-get-resource-prefix", Fx_get_resource_prefix, 1, 2, 0, /*
|
|
1644 Return the resource prefix for LOCALE on DEVICE.
|
|
1645 The resource prefix is the strings used to prefix resources if
|
|
1646 the LOCALE and DEVICE arguments were passed to `x-get-resource'.
|
|
1647 The returned value is a cons of a name prefix and a class prefix.
|
|
1648 For example, if LOCALE is a frame, the returned value might be
|
|
1649 \("xemacs.frame.FRAME-NAME" . "Emacs.EmacsLocaleType.EmacsFrame").
|
|
1650 If no valid X device for resourcing can be obtained, this function
|
|
1651 returns nil. (In such a case, `x-get-resource' would always return nil.)
|
|
1652 */
|
|
1653 (locale, device))
|
|
1654 {
|
|
1655 Display *display;
|
|
1656
|
771
|
1657 Dynarr_reset (name_Extbyte_dynarr );
|
|
1658 Dynarr_reset (class_Extbyte_dynarr);
|
428
|
1659
|
|
1660 x_get_resource_prefix (locale, device, &display,
|
771
|
1661 name_Extbyte_dynarr, class_Extbyte_dynarr);
|
428
|
1662 if (!display)
|
|
1663 return Qnil;
|
|
1664
|
867
|
1665 return Fcons (make_string ((Ibyte *) Dynarr_atp (name_Extbyte_dynarr, 0),
|
771
|
1666 Dynarr_length (name_Extbyte_dynarr)),
|
867
|
1667 make_string ((Ibyte *) Dynarr_atp (class_Extbyte_dynarr, 0),
|
771
|
1668 Dynarr_length (class_Extbyte_dynarr)));
|
428
|
1669 }
|
|
1670
|
|
1671 DEFUN ("x-put-resource", Fx_put_resource, 1, 2, 0, /*
|
|
1672 Add a resource to the resource database for DEVICE.
|
|
1673 RESOURCE-LINE specifies the resource to add and should be a
|
|
1674 standard resource specification.
|
|
1675 */
|
|
1676 (resource_line, device))
|
|
1677 {
|
|
1678 struct device *d = decode_device (device);
|
|
1679
|
|
1680 if (DEVICE_X_P (d))
|
|
1681 {
|
|
1682 XrmDatabase db = XtDatabase (DEVICE_X_DISPLAY (d));
|
771
|
1683 Extbyte *str, *colon_pos;
|
|
1684
|
|
1685 CHECK_STRING (resource_line);
|
|
1686 LISP_STRING_TO_EXTERNAL (resource_line, str,
|
|
1687 coding_system_of_xrm_database (db));
|
|
1688 if (!(colon_pos = strchr (str, ':')) || strchr (str, '\n'))
|
|
1689 invalid:
|
|
1690 syntax_error ("Invalid resource line", resource_line);
|
|
1691 if ((int)
|
|
1692 strspn (str,
|
|
1693 /* Only the following chars are allowed before the colon */
|
|
1694 " \t.*?abcdefghijklmnopqrstuvwxyz"
|
|
1695 "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_-")
|
|
1696 != colon_pos - str)
|
|
1697 goto invalid;
|
|
1698
|
428
|
1699 XrmPutLineResource (&db, str);
|
|
1700 }
|
|
1701
|
|
1702 return Qnil;
|
|
1703 }
|
|
1704
|
|
1705
|
|
1706 /************************************************************************/
|
|
1707 /* display information functions */
|
|
1708 /************************************************************************/
|
|
1709
|
|
1710 DEFUN ("default-x-device", Fdefault_x_device, 0, 0, 0, /*
|
|
1711 Return the default X device for resourcing.
|
|
1712 This is the first-created X device that still exists.
|
872
|
1713 See also `default-device'.
|
428
|
1714 */
|
|
1715 ())
|
|
1716 {
|
872
|
1717 return get_default_device (Qx);
|
428
|
1718 }
|
|
1719
|
|
1720 DEFUN ("x-display-visual-class", Fx_display_visual_class, 0, 1, 0, /*
|
|
1721 Return the visual class of the X display DEVICE is using.
|
|
1722 This can be altered from the default at startup using the XResource "EmacsVisual".
|
|
1723 The returned value will be one of the symbols `static-gray', `gray-scale',
|
|
1724 `static-color', `pseudo-color', `true-color', or `direct-color'.
|
|
1725 */
|
|
1726 (device))
|
|
1727 {
|
|
1728 Visual *vis = DEVICE_X_VISUAL (decode_x_device (device));
|
1204
|
1729 switch (vis->X_CLASSFIELD)
|
428
|
1730 {
|
|
1731 case StaticGray: return intern ("static-gray");
|
|
1732 case GrayScale: return intern ("gray-scale");
|
|
1733 case StaticColor: return intern ("static-color");
|
|
1734 case PseudoColor: return intern ("pseudo-color");
|
|
1735 case TrueColor: return intern ("true-color");
|
|
1736 case DirectColor: return intern ("direct-color");
|
|
1737 default:
|
563
|
1738 invalid_state ("display has an unknown visual class", Qunbound);
|
428
|
1739 return Qnil; /* suppress compiler warning */
|
|
1740 }
|
|
1741 }
|
|
1742
|
|
1743 DEFUN ("x-display-visual-depth", Fx_display_visual_depth, 0, 1, 0, /*
|
|
1744 Return the bitplane depth of the visual the X display DEVICE is using.
|
|
1745 */
|
|
1746 (device))
|
|
1747 {
|
|
1748 return make_int (DEVICE_X_DEPTH (decode_x_device (device)));
|
|
1749 }
|
|
1750
|
|
1751 static Lisp_Object
|
|
1752 x_device_system_metrics (struct device *d,
|
|
1753 enum device_metrics m)
|
|
1754 {
|
|
1755 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1756
|
|
1757 switch (m)
|
|
1758 {
|
|
1759 case DM_size_device:
|
|
1760 return Fcons (make_int (DisplayWidth (dpy, DefaultScreen (dpy))),
|
|
1761 make_int (DisplayHeight (dpy, DefaultScreen (dpy))));
|
|
1762 case DM_size_device_mm:
|
|
1763 return Fcons (make_int (DisplayWidthMM (dpy, DefaultScreen (dpy))),
|
|
1764 make_int (DisplayHeightMM (dpy, DefaultScreen (dpy))));
|
|
1765 case DM_num_bit_planes:
|
|
1766 return make_int (DisplayPlanes (dpy, DefaultScreen (dpy)));
|
|
1767 case DM_num_color_cells:
|
|
1768 return make_int (DisplayCells (dpy, DefaultScreen (dpy)));
|
1942
|
1769 case DM_num_screens:
|
|
1770 return make_int (ScreenCount (dpy));
|
|
1771 case DM_backing_store:
|
|
1772 switch (DoesBackingStore (DefaultScreenOfDisplay (dpy)))
|
|
1773 {
|
|
1774 case Always:
|
|
1775 return intern ("always");
|
|
1776 case WhenMapped:
|
|
1777 return intern ("when-mapped");
|
|
1778 default:
|
|
1779 return intern ("not-useful");
|
|
1780 }
|
|
1781 case DM_save_under:
|
|
1782 return (DoesSaveUnders (DefaultScreenOfDisplay (dpy)) == True)
|
|
1783 ? Qt : Qnil;
|
428
|
1784 default: /* No such device metric property for X devices */
|
|
1785 return Qunbound;
|
|
1786 }
|
|
1787 }
|
|
1788
|
|
1789 DEFUN ("x-server-vendor", Fx_server_vendor, 0, 1, 0, /*
|
|
1790 Return the vendor ID string of the X server DEVICE is on.
|
|
1791 Return the empty string if the vendor ID string cannot be determined.
|
|
1792 */
|
|
1793 (device))
|
|
1794 {
|
|
1795 Display *dpy = get_x_display (device);
|
2367
|
1796 Extbyte *vendor = ServerVendor (dpy);
|
428
|
1797
|
2367
|
1798 return build_ext_string (vendor ? vendor : "", Qx_hpc_encoding);
|
428
|
1799 }
|
|
1800
|
|
1801 DEFUN ("x-server-version", Fx_server_version, 0, 1, 0, /*
|
|
1802 Return the version numbers of the X server DEVICE is on.
|
|
1803 The returned value is a list of three integers: the major and minor
|
|
1804 version numbers of the X Protocol in use, and the vendor-specific release
|
|
1805 number. See also `x-server-vendor'.
|
|
1806 */
|
|
1807 (device))
|
|
1808 {
|
|
1809 Display *dpy = get_x_display (device);
|
|
1810
|
|
1811 return list3 (make_int (ProtocolVersion (dpy)),
|
|
1812 make_int (ProtocolRevision (dpy)),
|
|
1813 make_int (VendorRelease (dpy)));
|
|
1814 }
|
|
1815
|
|
1816 DEFUN ("x-valid-keysym-name-p", Fx_valid_keysym_name_p, 1, 1, 0, /*
|
|
1817 Return true if KEYSYM names a keysym that the X library knows about.
|
|
1818 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1819 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
|
1820 */
|
|
1821 (keysym))
|
|
1822 {
|
2367
|
1823 const Extbyte *keysym_ext;
|
428
|
1824
|
|
1825 CHECK_STRING (keysym);
|
442
|
1826 LISP_STRING_TO_EXTERNAL (keysym, keysym_ext, Qctext);
|
428
|
1827
|
|
1828 return XStringToKeysym (keysym_ext) ? Qt : Qnil;
|
|
1829 }
|
|
1830
|
|
1831 DEFUN ("x-keysym-hash-table", Fx_keysym_hash_table, 0, 1, 0, /*
|
440
|
1832 Return a hash table containing a key for all keysyms on DEVICE.
|
|
1833 DEVICE must be an X11 display device. See `x-keysym-on-keyboard-p'.
|
428
|
1834 */
|
|
1835 (device))
|
|
1836 {
|
|
1837 struct device *d = decode_device (device);
|
|
1838 if (!DEVICE_X_P (d))
|
563
|
1839 gui_error ("Not an X device", device);
|
428
|
1840
|
|
1841 return DEVICE_X_DATA (d)->x_keysym_map_hash_table;
|
|
1842 }
|
|
1843
|
|
1844 DEFUN ("x-keysym-on-keyboard-sans-modifiers-p", Fx_keysym_on_keyboard_sans_modifiers_p,
|
|
1845 1, 2, 0, /*
|
|
1846 Return true if KEYSYM names a key on the keyboard of DEVICE.
|
|
1847 More precisely, return true if pressing a physical key
|
|
1848 on the keyboard of DEVICE without any modifier keys generates KEYSYM.
|
|
1849 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1850 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
|
1851 The keysym name can be provided in two forms:
|
|
1852 - if keysym is a string, it must be the name as known to X windows.
|
|
1853 - if keysym is a symbol, it must be the name as known to XEmacs.
|
|
1854 The two names differ in capitalization and underscoring.
|
|
1855 */
|
|
1856 (keysym, device))
|
|
1857 {
|
|
1858 struct device *d = decode_device (device);
|
|
1859 if (!DEVICE_X_P (d))
|
563
|
1860 gui_error ("Not an X device", device);
|
428
|
1861
|
|
1862 return (EQ (Qsans_modifiers,
|
|
1863 Fgethash (keysym, DEVICE_X_KEYSYM_MAP_HASH_TABLE (d), Qnil)) ?
|
|
1864 Qt : Qnil);
|
|
1865 }
|
|
1866
|
|
1867
|
|
1868 DEFUN ("x-keysym-on-keyboard-p", Fx_keysym_on_keyboard_p, 1, 2, 0, /*
|
|
1869 Return true if KEYSYM names a key on the keyboard of DEVICE.
|
|
1870 More precisely, return true if some keystroke (possibly including modifiers)
|
|
1871 on the keyboard of DEVICE keys generates KEYSYM.
|
|
1872 Valid keysyms are listed in the files /usr/include/X11/keysymdef.h and in
|
|
1873 /usr/lib/X11/XKeysymDB, or whatever the equivalents are on your system.
|
|
1874 The keysym name can be provided in two forms:
|
|
1875 - if keysym is a string, it must be the name as known to X windows.
|
|
1876 - if keysym is a symbol, it must be the name as known to XEmacs.
|
|
1877 The two names differ in capitalization and underscoring.
|
2828
|
1878
|
|
1879 This function is not entirely trustworthy, in that Xlib compose processing
|
|
1880 can produce keysyms that XEmacs will not have seen when it examined the
|
|
1881 keysyms available on startup. So pressing `dead-diaeresis' and then 'a' may
|
|
1882 pass `adiaeresis' to XEmacs, or (in some implementations) even `U00E4',
|
|
1883 where `(x-keysym-on-keyboard-p 'adiaeresis)' and `(x-keysym-on-keyboard-p
|
|
1884 'U00E4)' would both have returned nil. Subsequent to XEmacs seeing a keysym
|
|
1885 it was previously unaware of, the predicate will take note of it, though.
|
428
|
1886 */
|
|
1887 (keysym, device))
|
|
1888 {
|
|
1889 struct device *d = decode_device (device);
|
|
1890 if (!DEVICE_X_P (d))
|
563
|
1891 gui_error ("Not an X device", device);
|
428
|
1892
|
|
1893 return (NILP (Fgethash (keysym, DEVICE_X_KEYSYM_MAP_HASH_TABLE (d), Qnil)) ?
|
|
1894 Qnil : Qt);
|
|
1895 }
|
|
1896
|
|
1897
|
|
1898 /************************************************************************/
|
|
1899 /* grabs and ungrabs */
|
|
1900 /************************************************************************/
|
|
1901
|
|
1902 DEFUN ("x-grab-pointer", Fx_grab_pointer, 0, 3, 0, /*
|
|
1903 Grab the pointer and restrict it to its current window.
|
|
1904 If optional DEVICE argument is nil, the default device will be used.
|
|
1905 If optional CURSOR argument is non-nil, change the pointer shape to that
|
|
1906 until `x-ungrab-pointer' is called (it should be an object returned by the
|
|
1907 `make-cursor-glyph' function).
|
|
1908 If the second optional argument IGNORE-KEYBOARD is non-nil, ignore all
|
|
1909 keyboard events during the grab.
|
|
1910 Returns t if the grab is successful, nil otherwise.
|
|
1911 */
|
|
1912 (device, cursor, ignore_keyboard))
|
|
1913 {
|
|
1914 Window w;
|
|
1915 int pointer_mode, result;
|
|
1916 struct device *d = decode_x_device (device);
|
|
1917
|
|
1918 if (!NILP (cursor))
|
|
1919 {
|
|
1920 CHECK_POINTER_GLYPH (cursor);
|
|
1921 cursor = glyph_image_instance (cursor, device, ERROR_ME, 0);
|
|
1922 }
|
|
1923
|
|
1924 if (!NILP (ignore_keyboard))
|
|
1925 pointer_mode = GrabModeSync;
|
|
1926 else
|
|
1927 pointer_mode = GrabModeAsync;
|
|
1928
|
|
1929 w = XtWindow (FRAME_X_TEXT_WIDGET (device_selected_frame (d)));
|
|
1930
|
|
1931 /* #### Possibly this needs to gcpro the cursor somehow, but it doesn't
|
|
1932 seem to cause a problem if XFreeCursor is called on a cursor in use
|
|
1933 in a grab; I suppose the X server counts the grab as a reference
|
|
1934 and doesn't free it until it exits? */
|
|
1935 result = XGrabPointer (DEVICE_X_DISPLAY (d), w,
|
|
1936 False,
|
|
1937 ButtonMotionMask |
|
|
1938 ButtonPressMask |
|
|
1939 ButtonReleaseMask |
|
|
1940 PointerMotionHintMask,
|
|
1941 GrabModeAsync, /* Keep pointer events flowing */
|
|
1942 pointer_mode, /* Stall keyboard events */
|
|
1943 w, /* Stay in this window */
|
|
1944 (NILP (cursor) ? 0
|
|
1945 : XIMAGE_INSTANCE_X_CURSOR (cursor)),
|
|
1946 CurrentTime);
|
|
1947 return (result == GrabSuccess) ? Qt : Qnil;
|
|
1948 }
|
|
1949
|
|
1950 DEFUN ("x-ungrab-pointer", Fx_ungrab_pointer, 0, 1, 0, /*
|
|
1951 Release a pointer grab made with `x-grab-pointer'.
|
|
1952 If optional first arg DEVICE is nil the default device is used.
|
|
1953 If it is t the pointer will be released on all X devices.
|
|
1954 */
|
|
1955 (device))
|
|
1956 {
|
|
1957 if (!EQ (device, Qt))
|
|
1958 {
|
|
1959 Display *dpy = get_x_display (device);
|
|
1960 XUngrabPointer (dpy, CurrentTime);
|
|
1961 }
|
|
1962 else
|
|
1963 {
|
|
1964 Lisp_Object devcons, concons;
|
|
1965
|
|
1966 DEVICE_LOOP_NO_BREAK (devcons, concons)
|
|
1967 {
|
|
1968 struct device *d = XDEVICE (XCAR (devcons));
|
|
1969
|
|
1970 if (DEVICE_X_P (d))
|
|
1971 XUngrabPointer (DEVICE_X_DISPLAY (d), CurrentTime);
|
|
1972 }
|
|
1973 }
|
|
1974
|
|
1975 return Qnil;
|
|
1976 }
|
|
1977
|
|
1978 DEFUN ("x-grab-keyboard", Fx_grab_keyboard, 0, 1, 0, /*
|
|
1979 Grab the keyboard on the given device (defaulting to the selected one).
|
|
1980 So long as the keyboard is grabbed, all keyboard events will be delivered
|
|
1981 to emacs -- it is not possible for other X clients to eavesdrop on them.
|
|
1982 Ungrab the keyboard with `x-ungrab-keyboard' (use an unwind-protect).
|
|
1983 Returns t if the grab is successful, nil otherwise.
|
|
1984 */
|
|
1985 (device))
|
|
1986 {
|
|
1987 struct device *d = decode_x_device (device);
|
|
1988 Window w = XtWindow (FRAME_X_TEXT_WIDGET (device_selected_frame (d)));
|
|
1989 Display *dpy = DEVICE_X_DISPLAY (d);
|
|
1990 Status status;
|
|
1991 XSync (dpy, False);
|
|
1992 status = XGrabKeyboard (dpy, w, True,
|
|
1993 /* I don't really understand sync-vs-async
|
|
1994 grabs, but this is what xterm does. */
|
|
1995 GrabModeAsync, GrabModeAsync,
|
|
1996 /* Use the timestamp of the last user action
|
|
1997 read by emacs proper; xterm uses CurrentTime
|
|
1998 but there's a comment that says "wrong"...
|
|
1999 (Despite the name this is the time of the
|
|
2000 last key or mouse event.) */
|
|
2001 DEVICE_X_MOUSE_TIMESTAMP (d));
|
|
2002 if (status == GrabSuccess)
|
|
2003 {
|
|
2004 /* The XUngrabKeyboard should generate a FocusIn back to this
|
|
2005 window but it doesn't unless we explicitly set focus to the
|
|
2006 window first (which should already have it. The net result
|
|
2007 is that without this call when x-ungrab-keyboard is called
|
|
2008 the selected frame ends up not having focus. */
|
|
2009 XSetInputFocus (dpy, w, RevertToParent, DEVICE_X_MOUSE_TIMESTAMP (d));
|
|
2010 return Qt;
|
|
2011 }
|
|
2012 else
|
|
2013 return Qnil;
|
|
2014 }
|
|
2015
|
|
2016 DEFUN ("x-ungrab-keyboard", Fx_ungrab_keyboard, 0, 1, 0, /*
|
|
2017 Release a keyboard grab made with `x-grab-keyboard'.
|
|
2018 */
|
|
2019 (device))
|
|
2020 {
|
|
2021 Display *dpy = get_x_display (device);
|
|
2022 XUngrabKeyboard (dpy, CurrentTime);
|
|
2023 return Qnil;
|
|
2024 }
|
|
2025
|
|
2026 DEFUN ("x-get-font-path", Fx_get_font_path, 0, 1, 0, /*
|
|
2027 Get the X Server's font path.
|
|
2028
|
|
2029 See also `x-set-font-path'.
|
|
2030 */
|
|
2031 (device))
|
|
2032 {
|
|
2033 Display *dpy = get_x_display (device);
|
|
2034 int ndirs_return;
|
2367
|
2035 const Extbyte **directories =
|
|
2036 (const Extbyte **) XGetFontPath (dpy, &ndirs_return);
|
428
|
2037 Lisp_Object font_path = Qnil;
|
|
2038
|
|
2039 if (!directories)
|
563
|
2040 gui_error ("Can't get X font path", device);
|
428
|
2041
|
|
2042 while (ndirs_return--)
|
|
2043 font_path = Fcons (build_ext_string (directories[ndirs_return],
|
440
|
2044 Qfile_name),
|
|
2045 font_path);
|
428
|
2046
|
|
2047 return font_path;
|
|
2048 }
|
|
2049
|
|
2050 DEFUN ("x-set-font-path", Fx_set_font_path, 1, 2, 0, /*
|
|
2051 Set the X Server's font path to FONT-PATH.
|
|
2052
|
|
2053 There is only one font path per server, not one per client. Use this
|
|
2054 sparingly. It uncaches all of the X server's font information.
|
|
2055
|
|
2056 Font directories should end in the path separator and should contain
|
|
2057 a file called fonts.dir usually created with the program mkfontdir.
|
|
2058
|
|
2059 Setting the FONT-PATH to nil tells the X server to use the default
|
|
2060 font path.
|
|
2061
|
|
2062 See also `x-get-font-path'.
|
|
2063 */
|
|
2064 (font_path, device))
|
|
2065 {
|
|
2066 Display *dpy = get_x_display (device);
|
2367
|
2067 Extbyte **directories;
|
428
|
2068 int i=0,ndirs=0;
|
|
2069
|
2367
|
2070 {
|
|
2071 EXTERNAL_LIST_LOOP_2 (path_entry, font_path)
|
|
2072 {
|
|
2073 CHECK_STRING (path_entry);
|
|
2074 ndirs++;
|
|
2075 }
|
|
2076 }
|
428
|
2077
|
2367
|
2078 directories = alloca_array (Extbyte *, ndirs);
|
428
|
2079
|
2367
|
2080 {
|
|
2081 EXTERNAL_LIST_LOOP_2 (path_entry, font_path)
|
|
2082 {
|
|
2083 LISP_STRING_TO_EXTERNAL (path_entry, directories[i++],
|
|
2084 Qfile_name);
|
|
2085 }
|
|
2086 }
|
428
|
2087
|
|
2088 expect_x_error (dpy);
|
2367
|
2089 XSetFontPath (dpy, directories, ndirs);
|
428
|
2090 signal_if_x_error (dpy, 1/*resumable_p*/);
|
|
2091
|
|
2092 return Qnil;
|
|
2093 }
|
|
2094
|
|
2095
|
|
2096 /************************************************************************/
|
|
2097 /* initialization */
|
|
2098 /************************************************************************/
|
|
2099
|
|
2100 void
|
|
2101 syms_of_device_x (void)
|
|
2102 {
|
3092
|
2103 #ifdef NEW_GC
|
|
2104 INIT_LRECORD_IMPLEMENTATION (x_device);
|
|
2105 #endif /* NEW_GC */
|
|
2106
|
428
|
2107 DEFSUBR (Fx_debug_mode);
|
|
2108 DEFSUBR (Fx_get_resource);
|
|
2109 DEFSUBR (Fx_get_resource_prefix);
|
|
2110 DEFSUBR (Fx_put_resource);
|
|
2111
|
|
2112 DEFSUBR (Fdefault_x_device);
|
|
2113 DEFSUBR (Fx_display_visual_class);
|
|
2114 DEFSUBR (Fx_display_visual_depth);
|
|
2115 DEFSUBR (Fx_server_vendor);
|
|
2116 DEFSUBR (Fx_server_version);
|
|
2117 DEFSUBR (Fx_valid_keysym_name_p);
|
|
2118 DEFSUBR (Fx_keysym_hash_table);
|
|
2119 DEFSUBR (Fx_keysym_on_keyboard_p);
|
|
2120 DEFSUBR (Fx_keysym_on_keyboard_sans_modifiers_p);
|
|
2121
|
|
2122 DEFSUBR (Fx_grab_pointer);
|
|
2123 DEFSUBR (Fx_ungrab_pointer);
|
|
2124 DEFSUBR (Fx_grab_keyboard);
|
|
2125 DEFSUBR (Fx_ungrab_keyboard);
|
|
2126
|
|
2127 DEFSUBR (Fx_get_font_path);
|
|
2128 DEFSUBR (Fx_set_font_path);
|
|
2129
|
563
|
2130 DEFSYMBOL (Qx_error);
|
|
2131 DEFSYMBOL (Qinit_pre_x_win);
|
|
2132 DEFSYMBOL (Qinit_post_x_win);
|
771
|
2133
|
|
2134 #ifdef MULE
|
|
2135 DEFSYMBOL (Qget_coding_system_from_locale);
|
|
2136 #endif
|
428
|
2137 }
|
|
2138
|
|
2139 void
|
|
2140 reinit_console_type_create_device_x (void)
|
|
2141 {
|
|
2142 /* Initialize variables to speed up X resource interactions */
|
2367
|
2143 const Ascbyte *valid_resource_chars =
|
428
|
2144 "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
|
2145 while (*valid_resource_chars)
|
|
2146 valid_resource_char_p[(unsigned int) (*valid_resource_chars++)] = 1;
|
|
2147
|
771
|
2148 name_Extbyte_dynarr = Dynarr_new (Extbyte);
|
|
2149 class_Extbyte_dynarr = Dynarr_new (Extbyte);
|
428
|
2150 }
|
|
2151
|
|
2152 void
|
|
2153 console_type_create_device_x (void)
|
|
2154 {
|
|
2155 reinit_console_type_create_device_x ();
|
|
2156 CONSOLE_HAS_METHOD (x, init_device);
|
|
2157 CONSOLE_HAS_METHOD (x, finish_init_device);
|
|
2158 CONSOLE_HAS_METHOD (x, mark_device);
|
|
2159 CONSOLE_HAS_METHOD (x, delete_device);
|
|
2160 CONSOLE_HAS_METHOD (x, device_system_metrics);
|
|
2161 }
|
|
2162
|
|
2163 void
|
|
2164 reinit_vars_of_device_x (void)
|
|
2165 {
|
|
2166 error_expected = 0;
|
|
2167 error_occurred = 0;
|
|
2168
|
|
2169 in_resource_setting = 0;
|
|
2170 }
|
|
2171
|
|
2172 void
|
|
2173 vars_of_device_x (void)
|
|
2174 {
|
|
2175 DEFVAR_LISP ("x-emacs-application-class", &Vx_emacs_application_class /*
|
|
2176 The X application class of the XEmacs process.
|
|
2177 This controls, among other things, the name of the `app-defaults' file
|
|
2178 that XEmacs will use. For changes to this variable to take effect, they
|
|
2179 must be made before the connection to the X server is initialized, that is,
|
|
2180 this variable may only be changed before emacs is dumped, or by setting it
|
|
2181 in the file lisp/term/x-win.el.
|
|
2182
|
2681
|
2183 If this variable is nil on startup, the application uses `XEmacs'. Versions
|
|
2184 previous to 21.5.21 examined the resource database and used `XEmacs' if any
|
|
2185 resources beginning with that string existed, and `Emacs' otherwise, for
|
2828
|
2186 greater backward compatibility. However, this has always tended to conflict
|
2681
|
2187 with GNU Emacs, so this behavior is deprecated--in the short term, you can
|
|
2188 restore it in a post-21.5.21 XEmacs by setting the
|
|
2189 USE_EMACS_AS_DEFAULT_APPLICATION_CLASS environment variable to some value,
|
|
2190 but in the medium and long term, you should migrate your X resources.
|
428
|
2191 */ );
|
|
2192 Vx_emacs_application_class = Qnil;
|
|
2193
|
|
2194 DEFVAR_LISP ("x-initial-argv-list", &Vx_initial_argv_list /*
|
|
2195 You don't want to know.
|
|
2196 This is used during startup to communicate the remaining arguments in
|
|
2197 `command-line-args-left' to the C code, which passes the args to
|
|
2198 the X initialization code, which removes some args, and then the
|
|
2199 args are placed back into `x-initial-arg-list' and thence into
|
|
2200 `command-line-args-left'. Perhaps `command-line-args-left' should
|
|
2201 just reside in C.
|
|
2202 */ );
|
|
2203 Vx_initial_argv_list = Qnil;
|
|
2204
|
771
|
2205 #ifdef MULE
|
428
|
2206 DEFVAR_LISP ("x-app-defaults-directory", &Vx_app_defaults_directory /*
|
|
2207 Used by the Lisp code to communicate to the low level X initialization
|
|
2208 where the localized init files are.
|
|
2209 */ );
|
|
2210 Vx_app_defaults_directory = Qnil;
|
|
2211 #endif
|
|
2212
|
|
2213 Fprovide (Qx);
|
|
2214 }
|