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