/slashem/src/options.c
C | 4384 lines | 4040 code | 147 blank | 197 comment | 531 complexity | 2faff6e57e06b493a4c0353918fb03ef MD5 | raw file
Possible License(s): LGPL-2.0
Large files files are truncated, but you can click here to view the full file
- /* SCCS Id: @(#)options.c 3.4 2003/11/14 */
- /* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
- /* NetHack may be freely redistributed. See license for details. */
- #ifdef OPTION_LISTS_ONLY /* (AMIGA) external program for opt lists */
- #include "config.h"
- #include "objclass.h"
- #include "flag.h"
- NEARDATA struct flag flags; /* provide linkage */
- NEARDATA struct instance_flags iflags; /* provide linkage */
- #define static
- #else
- #include "hack.h"
- #include "tcap.h"
- #include <ctype.h>
- #endif
- #if defined(GL_GRAPHICS) || defined(SDL_GRAPHICS)
- #include "winGL.h" /* Sdlgl_parse_options */
- #endif
- #include "filename.h"
- #define WINTYPELEN 16
- #ifdef DEFAULT_WC_TILED_MAP
- #define PREFER_TILED TRUE
- #else
- #define PREFER_TILED FALSE
- #endif
- /*
- * NOTE: If you add (or delete) an option, please update the short
- * options help (option_help()), the long options help (dat/opthelp),
- * and the current options setting display function (doset()),
- * and also the Guidebooks.
- *
- * The order matters. If an option is a an initial substring of another
- * option (e.g. time and timed_delay) the shorter one must come first.
- */
- static struct Bool_Opt
- {
- const char *name;
- boolean *addr, initvalue;
- int optflags;
- } boolopt[] = {
- #ifdef AMIGA
- {"altmeta", &flags.altmeta, TRUE, DISP_IN_GAME},
- #else
- {"altmeta", (boolean *)0, TRUE, DISP_IN_GAME},
- #endif
- {"ascii_map", &iflags.wc_ascii_map, !PREFER_TILED, SET_IN_GAME}, /*WC*/
- #ifdef MFLOPPY
- {"asksavedisk", &flags.asksavedisk, FALSE, SET_IN_GAME},
- #else
- {"asksavedisk", (boolean *)0, FALSE, SET_IN_GAME},
- #endif
- {"autodig", &flags.autodig, FALSE, SET_IN_GAME},
- {"autopickup", &flags.pickup, TRUE, SET_IN_GAME},
- {"autoquiver", &flags.autoquiver, FALSE, SET_IN_GAME},
- #if defined(MICRO) && !defined(AMIGA)
- {"BIOS", &iflags.BIOS, FALSE, SET_IN_FILE},
- #else
- {"BIOS", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef INSURANCE
- {"checkpoint", &flags.ins_chkpt, TRUE, SET_IN_GAME},
- #else
- {"checkpoint", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef MFLOPPY
- {"checkspace", &iflags.checkspace, TRUE, SET_IN_GAME},
- #else
- {"checkspace", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"cmdassist", &iflags.cmdassist, TRUE, SET_IN_GAME},
- # if defined(MICRO) || defined(WIN32) || defined(NEWT_GRAPHICS)
- {"color", &iflags.wc_color,TRUE, SET_IN_GAME}, /*WC*/
- # else /* systems that support multiple terminals, many monochrome */
- {"color", &iflags.wc_color, FALSE, SET_IN_GAME}, /*WC*/
- # endif
- {"confirm",&flags.confirm, TRUE, SET_IN_GAME},
- #if defined(TERMLIB) && !defined(MAC_GRAPHICS_ENV)
- {"DECgraphics", &iflags.DECgraphics, FALSE, SET_IN_GAME},
- #else
- {"DECgraphics", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"eight_bit_tty", &iflags.wc_eight_bit_input, FALSE, SET_IN_GAME}, /*WC*/
- #if defined(TTY_GRAPHICS) || defined(NEWT_GRAPHICS)
- {"extmenu", &iflags.extmenu, FALSE, SET_IN_GAME},
- #else
- {"extmenu", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef OPT_DISPMAP
- {"fast_map", &flags.fast_map, TRUE, SET_IN_GAME},
- #else
- {"fast_map", (boolean *)0, TRUE, SET_IN_FILE},
- #endif
- {"female", &flags.female, FALSE, DISP_IN_GAME},
- {"fixinv", &flags.invlet_constant, TRUE, SET_IN_GAME},
- #ifdef AMIFLUSH
- {"flush", &flags.amiflush, FALSE, SET_IN_GAME},
- #else
- {"flush", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"fullscreen", &iflags.wc2_fullscreen, FALSE, SET_IN_FILE},
- {"help", &flags.help, TRUE, SET_IN_GAME},
- {"hilite_pet", &iflags.wc_hilite_pet, FALSE, SET_IN_GAME}, /*WC*/
- #ifdef ASCIIGRAPH
- {"IBMgraphics", &iflags.IBMgraphics, FALSE, SET_IN_GAME},
- #else
- {"IBMgraphics", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifndef MAC
- {"ignintr", &flags.ignintr, FALSE, SET_IN_GAME},
- #else
- {"ignintr", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef SHOW_WEIGHT
- {"invweight", &flags.invweight, FALSE, SET_IN_GAME},
- #else
- {"invweight", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- /*WAC the keep savefile option...*/
- #ifdef KEEP_SAVE
- {"keep_savefile", &flags.keep_savefile, FALSE, SET_IN_FILE},
- #else
- {"keep_savefile", (boolean *)0, FALSE, DISP_IN_GAME},
- #endif
- {"large_font", &iflags.obsolete, FALSE, SET_IN_FILE}, /* OBSOLETE */
- {"legacy", &flags.legacy, TRUE, DISP_IN_GAME},
- {"lit_corridor", &flags.lit_corridor, FALSE, SET_IN_GAME},
- {"lootabc", &iflags.lootabc, FALSE, SET_IN_GAME},
- #ifdef MAC_GRAPHICS_ENV
- {"Macgraphics", &iflags.MACgraphics, TRUE, SET_IN_GAME},
- #else
- {"Macgraphics", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef MAIL
- {"mail", &flags.biff, TRUE, SET_IN_GAME},
- #else
- {"mail", (boolean *)0, TRUE, SET_IN_FILE},
- #endif
- #ifdef MENU_COLOR
- # ifdef MICRO
- {"menucolors", &iflags.use_menu_color, TRUE, SET_IN_GAME},
- # else
- {"menucolors", &iflags.use_menu_color, FALSE, SET_IN_GAME},
- # endif
- #else
- {"menucolors", (boolean *)0, FALSE, SET_IN_GAME},
- #endif
- {"menu_on_esc", &flags.menu_on_esc, TRUE, SET_IN_GAME},
- #ifdef WIZARD
- /* for menu debugging only*/
- {"menu_tab_sep", &iflags.menu_tab_sep, FALSE, SET_IN_GAME},
- #else
- {"menu_tab_sep", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef WIZARD
- {"mon_polycontrol", &iflags.mon_polycontrol, FALSE, SET_IN_GAME},
- #else
- {"mon_polycontrol", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"mouse_support", &iflags.wc_mouse_support, TRUE, DISP_IN_GAME}, /*WC*/
- #ifdef NEWS
- {"news", &iflags.news, TRUE, DISP_IN_GAME},
- #else
- {"news", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"null", &flags.null, TRUE, SET_IN_GAME},
- #ifdef MAC
- {"page_wait", &flags.page_wait, TRUE, SET_IN_GAME},
- #else
- {"page_wait", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"perm_invent", &flags.perm_invent, FALSE, SET_IN_GAME},
- {"pickup_thrown", &flags.pickup_thrown, TRUE, SET_IN_GAME},
- {"popup_dialog", &iflags.wc_popup_dialog, FALSE, SET_IN_GAME}, /*WC*/
- {"prayconfirm", &flags.prayconfirm, TRUE, SET_IN_GAME},
- {"preload_tiles", &iflags.wc_preload_tiles, TRUE, DISP_IN_GAME}, /*WC*/
- {"pushweapon", &flags.pushweapon, FALSE, SET_IN_GAME},
- {"radar", (boolean *)0, FALSE, SET_IN_FILE}, /* OBSOLETE */
- #if defined(MICRO) && !defined(AMIGA)
- {"rawio", &iflags.rawio, FALSE, DISP_IN_GAME},
- #else
- {"rawio", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"rest_on_space", &flags.rest_on_space, FALSE, SET_IN_GAME},
- {"safe_pet", &flags.safe_dog, TRUE, SET_IN_GAME},
- #if defined(OBJ_SANITY)
- {"sanity_check", &iflags.sanity_check, TRUE, SET_IN_GAME},
- #elif defined(WIZARD)
- {"sanity_check", &iflags.sanity_check, FALSE, SET_IN_GAME},
- #else
- {"sanity_check", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef EXP_ON_BOTL
- {"showexp", &flags.showexp, FALSE, SET_IN_GAME},
- #else
- {"showexp", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"showrace", &iflags.showrace, FALSE, SET_IN_GAME},
- #ifdef SCORE_ON_BOTL
- {"showscore", &flags.showscore, FALSE, SET_IN_GAME},
- #else
- {"showscore", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- /* WAC made the [ xx pts] dmg display optional */
- #ifdef SHOW_DMG
- {"showdmg", &flags.showdmg, FALSE, SET_IN_GAME},
- #else
- {"showdmg", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- #ifdef SHOW_WEIGHT
- {"showweight", &flags.showweight, FALSE, SET_IN_GAME},
- #else
- {"showweight", (boolean *)0, FALSE, SET_IN_FILE},
- #endif
- {"silent", &flags.silent, TRUE, SET_IN_GAME},
- {"softkeyboard", &iflags.wc2_softkeyboard, FALSE, SET_IN_FILE},
- {"sortpack", &flags.sortpack, TRUE, SET_IN_GAME},
- {"sound", &flags.soundok, TRUE, SET_IN_GAME},
- {"sparkle", &flags.sparkle, TRUE, SET_IN_GAME},
- {"standout", &flags.standout, FALSE, SET_IN_GAME},
- {"splash_screen", &iflags.wc_splash_screen, TRUE, DISP_IN_GAME}, /*WC*/
- {"tiled_map", &iflags.wc_tiled_map, PREFER_TILED, DISP_IN_GAME}, /*WC*/
- {"time", &flags.time, FALSE, SET_IN_GAME},
- #ifdef TIMED_DELAY
- {"timed_delay", &flags.nap, TRUE, SET_IN_GAME},
- #else
- {"timed_delay", (boolean *)0, FALSE, SET_IN_GAME},
- #endif
- {"tombstone",&flags.tombstone, TRUE, SET_IN_GAME},
- {"toptenwin",&flags.toptenwin, FALSE, SET_IN_GAME},
- {"travel", &iflags.travelcmd, TRUE, SET_IN_GAME},
- #ifdef WIN32CON
- {"use_inverse", &iflags.wc_inverse, TRUE, SET_IN_GAME}, /*WC*/
- #else
- {"use_inverse", &iflags.wc_inverse, FALSE, SET_IN_GAME}, /*WC*/
- #endif
- {"verbose", &flags.verbose, TRUE, SET_IN_GAME},
- {"wraptext", &iflags.wc2_wraptext, FALSE, SET_IN_GAME},
- {(char *)0, (boolean *)0, FALSE, 0}
- };
- genericptr_t nh_option_get_boolopt()
- {
- return (genericptr_t)boolopt;
- }
- /* compound options, for option_help() and external programs like Amiga
- * frontend */
- static struct Comp_Opt
- {
- const char *name, *descr;
- int size; /* for frontends and such allocating space --
- * usually allowed size of data in game, but
- * occasionally maximum reasonable size for
- * typing when game maintains information in
- * a different format */
- int optflags;
- } compopt[] = {
- { "align", "your starting alignment (lawful, neutral, or chaotic)",
- 8, DISP_IN_GAME },
- { "align_message", "message window alignment", 20, DISP_IN_GAME }, /*WC*/
- { "align_status", "status window alignment", 20, DISP_IN_GAME }, /*WC*/
- { "altkeyhandler", "alternate key handler", 20, DISP_IN_GAME },
- { "boulder", "the symbol to use for displaying boulders",
- 1, SET_IN_GAME },
- { "catname", "the name of your (first) cat (e.g., catname:Tabby)",
- PL_PSIZ, DISP_IN_GAME },
- { "disclose", "the kinds of information to disclose at end of game",
- sizeof(flags.end_disclose) * 2,
- SET_IN_GAME },
- { "dogname", "the name of your (first) dog (e.g., dogname:Fang)",
- PL_PSIZ, DISP_IN_GAME },
- { "dungeon", "the symbols to use in drawing the dungeon map",
- MAXDCHARS+1, SET_IN_FILE },
- { "effects", "the symbols to use in drawing special effects",
- MAXECHARS+1, SET_IN_FILE },
- { "font_map", "the font to use in the map window", 40, DISP_IN_GAME }, /*WC*/
- { "font_menu", "the font to use in menus", 40, DISP_IN_GAME }, /*WC*/
- { "font_message", "the font to use in the message window",
- 40, DISP_IN_GAME }, /*WC*/
- { "font_size_map", "the size of the map font", 20, DISP_IN_GAME }, /*WC*/
- { "font_size_menu", "the size of the menu font", 20, DISP_IN_GAME }, /*WC*/
- { "font_size_message", "the size of the message font", 20, DISP_IN_GAME }, /*WC*/
- { "font_size_status", "the size of the status font", 20, DISP_IN_GAME }, /*WC*/
- { "font_size_text", "the size of the text font", 20, DISP_IN_GAME }, /*WC*/
- { "font_status", "the font to use in status window", 40, DISP_IN_GAME }, /*WC*/
- { "font_text", "the font to use in text windows", 40, DISP_IN_GAME }, /*WC*/
- { "fruit", "the name of a fruit you enjoy eating",
- PL_FSIZ, SET_IN_GAME },
- { "gender", "your starting gender (male or female)",
- 8, DISP_IN_GAME },
- { "ghoulname", "the name of your (first) ghoul (e.g., ghoulname:Casper)",
- PL_PSIZ, DISP_IN_GAME },
- { "horsename", "the name of your (first) horse (e.g., horsename:Silver)",
- PL_PSIZ, DISP_IN_GAME },
- { "map_mode", "map display mode under Windows", 20, DISP_IN_GAME }, /*WC*/
- { "menucolor", "set menu colors", PL_PSIZ, SET_IN_FILE },
- { "menustyle", "user interface for object selection",
- MENUTYPELEN, SET_IN_GAME },
- { "menu_deselect_all", "deselect all items in a menu", 4, SET_IN_FILE },
- { "menu_deselect_page", "deselect all items on this page of a menu",
- 4, SET_IN_FILE },
- { "menu_first_page", "jump to the first page in a menu",
- 4, SET_IN_FILE },
- { "menu_headings", "bold, inverse, or underline headings", 9, SET_IN_GAME },
- { "menu_invert_all", "invert all items in a menu", 4, SET_IN_FILE },
- { "menu_invert_page", "invert all items on this page of a menu",
- 4, SET_IN_FILE },
- { "menu_last_page", "jump to the last page in a menu", 4, SET_IN_FILE },
- { "menu_next_page", "goto the next menu page", 4, SET_IN_FILE },
- { "menu_previous_page", "goto the previous menu page", 4, SET_IN_FILE },
- { "menu_search", "search for a menu item", 4, SET_IN_FILE },
- { "menu_select_all", "select all items in a menu", 4, SET_IN_FILE },
- { "menu_select_page", "select all items on this page of a menu",
- 4, SET_IN_FILE },
- { "monsters", "the symbols to use for monsters",
- MAXMCLASSES, SET_IN_FILE },
- { "msghistory", "number of top line messages to save",
- 5, DISP_IN_GAME },
- # ifdef TTY_GRAPHICS
- {"msg_window", "the type of message window required",1, SET_IN_GAME},
- # else
- {"msg_window", "the type of message window required", 1, SET_IN_FILE},
- # endif
- { "name", "your character's name (e.g., name:Merlin-W)",
- PL_NSIZ, DISP_IN_GAME },
- { "number_pad", "use the number pad", 1, SET_IN_GAME},
- { "objects", "the symbols to use for objects",
- MAXOCLASSES, SET_IN_FILE },
- { "packorder", "the inventory order of the items in your pack",
- MAXOCLASSES, SET_IN_GAME },
- #ifdef CHANGE_COLOR
- { "palette", "palette (00c/880/-fff is blue/yellow/reverse white)",
- 15 , SET_IN_GAME },
- # if defined(MAC)
- { "hicolor", "same as palette, only order is reversed",
- 15, SET_IN_FILE },
- # endif
- #endif
- { "pettype", "your preferred initial pet type", 4, DISP_IN_GAME },
- { "pickup_burden", "maximum burden picked up before prompt",
- 20, SET_IN_GAME },
- { "pickup_types", "types of objects to pick up automatically",
- MAXOCLASSES, SET_IN_GAME },
- { "player_selection", "choose character via dialog or prompts",
- 12, DISP_IN_GAME },
- { "race", "your starting race (e.g., Human, Elf)",
- PL_CSIZ, DISP_IN_GAME },
- { "role", "your starting role (e.g., Barbarian, Valkyrie)",
- PL_CSIZ, DISP_IN_GAME },
- { "runmode", "display frequency when `running' or `travelling'",
- sizeof "teleport", SET_IN_GAME },
- { "scores", "the parts of the score list you wish to see",
- 32, SET_IN_GAME },
- { "scroll_amount", "amount to scroll map when scroll_margin is reached",
- 20, DISP_IN_GAME }, /*WC*/
- { "scroll_margin", "scroll map when this far from the edge", 20, DISP_IN_GAME }, /*WC*/
- #ifdef MSDOS
- { "soundcard", "type of sound card to use", 20, SET_IN_FILE },
- #endif
- { "suppress_alert", "suppress alerts about version-specific features",
- 8, SET_IN_GAME },
- { "tile_width", "width of tiles", 20, DISP_IN_GAME}, /*WC*/
- { "tile_height", "height of tiles", 20, DISP_IN_GAME}, /*WC*/
- { "tile_file", "name of tile file", 70, DISP_IN_GAME}, /*WC*/
- { "tileset", "name of predefined tileset to use",
- PL_PSIZ, SET_IN_GAME },
- { "traps", "the symbols to use in drawing traps",
- MAXTCHARS+1, SET_IN_FILE },
- { "vary_msgcount", "show more old messages at a time", 20, DISP_IN_GAME }, /*WC*/
- #ifdef MSDOS
- { "video", "method of video updating", 20, SET_IN_FILE },
- #endif
- #ifdef VIDEOSHADES
- { "videocolors", "color mappings for internal screen routines",
- 40, DISP_IN_GAME },
- #ifdef MSDOS
- { "videoshades", "gray shades to map to black/gray/white",
- 32, DISP_IN_GAME },
- #endif
- #endif
- #ifdef WIN32CON
- {"subkeyvalue", "override keystroke value", 7, SET_IN_FILE},
- #endif
- { "windowcolors", "the foreground/background colors of windows", /*WC*/
- 80, DISP_IN_GAME },
- { "windowtype", "windowing system to use", WINTYPELEN, DISP_IN_GAME },
- { "wolfname", "the name of your (first) wolf (e.g., wolfname:Beast)",
- PL_PSIZ, DISP_IN_GAME },
- { (char *)0, (char *)0, 0, 0 }
- };
- static struct Bool_Tile_Opt
- {
- const char *name;
- unsigned long flag;
- unsigned long initvalue;
- } booltileopt[] = {
- {"transparent", TILESET_TRANSPARENT, 0},
- {"pseudo3D", TILESET_PSEUDO3D, 0},
- {(char *)0, 0, 0}
- };
- #ifdef OPTION_LISTS_ONLY
- #undef static
- #else /* use rest of file */
- static boolean need_redraw; /* for doset() */
- #if defined(TOS) && defined(TEXTCOLOR)
- extern boolean colors_changed; /* in tos.c */
- #endif
- #ifdef VIDEOSHADES
- extern char *shade[3]; /* in sys/msdos/video.c */
- extern char ttycolors[CLR_MAX]; /* in sys/msdos/video.c, win/tty/termcap.c*/
- #endif
- static char def_inv_order[MAXOCLASSES] = {
- COIN_CLASS, AMULET_CLASS, WEAPON_CLASS, ARMOR_CLASS, FOOD_CLASS,
- SCROLL_CLASS, SPBOOK_CLASS, POTION_CLASS, RING_CLASS, WAND_CLASS,
- TOOL_CLASS, GEM_CLASS, ROCK_CLASS, BALL_CLASS, CHAIN_CLASS, 0,
- };
- /*
- * Default menu manipulation command accelerators. These may _not_ be:
- *
- * + a number - reserved for counts
- * + an upper or lower case US ASCII letter - used for accelerators
- * + ESC - reserved for escaping the menu
- * + NULL, CR or LF - reserved for commiting the selection(s). NULL
- * is kind of odd, but the tty's xwaitforspace() will return it if
- * someone hits a <ret>.
- * + a default object class symbol - used for object class accelerators
- *
- * Standard letters (for now) are:
- *
- * < back 1 page
- * > forward 1 page
- * ^ first page
- * | last page
- * : search
- *
- * page all
- * , select .
- * \ deselect -
- * ~ invert @
- *
- * The command name list is duplicated in the compopt array.
- */
- typedef struct {
- const char *name;
- char cmd;
- } menu_cmd_t;
- #define NUM_MENU_CMDS 11
- static const menu_cmd_t default_menu_cmd_info[NUM_MENU_CMDS] = {
- /* 0*/ { "menu_first_page", MENU_FIRST_PAGE },
- { "menu_last_page", MENU_LAST_PAGE },
- { "menu_next_page", MENU_NEXT_PAGE },
- { "menu_previous_page", MENU_PREVIOUS_PAGE },
- { "menu_select_all", MENU_SELECT_ALL },
- /* 5*/ { "menu_deselect_all", MENU_UNSELECT_ALL },
- { "menu_invert_all", MENU_INVERT_ALL },
- { "menu_select_page", MENU_SELECT_PAGE },
- { "menu_deselect_page", MENU_UNSELECT_PAGE },
- { "menu_invert_page", MENU_INVERT_PAGE },
- /*10*/ { "menu_search", MENU_SEARCH },
- };
- /*
- * Allow the user to map incoming characters to various menu commands.
- * The accelerator list must be a valid C string.
- */
- #define MAX_MENU_MAPPED_CMDS 32 /* some number */
- char mapped_menu_cmds[MAX_MENU_MAPPED_CMDS+1]; /* exported */
- static char mapped_menu_op[MAX_MENU_MAPPED_CMDS+1];
- static short n_menu_mapped = 0;
- static boolean initial, from_file;
- STATIC_DCL void FDECL(doset_add_menu, (winid,const char *,int));
- STATIC_DCL void FDECL(nmcpy, (char *, const char *, int));
- STATIC_DCL void FDECL(escapes, (const char *, char *));
- STATIC_DCL void FDECL(rejectoption, (const char *));
- STATIC_DCL void FDECL(badoption, (const char *));
- STATIC_OVL void FDECL(badtileoption, (const char *));
- STATIC_DCL char *FDECL(string_for_opt, (char *,BOOLEAN_P));
- STATIC_OVL char *FDECL(string_for_tile_opt, (char *, BOOLEAN_P));
- STATIC_DCL char *FDECL(string_for_env_opt, (const char *, char *,BOOLEAN_P));
- STATIC_DCL void FDECL(bad_negation, (const char *,BOOLEAN_P));
- STATIC_DCL int FDECL(change_inv_order, (char *));
- STATIC_DCL void FDECL(oc_to_str, (char *, char *));
- STATIC_DCL void FDECL(graphics_opts, (char *,const char *,int,int));
- STATIC_DCL int FDECL(feature_alert_opts, (char *, const char *));
- STATIC_DCL const char *FDECL(get_compopt_value, (const char *, char *));
- STATIC_DCL boolean FDECL(special_handling, (const char *, BOOLEAN_P, BOOLEAN_P));
- STATIC_DCL void FDECL(warning_opts, (char *,const char *));
- STATIC_DCL void FDECL(duplicate_opt_detection, (const char *, int));
- STATIC_OVL void FDECL(wc_set_font_name, (int, char *));
- STATIC_OVL int FDECL(wc_set_window_colors, (char *));
- STATIC_OVL boolean FDECL(is_wc_option, (const char *));
- STATIC_OVL boolean FDECL(wc_supported, (const char *));
- STATIC_OVL boolean FDECL(is_wc2_option, (const char *));
- STATIC_OVL boolean FDECL(wc2_supported, (const char *));
- #ifdef AUTOPICKUP_EXCEPTIONS
- STATIC_DCL void FDECL(remove_autopickup_exception, (struct autopickup_exception *));
- STATIC_OVL int FDECL(count_ape_maps, (int *, int *));
- #endif
- /* check whether a user-supplied option string is a proper leading
- substring of a particular option name; option string might have
- a colon or equals sign and arbitrary value appended to it */
- boolean
- match_optname(user_string, opt_name, min_length, val_allowed)
- const char *user_string, *opt_name;
- int min_length;
- boolean val_allowed;
- {
- int len = (int)strlen(user_string);
- if (val_allowed) {
- const char *p = index(user_string, ':'),
- *q = index(user_string, '=');
- if (!p || (q && q < p)) p = q;
- while(p && p > user_string && isspace(*(p-1))) p--;
- if (p) len = (int)(p - user_string);
- }
- return (len >= min_length) && !strncmpi(opt_name, user_string, len);
- }
- /* most environment variables will eventually be printed in an error
- * message if they don't work, and most error message paths go through
- * BUFSZ buffers, which could be overflowed by a maliciously long
- * environment variable. if a variable can legitimately be long, or
- * if it's put in a smaller buffer, the responsible code will have to
- * bounds-check itself.
- */
- char *
- nh_getenv(ev)
- const char *ev;
- {
- char *getev = getenv(ev);
- if (getev && strlen(getev) <= (BUFSZ / 2))
- return getev;
- else
- return (char *)0;
- }
- void
- initoptions()
- {
- #ifndef MAC
- char *opts;
- #endif
- int i;
- /* initialize the random number generator */
- setrandom();
- /* for detection of configfile options specified multiple times */
- iflags.opt_booldup = iflags.opt_compdup = (int *)0;
-
- for (i = 0; boolopt[i].name; i++) {
- if (boolopt[i].addr)
- *(boolopt[i].addr) = boolopt[i].initvalue;
- }
- flags.end_own = FALSE;
- flags.end_top = 3;
- flags.end_around = 2;
- iflags.runmode = RUN_LEAP;
- iflags.msg_history = 20;
- #ifdef TTY_GRAPHICS
- iflags.prevmsg_window = 's';
- #endif
- iflags.menu_headings = ATR_INVERSE;
- /* Use negative indices to indicate not yet selected */
- flags.initrole = -1;
- flags.initrace = -1;
- flags.initgend = -1;
- flags.initalign = -1;
- /* Set the default monster and object class symbols. Don't use */
- /* memcpy() --- sizeof char != sizeof uchar on some machines. */
- for (i = 0; i < MAXOCLASSES; i++)
- oc_syms[i] = (uchar) def_oc_syms[i];
- for (i = 0; i < MAXMCLASSES; i++)
- monsyms[i] = (uchar) def_monsyms[i];
- for (i = 0; i < WARNCOUNT; i++)
- warnsyms[i] = def_warnsyms[i].sym;
- iflags.bouldersym = 0;
- iflags.travelcc.x = iflags.travelcc.y = -1;
- flags.warnlevel = 1;
- flags.warntype = 0L;
- /* assert( sizeof flags.inv_order == sizeof def_inv_order ); */
- (void)memcpy((genericptr_t)flags.inv_order,
- (genericptr_t)def_inv_order, sizeof flags.inv_order);
- flags.pickup_types[0] = '\0';
- flags.pickup_burden = MOD_ENCUMBER;
- for (i = 0; i < NUM_DISCLOSURE_OPTIONS; i++)
- flags.end_disclose[i] = DISCLOSE_PROMPT_DEFAULT_NO;
- switch_graphics(ASCII_GRAPHICS); /* set default characters */
- #if defined(UNIX) && defined(TTY_GRAPHICS)
- /*
- * Set defaults for some options depending on what we can
- * detect about the environment's capabilities.
- * This has to be done after the global initialization above
- * and before reading user-specific initialization via
- * config file/environment variable below.
- */
- /* this detects the IBM-compatible console on most 386 boxes */
- if ((opts = nh_getenv("TERM")) && !strncmp(opts, "AT", 2)) {
- switch_graphics(IBM_GRAPHICS);
- # ifdef TEXTCOLOR
- iflags.use_color = TRUE;
- # endif
- }
- #endif /* UNIX && TTY_GRAPHICS */
- #if defined(UNIX) || defined(VMS)
- # ifdef TTY_GRAPHICS
- /* detect whether a "vt" terminal can handle alternate charsets */
- if ((opts = nh_getenv("TERM")) &&
- !strncmpi(opts, "vt", 2) && AS && AE &&
- index(AS, '\016') && index(AE, '\017')) {
- switch_graphics(DEC_GRAPHICS);
- }
- # endif
- #endif /* UNIX || VMS */
- #ifdef MAC_GRAPHICS_ENV
- switch_graphics(MAC_GRAPHICS);
- #endif /* MAC_GRAPHICS_ENV */
- flags.menu_style = MENU_FULL;
- /* since this is done before init_objects(), do partial init here */
- objects[SLIME_MOLD].oc_name_idx = SLIME_MOLD;
- nmcpy(pl_fruit, OBJ_NAME(objects[SLIME_MOLD]), PL_FSIZ);
- #ifndef MAC
- opts = getenv(NETHACK_ENV_OPTIONS);
- if (!opts) opts = getenv("NETHACKOPTIONS");
- if (!opts) opts = getenv("HACKOPTIONS");
- if (opts) {
- if (*opts == '/' || *opts == '\\' || *opts == '@') {
- if (*opts == '@') opts++; /* @filename */
- /* looks like a filename */
- if (strlen(opts) < BUFSZ/2)
- read_config_file(opts);
- } else {
- read_config_file((char *)0);
- /* let the total length of options be long;
- * parseoptions() will check each individually
- */
- parseoptions(opts, TRUE, FALSE);
- }
- } else
- #endif
- read_config_file((char *)0);
- (void)fruitadd(pl_fruit);
- /* Remove "slime mold" from list of object names; this will */
- /* prevent it from being wished unless it's actually present */
- /* as a named (or default) fruit. Wishing for "fruit" will */
- /* result in the player's preferred fruit [better than "\033"]. */
- obj_descr[SLIME_MOLD].oc_name = "fruit";
- #if defined(GL_GRAPHICS) || defined(SDL_GRAPHICS)
- /* -AJA- SDL/GL support. Needs to happen after main config
- * file has been read.
- */
- opts = getenv(SDLGL_ENV_VAR);
- if (opts)
- Sdlgl_parse_options(opts, TRUE, FALSE);
- #endif
- return;
- }
- STATIC_OVL void
- nmcpy(dest, src, maxlen)
- char *dest;
- const char *src;
- int maxlen;
- {
- int count;
- for(count = 1; count < maxlen; count++) {
- if(*src == ',' || *src == '\0') break; /*exit on \0 terminator*/
- *dest++ = *src++;
- }
- *dest = 0;
- }
- /*
- * escapes: escape expansion for showsyms. C-style escapes understood include
- * \n, \b, \t, \r, \xnnn (hex), \onnn (octal), \nnn (decimal). The ^-prefix
- * for control characters is also understood, and \[mM] followed by any of the
- * previous forms or by a character has the effect of 'meta'-ing the value (so
- * that the alternate character set will be enabled).
- */
- STATIC_OVL void
- escapes(cp, tp)
- const char *cp;
- char *tp;
- {
- while (*cp)
- {
- int cval = 0, meta = 0;
- if (*cp == '\\' && index("mM", cp[1])) {
- meta = 1;
- cp += 2;
- }
- if (*cp == '\\' && index("0123456789xXoO", cp[1]))
- {
- const char *dp, *hex = "00112233445566778899aAbBcCdDeEfF";
- int dcount = 0;
- cp++;
- if (*cp == 'x' || *cp == 'X')
- for (++cp; (dp = index(hex, *cp)) && (dcount++ < 2); cp++)
- cval = (cval * 16) + (dp - hex) / 2;
- else if (*cp == 'o' || *cp == 'O')
- for (++cp; (index("01234567",*cp)) && (dcount++ < 3); cp++)
- cval = (cval * 8) + (*cp - '0');
- else
- for (; (index("0123456789",*cp)) && (dcount++ < 3); cp++)
- cval = (cval * 10) + (*cp - '0');
- }
- else if (*cp == '\\') /* C-style character escapes */
- {
- switch (*++cp)
- {
- case '\\': cval = '\\'; break;
- case 'n': cval = '\n'; break;
- case 't': cval = '\t'; break;
- case 'b': cval = '\b'; break;
- case 'r': cval = '\r'; break;
- default: cval = *cp;
- }
- cp++;
- }
- else if (*cp == '^') /* expand control-character syntax */
- {
- cval = (*++cp & 0x1f);
- cp++;
- }
- else
- cval = *cp++;
- if (meta)
- cval |= 0x80;
- *tp++ = cval;
- }
- *tp = '\0';
- }
- STATIC_OVL void
- rejectoption(optname)
- const char *optname;
- {
- #ifdef MICRO
- pline("\"%s\" settable only from %s.", optname, configfile);
- #else
- pline("%s can be set only from %s or %s.", optname,
- NETHACK_ENV_OPTIONS, configfile);
- #endif
- }
- STATIC_OVL void
- badoption(opts)
- const char *opts;
- {
- if (!initial) {
- if (!strncmp(opts, "h", 1) || !strncmp(opts, "?", 1))
- option_help();
- else
- pline("Bad syntax: %s. Enter \"?g\" for help.", opts);
- return;
- }
- #ifdef MAC
- else return;
- #endif
- if(from_file)
- raw_printf("Bad syntax in OPTIONS in %s: %s.", configfile, opts);
- else
- raw_printf("Bad syntax in %s: %s.", NETHACK_ENV_OPTIONS, opts);
- wait_synch();
- }
- STATIC_OVL void
- badauthoption(opts)
- const char *opts;
- {
- raw_printf("Bad syntax in AUTHENTICATION in %s: %s.", configfile, opts);
- wait_synch();
- }
- STATIC_OVL void
- badtileoption(opts)
- const char *opts;
- {
- raw_printf("Bad syntax in TILESET in %s: %s.", configfile, opts);
- wait_synch();
- }
- STATIC_OVL char *
- string_for_opt(opts, val_optional)
- char *opts;
- boolean val_optional;
- {
- char *colon, *equals;
- colon = index(opts, ':');
- equals = index(opts, '=');
- if (!colon || (equals && equals < colon)) colon = equals;
- if (!colon || !*++colon) {
- if (!val_optional) badoption(opts);
- return (char *)0;
- }
- return colon;
- }
- STATIC_OVL char *
- string_for_auth_opt(opts, val_optional)
- char *opts;
- boolean val_optional;
- {
- char *colon = string_for_opt(opts, TRUE);
- if (!colon && !val_optional) badauthoption(opts);
- return colon;
- }
- STATIC_OVL char *
- string_for_tile_opt(opts, val_optional)
- char *opts;
- boolean val_optional;
- {
- char *colon = string_for_opt(opts, TRUE);
- if (!colon && !val_optional) badtileoption(opts);
- return colon;
- }
- STATIC_OVL char *
- string_for_env_opt(optname, opts, val_optional)
- const char *optname;
- char *opts;
- boolean val_optional;
- {
- if(!initial) {
- rejectoption(optname);
- return (char *)0;
- }
- return string_for_opt(opts, val_optional);
- }
- STATIC_OVL void
- bad_negation(optname, with_parameter)
- const char *optname;
- boolean with_parameter;
- {
- pline_The("%s option may not %sbe negated.",
- optname,
- with_parameter ? "both have a value and " : "");
- }
- /*
- * Change the inventory order, using the given string as the new order.
- * Missing characters in the new order are filled in at the end from
- * the current inv_order, except for gold, which is forced to be first
- * if not explicitly present.
- *
- * This routine returns 1 unless there is a duplicate or bad char in
- * the string.
- */
- STATIC_OVL int
- change_inv_order(op)
- char *op;
- {
- int oc_sym, num;
- char *sp, buf[BUFSZ];
- num = 0;
- #ifndef GOLDOBJ
- if (!index(op, GOLD_SYM))
- buf[num++] = COIN_CLASS;
- #else
- /* !!!! probably unnecessary with gold as normal inventory */
- #endif
- for (sp = op; *sp; sp++) {
- oc_sym = def_char_to_objclass(*sp);
- /* reject bad or duplicate entries */
- if (oc_sym == MAXOCLASSES ||
- oc_sym == RANDOM_CLASS || oc_sym == ILLOBJ_CLASS ||
- !index(flags.inv_order, oc_sym) || index(sp+1, *sp))
- return 0;
- /* retain good ones */
- buf[num++] = (char) oc_sym;
- }
- buf[num] = '\0';
- /* fill in any omitted classes, using previous ordering */
- for (sp = flags.inv_order; *sp; sp++)
- if (!index(buf, *sp)) {
- buf[num++] = *sp;
- buf[num] = '\0'; /* explicitly terminate for next index() */
- }
- Strcpy(flags.inv_order, buf);
- return 1;
- }
- STATIC_OVL void
- graphics_opts(opts, optype, maxlen, offset)
- register char *opts;
- const char *optype;
- int maxlen, offset;
- {
- uchar translate[MAXPCHARS+1];
- int length, i;
- if (!(opts = string_for_env_opt(optype, opts, FALSE)))
- return;
- escapes(opts, opts);
- length = strlen(opts);
- if (length > maxlen) length = maxlen;
- /* match the form obtained from PC configuration files */
- for (i = 0; i < length; i++)
- translate[i] = (uchar) opts[i];
- assign_graphics(translate, length, maxlen, offset);
- }
- STATIC_OVL void
- warning_opts(opts, optype)
- register char *opts;
- const char *optype;
- {
- uchar translate[MAXPCHARS+1];
- int length, i;
- if (!(opts = string_for_env_opt(optype, opts, FALSE)))
- return;
- escapes(opts, opts);
- length = strlen(opts);
- if (length > WARNCOUNT) length = WARNCOUNT;
- /* match the form obtained from PC configuration files */
- for (i = 0; i < length; i++)
- translate[i] = (((i < WARNCOUNT) && opts[i]) ?
- (uchar) opts[i] : def_warnsyms[i].sym);
- assign_warnings(translate);
- }
- void
- assign_warnings(graph_chars)
- register uchar *graph_chars;
- {
- int i;
- for (i = 0; i < WARNCOUNT; i++)
- if (graph_chars[i]) warnsyms[i] = graph_chars[i];
- }
- STATIC_OVL int
- feature_alert_opts(op, optn)
- char *op;
- const char *optn;
- {
- char buf[BUFSZ];
- boolean rejectver = FALSE;
- unsigned long fnv = get_feature_notice_ver(op); /* version.c */
- if (fnv == 0L) return 0;
- if (fnv > get_current_feature_ver())
- rejectver = TRUE;
- else
- flags.suppress_alert = fnv;
- if (rejectver) {
- if (!initial)
- You_cant("disable new feature alerts for future versions.");
- else {
- Sprintf(buf,
- "\n%s=%s Invalid reference to a future version ignored",
- optn, op);
- badoption(buf);
- }
- return 0;
- }
- if (!initial) {
- Sprintf(buf, "%lu.%lu.%lu", FEATURE_NOTICE_VER_MAJ,
- FEATURE_NOTICE_VER_MIN, FEATURE_NOTICE_VER_PATCH);
- pline("Feature change alerts disabled for Slash'EM %s features and prior.",
- buf);
- }
- return 1;
- }
- void
- set_duplicate_opt_detection(on_or_off)
- int on_or_off;
- {
- int k, *optptr;
- if (on_or_off != 0) {
- /*-- ON --*/
- if (iflags.opt_booldup)
- impossible("iflags.opt_booldup already on (memory leak)");
- iflags.opt_booldup = (int *)alloc(SIZE(boolopt) * sizeof(int));
- optptr = iflags.opt_booldup;
- for (k = 0; k < SIZE(boolopt); ++k)
- *optptr++ = 0;
-
- if (iflags.opt_compdup)
- impossible("iflags.opt_compdup already on (memory leak)");
- iflags.opt_compdup = (int *)alloc(SIZE(compopt) * sizeof(int));
- optptr = iflags.opt_compdup;
- for (k = 0; k < SIZE(compopt); ++k)
- *optptr++ = 0;
- } else {
- /*-- OFF --*/
- if (iflags.opt_booldup) free((genericptr_t) iflags.opt_booldup);
- iflags.opt_booldup = (int *)0;
- if (iflags.opt_compdup) free((genericptr_t) iflags.opt_compdup);
- iflags.opt_compdup = (int *)0;
- }
- }
- STATIC_OVL void
- duplicate_opt_detection(opts, bool_or_comp)
- const char *opts;
- int bool_or_comp; /* 0 == boolean option, 1 == compound */
- {
- int i, *optptr;
- #if defined(MAC)
- /* the Mac has trouble dealing with the output of messages while
- * processing the config file. That should get fixed one day.
- * For now just return.
- */
- return;
- #endif
- if ((bool_or_comp == 0) && iflags.opt_booldup && initial && from_file) {
- for (i = 0; boolopt[i].name; i++) {
- if (match_optname(opts, boolopt[i].name, 3, FALSE)) {
- optptr = iflags.opt_booldup + i;
- if (*optptr == 1) {
- raw_printf(
- "\nWarning - Boolean option specified multiple times: %s.\n",
- opts);
- wait_synch();
- }
- *optptr += 1;
- break; /* don't match multiple options */
- }
- }
- } else if ((bool_or_comp == 1) && iflags.opt_compdup && initial && from_file) {
- for (i = 0; compopt[i].name; i++) {
- if (match_optname(opts, compopt[i].name, strlen(compopt[i].name), TRUE)) {
- optptr = iflags.opt_compdup + i;
- if (*optptr == 1) {
- raw_printf(
- "\nWarning - compound option specified multiple times: %s.\n",
- compopt[i].name);
- wait_synch();
- }
- *optptr += 1;
- break; /* don't match multiple options */
- }
- }
- }
- }
- #ifdef MENU_COLOR
- extern struct menucoloring *menu_colorings;
- static const struct {
- const char *name;
- const int color;
- } colornames[] = {
- {"black", CLR_BLACK},
- {"red", CLR_RED},
- {"green", CLR_GREEN},
- {"brown", CLR_BROWN},
- {"blue", CLR_BLUE},
- {"magenta", CLR_MAGENTA},
- {"cyan", CLR_CYAN},
- {"gray", CLR_GRAY},
- {"orange", CLR_ORANGE},
- {"lightgreen", CLR_BRIGHT_GREEN},
- {"yellow", CLR_YELLOW},
- {"lightblue", CLR_BRIGHT_BLUE},
- {"lightmagenta", CLR_BRIGHT_MAGENTA},
- {"lightcyan", CLR_BRIGHT_CYAN},
- {"white", CLR_WHITE}
- };
- static const struct {
- const char *name;
- const int attr;
- } attrnames[] = {
- {"none", ATR_NONE},
- {"bold", ATR_BOLD},
- {"dim", ATR_DIM},
- {"underline", ATR_ULINE},
- {"blink", ATR_BLINK},
- {"inverse", ATR_INVERSE}
- };
- /* parse '"regex_string"=color' and add it to menucoloring */
- boolean
- add_menu_coloring(str)
- char *str;
- {
- int i, c = NO_COLOR, a = ATR_NONE;
- struct menucoloring *tmp;
- char *tmps, *cs = strchr(str, '=');
- #ifdef POSIX_REGEX
- int errnum;
- char errbuf[80];
- #endif
- const char *err = (char *)0;
-
- if (!cs || !str) return FALSE;
-
- tmps = cs;
- tmps++;
- while (*tmps && isspace(*tmps)) tmps++;
- for (i = 0; i < SIZE(colornames); i++)
- if (strstri(tmps, colornames[i].name) == tmps) {
- c = colornames[i].color;
- break;
- }
- if ((i == SIZE(colornames)) && (*tmps >= '0' && *tmps <='9'))
- c = atoi(tmps);
-
- if (c > 15) return FALSE;
-
- tmps = strchr(str, '&');
- if (tmps) {
- tmps++;
- while (*tmps && isspace(*tmps)) tmps++;
- for (i = 0; i < SIZE(attrnames); i++)
- if (strstri(tmps, attrnames[i].name) == tmps) {
- a = attrnames[i].attr;
- break;
- }
- if ((i == SIZE(attrnames)) && (*tmps >= '0' && *tmps <='9'))
- a = atoi(tmps);
- }
-
- *cs = '\0';
- tmps = str;
- if ((*tmps == '"') || (*tmps == '\'')) {
- cs--;
- while (isspace(*cs)) cs--;
- if (*cs == *tmps) {
- *cs = '\0';
- tmps++;
- }
- }
-
- tmp = (struct menucoloring *)alloc(sizeof(struct menucoloring));
- #ifdef USE_REGEX_MATCH
- # ifdef GNU_REGEX
- tmp->match.translate = 0;
- tmp->match.fastmap = 0;
- tmp->match.buffer = 0;
- tmp->match.allocated = 0;
- tmp->match.regs_allocated = REGS_FIXED;
- err = re_compile_pattern(tmps, strlen(tmps), &tmp->match);
- # else
- # ifdef POSIX_REGEX
- errnum = regcomp(&tmp->match, tmps, REG_EXTENDED | REG_NOSUB);
- if (errnum != 0) {
- regerror(errnum, &tmp->match, errbuf, sizeof(errbuf));
- err = errbuf;
- }
- # endif
- # endif
- #else
- tmp->match = (char *)alloc(strlen(tmps)+1);
- (void) memcpy((genericptr_t)tmp->match, (genericptr_t)tmps, strlen(tmps)+1);
- #endif
- if (err) {
- raw_printf("\nMenucolor regex error: %s\n", err);
- wait_synch();
- free(tmp);
- return FALSE;
- } else {
- tmp->next = menu_colorings;
- tmp->color = c;
- tmp->attr = a;
- menu_colorings = tmp;
- return TRUE;
- }
- }
- #endif /* MENU_COLOR */
- void
- parseoptions(opts, tinitial, tfrom_file)
- register char *opts;
- boolean tinitial, tfrom_file;
- {
- register char *op;
- unsigned num;
- boolean negated;
- int i;
- const char *fullname;
- initial = tinitial;
- from_file = tfrom_file;
- if ((op = index(opts, ',')) != 0) {
- *op++ = 0;
- parseoptions(op, initial, from_file);
- }
- if (strlen(opts) > BUFSZ/2) {
- badoption("option too long");
- return;
- }
- /* strip leading and trailing white space */
- while (isspace((int)*opts)) opts++;
- op = eos(opts);
- while (--op >= opts && isspace((int)*op)) *op = '\0';
- if (!*opts) return;
- negated = FALSE;
- while ((*opts == '!') || !strncmpi(opts, "no", 2)) {
- if (*opts == '!') opts++; else opts += 2;
- negated = !negated;
- }
- /* variant spelling */
- if (match_optname(opts, "colour", 5, FALSE))
- Strcpy(opts, "color"); /* fortunately this isn't longer */
- if (!match_optname(opts, "subkeyvalue", 11, TRUE)) /* allow multiple */
- duplicate_opt_detection(opts, 1); /* 1 means compound opts */
- /* special boolean options */
- if (match_optname(opts, "female", 3, FALSE)) {
- if(!initial && flags.female == negated)
- pline("That is not anatomically possible.");
- else
- flags.initgend = flags.female = !negated;
- return;
- }
- if (match_optname(opts, "male", 4, FALSE)) {
- if(!initial && flags.female != negated)
- pline("That is not anatomically possible.");
- else
- flags.initgend = flags.female = negated;
- return;
- }
- #if defined(MICRO) && !defined(AMIGA)
- /* included for compatibility with old NetHack.cnf files */
- if (match_optname(opts, "IBM_", 4, FALSE)) {
- iflags.BIOS = !negated;
- return;
- }
- #endif /* MICRO */
- /* compound options */
- fullname = "pettype";
- if (match_optname(opts, fullname, 3, TRUE)) {
- if ((op = string_for_env_opt(fullname, opts, negated)) != 0) {
- if (negated) bad_negation(fullname, TRUE);
- else switch (*op) {
- case 'd': /* dog */
- case 'D':
- preferred_pet = 'd';
- break;
- case 'c': /* cat */
- case 'C':
- case 'f': /* feline */
- case 'F':
- preferred_pet = 'c';
- break;
- case 'n': /* no pet */
- case 'N':
- preferred_pet = 'n';
- break;
- default:
- pline("Unrecognized pet type '%s'.", op);
- break;
- }
- } else if (negated) preferred_pet = 'n';
- return;
- }
- fullname = "ghoulname";
- if (match_optname(opts, fullname, 3, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- nmcpy(ghoulname, op, PL_PSIZ);
- return;
- }
- fullname = "wolfname";
- if (match_optname(opts, fullname, 3, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- nmcpy(wolfname, op, PL_PSIZ);
- return;
- }
- fullname = "catname";
- if (match_optname(opts, fullname, 3, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- nmcpy(catname, op, PL_PSIZ);
- return;
- }
- fullname = "dogname";
- if (match_optname(opts, fullname, 3, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- nmcpy(dogname, op, PL_PSIZ);
- return;
- }
- fullname = "horsename";
- if (match_optname(opts, fullname, 5, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- nmcpy(horsename, op, PL_PSIZ);
- return;
- }
- /* menucolor:"regex_string"=color */
- fullname = "menucolor";
- if (match_optname(opts, fullname, 9, TRUE)) {
- #ifdef MENU_COLOR
- if (negated) bad_negation(fullname, FALSE);
- else if ((op = string_for_env_opt(fullname, opts, FALSE)) != 0)
- if (!add_menu_coloring(op))
- badoption(opts);
- #endif
- return;
- }
- fullname = "number_pad";
- if (match_optname(opts, fullname, 10, TRUE)) {
- boolean compat = (strlen(opts) <= 10);
- op = string_for_opt(opts, (compat || !initial));
- if (!op) {
- if (compat || negated || initial) {
- /* for backwards compatibility, "number_pad" without a
- value is a synonym for number_pad:1 */
- iflags.num_pad = !negated;
- if (iflags.num_pad) iflags.num_pad_mode = 0;
- number_pad(iflags.num_pad);
- }
- return;
- }
- if (negated) {
- bad_negation("number_pad", TRUE);
- return;
- }
- if (*op == '1' || *op == '2') {
- iflags.num_pad = 1;
- if (*op == '2') iflags.num_pad_mode = 1;
- else iflags.num_pad_mode = 0;
- number_pad(1);
- } else if (*op == '0') {
- iflags.num_pad = 0;
- iflags.num_pad_mode = 0;
- number_pad(0);
- } else badoption(opts);
- return;
- }
- fullname = "runmode";
- if (match_optname(opts, fullname, 4, TRUE)) {
- if (negated) {
- iflags.runmode = RUN_TPORT;
- } else if ((op = string_for_opt(opts, FALSE)) != 0) {
- if (!strncmpi(op, "teleport", strlen(op)))
- iflags.runmode = RUN_TPORT;
- else if (!strncmpi(op, "run", strlen(op)))
- iflags.runmode = RUN_LEAP;
- else if (!strncmpi(op, "walk", strlen(op)))
- iflags.runmode = RUN_STEP;
- else if (!strncmpi(op, "crawl", strlen(op)))
- iflags.runmode = RUN_CRAWL;
- else
- badoption(opts);
- }
- return;
- }
- fullname = "msghistory";
- if (match_optname(opts, fullname, 3, TRUE)) {
- op = string_for_env_opt(fullname, opts, negated);
- if ((negated && !op) || (!negated && op)) {
- iflags.msg_history = negated ? 0 : atoi(op);
- } else if (negated) bad_negation(fullname, TRUE);
- return;
- }
- fullname="msg_window";
- /* msg_window:single, combo, full or reversed */
- if (match_optname(opts, fullname, 4, TRUE)) {
- /* allow option to be silently ignored by non-tty ports */
- #ifdef TTY_GRAPHICS
- int tmp;
- if (!(op = string_for_opt(opts, TRUE))) {
- tmp = negated ? 's' : 'f';
- } else {
- if (negated) {
- bad_negation(fullname, TRUE);
- return;
- }
- tmp = tolower(*op);
- }
- switch (tmp) {
- case 's': /* single message history cycle (default if negated) */
- iflags.prevmsg_window = 's';
- break;
- case 'c': /* combination: two singles, then full page reversed */
- iflags.prevmsg_window = 'c';
- break;
- case 'f': /* full page (default if no opts) */
- iflags.prevmsg_window = 'f';
- break;
- case 'r': /* full page (reversed) */
- iflags.prevmsg_window = 'r';
- break;
- default:
- badoption(opts);
- }
- #endif
- return;
- }
- /* WINCAP
- * setting font options */
- fullname = "font";
- if (!strncmpi(opts, fullname, 4))
- {
- int wintype = -1;
- char *fontopts = opts + 4;
- if (!strncmpi(fontopts, "map", 3) ||
- !strncmpi(fontopts, "_map", 4))
- wintype = NHW_MAP;
- else if (!strncmpi(fontopts, "message", 7) ||
- !strncmpi(fontopts, "_message", 8))
- wintype = NHW_MESSAGE;
- else if (!strncmpi(fontopts, "text", 4) ||
- !strncmpi(fontopts, "_text", 5))
- wintype = NHW_TEXT;
- else if (!strncmpi(fontopts, "menu", 4) ||
- !strncmpi(fontopts, "_menu", 5))
- wintype = NHW_MENU;
- else if (!strncmpi(fontopts, "status", 6) ||
- !strncmpi(fontopts, "_status", 7))
- wintype = NHW_STATUS;
- else if (!strncmpi(fontopts, "_size", 5)) {
- if (!strncmpi(fontopts, "_size_map", 8))
- wintype = NHW_MAP;
- else if (!strncmpi(fontopts, "_size_message", 12))
- wintype = NHW_MESSAGE;
- else if (!strncmpi(fontopts, "_size_text", 9))
- wintype = NHW_TEXT;
- else if (!strncmpi(fontopts, "_size_menu", 9))
- wintype = NHW_MENU;
- else if (!strncmpi(fontopts, "_size_status", 11))
- wintype = NHW_STATUS;
- else {
- badoption(opts);
- return;
- }
- if (wintype > 0 && !negated &&
- (op = string_for_opt(opts, FALSE)) != 0) {
- switch(wintype) {
- case NHW_MAP:
- iflags.wc_fontsiz_map = atoi(op);
- break;
- case NHW_MESSAGE:
- iflags.wc_fontsiz_message = atoi(op);
- break;
- case NHW_TEXT:
- iflags.wc_fontsiz_text = atoi(op);
- break;
- case NHW_MENU:
- iflags.wc_fontsiz_menu = atoi(op);
- break;
- case NHW_STATUS:
- iflags.wc_fontsiz_status = atoi(op);
- break;
- }
- }
- return;
- } else {
- badoption(opts);
- }
- if (wintype > 0 &&
- (op = string_for_opt(opts, FALSE)) != 0) {
- wc_set_font_name(wintype, op);
- #ifdef MAC
- set_font_name (wintype, op);
- #endif
- return;
- } else if (negated) bad_negation(fullname, TRUE);
- return;
- }
- #ifdef CHANGE_COLOR
- if (match_optname(opts, "palette", 3, TRUE)
- # ifdef MAC
- || match_optname(opts, "hicolor", 3, TRUE)
- # endif
- ) {
- int color_number, color_incr;
- # ifdef MAC
- if (match_optname(opts, "hicolor", 3, TRUE)) {
- if (negated) {
- bad_negation("hicolor", FALSE);
- return;
- }
- color_number = CLR_MAX + 4; /* HARDCODED inverse number */
- color_incr = -1;
- } else {
- # endif
- if (negated) {
- bad_negation("palette", FALSE);
- return;
- }
- color_number = 0;
- color_incr = 1;
- # ifdef MAC
- }
- # endif
- if ((op = string_for_opt(opts, FALSE)) != (char *)0) {
- char *pt = op;
- int cnt, tmp, reverse;
- long rgb;
- while (*pt && color_number >= 0) {
- cnt = 3;
- rgb = 0L;
- if (*pt == '-') {
- reverse = 1;
- pt++;
- } else {
- reverse = 0;
- }
- while (cnt-- > 0) {
- if (*pt && *pt != '/') {
- # ifdef AMIGA
- rgb <<= 4;
- # else
- rgb <<= 8;
- # endif
- tmp = *(pt++);
- if (isalpha(tmp)) {
- tmp = (tmp + 9) & 0xf; /* Assumes ASCII... */
- } else {
- tmp &= 0xf; /* Digits in ASCII too... */
- }
- # ifndef AMIGA
- /* Add an extra so we fill f -> ff and 0 -> 00 */
- rgb += tmp << 4;
- # endif
- rgb += tmp;
- }
- }
- if (*pt == '/') {
- pt++;
- }
- change_color(color_number, rgb, reverse);
- color_number += color_incr;
- }
- }
- if (!initial) {
- need_redraw = TRUE;
- }
- return;
- }
- #endif /* CHANGE_COLOR */
- if (match_optname(opts, "fruit", 2, TRUE)) {
- char empty_str = '\0';
- op = string_for_opt(opts, negated);
- if (negated) {
- if (op) {
- bad_negation("fruit", TRUE);
- return;
- }
- op = &empty_str;
- goto goodfruit;
- }
- if (!op) return;
- if (!initial) {
- struct fruit *f;
- num = 0;
- for(f=ffruit; f; f=f->nextf) {
- if (!strcmp(op, f->fname)) goto goodfruit;
- num++;
- }
- if (num >= 100) {
- pline("Doing that so many times isn't very fruitful.");
- return;
- }
- }
- goodfruit:
- nmcpy(pl_fruit, op, PL_FSIZ);
- /* OBJ_NAME(objects[SLIME_MOLD]) won't work after initialization */
- if (!*pl_fruit)
- nmcpy(pl_fruit, "slime mold", PL_FSIZ);
- if (!initial)
- (void)fruitadd(pl_fruit);
- /* If initial, then initoptions is allowed to do it instead
- * of here (initoptions always has to do it even if there's
- * no fruit option at all. Also, we don't want people
- * setting multiple fruits in their options.)
- */
- return;
- }
- /* graphics:string */
- fullname = "graphics";
- if (match_optname(opts, fullname, 2, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else graphics_opts(opts, fullname, MAXPCHARS, 0);
- return;
- }
- fullname = "dungeon";
- if (match_optname(opts, fullname, 2, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else graphics_opts(opts, fullname, MAXDCHARS, 0);
- return;
- }
- fullname = "traps";
- if (match_optname(opts, fullname, 2, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else graphics_opts(opts, fullname, MAXTCHARS, MAXDCHARS);
- return;
- }
- fullname = "effects";
- if (match_optname(opts, fullname, 2, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else
- graphics_opts(opts, fullname, MAXECHARS, MAXDCHARS+MAXTCHARS);
- return;
- }
- /* objects:string */
- fullname = "objects";
- if (match_optname(opts, fullname, 7, TRUE)) {
- int length;
- if (negated) {
- bad_negation(fullname, FALSE);
- return;
- }
- if (!(opts = string_for_env_opt(fullname, opts, FALSE)))
- return;
- escapes(opts, opts);
- /*
- * Override the default object class symbols. The first
- * object in the object class is the "random object". I
- * don't want to use 0 as an object class, so the "random
- * object" is basically a place holder.
- *
- * The object class symbols have already been initialized in
- * initoptions().
- */
- length = strlen(opts);
- if (length >= MAXOCLASSES)
- length = MAXOCLASSES-1; /* don't count RANDOM_OBJECT */
- for (i = 0; i < length; i++)
- oc_syms[i+1] = (uchar) opts[i];
- return;
- }
- /* monsters:string */
- fullname = "monsters";
- if (match_optname(opts, fullname, 8, TRUE)) {
- int length;
- if (negated) {
- bad_negation(fullname, FALSE);
- return;
- }
- if (!(opts = string_for_env_opt(fullname, opts, FALSE)))
- return;
- escapes(opts, opts);
- /* Override default mon class symbols set in initoptions(). */
- length = strlen(opts);
- if (length >= MAXMCLASSES)
- length = MAXMCLASSES-1; /* mon class 0 unused */
- for (i = 0; i < length; i++)
- monsyms[i+1] = (uchar) opts[i];
- return;
- }
- fullname = "warnings";
- if (match_optname(opts, fullname, 5, TRUE)) {
- if (negated) bad_negation(fullname, FALSE);
- else warning_opts(opts, fullname);
- return;
- }
- /* boulder:symbol */
- fullname = "boulder";
- if (match_optname(opts, ful…
Large files files are truncated, but you can click here to view the full file