1 //-------------------------------------------------------------------------
2 /*
3 Copyright (C) 2016 EDuke32 developers and contributors
4 
5 This file is part of EDuke32.
6 
7 EDuke32 is free software; you can redistribute it and/or
8 modify it under the terms of the GNU General Public License version 2
9 as published by the Free Software Foundation.
10 
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
14 
15 See the GNU 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, write to the Free Software
19 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
20 */
21 //-------------------------------------------------------------------------
22 
23 #include "duke3d.h"
24 #include "scriplib.h"
25 #include "osdcmds.h"
26 #include "renderlayer.h"
27 #include "cmdline.h"
28 
29 #ifdef __ANDROID__
30 # include "android.h"
31 #endif
32 
33 #if defined RENDERTYPESDL && defined SDL_TARGET && SDL_TARGET > 1
34 # include "sdl_inc.h"
35 #endif
36 
37 // we load this in to get default button and key assignments
38 // as well as setting up function mappings
39 
40 #define __SETUP__   // JBF 20031211
41 #include "_functio.h"
42 
43 hashtable_t h_gamefuncs    = { NUMGAMEFUNCTIONS<<1, NULL };
44 
CONFIG_FunctionNameToNum(const char * func)45 int32_t CONFIG_FunctionNameToNum(const char *func)
46 {
47     int32_t i;
48 
49     if (!func)
50         return -1;
51 
52     i = hash_find(&h_gamefuncs,func);
53 
54     if (i < 0)
55     {
56         char *str = Bstrtolower(Xstrdup(func));
57         i = hash_find(&h_gamefuncs,str);
58         Bfree(str);
59 
60         return i;
61     }
62 
63     return i;
64 }
65 
66 
CONFIG_FunctionNumToName(int32_t func)67 char *CONFIG_FunctionNumToName(int32_t func)
68 {
69     if ((unsigned)func >= (unsigned)NUMGAMEFUNCTIONS)
70         return NULL;
71     return gamefunctions[func];
72 }
73 
74 
CONFIG_AnalogNameToNum(const char * func)75 int32_t CONFIG_AnalogNameToNum(const char *func)
76 {
77     if (!func)
78         return -1;
79 
80     if (!Bstrcasecmp(func,"analog_turning"))
81     {
82         return analog_turning;
83     }
84     if (!Bstrcasecmp(func,"analog_strafing"))
85     {
86         return analog_strafing;
87     }
88     if (!Bstrcasecmp(func,"analog_moving"))
89     {
90         return analog_moving;
91     }
92     if (!Bstrcasecmp(func,"analog_lookingupanddown"))
93     {
94         return analog_lookingupanddown;
95     }
96 
97     return -1;
98 }
99 
100 
CONFIG_AnalogNumToName(int32_t func)101 const char *CONFIG_AnalogNumToName(int32_t func)
102 {
103     switch (func)
104     {
105     case analog_turning:
106         return "analog_turning";
107     case analog_strafing:
108         return "analog_strafing";
109     case analog_moving:
110         return "analog_moving";
111     case analog_lookingupanddown:
112         return "analog_lookingupanddown";
113     }
114 
115     return NULL;
116 }
117 
118 
CONFIG_SetDefaultKeys(const char (* keyptr)[MAXGAMEFUNCLEN],bool lazy)119 void CONFIG_SetDefaultKeys(const char (*keyptr)[MAXGAMEFUNCLEN], bool lazy/*=false*/)
120 {
121     static char const s_gamefunc_[] = "gamefunc_";
122     int constexpr strlen_gamefunc_  = ARRAY_SIZE(s_gamefunc_) - 1;
123 
124     if (!lazy)
125     {
126         Bmemset(ud.config.KeyboardKeys, 0xff, sizeof(ud.config.KeyboardKeys));
127         CONTROL_ClearAllBinds();
128     }
129 
130     for (int i=0; i < ARRAY_SSIZE(gamefunctions); ++i)
131     {
132         if (gamefunctions[i][0] == '\0')
133             continue;
134 
135         auto &key = ud.config.KeyboardKeys[i];
136 
137         int const default0 = KB_StringToScanCode(keyptr[i<<1]);
138         int const default1 = KB_StringToScanCode(keyptr[(i<<1)+1]);
139 
140         // skip the function if the default key is already used
141         // or the function is assigned to another key
142         if (lazy && (key[0] != 0xff || (CONTROL_KeyIsBound(default0) && Bstrlen(CONTROL_KeyBinds[default0].cmdstr) > strlen_gamefunc_
143                         && CONFIG_FunctionNameToNum(CONTROL_KeyBinds[default0].cmdstr + strlen_gamefunc_) >= 0)))
144         {
145 #if 0 // defined(DEBUGGINGAIDS)
146             if (key[0] != 0xff)
147                 initprintf("Skipping %s bound to %s\n", keyptr[i<<1], CONTROL_KeyBinds[default0].cmdstr);
148 #endif
149             continue;
150         }
151 
152         key[0] = default0;
153         key[1] = default1;
154 
155         if (key[0])
156             CONTROL_FreeKeyBind(key[0]);
157 
158         if (key[1])
159             CONTROL_FreeKeyBind(key[1]);
160 
161         if (i == gamefunc_Show_Console)
162             OSD_CaptureKey(key[0]);
163         else
164             CONFIG_MapKey(i, key[0], 0, key[1], 0);
165     }
166 }
167 
168 
CONFIG_SetDefaults(void)169 void CONFIG_SetDefaults(void)
170 {
171     // JBF 20031211
172     int32_t i;
173 
174     ud.config.scripthandle = -1;
175 #ifdef __ANDROID__
176     droidinput.forward_sens = 5.f;
177     droidinput.strafe_sens = 5.f;
178     droidinput.pitch_sens = 5.f;
179     droidinput.yaw_sens = 5.f;
180     droidinput.hideStick = 0;
181     droidinput.gameControlsAlpha = 0.5;
182     droidinput.toggleCrouch = 1;
183     droidinput.quickSelectWeapon = 1;
184 
185     ud.setup.xdim = droidinfo.screen_width;
186     ud.setup.ydim = droidinfo.screen_height;
187 #else
188 # if defined RENDERTYPESDL && SDL_MAJOR_VERSION > 1
189     uint32_t inited = SDL_WasInit(SDL_INIT_VIDEO);
190     if (inited == 0)
191         SDL_Init(SDL_INIT_VIDEO);
192     else if (!(inited & SDL_INIT_VIDEO))
193         SDL_InitSubSystem(SDL_INIT_VIDEO);
194 
195     SDL_DisplayMode dm;
196     if (SDL_GetDesktopDisplayMode(0, &dm) == 0)
197     {
198         ud.setup.xdim = dm.w;
199         ud.setup.ydim = dm.h;
200     }
201     else
202 # endif
203     {
204         ud.setup.xdim = 1024;
205         ud.setup.ydim = 768;
206     }
207 #endif
208 
209 #ifdef USE_OPENGL
210     ud.setup.bpp = 32;
211 #else
212     ud.setup.bpp = 8;
213 #endif
214     ud.config.useprecache = 1;
215     ud.config.AmbienceToggle = 1;
216     ud.config.AutoAim = 1;
217     ud.config.FXVolume = 255;
218 #if defined(_WIN32)
219     ud.config.MixRate = 44100;
220 #elif defined __ANDROID__
221     ud.config.MixRate = droidinfo.audio_sample_rate;
222 #else
223     ud.config.MixRate = 48000;
224 #endif
225     ud.config.MouseBias = 0;
226     ud.config.MouseDeadZone = 0;
227     ud.config.MusicToggle = 1;
228     ud.config.MusicVolume = 195;
229     ud.config.MusicDevice = ASS_AutoDetect;
230     g_myAimMode = g_player[0].ps->aim_mode = 1;
231     ud.config.NumBits = 16;
232     ud.config.NumChannels = 2;
233 #if defined GEKKO || defined __OPENDINGUX__
234     ud.config.NumVoices = 32;
235 #else
236     ud.config.NumVoices = 64;
237 #endif
238 #ifdef ASS_REVERSESTEREO
239     ud.config.ReverseStereo = 0;
240 #endif
241     ud.auto_run = 1;
242     ud.config.ShowOpponentWeapons = 0;
243     ud.config.SoundToggle = 1;
244     ud.althud = 1;
245     ud.automsg = 0;
246     ud.autovote = 0;
247     ud.brightness = 8;
248     ud.camerasprite = -1;
249 
250 #if defined GEKKO || defined __OPENDINGUX__
251     ud.camera_time = 11;
252 #elif defined(__ANDROID__)
253     ud.camera_time = 7;
254 #else
255     ud.camera_time = 4;
256 #endif
257 
258     ud.color = 0;
259     ud.crosshair = 1;
260     ud.crosshairscale = 50;
261     ud.obituaries = 1;
262     ud.democams = 1;
263     ud.detail = 0;
264     ud.drawweapon = 1;
265     ud.idplayers = 1;
266     ud.levelstats = 0;
267     ud.lockout = 0;
268     ud.m_ffire = 1;
269     ud.m_marker = 1;
270     ud.menu_slidebarz = 65536;
271     ud.menu_slidebarmargin = RR ? 6 * 65536 : 65536;
272     ud.menu_slidecursorz = RR ? 32768 : 65536;
273     ud.mouseaiming = 0;
274     ud.mouseflip = 1;
275     ud.msgdisptime = 120;
276     ud.pwlockout[0] = '\0';
277     ud.runkey_mode = 0;
278     ud.screen_size = 4;
279     ud.screen_tilting = 1;
280     ud.shadows = 1;
281     ud.statusbarflags = STATUSBAR_NOSHRINK;
282     ud.statusbarmode = 1;
283     ud.statusbarscale = 100;
284     ud.team = 0;
285     ud.viewbob = 1;
286     ud.weaponsway = 1;
287     ud.weaponswitch = 3;	// new+empty
288     ud.angleinterpolation = 0;
289 #ifdef GEKKO
290     ud.setup.usejoystick = 1;
291 #else
292     ud.setup.usejoystick = 0;
293 #endif
294 
295     ud.setup.forcesetup = 1;
296     ud.setup.noautoload = 1;
297     ud.setup.fullscreen = 1;
298     ud.setup.usemouse = 1;
299 
300     ud.config.VoiceToggle = 5; // bitfield, 1 = local, 2 = dummy, 4 = other players in DM
301     ud.display_bonus_screen = 1;
302     ud.show_level_text = 1;
303     ud.configversion = 0;
304     ud.weaponscale = 100;
305     ud.textscale = 200;
306     ud.screenfade = 1;
307     ud.menubackground = 1;
308     ud.hudontop = 0;
309     ud.default_skill = 1;
310     ud.slidebar_paldisabled = 1;
311     ud.shadow_pal = 4;
312     ud.menu_scrollbartilenum = -1;
313     ud.menu_scrollbarz = 65536;
314     ud.menu_scrollcursorz = 65536;
315     ud.autosave = 1;
316     ud.autosavedeletion = 1;
317     ud.maxautosaves = 5;
318     ud.fov = 90;
319 
320     ud.config.CheckForUpdates = 1;
321 
322     Bstrcpy(ud.rtsname, G_DefaultRtsFile());
323 
324     Bstrcpy(szPlayerName, "Player");
325 
326     //if (RR)
327     //{
328     //    Bstrcpy(ud.ridecule[0], "Yer as ugly as a mud fence!");
329     //    Bstrcpy(ud.ridecule[1], "Duck you pecker-head!");
330     //    Bstrcpy(ud.ridecule[2], "You like that boy?");
331     //    Bstrcpy(ud.ridecule[3], "Yer lower than catfish crap!");
332     //    Bstrcpy(ud.ridecule[4], "Eat lead, you shit monkey!");
333     //    Bstrcpy(ud.ridecule[5], "You dumb-ass!");
334     //    Bstrcpy(ud.ridecule[6], "Yer slower'n a three legged dog!");
335     //    Bstrcpy(ud.ridecule[7], "Come on...Squeal like a pig!");
336     //    Bstrcpy(ud.ridecule[8], "Haw, haw, haw!");
337     //    Bstrcpy(ud.ridecule[9], "Now you gone and done it!");
338     //}
339     //else
340     //{
341         Bstrcpy(ud.ridecule[0], "An inspiration for birth control.");
342         Bstrcpy(ud.ridecule[1], "You're gonna die for that!");
343         Bstrcpy(ud.ridecule[2], "It hurts to be you.");
344         Bstrcpy(ud.ridecule[3], "Lucky son of a bitch.");
345         Bstrcpy(ud.ridecule[4], "Hmmm... payback time.");
346         Bstrcpy(ud.ridecule[5], "You bottom dwelling scum sucker.");
347         Bstrcpy(ud.ridecule[6], "Damn, you're ugly.");
348         Bstrcpy(ud.ridecule[7], "Ha ha ha... wasted!");
349         Bstrcpy(ud.ridecule[8], "You suck!");
350         Bstrcpy(ud.ridecule[9], "AARRRGHHHHH!!!");
351     //}
352 
353     // JBF 20031211
354 
355     if (RR)
356     {
357         Bstrcpy((char*)keydefaults[gamefunc_Holo_Duke<<1], "B");
358         Bstrcpy((char*)keydefaults[gamefunc_Jetpack<<1], "C");
359         Bstrcpy((char*)keydefaults[gamefunc_NightVision<<1], "Y");
360         Bstrcpy((char*)keydefaults[gamefunc_MedKit<<1], "R");
361         Bstrcpy((char*)keydefaults[gamefunc_Steroids<<1], "M");
362         Bstrcpy((char*)keydefaults[gamefunc_Show_Opponents_Weapon<<1], "V");
363 
364         Bstrcpy((char*)oldkeydefaults[gamefunc_Holo_Duke<<1], "B");
365         Bstrcpy((char*)oldkeydefaults[gamefunc_Jetpack<<1], "C");
366         Bstrcpy((char*)oldkeydefaults[gamefunc_NightVision<<1], "Y");
367         Bstrcpy((char*)oldkeydefaults[gamefunc_MedKit<<1], "W");
368         Bstrcpy((char*)oldkeydefaults[gamefunc_Steroids<<1], "M");
369         Bstrcpy((char*)oldkeydefaults[gamefunc_Show_Opponents_Weapon<<1], "E");
370         Bstrcpy((char*)oldkeydefaults[gamefunc_Show_Console<<1], "V");
371     }
372 
373     CONFIG_SetDefaultKeys(keydefaults);
374 
375     memset(ud.config.MouseFunctions, -1, sizeof(ud.config.MouseFunctions));
376     for (i=0; i<MAXMOUSEBUTTONS; i++)
377     {
378         ud.config.MouseFunctions[i][0] = CONFIG_FunctionNameToNum(mousedefaults[i]);
379         CONTROL_MapButton(ud.config.MouseFunctions[i][0], i, 0, controldevice_mouse);
380         if (i>=4) continue;
381         ud.config.MouseFunctions[i][1] = CONFIG_FunctionNameToNum(mouseclickeddefaults[i]);
382         CONTROL_MapButton(ud.config.MouseFunctions[i][1], i, 1, controldevice_mouse);
383     }
384 
385     for (i=0; i<MAXMOUSEAXES; i++)
386     {
387         CONTROL_SetAnalogAxisScale(i, DEFAULTMOUSEANALOGUESCALE, controldevice_mouse);
388 
389         ud.config.MouseAnalogueAxes[i] = CONFIG_AnalogNameToNum(mouseanalogdefaults[i]);
390         CONTROL_MapAnalogAxis(i, ud.config.MouseAnalogueAxes[i], controldevice_mouse);
391     }
392     CONTROL_MouseSensitivity = DEFAULTMOUSESENSITIVITY;
393 
394     memset(ud.config.JoystickFunctions, -1, sizeof(ud.config.JoystickFunctions));
395     for (i=0; i<MAXJOYBUTTONSANDHATS; i++)
396     {
397         ud.config.JoystickFunctions[i][0] = CONFIG_FunctionNameToNum(joystickdefaults[i]);
398         ud.config.JoystickFunctions[i][1] = CONFIG_FunctionNameToNum(joystickclickeddefaults[i]);
399         CONTROL_MapButton(ud.config.JoystickFunctions[i][0], i, 0, controldevice_joystick);
400         CONTROL_MapButton(ud.config.JoystickFunctions[i][1], i, 1, controldevice_joystick);
401     }
402 
403     memset(ud.config.JoystickDigitalFunctions, -1, sizeof(ud.config.JoystickDigitalFunctions));
404     for (i=0; i<MAXJOYAXES; i++)
405     {
406         ud.config.JoystickAnalogueScale[i] = DEFAULTJOYSTICKANALOGUESCALE;
407         ud.config.JoystickAnalogueDead[i] = DEFAULTJOYSTICKANALOGUEDEAD;
408         ud.config.JoystickAnalogueSaturate[i] = DEFAULTJOYSTICKANALOGUESATURATE;
409         CONTROL_SetAnalogAxisScale(i, ud.config.JoystickAnalogueScale[i], controldevice_joystick);
410 
411         ud.config.JoystickDigitalFunctions[i][0] = CONFIG_FunctionNameToNum(joystickdigitaldefaults[i*2]);
412         ud.config.JoystickDigitalFunctions[i][1] = CONFIG_FunctionNameToNum(joystickdigitaldefaults[i*2+1]);
413         CONTROL_MapDigitalAxis(i, ud.config.JoystickDigitalFunctions[i][0], 0, controldevice_joystick);
414         CONTROL_MapDigitalAxis(i, ud.config.JoystickDigitalFunctions[i][1], 1, controldevice_joystick);
415 
416         ud.config.JoystickAnalogueAxes[i] = CONFIG_AnalogNameToNum(joystickanalogdefaults[i]);
417         CONTROL_MapAnalogAxis(i, ud.config.JoystickAnalogueAxes[i], controldevice_joystick);
418     }
419 }
420 
421 
422 // wrapper for CONTROL_MapKey(), generates key bindings to reflect changes to keyboard setup
CONFIG_MapKey(int which,kb_scancode key1,kb_scancode oldkey1,kb_scancode key2,kb_scancode oldkey2)423 void CONFIG_MapKey(int which, kb_scancode key1, kb_scancode oldkey1, kb_scancode key2, kb_scancode oldkey2)
424 {
425     int const keys[] = { key1, key2, oldkey1, oldkey2 };
426     char buf[2*MAXGAMEFUNCLEN];
427 
428     if (which == gamefunc_Show_Console)
429         OSD_CaptureKey(key1);
430 
431     for (int k = 0; (unsigned)k < ARRAY_SIZE(keys); k++)
432     {
433         if (keys[k] == 0xff || !keys[k])
434             continue;
435 
436         int match = 0;
437 
438         for (; sctokeylut[match].key; match++)
439         {
440             if (keys[k] == sctokeylut[match].sc)
441                 break;
442         }
443 
444         tempbuf[0] = 0;
445 
446         for (int i=NUMGAMEFUNCTIONS-1; i>=0; i--)
447         {
448             if (ud.config.KeyboardKeys[i][0] == keys[k] || ud.config.KeyboardKeys[i][1] == keys[k])
449             {
450                 Bsprintf(buf, "gamefunc_%s; ", CONFIG_FunctionNumToName(i));
451                 Bstrcat(tempbuf,buf);
452             }
453         }
454 
455         int const len = Bstrlen(tempbuf);
456 
457         if (len >= 2)
458         {
459             tempbuf[len-2] = 0;  // cut off the trailing "; "
460             CONTROL_BindKey(keys[k], tempbuf, 1, sctokeylut[match].key ? sctokeylut[match].key : "<?>");
461         }
462         else
463         {
464             CONTROL_FreeKeyBind(keys[k]);
465         }
466     }
467 }
468 
469 
CONFIG_SetupMouse(void)470 void CONFIG_SetupMouse(void)
471 {
472     int32_t i;
473     char str[80];
474     char temp[80];
475     int32_t scale;
476 
477     if (ud.config.scripthandle < 0) return;
478 
479     for (i=0; i<MAXMOUSEBUTTONS; i++)
480     {
481         Bsprintf(str,"MouseButton%d",i);
482         temp[0] = 0;
483         if (!SCRIPT_GetString(ud.config.scripthandle,"Controls", str,temp))
484             ud.config.MouseFunctions[i][0] = CONFIG_FunctionNameToNum(temp);
485 
486         Bsprintf(str,"MouseButtonClicked%d",i);
487         temp[0] = 0;
488         if (!SCRIPT_GetString(ud.config.scripthandle,"Controls", str,temp))
489             ud.config.MouseFunctions[i][1] = CONFIG_FunctionNameToNum(temp);
490     }
491 
492     // map over the axes
493     for (i=0; i<MAXMOUSEAXES; i++)
494     {
495         Bsprintf(str,"MouseAnalogAxes%d",i);
496         temp[0] = 0;
497         if (!SCRIPT_GetString(ud.config.scripthandle, "Controls", str,temp))
498             if (CONFIG_AnalogNameToNum(temp) != -1 || (!temp[0] && CONFIG_FunctionNameToNum(temp) != -1))
499                 ud.config.MouseAnalogueAxes[i] = CONFIG_AnalogNameToNum(temp);
500     }
501 
502     {
503         tempbuf[0] = 0;
504         SCRIPT_GetString(ud.config.scripthandle, "Controls","Mouse_Sensitivity",&tempbuf[0]);
505         if (tempbuf[0]) CONTROL_MouseSensitivity = atof(tempbuf);
506     }
507 
508     for (i=0; i<MAXMOUSEBUTTONS; i++)
509     {
510         CONTROL_MapButton(ud.config.MouseFunctions[i][0], i, 0, controldevice_mouse);
511         CONTROL_MapButton(ud.config.MouseFunctions[i][1], i, 1,  controldevice_mouse);
512     }
513     for (i=0; i<MAXMOUSEAXES; i++)
514         CONTROL_MapAnalogAxis(i, ud.config.MouseAnalogueAxes[i], controldevice_mouse);
515 }
516 
517 
CONFIG_SetupJoystick(void)518 void CONFIG_SetupJoystick(void)
519 {
520     int32_t i;
521     char str[80];
522     char temp[80];
523     int32_t scale;
524 
525     if (ud.config.scripthandle < 0) return;
526 
527     for (i=0; i<MAXJOYBUTTONSANDHATS; i++)
528     {
529         Bsprintf(str,"JoystickButton%d",i);
530         temp[0] = 0;
531         if (!SCRIPT_GetString(ud.config.scripthandle,"Controls", str,temp))
532             ud.config.JoystickFunctions[i][0] = CONFIG_FunctionNameToNum(temp);
533 
534         Bsprintf(str,"JoystickButtonClicked%d",i);
535         temp[0] = 0;
536         if (!SCRIPT_GetString(ud.config.scripthandle,"Controls", str,temp))
537             ud.config.JoystickFunctions[i][1] = CONFIG_FunctionNameToNum(temp);
538     }
539 
540     // map over the axes
541     for (i=0; i<MAXJOYAXES; i++)
542     {
543         Bsprintf(str,"JoystickAnalogAxes%d",i);
544         temp[0] = 0;
545         if (!SCRIPT_GetString(ud.config.scripthandle, "Controls", str,temp))
546             if (CONFIG_AnalogNameToNum(temp) != -1 || (!temp[0] && CONFIG_FunctionNameToNum(temp) != -1))
547                 ud.config.JoystickAnalogueAxes[i] = CONFIG_AnalogNameToNum(temp);
548 
549         Bsprintf(str,"JoystickDigitalAxes%d_0",i);
550         temp[0] = 0;
551         if (!SCRIPT_GetString(ud.config.scripthandle, "Controls", str,temp))
552             if (CONFIG_FunctionNameToNum(temp) != -1 || (!temp[0] && CONFIG_FunctionNameToNum(temp) != -1))
553                 ud.config.JoystickDigitalFunctions[i][0] = CONFIG_FunctionNameToNum(temp);
554 
555         Bsprintf(str,"JoystickDigitalAxes%d_1",i);
556         temp[0] = 0;
557         if (!SCRIPT_GetString(ud.config.scripthandle, "Controls", str,temp))
558             if (CONFIG_FunctionNameToNum(temp) != -1 || (!temp[0] && CONFIG_FunctionNameToNum(temp) != -1))
559                 ud.config.JoystickDigitalFunctions[i][1] = CONFIG_FunctionNameToNum(temp);
560 
561         Bsprintf(str,"JoystickAnalogScale%d",i);
562         scale = ud.config.JoystickAnalogueScale[i];
563         SCRIPT_GetNumber(ud.config.scripthandle, "Controls", str,&scale);
564         ud.config.JoystickAnalogueScale[i] = scale;
565 
566         Bsprintf(str,"JoystickAnalogDead%d",i);
567         scale = ud.config.JoystickAnalogueDead[i];
568         SCRIPT_GetNumber(ud.config.scripthandle, "Controls", str,&scale);
569         ud.config.JoystickAnalogueDead[i] = scale;
570 
571         Bsprintf(str,"JoystickAnalogSaturate%d",i);
572         scale = ud.config.JoystickAnalogueSaturate[i];
573         SCRIPT_GetNumber(ud.config.scripthandle, "Controls", str,&scale);
574         ud.config.JoystickAnalogueSaturate[i] = scale;
575     }
576 
577     for (i=0; i<MAXJOYBUTTONSANDHATS; i++)
578     {
579         CONTROL_MapButton(ud.config.JoystickFunctions[i][0], i, 0, controldevice_joystick);
580         CONTROL_MapButton(ud.config.JoystickFunctions[i][1], i, 1,  controldevice_joystick);
581     }
582     for (i=0; i<MAXJOYAXES; i++)
583     {
584         CONTROL_MapAnalogAxis(i, ud.config.JoystickAnalogueAxes[i], controldevice_joystick);
585         CONTROL_MapDigitalAxis(i, ud.config.JoystickDigitalFunctions[i][0], 0, controldevice_joystick);
586         CONTROL_MapDigitalAxis(i, ud.config.JoystickDigitalFunctions[i][1], 1, controldevice_joystick);
587         CONTROL_SetAnalogAxisScale(i, ud.config.JoystickAnalogueScale[i], controldevice_joystick);
588     }
589 }
590 
591 
CONFIG_ReadSetup(void)592 int32_t CONFIG_ReadSetup(void)
593 {
594     int32_t dummy;
595     char commmacro[] = "CommbatMacro# ";
596     char tempbuf[1024];
597 
598     CONTROL_ClearAssignments();
599     CONFIG_SetDefaults();
600 
601     ud.config.setupread = 1;
602     pathsearchmode = 1;
603 
604     if (ud.config.scripthandle < 0)
605     {
606         if (buildvfs_exists(g_setupFileName))  // JBF 20031211
607             ud.config.scripthandle = SCRIPT_Load(g_setupFileName);
608 #if !defined(EDUKE32_TOUCH_DEVICES)
609         else if (buildvfs_exists(SETUPFILENAME))
610         {
611             int32_t i;
612             i=wm_ynbox("Import Configuration Settings", "The configuration file \"%s\" was not found. "
613                 "Import configuration data from \"%s\"?", g_setupFileName, SETUPFILENAME);
614             if (i) ud.config.scripthandle = SCRIPT_Load(SETUPFILENAME);
615         }
616         else if (buildvfs_exists("duke3d.cfg"))
617         {
618             int32_t i;
619             i=wm_ynbox("Import Configuration Settings", "The configuration file \"%s\" was not found. "
620                 "Import configuration data from \"duke3d.cfg\"?", g_setupFileName);
621             if (i) ud.config.scripthandle = SCRIPT_Load("duke3d.cfg");
622         }
623 #endif
624     }
625 
626     pathsearchmode = 0;
627 
628     if (ud.config.scripthandle < 0)
629         return -1;
630 
631     for (dummy = 0; dummy < MAXRIDECULE; dummy++)
632     {
633         commmacro[13] = dummy+'0';
634         SCRIPT_GetString(ud.config.scripthandle, "Comm Setup",commmacro,&ud.ridecule[dummy][0]);
635     }
636 
637     Bmemset(tempbuf, 0, sizeof(tempbuf));
638     SCRIPT_GetString(ud.config.scripthandle, "Comm Setup","PlayerName",&tempbuf[0]);
639 
640     char nameBuf[64];
641 
642     while (Bstrlen(OSD_StripColors(nameBuf, tempbuf)) > 10)
643         tempbuf[Bstrlen(tempbuf) - 1] = '\0';
644 
645     Bstrncpyz(szPlayerName, tempbuf, sizeof(szPlayerName));
646 
647     SCRIPT_GetString(ud.config.scripthandle, "Comm Setup","RTSName",&ud.rtsname[0]);
648 
649     SCRIPT_GetNumber(ud.config.scripthandle, "Setup", "ConfigVersion", &ud.configversion);
650     SCRIPT_GetNumber(ud.config.scripthandle, "Setup", "ForceSetup", &ud.setup.forcesetup);
651     SCRIPT_GetNumber(ud.config.scripthandle, "Setup", "NoAutoLoad", &ud.setup.noautoload);
652     SCRIPT_GetNumber(ud.config.scripthandle, "Setup", "CacheSize", &dummy);
653 
654     if (dummy > MAXCACHE1DSIZE)
655         MAXCACHE1DSIZE = dummy;
656 
657     if (g_noSetup == 0 && g_modDir[0] == '/')
658     {
659         struct Bstat st;
660         SCRIPT_GetString(ud.config.scripthandle, "Setup","ModDir",&g_modDir[0]);
661 
662         if (Bstat(g_modDir, &st))
663         {
664             if ((st.st_mode & S_IFDIR) != S_IFDIR)
665             {
666                 initprintf("Invalid mod dir in cfg!\n");
667                 Bsprintf(g_modDir,"/");
668             }
669         }
670     }
671 
672     if (g_grpNamePtr == NULL && g_addonNum == 0)
673     {
674         SCRIPT_GetStringPtr(ud.config.scripthandle, "Setup","SelectedGRP",&g_grpNamePtr);
675         if (g_grpNamePtr && !Bstrlen(g_grpNamePtr))
676             g_grpNamePtr = dup_filename(G_DefaultGrpFile());
677     }
678 
679     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "Out",&ud.lockout);
680     SCRIPT_GetString(ud.config.scripthandle, "Screen Setup","Password",&ud.pwlockout[0]);
681 
682     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "ScreenHeight", &ud.setup.ydim);
683     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "ScreenMode", &ud.setup.fullscreen);
684     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "ScreenWidth", &ud.setup.xdim);
685 
686     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "WindowPositioning", (int32_t *)&windowpos);
687 
688     windowx = -1;
689     windowy = -1;
690     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "WindowPosX", (int32_t *)&windowx);
691     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "WindowPosY", (int32_t *)&windowy);
692 
693     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "MaxRefreshFreq", (int32_t *)&maxrefreshfreq);
694     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "ScreenBPP", &ud.setup.bpp);
695 
696     if (ud.setup.bpp < 8) ud.setup.bpp = 32;
697 
698 #ifdef POLYMER
699     int32_t rendmode = 0;
700     SCRIPT_GetNumber(ud.config.scripthandle, "Screen Setup", "Polymer", &rendmode);
701     glrendmode = (rendmode > 0) ? REND_POLYMER : REND_POLYMOST;
702 #endif
703 
704     SCRIPT_GetNumber(ud.config.scripthandle, "Misc", "Executions",&ud.executions);
705 
706 #ifdef _WIN32
707     SCRIPT_GetNumber(ud.config.scripthandle, "Updates", "CheckForUpdates", &ud.config.CheckForUpdates);
708     SCRIPT_GetNumber(ud.config.scripthandle, "Updates", "LastUpdateCheck", &ud.config.LastUpdateCheck);
709 #endif
710 
711     ud.config.setupread = 1;
712     return 0;
713 }
714 
715 
CONFIG_WriteSettings(void)716 void CONFIG_WriteSettings(void) // save binds and aliases to <cfgname>_settings.cfg
717 {
718     int32_t i;
719     BFILE *fp;
720     char *ptr = Xstrdup(g_setupFileName);
721     char tempbuf[128];
722 
723     if (!Bstrcmp(g_setupFileName, SETUPFILENAME))
724         Bsprintf(tempbuf, "settings.cfg");
725     else Bsprintf(tempbuf, "%s_settings.cfg", strtok(ptr, "."));
726 
727     fp = Bfopen(tempbuf, "wt");
728 
729     if (fp)
730     {
731         Bfprintf(fp,"// this file is automatically generated by %s\n", AppProperName);
732         Bfprintf(fp,"unbindall\n");
733 
734         for (i=0; i<MAXBOUNDKEYS+MAXMOUSEBUTTONS; i++)
735             if (CONTROL_KeyIsBound(i))
736                 Bfprintf(fp,"bind \"%s\"%s \"%s\"\n",CONTROL_KeyBinds[i].key,
737                 CONTROL_KeyBinds[i].repeat?"":" norepeat",CONTROL_KeyBinds[i].cmdstr);
738 
739         for (int i=0; i<NUMGAMEFUNCTIONS; ++i)
740         {
741             char const * name = CONFIG_FunctionNumToName(i);
742             if (name && name[0] != '\0' && (ud.config.KeyboardKeys[i][0] == 0xff || !ud.config.KeyboardKeys[i][0]))
743             {
744                 Bfprintf(fp,"unbound %s\n", name);
745             }
746         }
747 
748         OSD_WriteAliases(fp);
749 
750         if (g_crosshairSum != -1 && g_crosshairSum != DefaultCrosshairColors.r+(DefaultCrosshairColors.g<<8)+(DefaultCrosshairColors.b<<16))
751             Bfprintf(fp, "crosshaircolor %d %d %d\n", CrosshairColors.r, CrosshairColors.g, CrosshairColors.b);
752 
753         OSD_WriteCvars(fp);
754 
755         Bfclose(fp);
756 
757         if (!Bstrcmp(g_setupFileName, SETUPFILENAME))
758             OSD_Printf("Wrote settings.cfg\n");
759         else OSD_Printf("Wrote %s_settings.cfg\n",ptr);
760 
761         Bfree(ptr);
762         return;
763     }
764 
765     if (!Bstrcmp(g_setupFileName, SETUPFILENAME))
766         OSD_Printf("Error writing settings.cfg: %s\n", strerror(errno));
767     else OSD_Printf("Error writing %s_settings.cfg: %s\n",ptr,strerror(errno));
768 
769     Bfree(ptr);
770 }
771 
CONFIG_WriteSetup(uint32_t flags)772 void CONFIG_WriteSetup(uint32_t flags)
773 {
774     int32_t dummy;
775 
776     if (!ud.config.setupread) return;
777 
778     if (ud.config.scripthandle < 0)
779         ud.config.scripthandle = SCRIPT_Init(g_setupFileName);
780 
781     SCRIPT_PutNumber(ud.config.scripthandle, "Misc", "Executions",ud.executions,FALSE,FALSE);
782 
783     SCRIPT_PutNumber(ud.config.scripthandle, "Setup","ConfigVersion",BYTEVERSION_EDUKE32,FALSE,FALSE);
784     SCRIPT_PutNumber(ud.config.scripthandle, "Setup", "ForceSetup", ud.setup.forcesetup, FALSE, FALSE);
785     SCRIPT_PutNumber(ud.config.scripthandle, "Setup", "NoAutoLoad", ud.setup.noautoload, FALSE, FALSE);
786     SCRIPT_PutNumber(ud.config.scripthandle, "Setup", "CacheSize", MAXCACHE1DSIZE, FALSE, FALSE);
787 
788 #ifdef POLYMER
789     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "Polymer",glrendmode == REND_POLYMER,FALSE,FALSE);
790 #endif
791 
792     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "ScreenBPP", ud.setup.bpp, FALSE, FALSE);
793     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "ScreenHeight", ud.setup.ydim, FALSE, FALSE);
794     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "ScreenMode", ud.setup.fullscreen, FALSE, FALSE);
795     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "ScreenWidth", ud.setup.xdim, FALSE, FALSE);
796 
797     if (g_grpNamePtr && !g_addonNum)
798         SCRIPT_PutString(ud.config.scripthandle, "Setup","SelectedGRP",g_grpNamePtr);
799 
800 #ifdef STARTUP_SETUP_WINDOW
801     if (g_noSetup == 0)
802         SCRIPT_PutString(ud.config.scripthandle, "Setup","ModDir",&g_modDir[0]);
803 #endif
804     // exit early after only updating the values that can be changed from the startup window
805     if (flags & 1)
806     {
807         SCRIPT_Save(ud.config.scripthandle, g_setupFileName);
808         SCRIPT_Free(ud.config.scripthandle);
809         return;
810     }
811 
812     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "WindowPositioning", windowpos, FALSE, FALSE);
813     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "WindowPosX", windowx, FALSE, FALSE);
814     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "WindowPosY", windowy, FALSE, FALSE);
815     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "MaxRefreshFreq", maxrefreshfreq, FALSE, FALSE);
816 
817     SCRIPT_PutNumber(ud.config.scripthandle, "Screen Setup", "Out",ud.lockout,FALSE,FALSE);
818     SCRIPT_PutString(ud.config.scripthandle, "Screen Setup", "Password",ud.pwlockout);
819 
820 #ifdef _WIN32
821     SCRIPT_PutNumber(ud.config.scripthandle, "Updates", "CheckForUpdates", ud.config.CheckForUpdates, FALSE, FALSE);
822     SCRIPT_PutNumber(ud.config.scripthandle, "Updates", "LastUpdateCheck", ud.config.LastUpdateCheck, FALSE, FALSE);
823 #endif
824 
825     if (ud.setup.usemouse)
826     {
827         for (dummy=0; dummy<MAXMOUSEBUTTONS; dummy++)
828         {
829             if (CONFIG_FunctionNumToName(ud.config.MouseFunctions[dummy][0]))
830             {
831                 Bsprintf(buf, "MouseButton%d", dummy);
832                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.MouseFunctions[dummy][0]));
833             }
834 
835             if (dummy >= (MAXMOUSEBUTTONS-2)) continue;
836 
837             if (CONFIG_FunctionNumToName(ud.config.MouseFunctions[dummy][1]))
838             {
839                 Bsprintf(buf, "MouseButtonClicked%d", dummy);
840                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.MouseFunctions[dummy][1]));
841             }
842         }
843 
844         for (dummy=0; dummy<MAXMOUSEAXES; dummy++)
845         {
846             if (CONFIG_AnalogNumToName(ud.config.MouseAnalogueAxes[dummy]))
847             {
848                 Bsprintf(buf, "MouseAnalogAxes%d", dummy);
849                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_AnalogNumToName(ud.config.MouseAnalogueAxes[dummy]));
850             }
851         }
852     }
853 
854     if (ud.setup.usejoystick)
855     {
856         for (dummy=0; dummy<MAXJOYBUTTONSANDHATS; dummy++)
857         {
858             if (CONFIG_FunctionNumToName(ud.config.JoystickFunctions[dummy][0]))
859             {
860                 Bsprintf(buf, "JoystickButton%d", dummy);
861                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.JoystickFunctions[dummy][0]));
862             }
863 
864             if (CONFIG_FunctionNumToName(ud.config.JoystickFunctions[dummy][1]))
865             {
866                 Bsprintf(buf, "JoystickButtonClicked%d", dummy);
867                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.JoystickFunctions[dummy][1]));
868             }
869         }
870         for (dummy=0; dummy<MAXJOYAXES; dummy++)
871         {
872             if (CONFIG_AnalogNumToName(ud.config.JoystickAnalogueAxes[dummy]))
873             {
874                 Bsprintf(buf, "JoystickAnalogAxes%d", dummy);
875                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_AnalogNumToName(ud.config.JoystickAnalogueAxes[dummy]));
876             }
877 
878             if (CONFIG_FunctionNumToName(ud.config.JoystickDigitalFunctions[dummy][0]))
879             {
880                 Bsprintf(buf, "JoystickDigitalAxes%d_0", dummy);
881                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.JoystickDigitalFunctions[dummy][0]));
882             }
883 
884             if (CONFIG_FunctionNumToName(ud.config.JoystickDigitalFunctions[dummy][1]))
885             {
886                 Bsprintf(buf, "JoystickDigitalAxes%d_1", dummy);
887                 SCRIPT_PutString(ud.config.scripthandle, "Controls", buf, CONFIG_FunctionNumToName(ud.config.JoystickDigitalFunctions[dummy][1]));
888             }
889 
890             if (ud.config.JoystickAnalogueScale[dummy] != DEFAULTJOYSTICKANALOGUESCALE)
891             {
892                 Bsprintf(buf, "JoystickAnalogScale%d", dummy);
893                 SCRIPT_PutNumber(ud.config.scripthandle, "Controls", buf, ud.config.JoystickAnalogueScale[dummy], FALSE, FALSE);
894             }
895 
896             if (ud.config.JoystickAnalogueDead[dummy] != DEFAULTJOYSTICKANALOGUEDEAD)
897             {
898                 Bsprintf(buf, "JoystickAnalogDead%d", dummy);
899                 SCRIPT_PutNumber(ud.config.scripthandle, "Controls", buf, ud.config.JoystickAnalogueDead[dummy], FALSE, FALSE);
900             }
901 
902             if (ud.config.JoystickAnalogueSaturate[dummy] != DEFAULTJOYSTICKANALOGUESATURATE)
903             {
904                 Bsprintf(buf, "JoystickAnalogSaturate%d", dummy);
905                 SCRIPT_PutNumber(ud.config.scripthandle, "Controls", buf, ud.config.JoystickAnalogueSaturate[dummy], FALSE, FALSE);
906             }
907         }
908     }
909 
910     SCRIPT_PutString(ud.config.scripthandle, "Comm Setup","PlayerName",&szPlayerName[0]);
911 
912     SCRIPT_PutString(ud.config.scripthandle, "Comm Setup","RTSName",&ud.rtsname[0]);
913 
914     char commmacro[] = "CommbatMacro# ";
915 
916     for (dummy = 0; dummy < MAXRIDECULE; dummy++)
917     {
918         commmacro[13] = dummy+'0';
919         SCRIPT_PutString(ud.config.scripthandle, "Comm Setup",commmacro,&ud.ridecule[dummy][0]);
920     }
921 
922     SCRIPT_Save(ud.config.scripthandle, g_setupFileName);
923 
924     if ((flags & 2) == 0)
925         SCRIPT_Free(ud.config.scripthandle);
926 
927     OSD_Printf("Wrote %s\n",g_setupFileName);
928     CONFIG_WriteSettings();
929     Bfflush(NULL);
930 }
931 
CONFIG_GetMapEntryName(char m[],char const * const mapname)932 static const char *CONFIG_GetMapEntryName(char m[], char const * const mapname)
933 {
934     strcpy(m, mapname);
935 
936     char *p = strrchr(m, '/');
937     if (!p) p = strrchr(m, '\\');
938     if (p) Bmemmove(m, p, Bstrlen(p)+1);
939     for (p=m; *p; p++) *p = tolower(*p);
940 
941     // cheap hack because SCRIPT_GetNumber doesn't like the slashes
942     p = m;
943     while (*p == '/') p++;
944 
945     return p;
946 }
947 
CONFIG_GetMD4EntryName(char m[],uint8_t const * const md4)948 static void CONFIG_GetMD4EntryName(char m[], uint8_t const * const md4)
949 {
950     sprintf(m, "MD4_%08x%08x%08x%08x",
951             B_BIG32(B_UNBUF32(&md4[0])), B_BIG32(B_UNBUF32(&md4[4])),
952             B_BIG32(B_UNBUF32(&md4[8])), B_BIG32(B_UNBUF32(&md4[12])));
953 }
954 
CONFIG_GetMapBestTime(char const * const mapname,uint8_t const * const mapmd4)955 int32_t CONFIG_GetMapBestTime(char const * const mapname, uint8_t const * const mapmd4)
956 {
957     if (!ud.config.setupread || ud.config.scripthandle < 0)
958         return -1;
959 
960     char m[37];
961     CONFIG_GetMD4EntryName(m, mapmd4);
962 
963     int32_t t = -1;
964     if (SCRIPT_GetNumber(ud.config.scripthandle, "MapTimes", m, &t))
965     {
966         // fall back to map filenames
967         char m2[BMAX_PATH];
968         char const * const p = CONFIG_GetMapEntryName(m2, mapname);
969         SCRIPT_GetNumber(ud.config.scripthandle, "MapTimes", p, &t);
970     }
971     return t;
972 }
973 
CONFIG_SetMapBestTime(uint8_t const * const mapmd4,int32_t const tm)974 int32_t CONFIG_SetMapBestTime(uint8_t const * const mapmd4, int32_t const tm)
975 {
976     if (ud.config.scripthandle < 0) ud.config.scripthandle = SCRIPT_Init(g_setupFileName);
977     if (ud.config.scripthandle < 0) return -1;
978 
979     char m[37];
980     CONFIG_GetMD4EntryName(m, mapmd4);
981 
982     SCRIPT_PutNumber(ud.config.scripthandle, "MapTimes", m, tm, FALSE, FALSE);
983     return 0;
984 }
985 
986