X-Git-Url: http://winboard.nl/cgi-bin?a=blobdiff_plain;f=xoptions.c;h=bfc487c585b686def414ac3fc6c3572682a5d35e;hb=8bd006fe9af8f3d1469357d5efa7ff46a78773e0;hp=b30719d29c65f21fe3e60e5b1bd037e28946ff3f;hpb=ca3a88fcfe0d10b8d841251166afbf5cc1a02886;p=xboard.git diff --git a/xoptions.c b/xoptions.c index b30719d..bfc487c 100644 --- a/xoptions.c +++ b/xoptions.c @@ -1,7 +1,7 @@ /* * xoptions.c -- Move list window, part of X front end for XBoard * - * Copyright 2000, 2009, 2010 Free Software Foundation, Inc. + * Copyright 2000, 2009, 2010, 2011 Free Software Foundation, Inc. * ------------------------------------------------------------------------ * * GNU XBoard is free software: you can redistribute it and/or modify @@ -80,6 +80,8 @@ extern char *getenv(); #endif extern void SendToProgram P((char *message, ChessProgramState *cps)); +FILE * XsraSelFile P((Widget w, char *prompt, char *ok, char *cancel, char *failed, + char *init_path, char *mode, int (*show_entry)(), char **name_return)); extern Widget formWidget, shellWidget, boardWidget, menuBarWidget; extern Display *xDisplay; @@ -90,6 +92,7 @@ extern Window xBoardWindow; extern Arg layoutArgs[2], formArgs[2]; Pixel timerForegroundPixel, timerBackgroundPixel; extern int searchTime; +extern int lineGap; // [HGM] the following code for makng menu popups was cloned from the FileNamePopUp routines @@ -97,14 +100,18 @@ static Widget previous = NULL; void SetFocus(Widget w, XtPointer data, XEvent *event, Boolean *b) { - Arg args; + Arg args[2]; + char *s; if(previous) { - XtSetArg(args, XtNdisplayCaret, False); - XtSetValues(previous, &args, 1); + XtSetArg(args[0], XtNdisplayCaret, False); + XtSetValues(previous, args, 1); } - XtSetArg(args, XtNdisplayCaret, True); - XtSetValues(w, &args, 1); + XtSetArg(args[0], XtNstring, &s); + XtGetValues(w, args, 1); + XtSetArg(args[0], XtNdisplayCaret, True); + XtSetArg(args[1], XtNinsertPosition, strlen(s)); + XtSetValues(w, args, 2); XtSetKeyboardFocus((Widget) data, w); previous = w; } @@ -966,21 +973,23 @@ struct NewVarButton buttonDesc[] = { {N_("give-away"), "#FFFFBF", 0, VariantGiveaway}, {N_("losers"), "#FFFFBF", 0, VariantLosers}, {N_("fairy"), "#BFBFBF", 0, VariantFairy}, + {N_("Seirawan"), "#FFBFBF", 0, VariantSChess}, {N_("Superchess"), "#FFBFBF", 0, VariantSuper}, {N_("crazyhouse"), "#FFBFBF", 0, VariantCrazyhouse}, {N_("bughouse"), "#FFBFBF", 0, VariantBughouse}, {N_("shogi (9x9)"), "#BFFFFF", 0, VariantShogi}, {N_("xiangqi (9x10)"), "#BFFFFF", 0, VariantXiangqi}, {N_("courier (12x8)"), "#BFFFBF", 0, VariantCourier}, - {N_("janus (10x8)"), "#BFBFFF", 0, VariantJanus}, {N_("Capablanca (10x8)"), "#BFBFFF", 0, VariantCapablanca}, - {N_("CRC (10x8)"), "#BFBFFF", 0, VariantCapaRandom}, #ifdef GOTHIC {N_("Gothic (10x8)"), "#BFBFFF", 0, VariantGothic}, #endif + {N_("janus (10x8)"), "#BFBFFF", 0, VariantJanus}, + {N_("CRC (10x8)"), "#BFBFFF", 0, VariantCapaRandom}, #ifdef FALCON {N_("Falcon (10x8)"), "#BFBFFF", 0, VariantFalcon}, #endif + {N_("Spartan"), "#FF0000", 0, VariantSpartan}, {NULL, 0, 0, (VariantClass) 0} }; @@ -1086,6 +1095,7 @@ void NewVariantPopUp() XtSetArg(args[j], XtNradioData, i+1); j++; XtSetArg(args[j], XtNbackground, buttonColor); j++; XtSetArg(args[j], XtNstate, gameInfo.variant == buttonDesc[i].variant); j++; + XtSetArg(args[j], XtNsensitive, appData.noChessProgram || strstr(first.variants, VariantName(buttonDesc[i].variant))); j++; buttonDesc[i].handle = last = XtCreateManagedWidget(buttonDesc[i].name, toggleWidgetClass, form, args, j); } @@ -1386,7 +1396,7 @@ void SpinCallback(w, client_data, call_data) { String name, val; Arg args[16]; - char buf[MSG_SIZ]; + char buf[MSG_SIZ], *p; int j; int data = (intptr_t) client_data; @@ -1397,6 +1407,17 @@ void SpinCallback(w, client_data, call_data) XtSetArg(args[0], XtNstring, &val); XtGetValues(currentCps->option[data].handle, args, 1); sscanf(val, "%d", &j); + if (strcmp(name, "browse") == 0) { + if(XsraSelFile(SettingsShell, currentCps->option[data].name, NULL, NULL, "", "", + currentCps->option[data].type == PathName ? "p" : "f", NULL, &p)) { + int len = strlen(p); + if(len && p[len-1] == '/') p[len-1] = NULLCHAR; + XtSetArg(args[0], XtNstring, p); + XtSetValues(currentCps->option[data].handle, args, 1); + } + SetFocus(currentCps->option[data].handle, SettingsShell, (XEvent*) NULL, False); + return; + } else if (strcmp(name, "+") == 0) { if(++j > currentCps->option[data].max) return; } else @@ -1749,6 +1770,571 @@ void SecondSettingsProc(w, event, prms, nprms) SettingsPopUp(&second); } +//----------------------------Generic dialog -------------------------------------------- + +// cloned from Engine Settings dialog + +typedef void ButtonCallback(int n); + +static Option *currentOption; +int MakeColors P((void)); +void CreateGCs P((int redo)); +void CreateXPMBoard P((char *s, int kind)); +void CreateXPMPieces P((void)); +void GenericReadout(); + +void IcsOptionsOK(int n) +{ + ParseIcsTextColors(); +} + +Option icsOptions[] = { +{ 0, 0, 0, NULL, (void*) &appData.autoKibitz, "", NULL, CheckBox, _("Auto-Kibitz") }, +{ 0, 0, 0, NULL, (void*) &appData.autoComment, "", NULL, CheckBox, _("Auto-Comment") }, +{ 0, 0, 0, NULL, (void*) &appData.autoObserve, "", NULL, CheckBox, _("Auto-Observe") }, +{ 0, 0, 0, NULL, (void*) &appData.autoRaiseBoard, "", NULL, CheckBox, _("Auto-Raise Board") }, +{ 0, 0, 0, NULL, (void*) &appData.bgObserve, "", NULL, CheckBox, _("Background Observe while Playing") }, +{ 0, 0, 0, NULL, (void*) &appData.dualBoard, "", NULL, CheckBox, _("Dual Board for Background-Observed Game") }, +{ 0, 0, 0, NULL, (void*) &appData.getMoveList, "", NULL, CheckBox, _("Get Move List") }, +{ 0, 0, 0, NULL, (void*) &appData.quietPlay, "", NULL, CheckBox, _("Quiet Play") }, +{ 0, 0, 0, NULL, (void*) &appData.seekGraph, "", NULL, CheckBox, _("Seek Graph") }, +{ 0, 0, 0, NULL, (void*) &appData.autoRefresh, "", NULL, CheckBox, _("Auto-Refresh Seek Graph") }, +{ 0, 0, 0, NULL, (void*) &appData.premove, "", NULL, CheckBox, _("Premove") }, +{ 0, 0, 0, NULL, (void*) &appData.premoveWhite, "", NULL, CheckBox, _("Premove for White") }, +{ 0, 0, 0, NULL, (void*) &appData.premoveWhiteText, "", NULL, TextBox, _("First White Move:") }, +{ 0, 0, 0, NULL, (void*) &appData.premoveBlack, "", NULL, CheckBox, _("Premove for Black") }, +{ 0, 0, 0, NULL, (void*) &appData.premoveBlackText, "", NULL, TextBox, _("First Black Move:") }, +{ 0, 0, 0, NULL, NULL, NULL, NULL, Break, "" }, +{ 0, 0, 0, NULL, (void*) &appData.icsAlarm, "", NULL, CheckBox, _("Alarm") }, +{ 0, 0, 100000000, NULL, (void*) &appData.icsAlarmTime, "", NULL, Spin, _("Alarm Time (msec):") }, +//{ 0, 0, 0, NULL, (void*) &appData.chatBoxes, "", NULL, TextBox, _("Startup Chat Boxes:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorize, "", NULL, CheckBox, _("Colorize Messages") }, +{ 0, 0, 0, NULL, (void*) &appData.colorShout, "", NULL, TextBox, _("Shout Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorSShout, "", NULL, TextBox, _("S-Shout Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorChannel1, "", NULL, TextBox, _("Channel #1 Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorChannel, "", NULL, TextBox, _("Other Channel Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorKibitz, "", NULL, TextBox, _("Kibitz Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorTell, "", NULL, TextBox, _("Tell Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorChallenge, "", NULL, TextBox, _("Challenge Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorRequest, "", NULL, TextBox, _("Request Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &appData.colorSeek, "", NULL, TextBox, _("Seek Text Colors:") }, +{ 0, 0, 0, NULL, (void*) &IcsOptionsOK, "", NULL, EndMark , "" } +}; + +Option loadOptions[] = { +{ 0, 0, 0, NULL, (void*) &appData.autoDisplayTags, "", NULL, CheckBox, _("Auto-Display Tags") }, +{ 0, 0, 0, NULL, (void*) &appData.autoDisplayComment, "", NULL, CheckBox, _("Auto-Display Comment") }, +{ 0, 0, 0, NULL, NULL, NULL, NULL, Label, _("Auto-Play speed of loaded games\n(0 = instant, -1 = off):") }, +{ 0, -1, 10000000, NULL, (void*) &appData.timeDelay, "", NULL, Fractional, _("Seconds per Move:") }, +{ 0, 0, 0, NULL, NULL, "", NULL, EndMark , "" } +}; + +Option saveOptions[] = { +{ 0, 0, 0, NULL, (void*) &appData.autoSaveGames, "", NULL, CheckBox, _("Auto-Save Games") }, +{ 0, 0, 0, NULL, (void*) &appData.saveGameFile, "", NULL, FileName, _("Save Games on File:") }, +{ 0, 0, 0, NULL, (void*) &appData.savePositionFile, "", NULL, FileName, _("Save Final Positions on File:") }, +{ 0, 0, 0, NULL, (void*) &appData.pgnEventHeader, "", NULL, TextBox, _("PGN Event Header:") }, +{ 0, 0, 0, NULL, (void*) &appData.oldSaveStyle, "", NULL, CheckBox, _("Old Save Style (as opposed to PGN)") }, +{ 0, 0, 0, NULL, (void*) &appData.saveExtendedInfoInPGN, "", NULL, CheckBox, _("Save Score/Depth Info in PGN") }, +{ 0, 0, 0, NULL, (void*) &appData.saveOutOfBookInfo, "", NULL, CheckBox, _("Save Out-of-Book Info in PGN ") }, +{ 0, 1, 0, NULL, NULL, "", NULL, EndMark , "" } +}; + +void SetColor(char *colorName, Widget box) +{ + Arg args[5]; + Pixel buttonColor; + XrmValue vFrom, vTo; + if (!appData.monoMode) { + vFrom.addr = (caddr_t) colorName; + vFrom.size = strlen(colorName); + XtConvert(shellWidget, XtRString, &vFrom, XtRPixel, &vTo); + if (vTo.addr == NULL) { + buttonColor = (Pixel) -1; + } else { + buttonColor = *(Pixel *) vTo.addr; + } + } + XtSetArg(args[0], XtNbackground, buttonColor);; + XtSetValues(box, args, 1); +} + +void AdjustColor(int i) +{ + int n = currentOption[i].value, col, j, r, g, b, step = 10; + char *s, buf[MSG_SIZ]; // color string + Arg args[5]; + XtSetArg(args[0], XtNstring, &s); + XtGetValues(currentOption[i-n-1].handle, args, 1); + if(sscanf(s, "#%x", &col) != 1) return; // malformed + b = col & 0xFF; g = col & 0xFF00; r = col & 0xFF0000; + switch(n) { + case 1: g -= 0x100*step; b -= step; break; + case 2: r -= 0x10000*step; b -= step; break; + case 3: g -= 0x100*step; r -= 0x10000*step; break; + case 4: r += 0x10000*step; g += 0x100*step; b += step; break; + } + if(r < 0) r = 0; if(g < 0) g = 0; if(b < 0) b = 0; + if(r > 0xFF0000) r = 0xFF0000; if(g > 0xFF00) g = 0xFF00; if(b > 0xFF) b = 0xFF; + col = r | g | b; + snprintf(buf, MSG_SIZ, "#%06x", col); + for(j=1; j<7; j++) if(buf[j] >= 'a') buf[j] -= 32; // capitalize + SetColor(buf, currentOption[i-n].handle); + XtSetArg(args[0], XtNstring, buf); + XtSetValues(currentOption[i-n-1].handle, args, 1); +} + +void BoardOptionsOK(int n) +{ + if(appData.overrideLineGap >= 0) lineGap = appData.overrideLineGap; + MakeColors(); CreateGCs(True); + CreateXPMPieces(); + CreateXPMBoard(appData.liteBackTextureFile, 1); + CreateXPMBoard(appData.darkBackTextureFile, 0); + InitDrawingSizes(-1, 0); + DrawPosition(True, NULL); +} + +Option boardOptions[] = { +{ 0, 0, 70, NULL, (void*) &appData.whitePieceColor, "", NULL, TextBox, _("White Piece Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 70, NULL, (void*) &appData.blackPieceColor, "", NULL, TextBox, _("Black Piece Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 70, NULL, (void*) &appData.lightSquareColor, "", NULL, TextBox, _("Light Square Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 70, NULL, (void*) &appData.darkSquareColor, "", NULL, TextBox, _("Dark Square Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 70, NULL, (void*) &appData.highlightSquareColor, "", NULL, TextBox, _("Highlight Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 70, NULL, (void*) &appData.premoveHighlightColor, "", NULL, TextBox, _("Premove Highlight Color:") }, +{ 1000, 1, 0, NULL, NULL, NULL, NULL, Button, " " }, +{ 1, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "R" }, +{ 2, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "G" }, +{ 3, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "B" }, +{ 4, 1, 0, NULL, (void*) &AdjustColor, NULL, NULL, Button, "W" }, +{ 0, 0, 0, NULL, (void*) &appData.upsideDown, "", NULL, CheckBox, _("Flip Pieces Shogi Style") }, +{ 0, 0, 0, NULL, (void*) &appData.allWhite, "", NULL, CheckBox, _("Use Outline Pieces for Black") }, +{ 0, 0, 0, NULL, (void*) &appData.monoMode, "", NULL, CheckBox, _("Mono Mode") }, +{ 0,-1, 5, NULL, (void*) &appData.overrideLineGap, "", NULL, Spin, _("Line Gap ( -1 = default for board size):") }, +{ 0, 0, 0, NULL, (void*) &appData.liteBackTextureFile, "", NULL, FileName, _("Light-Squares Texture File:") }, +{ 0, 0, 0, NULL, (void*) &appData.darkBackTextureFile, "", NULL, FileName, _("Dark-Squares Texture File:") }, +{ 0, 0, 0, NULL, (void*) &appData.bitmapDirectory, "", NULL, PathName, _("Directory with Bitmap Pieces:") }, +{ 0, 0, 0, NULL, (void*) &appData.pixmapDirectory, "", NULL, PathName, _("Directory with Pixmap Pieces:") }, +{ 0, 0, 0, NULL, (void*) &BoardOptionsOK, "", NULL, EndMark , "" } +}; + +void GenericReadout() +{ + int i, j; + String name, val; + Arg args[16]; + char buf[MSG_SIZ]; + float x; + for(i=0; ; i++) { // send all options that had to be OK-ed to engine + switch(currentOption[i].type) { + case TextBox: + case FileName: + case PathName: + XtSetArg(args[0], XtNstring, &val); + XtGetValues(currentOption[i].handle, args, 1); + if(*(char**) currentOption[i].target == NULL || strcmp(*(char**) currentOption[i].target, val)) { + safeStrCpy(currentOption[i].name + 100, val, MSG_SIZ-100); // text value kept in pivate storage for each option + *(char**) currentOption[i].target = currentOption[i].name + 100; // option gets to point to that + } + break; + case Spin: + case Fractional: + XtSetArg(args[0], XtNstring, &val); + XtGetValues(currentOption[i].handle, args, 1); + sscanf(val, "%f", &x); + if(x > currentOption[i].max) x = currentOption[i].max; + if(x < currentOption[i].min) x = currentOption[i].min; + if(currentOption[i].value != x) { + currentOption[i].value = x; + if(currentOption[i].type == Spin) *(int*) currentOption[i].target = x; + else *(float*) currentOption[i].target = x; + } + break; + case CheckBox: + j = 0; + XtSetArg(args[0], XtNstate, &j); + XtGetValues(currentOption[i].handle, args, 1); + if(currentOption[i].value != j) { + currentOption[i].value = j; + *(Boolean*) currentOption[i].target = j; + } + break; + case ComboBox: + val = ((char**)currentOption[i].choice)[values[i]]; + if(val && (*(char**) currentOption[i].target == NULL || strcmp(*(char**) currentOption[i].target, val))) { + if(*(char**) currentOption[i].target) free(*(char**) currentOption[i].target); + *(char**) currentOption[i].target = strdup(val); + } + break; + case EndMark: + if(currentOption[i].target) // callback for implementing necessary actions on OK (like redraw) + ((ButtonCallback*) currentOption[i].target)(i); + break; + default: + printf("GenericReadout: unexpected case in switch.\n"); + case Button: + case Label: + break; + } + if(currentOption[i].type == EndMark) break; + } +} + +void GenericCallback(w, client_data, call_data) + Widget w; + XtPointer client_data, call_data; +{ + String name, val; + Arg args[16]; + char buf[MSG_SIZ]; + int i, j; + int data = (intptr_t) client_data; + + XtSetArg(args[0], XtNlabel, &name); + XtGetValues(w, args, 1); + + if (strcmp(name, _("cancel")) == 0) { + SettingsPopDown(); + return; + } + if (strcmp(name, _("OK")) == 0) { // save buttons imply OK + GenericReadout(); + SettingsPopDown(); + return; + } + ((ButtonCallback*) currentOption[data].target)(data); +} + +void +GenericPopUp(Option *option, char *title) +{ + Arg args[16]; + Widget popup, layout, dialog, edit=NULL, form, last, b_ok, b_cancel, leftMargin = NULL, textField = NULL; + Window root, child; + int x, y, i, j, height=999, width=1, h, c, w; + int win_x, win_y, maxWidth, maxTextWidth; + unsigned int mask; + char def[MSG_SIZ], *msg; + static char pane[6] = "paneX"; + Widget texts[100], forelast = NULL, anchor, widest, lastrow = NULL; + + currentOption = option; // make available to callback + // kludge: fake address of a ChessProgramState struct that contains the options, so Spin and Combo callbacks work on it + currentCps = (ChessProgramState *) ((char *) option - ((char *)&first.option - (char *)&first)); + +// if(cps->nrOptions > 50) width = 4; else if(cps->nrOptions>24) width = 2; else width = 1; +// height = cps->nrOptions / width + 1; + i = 0; + XtSetArg(args[i], XtNresizable, True); i++; + SettingsShell = popup = + XtCreatePopupShell(title, transientShellWidgetClass, + shellWidget, args, i); + + layout = + XtCreateManagedWidget(layoutName, formWidgetClass, popup, + layoutArgs, XtNumber(layoutArgs)); + for(c=0; c maxWidth) maxWidth = w; + widest = texts[h]; + } else { + if(w > maxTextWidth) maxTextWidth = w; + if(!widest) widest = texts[h]; + } + } + } + if(maxTextWidth + 110 < maxWidth) + maxTextWidth = maxWidth - 110; + else maxWidth = maxTextWidth + 110; + for(h=0; h