Mercurial > hg > xemacs-beta
annotate src/dialog-x.c @ 5001:714d854d00e9
Automatic merge
author | Ben Wing <ben@xemacs.org> |
---|---|
date | Sun, 07 Feb 2010 05:23:07 -0600 |
parents | 3c3c1d139863 |
children | ae48681c47fa |
rev | line source |
---|---|
428 | 1 /* Implements elisp-programmable dialog boxes -- X interface. |
2 Copyright (C) 1993, 1994 Free Software Foundation, Inc. | |
3 Copyright (C) 1995 Tinker Systems and INS Engineering Corp. | |
1346 | 4 Copyright (C) 2000, 2002, 2003 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 /* This file Mule-ized by Ben Wing, 7-8-00. */ |
26 | |
428 | 27 #include <config.h> |
28 #include "lisp.h" | |
29 | |
30 #include "buffer.h" | |
31 #include "commands.h" /* zmacs_regions */ | |
32 #include "events.h" | |
872 | 33 #include "frame-impl.h" |
428 | 34 #include "gui.h" |
35 #include "opaque.h" | |
36 #include "window.h" | |
37 | |
872 | 38 #include "console-x-impl.h" |
39 | |
40 #include "EmacsFrame.h" | |
428 | 41 |
42 static void | |
43 maybe_run_dbox_text_callback (LWLIB_ID id) | |
44 { | |
45 widget_value *wv; | |
46 int got_some; | |
47 wv = xmalloc_widget_value (); | |
436 | 48 wv->name = xstrdup ("value"); |
428 | 49 got_some = lw_get_some_values (id, wv); |
50 if (got_some) | |
51 { | |
52 Lisp_Object text_field_callback; | |
442 | 53 Extbyte *text_field_value = wv->value; |
826 | 54 text_field_callback = VOID_TO_LISP (wv->call_data); |
442 | 55 text_field_callback = XCAR (XCDR (text_field_callback)); |
428 | 56 if (text_field_value) |
57 { | |
442 | 58 void *tmp = |
59 LISP_TO_VOID (cons3 (Qnil, | |
60 list2 (text_field_callback, | |
4953
304aebb79cd3
function renamings to track names of char typedefs
Ben Wing <ben@xemacs.org>
parents:
3466
diff
changeset
|
61 build_extstring (text_field_value, |
442 | 62 Qlwlib_encoding)), |
63 Qnil)); | |
428 | 64 popup_selection_callback (0, id, (XtPointer) tmp); |
65 } | |
66 } | |
436 | 67 /* This code tried to optimize, newing/freeing. This is generally |
442 | 68 unsafe so we will always strdup and always use |
436 | 69 free_widget_value_tree. */ |
70 free_widget_value_tree (wv); | |
428 | 71 } |
72 | |
73 static void | |
74 dbox_selection_callback (Widget widget, LWLIB_ID id, XtPointer client_data) | |
75 { | |
76 /* This is called with client_data == -1 when WM_DELETE_WINDOW is sent | |
77 instead of a button being selected. */ | |
78 struct device *d = get_device_from_display (XtDisplay (widget)); | |
79 struct frame *f = 0; | |
80 Widget cur_widget = widget; | |
81 | |
82 /* The parent which is actually connected to our EmacsFrame may be a | |
83 ways up the tree. */ | |
84 while (!f && cur_widget) | |
85 { | |
86 f = x_any_window_to_frame (d, XtWindow (cur_widget)); | |
87 cur_widget = XtParent (cur_widget); | |
88 } | |
89 | |
90 if (popup_handled_p (id)) | |
91 return; | |
92 assert (popup_up_p != 0); | |
93 ungcpro_popup_callbacks (id); | |
94 popup_up_p--; | |
95 maybe_run_dbox_text_callback (id); | |
96 popup_selection_callback (widget, id, client_data); | |
442 | 97 /* #### need to error-protect! will do so when i merge in |
98 my working ws */ | |
99 va_run_hook_with_args (Qdelete_dialog_box_hook, 1, make_int (id)); | |
428 | 100 lw_destroy_all_widgets (id); |
101 | |
102 /* The Motif dialog box sets the keyboard focus to itself. When it | |
103 goes away we have to take care of getting the focus back | |
104 ourselves. */ | |
105 #ifdef EXTERNAL_WIDGET | |
106 /* #### Not sure if this special case is necessary. */ | |
3466 | 107 if (f && !FRAME_X_EXTERNAL_WINDOW_P (f)) |
428 | 108 #else |
109 if (f) | |
110 #endif | |
111 lw_set_keyboard_focus (FRAME_X_SHELL_WIDGET (f), FRAME_X_TEXT_WIDGET (f)); | |
112 } | |
113 | |
442 | 114 static const Extbyte * const button_names [] = { |
428 | 115 "button1", "button2", "button3", "button4", "button5", |
116 "button6", "button7", "button8", "button9", "button10" }; | |
117 | |
118 static widget_value * | |
442 | 119 dbox_descriptor_to_widget_value (Lisp_Object keys) |
428 | 120 { |
121 /* This function can GC */ | |
122 int lbuttons = 0, rbuttons = 0; | |
123 int partition_seen = 0; | |
124 int text_field_p = 0; | |
125 int allow_text_p = 1; | |
126 widget_value *prev = 0, *kids = 0; | |
127 int n = 0; | |
853 | 128 int count; |
428 | 129 Lisp_Object wv_closure, gui_item; |
444 | 130 Lisp_Object question = Qnil; |
131 Lisp_Object buttons = Qnil; | |
2286 | 132 /* Lisp_Object title = Qnil; #### currently unused */ |
428 | 133 |
442 | 134 { |
135 EXTERNAL_PROPERTY_LIST_LOOP_3 (key, value, keys) | |
136 { | |
137 if (EQ (key, Q_question)) | |
138 { | |
139 CHECK_STRING (value); | |
140 question = value; | |
141 } | |
142 else if (EQ (key, Q_title)) | |
143 { | |
144 CHECK_STRING (value); | |
2286 | 145 /* title = value; */ |
442 | 146 } |
147 else if (EQ (key, Q_buttons)) | |
148 { | |
149 CHECK_LIST (value); | |
150 buttons = value; | |
151 } | |
152 else | |
563 | 153 invalid_constant ("Unrecognized question-dialog keyword", key); |
442 | 154 } |
155 } | |
156 | |
157 if (NILP (question)) | |
563 | 158 sferror ("Dialog descriptor provides no question", keys); |
428 | 159 |
160 /* Inhibit GC during this conversion. The reasons for this are | |
161 the same as in menu_item_descriptor_to_widget_value(); see | |
162 the large comment above that function. */ | |
163 | |
853 | 164 count = begin_gc_forbidden (); |
428 | 165 |
166 kids = prev = xmalloc_widget_value (); | |
167 | |
168 /* Also make sure that we free the partially-created widget_value | |
169 tree on Lisp error. */ | |
170 | |
171 wv_closure = make_opaque_ptr (kids); | |
172 record_unwind_protect (widget_value_unwind, wv_closure); | |
436 | 173 prev->name = xstrdup ("message"); |
4981
4aebb0131297
Cleanups/renaming of EXTERNAL_TO_C_STRING and friends
Ben Wing <ben@xemacs.org>
parents:
4953
diff
changeset
|
174 prev->value = LISP_STRING_TO_EXTERNAL_MALLOC (question, Qlwlib_encoding); |
428 | 175 prev->enabled = 1; |
176 | |
442 | 177 { |
178 EXTERNAL_LIST_LOOP_2 (button, buttons) | |
179 { | |
180 widget_value *wv; | |
428 | 181 |
442 | 182 if (NILP (button)) |
183 { | |
184 if (partition_seen) | |
563 | 185 sferror ("More than one partition (nil) seen in dbox spec", |
442 | 186 keys); |
187 partition_seen = 1; | |
188 continue; | |
189 } | |
190 CHECK_VECTOR (button); | |
191 wv = xmalloc_widget_value (); | |
428 | 192 |
442 | 193 gui_item = gui_parse_item_keywords (button); |
194 if (!button_item_to_widget_value (Qdialog, | |
195 gui_item, wv, allow_text_p, 1, 0, 1)) | |
196 { | |
197 free_widget_value_tree (wv); | |
198 continue; | |
199 } | |
428 | 200 |
442 | 201 if (wv->type == TEXT_TYPE) |
202 { | |
203 text_field_p = 1; | |
204 allow_text_p = 0; /* only allow one */ | |
205 } | |
206 else /* it's a button */ | |
207 { | |
208 allow_text_p = 0; /* only allow text field at the front */ | |
209 if (wv->value) | |
4976
16112448d484
Rename xfree(FOO, TYPE) -> xfree(FOO)
Ben Wing <ben@xemacs.org>
parents:
4953
diff
changeset
|
210 xfree (wv->value); |
442 | 211 wv->value = wv->name; /* what a mess... */ |
212 wv->name = xstrdup (button_names [n]); | |
428 | 213 |
442 | 214 if (partition_seen) |
215 rbuttons++; | |
216 else | |
217 lbuttons++; | |
218 n++; | |
428 | 219 |
442 | 220 if (lbuttons > 9 || rbuttons > 9) |
563 | 221 sferror ("Too many buttons (9)", |
442 | 222 keys); /* #### this leaks */ |
223 } | |
428 | 224 |
442 | 225 prev->next = wv; |
226 prev = wv; | |
227 } | |
228 } | |
428 | 229 |
230 if (n == 0) | |
563 | 231 sferror ("Dialog boxes must have some buttons", keys); |
442 | 232 |
428 | 233 { |
442 | 234 Extbyte type = (text_field_p ? 'P' : 'Q'); |
235 static Extbyte tmp_dbox_name [255]; | |
236 | |
428 | 237 widget_value *dbox; |
238 sprintf (tmp_dbox_name, "%c%dBR%d", type, lbuttons + rbuttons, rbuttons); | |
239 dbox = xmalloc_widget_value (); | |
436 | 240 dbox->name = xstrdup (tmp_dbox_name); |
428 | 241 dbox->contents = kids; |
242 | |
243 /* No more need to free the half-filled-in structures. */ | |
244 set_opaque_ptr (wv_closure, 0); | |
771 | 245 unbind_to (count); |
428 | 246 return dbox; |
247 } | |
248 } | |
249 | |
442 | 250 static Lisp_Object |
251 x_make_dialog_box_internal (struct frame* f, Lisp_Object type, | |
252 Lisp_Object keys) | |
428 | 253 { |
254 int dbox_id; | |
255 widget_value *data; | |
2286 | 256 Widget parent; |
428 | 257 |
442 | 258 if (!EQ (type, Qquestion)) |
563 | 259 signal_error (Qunimplemented, "Dialog box type", type); |
442 | 260 |
261 data = dbox_descriptor_to_widget_value (keys); | |
428 | 262 |
263 parent = FRAME_X_SHELL_WIDGET (f); | |
264 | |
265 dbox_id = new_lwlib_id (); | |
2286 | 266 (void) lw_create_widget (data->name, "dialog", dbox_id, data, parent, 1, 0, |
428 | 267 dbox_selection_callback, 0); |
268 lw_modify_all_widgets (dbox_id, data, True); | |
269 lw_modify_all_widgets (dbox_id, data->contents, True); | |
270 free_popup_widget_value_tree (data); | |
271 | |
272 gcpro_popup_callbacks (dbox_id); | |
273 | |
274 /* Setting zmacs-region-stays is necessary here because executing a | |
275 command from a dialog is really a two-command process: the first | |
276 command (bound to the button-click) simply pops up the dialog, | |
277 and returns. This causes a sequence of magic-events (destined | |
278 for the dialog widget) to begin. Eventually, a dialog item is | |
279 selected, and a misc-user-event blip is pushed onto the end of | |
280 the input stream, which is then executed by the event loop. | |
281 | |
282 So there are two command-events, with a bunch of magic-events | |
283 between them. We don't want the *first* command event to alter | |
284 the state of the region, so that the region can be available as | |
285 an argument for the second command. */ | |
286 if (zmacs_regions) | |
287 zmacs_region_stays = 1; | |
288 | |
289 popup_up_p++; | |
290 lw_pop_up_all_widgets (dbox_id); | |
442 | 291 |
292 /* #### this could (theoretically) cause problems if we are up for | |
293 a REALLY REALLY long time -- too big to fit into lisp integer. */ | |
294 return make_int (dbox_id); | |
428 | 295 } |
296 | |
297 void | |
298 syms_of_dialog_x (void) | |
299 { | |
300 } | |
301 | |
302 void | |
303 console_type_create_dialog_x (void) | |
304 { | |
442 | 305 CONSOLE_HAS_METHOD (x, make_dialog_box_internal); |
428 | 306 } |
307 | |
308 void | |
309 vars_of_dialog_x (void) | |
310 { | |
311 #if defined (LWLIB_DIALOGS_LUCID) | |
312 Fprovide (intern ("lucid-dialogs")); | |
313 #elif defined (LWLIB_DIALOGS_MOTIF) | |
314 Fprovide (intern ("motif-dialogs")); | |
315 #elif defined (LWLIB_DIALOGS_ATHENA) | |
316 Fprovide (intern ("athena-dialogs")); | |
317 #endif | |
318 } |