Match handles with multiple titles for channel Chat Boxes
[xboard.git] / xoptions.c
1 /*
2  * xoptions.c -- Move list window, part of X front end for XBoard
3  *
4  * Copyright 2000, 2009, 2010 Free Software Foundation, Inc.
5  * ------------------------------------------------------------------------
6  *
7  * GNU XBoard is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or (at
10  * your option) any later version.
11  *
12  * GNU XBoard is distributed in the hope that it will be useful, but
13  * WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15  * General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see http://www.gnu.org/licenses/.  *
19  *
20  *------------------------------------------------------------------------
21  ** See the file ChangeLog for a revision history.  */
22
23 // [HGM] this file is the counterpart of woptions.c, containing xboard popup menus
24 // similar to those of WinBoard, to set the most common options interactively.
25
26 #include "config.h"
27
28 #include <stdio.h>
29 #include <ctype.h>
30 #include <errno.h>
31 #include <sys/types.h>
32
33 #if STDC_HEADERS
34 # include <stdlib.h>
35 # include <string.h>
36 #else /* not STDC_HEADERS */
37 extern char *getenv();
38 # if HAVE_STRING_H
39 #  include <string.h>
40 # else /* not HAVE_STRING_H */
41 #  include <strings.h>
42 # endif /* not HAVE_STRING_H */
43 #endif /* not STDC_HEADERS */
44
45 #if HAVE_UNISTD_H
46 # include <unistd.h>
47 #endif
48 #include <stdint.h>
49
50 #include <X11/Intrinsic.h>
51 #include <X11/StringDefs.h>
52 #include <X11/Shell.h>
53 #include <X11/Xaw/Dialog.h>
54 #include <X11/Xaw/Form.h>
55 #include <X11/Xaw/List.h>
56 #include <X11/Xaw/Label.h>
57 #include <X11/Xaw/SimpleMenu.h>
58 #include <X11/Xaw/SmeBSB.h>
59 #include <X11/Xaw/SmeLine.h>
60 #include <X11/Xaw/Box.h>
61 #include <X11/Xaw/Paned.h>
62 #include <X11/Xaw/MenuButton.h>
63 #include <X11/cursorfont.h>
64 #include <X11/Xaw/Text.h>
65 #include <X11/Xaw/AsciiText.h>
66 #include <X11/Xaw/Viewport.h>
67 #include <X11/Xaw/Toggle.h>
68
69 #include "common.h"
70 #include "backend.h"
71 #include "xboard.h"
72 #include "gettext.h"
73
74 #ifdef ENABLE_NLS
75 # define  _(s) gettext (s)
76 # define N_(s) gettext_noop (s)
77 #else
78 # define  _(s) (s)
79 # define N_(s)  s
80 #endif
81
82 extern void SendToProgram P((char *message, ChessProgramState *cps));
83
84 extern Widget formWidget, shellWidget, boardWidget, menuBarWidget;
85 extern Display *xDisplay;
86 extern int squareSize;
87 extern Pixmap xMarkPixmap;
88 extern char *layoutName;
89 extern Window xBoardWindow;
90 extern Arg layoutArgs[2], formArgs[2];
91 Pixel timerForegroundPixel, timerBackgroundPixel;
92
93 // [HGM] the following code for makng menu popups was cloned from the FileNamePopUp routines
94
95 static Widget previous = NULL;
96
97 void SetFocus(Widget w, XtPointer data, XEvent *event, Boolean *b)
98 {
99     Arg args;
100
101     if(previous) {
102         XtSetArg(args, XtNdisplayCaret, False);
103         XtSetValues(previous, &args, 1);
104     }
105     XtSetArg(args, XtNdisplayCaret, True);
106     XtSetValues(w, &args, 1);
107     XtSetKeyboardFocus((Widget) data, w);
108     previous = w;
109 }
110
111 //--------------------------- New Shuffle Game --------------------------------------------
112 extern int shuffleOpenings;
113 extern int startedFromPositionFile;
114 int shuffleUp;
115 Widget shuffleShell;
116
117 void ShufflePopDown()
118 {
119     if (!shuffleUp) return;
120     XtPopdown(shuffleShell);
121     XtDestroyWidget(shuffleShell);
122     shuffleUp = False;
123     ModeHighlight();
124 }
125
126 void ShuffleCallback(w, client_data, call_data)
127      Widget w;
128      XtPointer client_data, call_data;
129 {
130     String name;
131     Widget w2;
132     Arg args[16];
133     char buf[80];
134     
135     XtSetArg(args[0], XtNlabel, &name);
136     XtGetValues(w, args, 1);
137     
138     if (strcmp(name, _("cancel")) == 0) {
139         ShufflePopDown();
140         return;
141     }
142     if (strcmp(name, _("off")) == 0) {
143         ShufflePopDown();
144         shuffleOpenings = False; // [HGM] should be moved to New Variant menu, once we have it!
145         ResetGameEvent();
146         return;
147     }
148     if (strcmp(name, _("random")) == 0) {
149         sprintf(buf, "%d", rand());
150         XtSetArg(args[0],XtNvalue, buf); // erase bad (non-numeric) value
151         XtSetValues(XtParent(w), args, 1);
152         return;
153     }
154     if (strcmp(name, _("ok")) == 0) {
155         int nr; String name;
156         name = XawDialogGetValueString(w2 = XtParent(w));
157         if(sscanf(name ,"%d",&nr) != 1) {
158             sprintf(buf, "%d", appData.defaultFrcPosition);
159             XtSetArg(args[0],XtNvalue, buf); // erase bad (non-numeric) value
160             XtSetValues(w2, args, 1);
161             return;
162         }
163         appData.defaultFrcPosition = nr;
164         shuffleOpenings = True;
165         ShufflePopDown();
166         ResetGameEvent();
167         return;
168     }
169 }
170
171 void ShufflePopUp()
172 {
173     Arg args[16];
174     Widget popup, layout, dialog, edit;
175     Window root, child;
176     int x, y, i;
177     int win_x, win_y;
178     unsigned int mask;
179     char def[80];
180     
181     i = 0;
182     XtSetArg(args[i], XtNresizable, True); i++;
183     XtSetArg(args[i], XtNwidth, DIALOG_SIZE); i++;
184     shuffleShell = popup =
185       XtCreatePopupShell(_("New Shuffle Game"), transientShellWidgetClass,
186                          shellWidget, args, i);
187     
188     layout =
189       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
190                             layoutArgs, XtNumber(layoutArgs));
191   
192     sprintf(def, "%d\n", appData.defaultFrcPosition);
193     i = 0;
194     XtSetArg(args[i], XtNlabel, _("Start-position number:")); i++;
195     XtSetArg(args[i], XtNvalue, def); i++;
196     XtSetArg(args[i], XtNborderWidth, 0); i++;
197     dialog = XtCreateManagedWidget(_("Shuffle"), dialogWidgetClass,
198                                    layout, args, i);
199     
200 //    XtSetArg(args[0], XtNeditType, XawtextEdit);  // [HGM] can't get edit to work decently
201 //    XtSetArg(args[1], XtNuseStringInPlace, False);
202 //    XtSetValues(dialog, args, 2);
203
204     XawDialogAddButton(dialog, _("ok"), ShuffleCallback, (XtPointer) dialog);
205     XawDialogAddButton(dialog, _("cancel"), ShuffleCallback, (XtPointer) dialog);
206     XawDialogAddButton(dialog, _("random"), ShuffleCallback, (XtPointer) dialog);
207     XawDialogAddButton(dialog, _("off"), ShuffleCallback, (XtPointer) dialog);
208     
209     XtRealizeWidget(popup);
210     CatchDeleteWindow(popup, "ShufflePopDown");
211     
212     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
213                   &x, &y, &win_x, &win_y, &mask);
214     
215     XtSetArg(args[0], XtNx, x - 10);
216     XtSetArg(args[1], XtNy, y - 30);
217     XtSetValues(popup, args, 2);
218     
219     XtPopup(popup, XtGrabExclusive);
220     shuffleUp = True;
221     
222     edit = XtNameToWidget(dialog, "*value");
223
224     XtSetKeyboardFocus(popup, edit);
225 }
226
227 void ShuffleMenuProc(w, event, prms, nprms)
228      Widget w;
229      XEvent *event;
230      String *prms;
231      Cardinal *nprms;
232 {
233 //    if (gameMode == AnalyzeMode || gameMode == AnalyzeFile) {
234 //      Reset(FALSE, TRUE);
235 //    }
236     ShufflePopUp();
237 }
238
239 //--------------------------- Time-Control Menu Popup ----------------------------------
240 int TimeControlUp;
241 Widget TimeControlShell;
242 int tcInc;
243 Widget tcMess1, tcMess2, tcData, tcTime, tcOdds1, tcOdds2;
244 int tcIncrement, tcMoves;
245
246 void TimeControlPopDown()
247 {
248     if (!TimeControlUp) return;
249     XtPopdown(TimeControlShell);
250     XtDestroyWidget(TimeControlShell);
251     TimeControlUp = False;
252     ModeHighlight();
253 }
254
255 void TimeControlCallback(w, client_data, call_data)
256      Widget w;
257      XtPointer client_data, call_data;
258 {
259     String name, txt;
260     Widget w2;
261     Arg args[16];
262     char buf[80];
263     int j;
264
265     XtSetArg(args[0], XtNlabel, &name);
266     XtGetValues(w, args, 1);
267     
268     if (strcmp(name, _("classical")) == 0) {
269         if(!tcInc) return;
270         j=0;
271         XtSetArg(args[j], XtNlabel, _("minutes for each")); j++;
272         XtSetValues(tcMess1, args, j);
273         j=0;
274         XtSetArg(args[j], XtNlabel, _("moves")); j++;
275         XtSetValues(tcMess2, args, j);
276         j=0;
277         XtSetArg(args[j], XtNstring, &name); j++;
278         XtGetValues(tcData, args, j);
279         tcIncrement = 0; sscanf(name, "%d", &tcIncrement);
280         sprintf(buf, "%d", tcMoves);
281         j=0;
282         XtSetArg(args[j], XtNstring, buf); j++;
283         XtSetValues(tcData, args, j);
284         tcInc = False;
285         return;
286     }
287     if (strcmp(name, _("incremental")) == 0) {
288         if(tcInc) return;
289         j=0;
290         XtSetArg(args[j], XtNlabel, _("minutes, plus")); j++;
291         XtSetValues(tcMess1, args, j);
292         j=0;
293         XtSetArg(args[j], XtNlabel, _("sec/move")); j++;
294         XtSetValues(tcMess2, args, j);
295         j=0;
296         XtSetArg(args[j], XtNstring, &name); j++;
297         XtGetValues(tcData, args, j);
298         tcMoves = appData.movesPerSession; sscanf(name, "%d", &tcMoves);
299         sprintf(buf, "%d", tcIncrement);
300         j=0;
301         XtSetArg(args[j], XtNstring, buf); j++;
302         XtSetValues(tcData, args, j);
303         tcInc = True;
304         return;
305     }
306     if (strcmp(name, _(" OK ")) == 0) {
307         int inc, mps, tc, ok;
308         XtSetArg(args[0], XtNstring, &txt);
309         XtGetValues(tcData, args, 1);
310         if(tcInc) {
311             ok = sscanf(txt, "%d", &inc); mps = 0;
312             if(!ok && txt[0] == 0) { inc = 0; ok = 1; } // accept empty string as zero
313             ok &= (inc >= 0);
314         } else {
315             ok = sscanf(txt, "%d", &mps); inc = -1;
316             ok &= (mps > 0);
317         }
318         if(ok != 1) {
319             XtSetArg(args[0], XtNstring, ""); // erase any offending input
320             XtSetValues(tcData, args, 1);
321             return;
322         }
323         XtSetArg(args[0], XtNstring, &txt);
324         XtGetValues(tcTime, args, 1);
325         if(!ParseTimeControl(txt, inc, mps)) {
326             XtSetArg(args[0], XtNstring, ""); // erase any offending input
327             XtSetValues(tcTime, args, 1);
328             DisplayError(_("Bad Time-Control String"), 0);
329             return;
330         }
331         appData.movesPerSession = mps;
332         appData.timeIncrement = inc;
333         appData.timeControl = strdup(txt);
334         XtSetArg(args[0], XtNstring, &txt);
335         XtGetValues(tcOdds1, args, 1);
336         appData.firstTimeOdds = first.timeOdds 
337                 = (sscanf(txt, "%d", &j) == 1 && j > 0) ? j : 1;
338         XtGetValues(tcOdds2, args, 1);
339         appData.secondTimeOdds = second.timeOdds 
340                 = (sscanf(txt, "%d", &j) == 1 && j > 0) ? j : 1;
341
342         Reset(True, True);
343         TimeControlPopDown();
344         return;
345     }
346 }
347
348 void TimeControlPopUp()
349 {
350     Arg args[16];
351     Widget popup, layout, form, edit, b_ok, b_cancel, b_clas, b_inc, mess;
352     Window root, child;
353     int x, y, i, j;
354     int win_x, win_y;
355     unsigned int mask;
356     char def[80];
357     
358     tcInc = (appData.timeIncrement >= 0);
359     tcMoves = appData.movesPerSession; tcIncrement = appData.timeIncrement;
360     if(!tcInc) tcIncrement = 0;
361     sprintf(def, "%d", tcInc ? tcIncrement : tcMoves);
362
363     i = 0;
364     XtSetArg(args[i], XtNresizable, True); i++;
365 //    XtSetArg(args[i], XtNwidth, DIALOG_SIZE); i++;
366     TimeControlShell = popup =
367       XtCreatePopupShell(_("TimeControl Menu"), transientShellWidgetClass,
368                          shellWidget, args, i);
369     
370     layout =
371       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
372                             layoutArgs, XtNumber(layoutArgs));
373   
374     form =
375       XtCreateManagedWidget(layoutName, formWidgetClass, layout,
376                             formArgs, XtNumber(formArgs));
377   
378     j = 0;
379 //    XtSetArg(args[j], XtNwidth,     (XtArgVal) 300); j++;
380 //    XtSetArg(args[j], XtNheight,    (XtArgVal) 85); j++;
381     XtSetValues(popup, args, j);
382
383     j= 0;
384     XtSetArg(args[j], XtNborderWidth, 1); j++;
385     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
386     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
387     XtSetArg(args[j], XtNstring, appData.timeControl);  j++;
388     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
389     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
390     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
391     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
392     XtSetArg(args[j], XtNright, XtChainRight);  j++;
393     XtSetArg(args[j], XtNresizable, True);  j++;
394     XtSetArg(args[j], XtNwidth,  85);  j++;
395 //    XtSetArg(args[j], XtNheight, 20);  j++;
396     tcTime = XtCreateManagedWidget("TC", asciiTextWidgetClass, form, args, j);
397     XtAddEventHandler(tcTime, ButtonPressMask, False, SetFocus, (XtPointer) popup);
398
399     j= 0;
400     XtSetArg(args[j], XtNlabel, tcInc ? _("   minutes, plus   ") : _("minutes for each")); j++;
401     XtSetArg(args[j], XtNborderWidth, 0); j++;
402     XtSetArg(args[j], XtNfromHoriz, tcTime); j++;
403     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
404     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
405     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
406     XtSetArg(args[j], XtNright, XtChainRight);  j++;
407   //  XtSetArg(args[j], XtNwidth,  100);  j++;
408   //  XtSetArg(args[j], XtNheight, 20);  j++;
409     tcMess1 = XtCreateManagedWidget("TCtext", labelWidgetClass, form, args, j);
410
411     j= 0;
412     XtSetArg(args[j], XtNborderWidth, 1); j++;
413     XtSetArg(args[j], XtNfromHoriz, tcMess1); j++;
414     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
415     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
416     XtSetArg(args[j], XtNstring, def);  j++;
417     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
418     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
419     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
420     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
421     XtSetArg(args[j], XtNright, XtChainRight);  j++;
422     XtSetArg(args[j], XtNresizable, True);  j++;
423     XtSetArg(args[j], XtNwidth,  40);  j++;
424 //    XtSetArg(args[j], XtNheight, 20);  j++;
425     tcData = XtCreateManagedWidget("MPS", asciiTextWidgetClass, form, args, j);
426     XtAddEventHandler(tcData, ButtonPressMask, False, SetFocus, (XtPointer) popup);
427
428     j= 0;
429     XtSetArg(args[j], XtNlabel, tcInc ? _("sec/move") : _("moves     ")); j++;
430     XtSetArg(args[j], XtNjustify, XtJustifyLeft); j++;
431     XtSetArg(args[j], XtNborderWidth, 0); j++;
432     XtSetArg(args[j], XtNfromHoriz, tcData); j++;
433     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
434     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
435     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
436     XtSetArg(args[j], XtNright, XtChainRight);  j++;
437 //    XtSetArg(args[j], XtNwidth,  80);  j++;
438 //    XtSetArg(args[j], XtNheight, 20);  j++;
439     tcMess2 = XtCreateManagedWidget("MPStext", labelWidgetClass,
440                                    form, args, j);
441
442     j= 0;
443     XtSetArg(args[j], XtNborderWidth, 1); j++;
444     XtSetArg(args[j], XtNfromVert, tcTime); j++;
445     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
446     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
447     XtSetArg(args[j], XtNstring, "1");  j++;
448     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
449     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
450     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
451     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
452     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
453     XtSetArg(args[j], XtNresizable, True);  j++;
454     XtSetArg(args[j], XtNwidth,  40);  j++;
455 //    XtSetArg(args[j], XtNheight, 20);  j++;
456     tcOdds1 = XtCreateManagedWidget("Odds1", asciiTextWidgetClass, form, args, j);
457     XtAddEventHandler(tcOdds1, ButtonPressMask, False, SetFocus, (XtPointer) popup);
458
459     j= 0;
460     XtSetArg(args[j], XtNborderWidth, 1); j++;
461     XtSetArg(args[j], XtNfromVert, tcTime); j++;
462     XtSetArg(args[j], XtNfromHoriz, tcOdds1); j++;
463     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
464     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
465     XtSetArg(args[j], XtNstring, "1");  j++;
466     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
467     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
468     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
469     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
470     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
471     XtSetArg(args[j], XtNresizable, True);  j++;
472     XtSetArg(args[j], XtNwidth,  40);  j++;
473 //    XtSetArg(args[j], XtNheight, 20);  j++;
474     tcOdds2 = XtCreateManagedWidget("Odds2", asciiTextWidgetClass, form, args, j);
475     XtAddEventHandler(tcOdds2, ButtonPressMask, False, SetFocus, (XtPointer) popup);
476
477     j= 0;
478     XtSetArg(args[j], XtNlabel, _("Engine #1 and #2 Time-Odds Factors")); j++;
479     XtSetArg(args[j], XtNjustify, XtJustifyLeft); j++;
480     XtSetArg(args[j], XtNborderWidth, 0); j++;
481     XtSetArg(args[j], XtNfromVert, tcTime); j++;
482     XtSetArg(args[j], XtNfromHoriz, tcOdds2); j++;
483     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
484     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
485     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
486     XtSetArg(args[j], XtNright, XtChainRight);  j++;
487 //    XtSetArg(args[j], XtNwidth,  200);  j++;
488 //    XtSetArg(args[j], XtNheight, 20);  j++;
489     mess = XtCreateManagedWidget("Oddstext", labelWidgetClass,
490                                    form, args, j);
491     j=0;
492     XtSetArg(args[j], XtNfromVert, tcOdds1);  j++;
493     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
494     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
495     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
496     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
497     b_clas= XtCreateManagedWidget(_("classical"), commandWidgetClass,
498                                    form, args, j);   
499     XtAddCallback(b_clas, XtNcallback, TimeControlCallback, (XtPointer) 0);
500
501     j=0;
502     XtSetArg(args[j], XtNfromVert, tcOdds1);  j++;
503     XtSetArg(args[j], XtNfromHoriz, b_clas);  j++;
504     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
505     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
506     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
507     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
508     b_inc = XtCreateManagedWidget(_("incremental"), commandWidgetClass,
509                                    form, args, j);   
510     XtAddCallback(b_inc, XtNcallback, TimeControlCallback, (XtPointer) 0);
511
512     j=0;
513     XtSetArg(args[j], XtNfromVert, tcOdds1);  j++;
514     XtSetArg(args[j], XtNfromHoriz, tcData);  j++;
515     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
516     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
517     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
518     XtSetArg(args[j], XtNright, XtChainRight);  j++;
519     b_ok= XtCreateManagedWidget(_(" OK "), commandWidgetClass,
520                                    form, args, j);   
521     XtAddCallback(b_ok, XtNcallback, TimeControlCallback, (XtPointer) 0);
522
523     j=0;
524     XtSetArg(args[j], XtNfromVert, tcOdds1);  j++;
525     XtSetArg(args[j], XtNfromHoriz, b_ok);  j++;
526     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
527     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
528     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
529     XtSetArg(args[j], XtNright, XtChainRight);  j++;
530     b_cancel= XtCreateManagedWidget(_("cancel"), commandWidgetClass,
531                                    form, args, j);   
532     XtAddCallback(b_cancel, XtNcallback, TimeControlPopDown, (XtPointer) 0);
533
534     XtRealizeWidget(popup);
535     CatchDeleteWindow(popup, "TimeControlPopDown");
536     
537     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
538                   &x, &y, &win_x, &win_y, &mask);
539     
540     XtSetArg(args[0], XtNx, x - 10);
541     XtSetArg(args[1], XtNy, y - 30);
542     XtSetValues(popup, args, 2);
543     
544     XtPopup(popup, XtGrabExclusive);
545     TimeControlUp = True;
546     
547     previous = NULL;
548     SetFocus(tcTime, popup, (XEvent*) NULL, False);
549 //    XtSetKeyboardFocus(popup, tcTime);
550 }
551
552 void TimeControlProc(w, event, prms, nprms)
553      Widget w;
554      XEvent *event;
555      String *prms;
556      Cardinal *nprms;
557 {
558    TimeControlPopUp();
559 }
560
561 //--------------------------- Engine-Options Menu Popup ----------------------------------
562 int EngineUp;
563 Widget EngineShell;
564 extern int adjudicateLossThreshold;
565
566 Widget engDrawMoves, engThreshold, engRule, engRepeat;
567
568 void EnginePopDown()
569 {
570     if (!EngineUp) return;
571     XtPopdown(EngineShell);
572     XtDestroyWidget(EngineShell);
573     EngineUp = False;
574     ModeHighlight();
575 }
576
577 int ReadToggle(Widget w)
578 {
579     Arg args; Boolean res;
580
581     XtSetArg(args, XtNstate, &res);
582     XtGetValues(w, &args, 1);
583
584     return res;
585 }
586
587 Widget w1, w2, w3, w4, w5, w6, w7, w8;
588
589 void EngineCallback(w, client_data, call_data)
590      Widget w;
591      XtPointer client_data, call_data;
592 {
593     String name;
594     Widget s2;
595     Arg args[16];
596     char buf[80];
597     int j;
598     
599     XtSetArg(args[0], XtNlabel, &name);
600     XtGetValues(w, args, 1);
601     
602     if (strcmp(name, _("OK")) == 0) {
603         // read all switches
604         appData.periodicUpdates = ReadToggle(w1);
605 //      appData.hideThinkingFromHuman = ReadToggle(w2);
606         first.scoreIsAbsolute  = appData.firstScoreIsAbsolute  = ReadToggle(w3);
607         second.scoreIsAbsolute = appData.secondScoreIsAbsolute = ReadToggle(w4);
608         appData.testClaims    = ReadToggle(w5);
609         appData.checkMates    = ReadToggle(w6);
610         appData.materialDraws = ReadToggle(w7);
611         appData.trivialDraws  = ReadToggle(w8);
612
613         // adjust setting in other menu for duplicates 
614         // (perhaps duplicates should be removed from general Option Menu?)
615 //      XtSetArg(args[0], XtNleftBitmap, appData.showThinking ? xMarkPixmap : None);
616 //      XtSetValues(XtNameToWidget(menuBarWidget,
617 //                                 "menuOptions.Show Thinking"), args, 1);
618
619         // read out numeric controls, simply ignore bad formats for now
620         XtSetArg(args[0], XtNstring, &name);
621         XtGetValues(engDrawMoves, args, 1);
622         if(sscanf(name, "%d", &j) == 1) appData.adjudicateDrawMoves = j;
623         XtGetValues(engThreshold, args, 1);
624         if(sscanf(name, "%d", &j) == 1) 
625                 adjudicateLossThreshold = appData.adjudicateLossThreshold = -j; // inverted!
626         XtGetValues(engRule, args, 1);
627         if(sscanf(name, "%d", &j) == 1) appData.ruleMoves = j;
628         XtGetValues(engRepeat, args, 1);
629         if(sscanf(name, "%d", &j) == 1) appData.drawRepeats = j;
630
631         EnginePopDown();
632         ShowThinkingEvent(); // [HGM] thinking: score adjudication might need thinking output
633         return;
634     }
635 }
636
637 void EnginePopUp()
638 {
639     Arg args[16];
640     Widget popup, layout, form, edit, b_ok, b_cancel, b_clas, b_inc, s1; 
641     Window root, child;
642     int x, y, i, j, width;
643     int win_x, win_y;
644     unsigned int mask;
645     char def[80];
646     
647     tcInc = (appData.timeIncrement >= 0);
648     tcMoves = appData.movesPerSession; tcIncrement = appData.timeIncrement;
649     if(!tcInc) tcIncrement = 0;
650     sprintf(def, "%d", tcInc ? tcIncrement : tcMoves);
651
652     i = 0;
653     XtSetArg(args[i], XtNresizable, True); i++;
654 //    XtSetArg(args[i], XtNwidth, DIALOG_SIZE); i++;
655     EngineShell = popup =
656       XtCreatePopupShell(_("Adjudications"), transientShellWidgetClass,
657                          shellWidget, args, i);
658     
659     layout =
660       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
661                             layoutArgs, XtNumber(layoutArgs));
662   
663     form =
664       XtCreateManagedWidget(layoutName, formWidgetClass, layout,
665                             formArgs, XtNumber(formArgs));
666   
667     j = 0;
668 //    XtSetArg(args[j], XtNwidth,     (XtArgVal) 250); j++;
669 //    XtSetArg(args[j], XtNheight,    (XtArgVal) 400); j++;
670 //    XtSetValues(popup, args, j);
671
672     j = 0;
673 //    XtSetArg(args[j], XtNwidth,       (XtArgVal) 250); j++;
674 //    XtSetArg(args[j], XtNheight,      (XtArgVal) 20); j++;
675     XtSetArg(args[j], XtNleft,        (XtArgVal) XtChainLeft); j++;
676     XtSetArg(args[j], XtNright,       (XtArgVal) XtChainRight); j++;
677     XtSetArg(args[j], XtNstate,       appData.periodicUpdates); j++;
678 //    XtSetArg(args[j], XtNjustify,     (XtArgVal) XtJustifyLeft); j++;
679     w1 = XtCreateManagedWidget(_("Periodic Updates (Analysis Mode)"), toggleWidgetClass, form, args, j);
680
681     XtSetArg(args[j], XtNwidth,       (XtArgVal) &width);
682     XtGetValues(w1, &args[j], 1);
683
684 //    XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) w1);
685 //    XtSetArg(args[j-3], XtNstate,       appData.hideThinkingFromHuman);
686 //    w2 = XtCreateManagedWidget(_("Hide Thinking from Human"), toggleWidgetClass, form, args, j);
687
688     XtSetArg(args[j], XtNwidth,       (XtArgVal) width); j++;
689     XtSetArg(args[j-2], XtNstate,     appData.firstScoreIsAbsolute);
690     XtSetArg(args[j], XtNfromVert,    (XtArgVal) w1); j++;
691     w3 = XtCreateManagedWidget(_("Engine #1 Score is Absolute"), toggleWidgetClass, form, args, j);
692
693     XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) w3);
694     XtSetArg(args[j-3], XtNstate,       appData.secondScoreIsAbsolute);
695     w4 = XtCreateManagedWidget(_("Engine #2 Score is Absolute"), toggleWidgetClass, form, args, j);
696
697     s1 = XtCreateManagedWidget(_("\nEngine-Engine Adjudications:"), labelWidgetClass, form, args, 3);
698
699     XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) s1);
700     XtSetArg(args[j-3], XtNstate,       appData.testClaims);
701     w5 = XtCreateManagedWidget(_("Verify Engine Result Claims"), toggleWidgetClass, form, args, j);
702
703     XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) w5);
704     XtSetArg(args[j-3], XtNstate,       appData.checkMates);
705     w6 = XtCreateManagedWidget(_("Detect All Mates"), toggleWidgetClass, form, args, j);
706
707     XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) w6);
708     XtSetArg(args[j-3], XtNstate,       appData.materialDraws);
709     w7 = XtCreateManagedWidget(_("Draw when Insuff. Mating Material"), toggleWidgetClass, form, args, j);
710
711     XtSetArg(args[j-1], XtNfromVert,  (XtArgVal) w7);
712     XtSetArg(args[j-3], XtNstate,       appData.trivialDraws);
713     w8 = XtCreateManagedWidget(_("Adjudicate Trivial Draws"), toggleWidgetClass, form, args, j);
714
715     XtSetArg(args[0], XtNfromVert,  (XtArgVal) w4);
716     XtSetArg(args[1], XtNborderWidth, (XtArgVal) 0);
717     XtSetValues(s1, args, 2);
718
719     sprintf(def, "%d", appData.adjudicateDrawMoves);
720     j= 0;
721     XtSetArg(args[j], XtNborderWidth, 1); j++;
722     XtSetArg(args[j], XtNfromVert, w8); j++;
723     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
724     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
725     XtSetArg(args[j], XtNstring, def);  j++;
726     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
727     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
728     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
729     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
730     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
731     XtSetArg(args[j], XtNresizable, True);  j++;
732     XtSetArg(args[j], XtNwidth,  60);  j++;
733 //    XtSetArg(args[j], XtNheight, 20);  j++;
734     engDrawMoves = XtCreateManagedWidget("Length", asciiTextWidgetClass, form, args, j);
735     XtAddEventHandler(engDrawMoves, ButtonPressMask, False, SetFocus, (XtPointer) popup);
736
737     j= 0;
738     XtSetArg(args[j], XtNlabel, _(" moves maximum, then draw")); j++;
739     XtSetArg(args[j], XtNjustify,     (XtArgVal) XtJustifyLeft); j++;
740     XtSetArg(args[j], XtNborderWidth, 0); j++;
741     XtSetArg(args[j], XtNfromVert, w8); j++;
742     XtSetArg(args[j], XtNfromHoriz, engDrawMoves); j++;
743     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
744     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
745     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
746     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
747 //    XtSetArg(args[j], XtNwidth,  170);  j++;
748 //    XtSetArg(args[j], XtNheight, 20);  j++;
749     tcMess1 = XtCreateManagedWidget("TCtext", labelWidgetClass, form, args, j);
750
751     sprintf(def, "%d", -appData.adjudicateLossThreshold); // inverted!
752     j= 0;
753     XtSetArg(args[j], XtNborderWidth, 1); j++;
754     XtSetArg(args[j], XtNfromVert, engDrawMoves); j++;
755     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
756     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
757     XtSetArg(args[j], XtNstring, def);  j++;
758     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
759     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
760     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
761     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
762     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
763     XtSetArg(args[j], XtNresizable, True);  j++;
764     XtSetArg(args[j], XtNwidth,  60);  j++;
765 //    XtSetArg(args[j], XtNheight, 20);  j++;
766     engThreshold = XtCreateManagedWidget("Threshold", asciiTextWidgetClass, form, args, j);
767     XtAddEventHandler(engThreshold, ButtonPressMask, False, SetFocus, (XtPointer) popup);
768
769     j= 0;
770     XtSetArg(args[j], XtNlabel, _("-centiPawn lead is win")); j++;
771     XtSetArg(args[j], XtNjustify, XtJustifyLeft); j++;
772     XtSetArg(args[j], XtNborderWidth, 0); j++;
773     XtSetArg(args[j], XtNfromVert, engDrawMoves); j++;
774     XtSetArg(args[j], XtNfromHoriz, engThreshold); j++;
775     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
776     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
777     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
778     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
779 //    XtSetArg(args[j], XtNwidth,  150);  j++;
780 //    XtSetArg(args[j], XtNheight, 20);  j++;
781     tcMess2 = XtCreateManagedWidget("MPStext", labelWidgetClass, form, args, j);
782
783     sprintf(def, "%d", appData.ruleMoves);
784     j= 0;
785     XtSetArg(args[j], XtNborderWidth, 1); j++;
786     XtSetArg(args[j], XtNfromVert, engThreshold); j++;
787     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
788     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
789     XtSetArg(args[j], XtNstring, def);  j++;
790     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
791     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
792     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
793     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
794     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
795     XtSetArg(args[j], XtNresizable, True);  j++;
796     XtSetArg(args[j], XtNwidth,  30);  j++;
797 //    XtSetArg(args[j], XtNheight, 20);  j++;
798     engRule = XtCreateManagedWidget("Rule", asciiTextWidgetClass, form, args, j);
799     XtAddEventHandler(engRule, ButtonPressMask, False, SetFocus, (XtPointer) popup);
800
801     j= 0;
802     XtSetArg(args[j], XtNlabel, _("-move rule applied")); j++;
803     XtSetArg(args[j], XtNjustify,     (XtArgVal) XtJustifyLeft); j++;
804     XtSetArg(args[j], XtNborderWidth, 0); j++;
805     XtSetArg(args[j], XtNfromVert, engThreshold); j++;
806     XtSetArg(args[j], XtNfromHoriz, engRule); j++;
807     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
808     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
809     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
810     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
811 //    XtSetArg(args[j], XtNwidth,  130);  j++;
812 //    XtSetArg(args[j], XtNheight, 20);  j++;
813     tcMess1 = XtCreateManagedWidget("TCtext", labelWidgetClass, form, args, j);
814
815     sprintf(def, "%d", appData.drawRepeats);
816     j= 0;
817     XtSetArg(args[j], XtNborderWidth, 1); j++;
818     XtSetArg(args[j], XtNfromVert, engRule); j++;
819     XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
820     XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
821     XtSetArg(args[j], XtNstring, def);  j++;
822     XtSetArg(args[j], XtNdisplayCaret, False);  j++;
823     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
824     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
825     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
826     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
827     XtSetArg(args[j], XtNresizable, True);  j++;
828     XtSetArg(args[j], XtNwidth,  30);  j++;
829 //    XtSetArg(args[j], XtNheight, 20);  j++;
830     engRepeat = XtCreateManagedWidget("Repeats", asciiTextWidgetClass, form, args, j);
831     XtAddEventHandler(engRepeat, ButtonPressMask, False, SetFocus, (XtPointer) popup);
832
833     j= 0;
834     XtSetArg(args[j], XtNlabel, _("-fold repeat is draw")); j++;
835     XtSetArg(args[j], XtNjustify, XtJustifyLeft); j++;
836     XtSetArg(args[j], XtNborderWidth, 0); j++;
837     XtSetArg(args[j], XtNfromVert, engRule); j++;
838     XtSetArg(args[j], XtNfromHoriz, engRepeat); j++;
839     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
840     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
841     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
842     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
843 //    XtSetArg(args[j], XtNwidth,  130);  j++;
844 //    XtSetArg(args[j], XtNheight, 20);  j++;
845     tcMess2 = XtCreateManagedWidget("MPStext", labelWidgetClass, form, args, j);
846
847     j=0;
848     XtSetArg(args[j], XtNfromVert, engRepeat);  j++;
849     XtSetArg(args[j], XtNfromHoriz, tcMess2);  j++;
850     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
851     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
852     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
853     XtSetArg(args[j], XtNright, XtChainRight);  j++;
854     b_ok= XtCreateManagedWidget(_("OK"), commandWidgetClass, form, args, j);   
855     XtAddCallback(b_ok, XtNcallback, EngineCallback, (XtPointer) 0);
856
857     j=0;
858     XtSetArg(args[j], XtNfromVert, engRepeat);  j++;
859     XtSetArg(args[j], XtNfromHoriz, b_ok);  j++;
860     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
861     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
862     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
863     XtSetArg(args[j], XtNright, XtChainRight);  j++;
864     b_cancel= XtCreateManagedWidget(_("cancel"), commandWidgetClass,
865                                    form, args, j);   
866     XtAddCallback(b_cancel, XtNcallback, EnginePopDown, (XtPointer) 0);
867
868     XtRealizeWidget(popup);
869     CatchDeleteWindow(popup, "EnginePopDown");
870     
871     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
872                   &x, &y, &win_x, &win_y, &mask);
873     
874     XtSetArg(args[0], XtNx, x - 10);
875     XtSetArg(args[1], XtNy, y - 30);
876     XtSetValues(popup, args, 2);
877     
878     XtPopup(popup, XtGrabExclusive);
879     EngineUp = True;
880     
881     previous = NULL;
882     SetFocus(engThreshold, popup, (XEvent*) NULL, False);
883 }
884
885 void EngineMenuProc(w, event, prms, nprms)
886      Widget w;
887      XEvent *event;
888      String *prms;
889      Cardinal *nprms;
890 {
891    EnginePopUp();
892 }
893
894 //--------------------------- New-Variant Menu PopUp -----------------------------------
895 struct NewVarButton {
896   char   *name;
897   char *color;
898   Widget handle;
899   VariantClass variant;
900 };
901
902 struct NewVarButton buttonDesc[] = {
903     {N_("normal"),            "#FFFFFF", 0, VariantNormal},
904     {N_("FRC"),               "#FFFFFF", 0, VariantFischeRandom},
905     {N_("wild castle"),       "#FFFFFF", 0, VariantWildCastle},
906     {N_("no castle"),         "#FFFFFF", 0, VariantNoCastle},
907     {N_("knightmate"),        "#FFFFFF", 0, VariantKnightmate},
908     {N_("berolina"),          "#FFFFFF", 0, VariantBerolina},
909     {N_("cylinder"),          "#FFFFFF", 0, VariantCylinder},
910     {N_("shatranj"),          "#FFFFFF", 0, VariantShatranj},
911     {N_("makruk"),            "#FFFFFF", 0, VariantMakruk},
912     {N_("atomic"),            "#FFFFFF", 0, VariantAtomic},
913     {N_("two kings"),         "#FFFFFF", 0, VariantTwoKings},
914     {N_("3-checks"),          "#FFFFFF", 0, Variant3Check},
915     {N_("suicide"),           "#FFFFBF", 0, VariantSuicide},
916     {N_("give-away"),         "#FFFFBF", 0, VariantGiveaway},
917     {N_("losers"),            "#FFFFBF", 0, VariantLosers},
918     {N_("fairy"),             "#BFBFBF", 0, VariantFairy},
919     {N_("Superchess"),        "#FFBFBF", 0, VariantSuper},
920     {N_("crazyhouse"),        "#FFBFBF", 0, VariantCrazyhouse},
921     {N_("bughouse"),          "#FFBFBF", 0, VariantBughouse},
922     {N_("shogi (9x9)"),       "#BFFFFF", 0, VariantShogi},
923     {N_("xiangqi (9x10)"),    "#BFFFFF", 0, VariantXiangqi},
924     {N_("courier (12x8)"),    "#BFFFBF", 0, VariantCourier},
925     {N_("janus (10x8)"),      "#BFBFFF", 0, VariantJanus},
926     {N_("Capablanca (10x8)"), "#BFBFFF", 0, VariantCapablanca},
927     {N_("CRC (10x8)"),        "#BFBFFF", 0, VariantCapaRandom},
928 #ifdef GOTHIC
929     {N_("Gothic (10x8)"),     "#BFBFFF", 0, VariantGothic},
930 #endif
931 #ifdef FALCON
932     {N_("Falcon (10x8)"),     "#BFBFFF", 0, VariantFalcon},
933 #endif
934     {NULL,                0, 0, (VariantClass) 0}
935 };
936
937 int NewVariantUp;
938 Widget NewVariantShell;
939
940 void NewVariantPopDown()
941 {
942     if (!NewVariantUp) return;
943     XtPopdown(NewVariantShell);
944     XtDestroyWidget(NewVariantShell);
945     NewVariantUp = False;
946     ModeHighlight();
947 }
948
949 void NewVariantCallback(w, client_data, call_data)
950      Widget w;
951      XtPointer client_data, call_data;
952 {
953     String name;
954     Widget w2;
955     Arg args[16];
956     char buf[80];
957     VariantClass v;
958     
959     XtSetArg(args[0], XtNlabel, &name);
960     XtGetValues(w, args, 1);
961     
962     if (strcmp(name, _("  OK  ")) == 0) {
963         int nr = (intptr_t) XawToggleGetCurrent(buttonDesc[0].handle) - 1;
964         if(nr < 0) return;
965         v = buttonDesc[nr].variant;
966         if(!appData.noChessProgram) { 
967             char *name = VariantName(v), buf[MSG_SIZ];
968             if (first.protocolVersion > 1 && StrStr(first.variants, name) == NULL) {
969                 /* [HGM] in protocol 2 we check if variant is suported by engine */
970                 sprintf(buf, _("Variant %s not supported by %s"), name, first.tidy);
971                 DisplayError(buf, 0);
972 //              NewVariantPopDown();
973                 return; /* ignore OK if first engine does not support it */
974             } else
975             if (second.initDone && second.protocolVersion > 1 && StrStr(second.variants, name) == NULL) {
976                 sprintf(buf, _("Warning: second engine (%s) does not support this!"), second.tidy);
977                 DisplayError(buf, 0);   /* use of second engine is optional; only warn user */
978             }
979         }
980
981         gameInfo.variant = v;
982         appData.variant = VariantName(v);
983
984         shuffleOpenings = FALSE; /* [HGM] shuffle: possible shuffle reset when we switch */
985         startedFromPositionFile = FALSE; /* [HGM] loadPos: no longer valid in new variant */
986         appData.pieceToCharTable = NULL;
987         Reset(True, True);
988         NewVariantPopDown();
989         return;
990     }
991 }
992
993 void NewVariantPopUp()
994 {
995     Arg args[16];
996     Widget popup, layout, dialog, edit, form, last = NULL, b_ok, b_cancel;
997     Window root, child;
998     int x, y, i, j;
999     int win_x, win_y;
1000     unsigned int mask;
1001     char def[80];
1002     XrmValue vFrom, vTo;
1003
1004     i = 0;
1005     XtSetArg(args[i], XtNresizable, True); i++;
1006 //    XtSetArg(args[i], XtNwidth, 250); i++;
1007 //    XtSetArg(args[i], XtNheight, 300); i++;
1008     NewVariantShell = popup =
1009       XtCreatePopupShell(_("NewVariant Menu"), transientShellWidgetClass,
1010                          shellWidget, args, i);
1011     
1012     layout =
1013       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
1014                             layoutArgs, XtNumber(layoutArgs));
1015   
1016     form =
1017       XtCreateManagedWidget("form", formWidgetClass, layout,
1018                             formArgs, XtNumber(formArgs));
1019   
1020     for(i = 0; buttonDesc[i].name != NULL; i++) {
1021         Pixel buttonColor;
1022         if (!appData.monoMode) {
1023             vFrom.addr = (caddr_t) buttonDesc[i].color;
1024             vFrom.size = strlen(buttonDesc[i].color);
1025             XtConvert(shellWidget, XtRString, &vFrom, XtRPixel, &vTo);
1026             if (vTo.addr == NULL) {
1027                 buttonColor = (Pixel) -1;
1028             } else {
1029                 buttonColor = *(Pixel *) vTo.addr;
1030             }
1031         }
1032     
1033         j = 0;
1034         XtSetArg(args[j], XtNradioGroup, last); j++;
1035         XtSetArg(args[j], XtNwidth, 125); j++;
1036 //      XtSetArg(args[j], XtNheight, 16); j++;
1037         XtSetArg(args[j], XtNfromVert, i == 15 ? NULL : last); j++;
1038         XtSetArg(args[j], XtNfromHoriz, i < 15 ? NULL : buttonDesc[i-15].handle); j++;
1039         XtSetArg(args[j], XtNradioData, i+1); j++;
1040         XtSetArg(args[j], XtNbackground, buttonColor); j++;
1041         XtSetArg(args[j], XtNstate, gameInfo.variant == buttonDesc[i].variant); j++;
1042         buttonDesc[i].handle = last =
1043             XtCreateManagedWidget(buttonDesc[i].name, toggleWidgetClass, form, args, j);
1044     }
1045
1046     j=0;
1047     XtSetArg(args[j], XtNfromVert, buttonDesc[12].handle);  j++;
1048     XtSetArg(args[j], XtNfromHoriz, buttonDesc[12].handle);  j++;
1049     XtSetArg(args[j], XtNheight, 35); j++;
1050 //    XtSetArg(args[j], XtNwidth, 60); j++;
1051     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
1052     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
1053     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
1054     XtSetArg(args[j], XtNright, XtChainRight);  j++;
1055     b_cancel= XtCreateManagedWidget(_("CANCEL"), commandWidgetClass, form, args, j);   
1056     XtAddCallback(b_cancel, XtNcallback, NewVariantPopDown, (XtPointer) 0);
1057
1058     j=0;
1059     XtSetArg(args[j], XtNfromHoriz, b_cancel);  j++;
1060     XtSetArg(args[j], XtNfromVert, buttonDesc[12].handle);  j++;
1061     XtSetArg(args[j], XtNheight, 35); j++;
1062 //    XtSetArg(args[j], XtNwidth, 60); j++;
1063     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
1064     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
1065     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
1066     XtSetArg(args[j], XtNright, XtChainRight);  j++;
1067     b_ok= XtCreateManagedWidget(_("  OK  "), commandWidgetClass, form, args, j);   
1068     XtAddCallback(b_ok, XtNcallback, NewVariantCallback, (XtPointer) 0);
1069
1070     j=0;
1071     XtSetArg(args[j], XtNfromVert, buttonDesc[14].handle);  j++;
1072 //    XtSetArg(args[j], XtNheight, 70); j++;
1073     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
1074     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
1075     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
1076     XtSetArg(args[j], XtNright, XtChainRight);  j++;
1077     XtSetArg(args[j], XtNlabel, _("WARNING: variants with un-orthodox\n"
1078                                   "pieces only have built-in bitmaps\n"
1079                                   "for -boardSize middling, bulky and\n"
1080                                   "petite, and substitute king or amazon\n"
1081                                   "for missing bitmaps. (See manual.)")); j++;
1082     XtCreateManagedWidget("warning", labelWidgetClass, form, args, j);
1083
1084             XtRealizeWidget(popup);
1085     CatchDeleteWindow(popup, "NewVariantPopDown");
1086     
1087     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
1088                   &x, &y, &win_x, &win_y, &mask);
1089     
1090     XtSetArg(args[0], XtNx, x - 10);
1091     XtSetArg(args[1], XtNy, y - 30);
1092     XtSetValues(popup, args, 2);
1093     
1094     XtPopup(popup, XtGrabExclusive);
1095     NewVariantUp = True;
1096 }
1097
1098 void NewVariantProc(w, event, prms, nprms)
1099      Widget w;
1100      XEvent *event;
1101      String *prms;
1102      Cardinal *nprms;
1103 {
1104    NewVariantPopUp();
1105 }
1106
1107 //--------------------------- UCI Menu Popup ------------------------------------------
1108 int UciUp;
1109 Widget UciShell;
1110
1111 struct UciControl {
1112   char *name;
1113   Widget handle;
1114   void *ptr;
1115 };
1116
1117 struct UciControl controlDesc[] = {
1118   {N_("maximum nr of CPUs:"), 0, &appData.smpCores},
1119   {N_("Polyglot Directory:"), 0, &appData.polyglotDir},
1120   {N_("Hash Size (MB):"),     0, &appData.defaultHashSize},
1121   {N_("EGTB Path:"),          0, &appData.defaultPathEGTB},
1122   {N_("EGTB Cache (MB):"),    0, &appData.defaultCacheSizeEGTB},
1123   {N_("Polyglot Book:"),      0, &appData.polyglotBook},
1124   {NULL, 0, NULL},
1125 };
1126
1127 void UciPopDown()
1128 {
1129     if (!UciUp) return;
1130     XtPopdown(UciShell);
1131     XtDestroyWidget(UciShell);
1132     UciUp = False;
1133     ModeHighlight();
1134 }
1135
1136 void UciCallback(w, client_data, call_data)
1137      Widget w;
1138      XtPointer client_data, call_data;
1139 {
1140     String name;
1141     Arg args[16];
1142     char buf[80];
1143     int oldCores = appData.smpCores, ponder = 0;
1144     
1145     XtSetArg(args[0], XtNlabel, &name);
1146     XtGetValues(w, args, 1);
1147     
1148     if (strcmp(name, _("OK")) == 0) {
1149         int nr, i, j; String name;
1150         for(i=0; i<6; i++) {
1151             XtSetArg(args[0], XtNstring, &name);
1152             XtGetValues(controlDesc[i].handle, args, 1);
1153             if(i&1) {
1154                 if(name)
1155                     *(char**) controlDesc[i].ptr = strdup(name);
1156             } else {
1157                 if(sscanf(name, "%d", &j) == 1) 
1158                     *(int*) controlDesc[i].ptr = j;
1159             }
1160         }
1161         XtSetArg(args[0], XtNstate, &appData.usePolyglotBook);
1162         XtGetValues(w1, args, 1);
1163         XtSetArg(args[0], XtNstate, &appData.firstHasOwnBookUCI);
1164         XtGetValues(w2, args, 1);
1165         XtSetArg(args[0], XtNstate, &appData.secondHasOwnBookUCI);
1166         XtGetValues(w3, args, 1);
1167         XtSetArg(args[0], XtNstate, &ponder);
1168         XtGetValues(w4, args, 1);
1169
1170         // adjust setting in other menu for duplicates 
1171         // (perhaps duplicates should be removed from general Option Menu?)
1172         XtSetArg(args[0], XtNleftBitmap, ponder ? xMarkPixmap : None);
1173         XtSetValues(XtNameToWidget(menuBarWidget,
1174                                    "menuOptions.Ponder Next Move"), args, 1);
1175
1176         // make sure changes are sent to first engine by re-initializing it
1177         // if it was already started pre-emptively at end of previous game
1178         if(gameMode == BeginningOfGame) Reset(True, True); else {
1179             // Some changed setting need immediate sending always.
1180             PonderNextMoveEvent(ponder);
1181             if(oldCores != appData.smpCores)
1182                 NewSettingEvent(False, "cores", appData.smpCores);
1183       }
1184       UciPopDown();
1185       return;
1186     }
1187 }
1188
1189 void UciPopUp()
1190 {
1191     Arg args[16];
1192     Widget popup, layout, dialog, edit, form, b_ok, b_cancel, last = NULL, new, upperLeft;
1193     Window root, child;
1194     int x, y, i, j;
1195     int win_x, win_y;
1196     unsigned int mask;
1197     char def[80];
1198     
1199     i = 0;
1200     XtSetArg(args[i], XtNresizable, True); i++;
1201 //    XtSetArg(args[i], XtNwidth, 300); i++;
1202     UciShell = popup =
1203       XtCreatePopupShell(_("Engine Settings"), transientShellWidgetClass,
1204                          shellWidget, args, i);
1205     
1206     layout =
1207       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
1208                             layoutArgs, XtNumber(layoutArgs));
1209   
1210     
1211     form =
1212       XtCreateManagedWidget("form", formWidgetClass, layout,
1213                             formArgs, XtNumber(formArgs));
1214   
1215     j = 0;
1216     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
1217     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
1218     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
1219 //    XtSetArg(args[j], XtNheight, 20); j++;
1220     for(i = 0; controlDesc[i].name != NULL; i++) {
1221         j = 3;
1222         XtSetArg(args[j], XtNfromVert, last); j++;
1223 //      XtSetArg(args[j], XtNwidth, 130); j++;
1224         XtSetArg(args[j], XtNjustify, XtJustifyLeft); j++;
1225         XtSetArg(args[j], XtNright, XtChainLeft);  j++;
1226         XtSetArg(args[j], XtNborderWidth, 0); j++;
1227         new = XtCreateManagedWidget(controlDesc[i].name, labelWidgetClass, form, args, j);
1228         if(i==0) upperLeft = new;
1229
1230         j = 4;
1231         XtSetArg(args[j], XtNborderWidth, 1); j++;
1232         XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
1233         XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
1234         XtSetArg(args[j], XtNdisplayCaret, False);  j++;
1235         XtSetArg(args[j], XtNright, XtChainRight);  j++;
1236         XtSetArg(args[j], XtNresizable, True);  j++;
1237         XtSetArg(args[j], XtNwidth, i&1 ? 245 : 50); j++;
1238         if(i&1) {
1239             XtSetArg(args[j], XtNstring, * (char**) controlDesc[i].ptr ? 
1240                                          * (char**) controlDesc[i].ptr : ""); j++;
1241         } else {
1242             sprintf(def, "%d", * (int*) controlDesc[i].ptr);
1243             XtSetArg(args[j], XtNstring, def); j++;
1244         }
1245         XtSetArg(args[j], XtNfromHoriz, upperLeft); j++;
1246         controlDesc[i].handle = last =
1247             XtCreateManagedWidget("text", asciiTextWidgetClass, form, args, j);
1248         XtAddEventHandler(last, ButtonPressMask, False, SetFocus, (XtPointer) popup);
1249     }
1250
1251     j=0;
1252     XtSetArg(args[j], XtNfromHoriz, controlDesc[0].handle);  j++;
1253     XtSetArg(args[j], XtNbottom, XtChainTop);  j++;
1254     XtSetArg(args[j], XtNtop, XtChainTop);  j++;
1255     XtSetArg(args[j], XtNleft, XtChainRight);  j++;
1256     XtSetArg(args[j], XtNright, XtChainRight);  j++;
1257     XtSetArg(args[j], XtNstate, appData.ponderNextMove);  j++;
1258     w4 = XtCreateManagedWidget(_("Ponder"), toggleWidgetClass, form, args, j);   
1259
1260     j=0;
1261     XtSetArg(args[j], XtNfromVert, last);  j++;
1262     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
1263     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
1264     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
1265     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
1266     b_ok = XtCreateManagedWidget(_("OK"), commandWidgetClass, form, args, j);   
1267     XtAddCallback(b_ok, XtNcallback, UciCallback, (XtPointer) 0);
1268
1269     XtSetArg(args[j], XtNfromHoriz, b_ok);  j++;
1270     b_cancel = XtCreateManagedWidget(_("cancel"), commandWidgetClass, form, args, j);   
1271     XtAddCallback(b_cancel, XtNcallback, UciPopDown, (XtPointer) 0);
1272
1273     j = 5;
1274     XtSetArg(args[j], XtNfromHoriz, upperLeft);  j++;
1275     XtSetArg(args[j], XtNstate, appData.usePolyglotBook);  j++;
1276     w1 = XtCreateManagedWidget(_(" use book "), toggleWidgetClass, form, args, j);   
1277 //    XtAddCallback(w1, XtNcallback, UciCallback, (XtPointer) 0);
1278
1279     j = 5;
1280     XtSetArg(args[j], XtNfromHoriz, w1);  j++;
1281     XtSetArg(args[j], XtNstate, appData.firstHasOwnBookUCI);  j++;
1282     w2 = XtCreateManagedWidget(_("own book 1"), toggleWidgetClass, form, args, j);   
1283 //    XtAddCallback(w2, XtNcallback, UciCallback, (XtPointer) 0);
1284
1285     j = 5;
1286     XtSetArg(args[j], XtNfromHoriz, w2);  j++;
1287     XtSetArg(args[j], XtNstate, appData.secondHasOwnBookUCI);  j++;
1288     w3 = XtCreateManagedWidget(_("own book 2"), toggleWidgetClass, form, args, j);   
1289 //    XtAddCallback(w3, XtNcallback, UciCallback, (XtPointer) 0);
1290
1291     XtRealizeWidget(popup);
1292     CatchDeleteWindow(popup, "UciPopDown");
1293     
1294     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
1295                   &x, &y, &win_x, &win_y, &mask);
1296     
1297     XtSetArg(args[0], XtNx, x - 10);
1298     XtSetArg(args[1], XtNy, y - 30);
1299     XtSetValues(popup, args, 2);
1300     
1301     XtPopup(popup, XtGrabExclusive);
1302     UciUp = True;
1303
1304     previous = NULL;
1305     SetFocus(controlDesc[2].handle, popup, (XEvent*) NULL, False);
1306 //    XtSetKeyboardFocus(popup, controlDesc[1].handle);
1307 }
1308
1309 void UciMenuProc(w, event, prms, nprms)
1310      Widget w;
1311      XEvent *event;
1312      String *prms;
1313      Cardinal *nprms;
1314 {
1315    UciPopUp();
1316 }
1317
1318 //--------------------------- Engine-specific options menu ----------------------------------
1319
1320 int SettingsUp;
1321 Widget SettingsShell;
1322 int values[MAX_OPTIONS];
1323 ChessProgramState *currentCps;
1324
1325 void SettingsPopDown()
1326 {
1327     if (!SettingsUp) return;
1328     XtPopdown(SettingsShell);
1329     XtDestroyWidget(SettingsShell);
1330     SettingsUp = False;
1331     ModeHighlight();
1332 }
1333
1334 void SpinCallback(w, client_data, call_data)
1335      Widget w;
1336      XtPointer client_data, call_data;
1337 {
1338     String name, val;
1339     Widget w2;
1340     Arg args[16];
1341     char buf[MSG_SIZ];
1342     int i, j;
1343     int data = (intptr_t) client_data;
1344     
1345     XtSetArg(args[0], XtNlabel, &name);
1346     XtGetValues(w, args, 1);
1347     
1348     j = 0;
1349     XtSetArg(args[0], XtNstring, &val);
1350     XtGetValues(currentCps->option[data].handle, args, 1);
1351     sscanf(val, "%d", &j);
1352     if (strcmp(name, "+") == 0) {
1353         if(++j > currentCps->option[data].max) return;
1354     } else
1355     if (strcmp(name, "-") == 0) {
1356         if(--j < currentCps->option[data].min) return;
1357     } else return;
1358     sprintf(buf, "%d", j);
1359     XtSetArg(args[0], XtNstring, buf);
1360     XtSetValues(currentCps->option[data].handle, args, 1);
1361 }
1362
1363 void SettingsCallback(w, client_data, call_data)
1364      Widget w;
1365      XtPointer client_data, call_data;
1366 {
1367     String name, val;
1368     Widget w2;
1369     Arg args[16];
1370     char buf[MSG_SIZ];
1371     int i, j;
1372     int data = (intptr_t) client_data;
1373     
1374     XtSetArg(args[0], XtNlabel, &name);
1375     XtGetValues(w, args, 1);
1376     
1377     if (strcmp(name, _("cancel")) == 0) {
1378         SettingsPopDown();
1379         return;
1380     }
1381     if (strcmp(name, _("OK")) == 0 || data) { // save buttons imply OK
1382         int nr;
1383
1384         for(i=0; i<currentCps->nrOptions; i++) { // send all options that had to be OK-ed to engine
1385             switch(currentCps->option[i].type) {
1386                 case TextBox:
1387                     XtSetArg(args[0], XtNstring, &val);
1388                     XtGetValues(currentCps->option[i].handle, args, 1);
1389                     if(strcmp(currentCps->option[i].textValue, val)) {
1390                         strcpy(currentCps->option[i].textValue, val);
1391                         sprintf(buf, "option %s=%s\n", currentCps->option[i].name, val);
1392                         SendToProgram(buf, currentCps);
1393                     }
1394                     break;
1395                 case Spin:
1396                     XtSetArg(args[0], XtNstring, &val);
1397                     XtGetValues(currentCps->option[i].handle, args, 1);
1398                     sscanf(val, "%d", &j);
1399                     if(j > currentCps->option[i].max) j = currentCps->option[i].max;
1400                     if(j < currentCps->option[i].min) j = currentCps->option[i].min;
1401                     if(currentCps->option[i].value != j) {
1402                         currentCps->option[i].value = j;
1403                         sprintf(buf, "option %s=%d\n", currentCps->option[i].name, j);
1404                         SendToProgram(buf, currentCps);
1405                     }
1406                     break;
1407                 case CheckBox:
1408                     j = 0;
1409                     XtSetArg(args[0], XtNstate, &j);
1410                     XtGetValues(currentCps->option[i].handle, args, 1);
1411                     if(currentCps->option[i].value != j) {
1412                         currentCps->option[i].value = j;
1413                         sprintf(buf, "option %s=%d\n", currentCps->option[i].name, j);
1414                         SendToProgram(buf, currentCps);
1415                     }
1416                     break;
1417                 case ComboBox:
1418                     if(currentCps->option[i].value != values[i]) {
1419                         currentCps->option[i].value = values[i];
1420                         sprintf(buf, "option %s=%s\n", currentCps->option[i].name, 
1421                                 ((char**)currentCps->option[i].textValue)[values[i]]);
1422                         SendToProgram(buf, currentCps);
1423                     }
1424                     break;
1425             }
1426         }
1427         if(data) { // send save-button command to engine
1428             sprintf(buf, "option %s\n", name);
1429             SendToProgram(buf, currentCps);
1430         }
1431         SettingsPopDown();
1432         return;
1433     }
1434     sprintf(buf, "option %s\n", name);
1435     SendToProgram(buf, currentCps);
1436 }
1437
1438 void ComboSelect(w, addr, index) // callback for all combo items
1439      Widget w;
1440      caddr_t addr;
1441      caddr_t index;
1442 {
1443     Arg args[16];
1444     int i = ((intptr_t)addr)>>8;
1445     int j = 255 & (intptr_t) addr;
1446
1447     values[i] = j; // store in temporary, for transfer at OK
1448     XtSetArg(args[0], XtNlabel, ((char**)currentCps->option[i].textValue)[j]);
1449     XtSetValues(currentCps->option[i].handle, args, 1);
1450 }
1451
1452 void CreateComboPopup(parent, name, n, mb)
1453      Widget parent;
1454      String name;
1455      int n;
1456      char *mb[];
1457 {
1458     int i=0, j;
1459     Widget menu, entry;
1460     Arg args[16];
1461
1462     menu = XtCreatePopupShell(name, simpleMenuWidgetClass,
1463                               parent, NULL, 0);
1464     j = 0;
1465     XtSetArg(args[j], XtNwidth, 100);  j++;
1466 //    XtSetArg(args[j], XtNright, XtChainRight);  j++;
1467     while (mb[i] != NULL) {
1468             entry = XtCreateManagedWidget(mb[i], smeBSBObjectClass,
1469                                           menu, args, j);
1470             XtAddCallback(entry, XtNcallback,
1471                           (XtCallbackProc) ComboSelect,
1472                           (caddr_t)(intptr_t) (256*n+i));
1473         i++;
1474     }
1475 }       
1476
1477 void SettingsPopUp(ChessProgramState *cps)
1478 {
1479     Arg args[16];
1480     Widget popup, layout, dialog, edit=NULL, form, oldform, last, b_ok, b_cancel, leftMargin = NULL;
1481     Window root, child;
1482     int x, y, i, j, height, width, h, c;
1483     int win_x, win_y;
1484     unsigned int mask;
1485     char def[80], *p, *q;
1486     static char pane[6] = "paneX";
1487
1488     // to do: start up second engine if needed
1489     if(!cps->initDone || !cps->nrOptions) return; // nothing to be done
1490     currentCps = cps;
1491
1492     if(cps->nrOptions > 50) width = 4; else if(cps->nrOptions>24) width = 2; else width = 1;
1493     height = cps->nrOptions / width + 1;
1494      i = 0;
1495     XtSetArg(args[i], XtNresizable, True); i++;
1496     SettingsShell = popup =
1497       XtCreatePopupShell(_("Settings Menu"), transientShellWidgetClass,
1498                          shellWidget, args, i);
1499     
1500     layout =
1501       XtCreateManagedWidget(layoutName, formWidgetClass, popup,
1502                             layoutArgs, XtNumber(layoutArgs));
1503   for(c=0; c<width; c++) {
1504     pane[4] = 'A'+c;
1505     form =
1506       XtCreateManagedWidget(pane, formWidgetClass, layout,
1507                             formArgs, XtNumber(formArgs));
1508     j=0;
1509     XtSetArg(args[j], XtNfromHoriz, leftMargin);  j++;
1510     XtSetValues(form, args, j);
1511     leftMargin = form;
1512  
1513     last = NULL;
1514     for(h=0; h<height; h++) {
1515         i = h + c*height;
1516         if(i >= cps->nrOptions) break;
1517         switch(cps->option[i].type) {
1518           case Spin:
1519             sprintf(def, "%d", cps->option[i].value);
1520           case TextBox:
1521             j=0;
1522             XtSetArg(args[j], XtNfromVert, last);  j++;
1523             XtSetArg(args[j], XtNborderWidth, 0);  j++;
1524             XtSetArg(args[j], XtNjustify, XtJustifyLeft);  j++;
1525             dialog = XtCreateManagedWidget(cps->option[i].name, labelWidgetClass, form, args, j);   
1526             j=0;
1527             XtSetArg(args[j], XtNfromVert, last);  j++;
1528             XtSetArg(args[j], XtNfromHoriz, dialog);  j++;
1529             XtSetArg(args[j], XtNborderWidth, 1); j++;
1530             XtSetArg(args[j], XtNwidth, cps->option[i].type == Spin ? 40 : 100); j++;
1531             XtSetArg(args[j], XtNeditType, XawtextEdit);  j++;
1532             XtSetArg(args[j], XtNuseStringInPlace, False);  j++;
1533             XtSetArg(args[j], XtNdisplayCaret, False);  j++;
1534             XtSetArg(args[j], XtNright, XtChainRight);  j++;
1535             XtSetArg(args[j], XtNresizable, True);  j++;
1536             XtSetArg(args[j], XtNstring, cps->option[i].type==Spin ? def : cps->option[i].textValue);  j++;
1537             edit = last;
1538             cps->option[i].handle = (void*)
1539                 (last = XtCreateManagedWidget("text", asciiTextWidgetClass, form, args, j));   
1540             XtAddEventHandler(last, ButtonPressMask, False, SetFocus, (XtPointer) popup);
1541             if(cps->option[i].type == TextBox) break;
1542
1543             // add increment and decrement controls for spin
1544             j=0;
1545             XtSetArg(args[j], XtNfromVert, edit);  j++;
1546             XtSetArg(args[j], XtNfromHoriz, last);  j++;
1547             XtSetArg(args[j], XtNheight, 10);  j++;
1548             XtSetArg(args[j], XtNwidth, 20);  j++;
1549             edit = XtCreateManagedWidget("+", commandWidgetClass, form, args, j);
1550             XtAddCallback(edit, XtNcallback, SpinCallback,
1551                           (XtPointer)(intptr_t) i);
1552
1553             j=0;
1554             XtSetArg(args[j], XtNfromVert, edit);  j++;
1555             XtSetArg(args[j], XtNfromHoriz, last);  j++;
1556             XtSetArg(args[j], XtNheight, 10);  j++;
1557             XtSetArg(args[j], XtNwidth, 20);  j++;
1558             last = XtCreateManagedWidget("-", commandWidgetClass, form, args, j);
1559             XtAddCallback(last, XtNcallback, SpinCallback,
1560                           (XtPointer)(intptr_t) i);
1561             break;
1562           case CheckBox:
1563             j=0;
1564             XtSetArg(args[j], XtNfromVert, last);  j++;
1565             XtSetArg(args[j], XtNwidth, 10);  j++;
1566             XtSetArg(args[j], XtNheight, 10);  j++;
1567             XtSetArg(args[j], XtNstate, cps->option[i].value);  j++;
1568             cps->option[i].handle = (void*) 
1569                 (dialog = XtCreateManagedWidget(" ", toggleWidgetClass, form, args, j));   
1570             j=0;
1571             XtSetArg(args[j], XtNfromVert, last);  j++;
1572             XtSetArg(args[j], XtNfromHoriz, dialog);  j++;
1573             XtSetArg(args[j], XtNborderWidth, 0);  j++;
1574             XtSetArg(args[j], XtNjustify, XtJustifyLeft);  j++;
1575             last = XtCreateManagedWidget(cps->option[i].name, labelWidgetClass, form, args, j);
1576             break;
1577           case SaveButton:
1578           case Button:
1579             j=0;
1580             XtSetArg(args[j], XtNfromVert, last);  j++;
1581             XtSetArg(args[j], XtNstate, cps->option[i].value);  j++;
1582             cps->option[i].handle = (void*) 
1583                 (last = XtCreateManagedWidget(cps->option[i].name, commandWidgetClass, form, args, j));   
1584             XtAddCallback(last, XtNcallback, SettingsCallback,
1585                           (XtPointer)(intptr_t) (cps->option[i].type == SaveButton));
1586             break;
1587           case ComboBox:
1588             j=0;
1589             XtSetArg(args[j], XtNfromVert, last);  j++;
1590             XtSetArg(args[j], XtNborderWidth, 0);  j++;
1591             XtSetArg(args[j], XtNjustify, XtJustifyLeft);  j++;
1592             dialog = XtCreateManagedWidget(cps->option[i].name, labelWidgetClass, form, args, j);
1593
1594             j=0;
1595             XtSetArg(args[j], XtNfromVert, last);  j++;
1596             XtSetArg(args[j], XtNfromHoriz, dialog);  j++;
1597             XtSetArg(args[j], XtNwidth, 100);  j++;
1598             XtSetArg(args[j], XtNmenuName, XtNewString(cps->option[i].name));  j++;
1599             XtSetArg(args[j], XtNlabel, ((char**)cps->option[i].textValue)[cps->option[i].value]);  j++;
1600             cps->option[i].handle = (void*) 
1601                 (last = XtCreateManagedWidget(" ", menuButtonWidgetClass, form, args, j));   
1602             CreateComboPopup(last, cps->option[i].name, i, (char **) cps->option[i].textValue);
1603             values[i] = cps->option[i].value;
1604             break;
1605         }
1606     }
1607   }
1608     j=0;
1609     XtSetArg(args[j], XtNfromVert, last);  j++;
1610     XtSetArg(args[j], XtNbottom, XtChainBottom);  j++;
1611     XtSetArg(args[j], XtNtop, XtChainBottom);  j++;
1612     XtSetArg(args[j], XtNleft, XtChainLeft);  j++;
1613     XtSetArg(args[j], XtNright, XtChainLeft);  j++;
1614     b_ok = XtCreateManagedWidget(_("OK"), commandWidgetClass, form, args, j);   
1615     XtAddCallback(b_ok, XtNcallback, SettingsCallback, (XtPointer) 0);
1616
1617     XtSetArg(args[j], XtNfromHoriz, b_ok);  j++;
1618     b_cancel = XtCreateManagedWidget(_("cancel"), commandWidgetClass, form, args, j);   
1619     XtAddCallback(b_cancel, XtNcallback, SettingsPopDown, (XtPointer) 0);
1620
1621     XtRealizeWidget(popup);
1622     CatchDeleteWindow(popup, "SettingsPopDown");
1623     
1624     XQueryPointer(xDisplay, xBoardWindow, &root, &child,
1625                   &x, &y, &win_x, &win_y, &mask);
1626     
1627     XtSetArg(args[0], XtNx, x - 10);
1628     XtSetArg(args[1], XtNy, y - 30);
1629     XtSetValues(popup, args, 2);
1630     
1631     XtPopup(popup, XtGrabExclusive);
1632     SettingsUp = True;
1633
1634     previous = NULL;
1635     if(edit)SetFocus(edit, popup, (XEvent*) NULL, False);
1636 }
1637
1638 void FirstSettingsProc(w, event, prms, nprms)
1639      Widget w;
1640      XEvent *event;
1641      String *prms;
1642      Cardinal *nprms;
1643 {
1644    SettingsPopUp(&first);
1645 }
1646
1647 void SecondSettingsProc(w, event, prms, nprms)
1648      Widget w;
1649      XEvent *event;
1650      String *prms;
1651      Cardinal *nprms;
1652 {
1653    SettingsPopUp(&second);
1654 }
1655
1656 //---------------------------- Chat Windows ----------------------------------------------
1657
1658 void OutputChatMessage(int partner, char *mess)
1659 {
1660     return; // dummy
1661 }
1662