]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/menu/xonotic/util.qc
Merge branch 'master' into Mario/crouch
[xonotic/xonotic-data.pk3dir.git] / qcsrc / menu / xonotic / util.qc
1 #include "util.qh"
2 #include "dialog.qh"
3
4 #include "../item.qh"
5
6 #include "../menu.qh"
7 #include <common/campaign_common.qh>
8 #include <common/constants.qh>
9 #include <common/mapinfo.qh>
10 #include <common/util.qh>
11 #include <common/command/_mod.qh>
12
13 float GL_CheckExtension(string ext)
14 {
15         return strhasword(cvar_string("gl_info_extensions"), ext);
16 }
17
18 float GL_Have_TextureCompression()
19 {
20         return (GL_CheckExtension("GL_EXT_texture_compression_s3tc") && GL_CheckExtension("GL_ARB_texture_compression"));
21 }
22
23 .entity parent, firstChild, nextSibling;
24 void forAllDescendants(entity root, void(entity, entity) funcPre, void(entity, entity) funcPost, entity pass)
25 {
26         depthfirst(root, parent, firstChild, nextSibling, funcPre, funcPost, pass);
27 }
28
29 .string cvarName;
30 void SUB_Null_ee(entity e1, entity e2)
31 {
32 }
33
34 .void(entity) saveCvars;
35 void saveCvarsOf(entity ignore, entity e)
36 {
37         if(e.saveCvars)
38                 e.saveCvars(e);
39 }
40
41 .void(entity) loadCvars;
42 void loadCvarsOf(entity ignore, entity e)
43 {
44         if(e.loadCvars)
45                 e.loadCvars(e);
46 }
47 void saveAllCvars(entity root)
48 {
49         forAllDescendants(root, saveCvarsOf, SUB_Null_ee, NULL);
50 }
51 void loadAllCvars(entity root)
52 {
53         forAllDescendants(root, loadCvarsOf, SUB_Null_ee, NULL);
54 }
55
56 .string cvarNames_Multi;
57 .void(entity me) saveCvars_Multi;
58 string getCvarsMulti(entity me)
59 {
60         if (me.cvarNames_Multi)
61                 return me.cvarNames_Multi;
62         return string_null;
63 }
64 void saveCvarsMulti(entity me)
65 {
66         float n, i;
67         string s, cvarname;
68
69         me.saveCvars_Multi(me);
70         s = cvar_string(me.cvarName);
71
72         n = tokenize_console(me.cvarNames_Multi);
73         for(i = 0; i < n; ++i)
74         {
75                 // cvars prefixed with ! get saved with the inverted value
76                 if(substring(argv(i), 0, 1) == "!")
77                 {
78                         cvarname = substring(argv(i), 1, strlen(argv(i)));
79                         cvar_set(cvarname, ((s == "0") ? "1" : "0"));
80                 }
81                 else
82                 {
83                         cvarname = argv(i);
84                         cvar_set(cvarname, s);
85                 }
86
87                 CheckSendCvars(me, cvarname);
88         }
89 }
90 void makeMulti(entity e, string otherCvars)
91 {
92         e.cvarNames_Multi = otherCvars;
93         e.saveCvars_Multi = e.saveCvars;
94         e.saveCvars = saveCvarsMulti;
95 }
96
97 .void(entity me) saveCvars_Callback;
98 .entity saveCvars_Callback_ent;
99 .void(entity me, entity cb) saveCvars_Callback_func;
100 void saveCvarsCallback(entity me)
101 {
102         me.saveCvars_Callback(me);
103         me.saveCvars_Callback_func(me.saveCvars_Callback_ent, me);
104 }
105 void makeCallback(entity e, entity cbent, void(entity, entity) cbfunc)
106 {
107         e.saveCvars_Callback = e.saveCvars;
108         e.saveCvars = saveCvarsCallback;
109         e.saveCvars_Callback_ent = cbent;
110         e.saveCvars_Callback_func = cbfunc;
111 }
112
113 .void(entity) draw_setDependent;
114 .string cvar_setDependent;
115 .float cvarMin_setDependent;
116 .float cvarMax_setDependent;
117 .string cvar2_setDependent;
118 .float cvar2Min_setDependent;
119 .float cvar2Max_setDependent;
120 .string cvar3_setDependent;
121 .float cvar3Min_setDependent;
122 .float cvar3Max_setDependent;
123 .float op_setDependent;
124 .string cvarString_setDependent;
125 .string cvarValue_setDependent;
126 .float(entity) func_setDependent;
127 .bool disabled;
128 void setDependent_Check(entity e)
129 {
130         float f;
131         string s;
132         if(e.func_setDependent)
133         {
134                 e.disabled = !(e.func_setDependent(e));
135         }
136         else if(e.cvarString_setDependent)
137         {
138                 s = cvar_string(e.cvarString_setDependent);
139                 e.disabled = (cvar_string(e.cvarString_setDependent) == e.cvarValue_setDependent);
140         }
141         else
142         {
143                 if(e.cvar_setDependent)
144                 {
145                         f = cvar(e.cvar_setDependent);
146                         if(e.cvarMin_setDependent <= e.cvarMax_setDependent)
147                                 e.disabled = ((f < e.cvarMin_setDependent) || (f > e.cvarMax_setDependent));
148                         else
149                                 e.disabled = ((f >= e.cvarMax_setDependent) && (f <= e.cvarMin_setDependent));
150                 }
151                 if(e.cvar2_setDependent)
152                 {
153                         f = cvar(e.cvar2_setDependent);
154                         if(e.cvar2Min_setDependent <= e.cvar2Max_setDependent)
155                                 e.disabled = (e.disabled + ((f < e.cvar2Min_setDependent) || (f > e.cvar2Max_setDependent)) > e.op_setDependent);
156                         else
157                                 e.disabled = (e.disabled + ((f >= e.cvar2Max_setDependent) && (f <= e.cvar2Min_setDependent)) > e.op_setDependent);
158                 }
159                 if(e.cvar3_setDependent)
160                 {
161                         f = cvar(e.cvar3_setDependent);
162                         if(e.cvar3Min_setDependent <= e.cvar3Max_setDependent)
163                                 e.disabled = (e.disabled + ((f < e.cvar3Min_setDependent) || (f > e.cvar3Max_setDependent)) > e.op_setDependent);
164                         else
165                                 e.disabled = (e.disabled + ((f >= e.cvar3Max_setDependent) && (f <= e.cvar3Min_setDependent)) > e.op_setDependent);
166                 }
167         }
168 }
169 void setDependent_Draw(entity e)
170 {
171         setDependent_Check(e);
172         e.draw_setDependent(e);
173 }
174 .void(entity) draw;
175 void setDependent(entity e, string theCvarName, float theCvarMin, float theCvarMax)
176 {
177         e.draw_setDependent = e.draw;
178         e.cvar_setDependent = theCvarName;
179         e.cvarMin_setDependent = theCvarMin;
180         e.cvarMax_setDependent = theCvarMax;
181         e.cvar2_setDependent = string_null;
182         e.cvar3_setDependent = string_null;
183         e.func_setDependent = func_null;
184         e.draw = setDependent_Draw;
185         setDependent_Check(e);
186 }
187 void setDependentStringNotEqual(entity e, string theCvarName, string theCvarValue)
188 {
189         e.draw_setDependent = e.draw;
190         e.cvarString_setDependent = theCvarName;
191         e.cvarValue_setDependent = theCvarValue;
192         e.cvar_setDependent = string_null;
193         e.cvar2_setDependent = string_null;
194         e.cvar3_setDependent = string_null;
195         e.func_setDependent = func_null;
196         e.draw = setDependent_Draw;
197         setDependent_Check(e);
198 }
199 void setDependentAND(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max)
200 {
201         e.draw_setDependent = e.draw;
202         e.cvar_setDependent = theCvarName;
203         e.cvarMin_setDependent = theCvarMin;
204         e.cvarMax_setDependent = theCvarMax;
205         e.cvar2_setDependent = theCvar2Name;
206         e.cvar2Min_setDependent = theCvar2Min;
207         e.cvar2Max_setDependent = theCvar2Max;
208         e.cvar3_setDependent = string_null;
209         e.op_setDependent = 0;
210         e.func_setDependent = func_null;
211         e.draw = setDependent_Draw;
212         setDependent_Check(e);
213 }
214 void setDependentOR(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max)
215 {
216         e.draw_setDependent = e.draw;
217         e.cvar_setDependent = theCvarName;
218         e.cvarMin_setDependent = theCvarMin;
219         e.cvarMax_setDependent = theCvarMax;
220         e.cvar2_setDependent = theCvar2Name;
221         e.cvar2Min_setDependent = theCvar2Min;
222         e.cvar2Max_setDependent = theCvar2Max;
223         e.cvar3_setDependent = string_null;
224         e.op_setDependent = 1;
225         e.func_setDependent = func_null;
226         e.draw = setDependent_Draw;
227         setDependent_Check(e);
228 }
229 void setDependentAND3(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max, string theCvar3Name, float theCvar3Min, float theCvar3Max)
230 {
231         e.draw_setDependent = e.draw;
232         e.cvar_setDependent = theCvarName;
233         e.cvarMin_setDependent = theCvarMin;
234         e.cvarMax_setDependent = theCvarMax;
235         e.cvar2_setDependent = theCvar2Name;
236         e.cvar2Min_setDependent = theCvar2Min;
237         e.cvar2Max_setDependent = theCvar2Max;
238         e.cvar3_setDependent = theCvar3Name;
239         e.cvar3Min_setDependent = theCvar3Min;
240         e.cvar3Max_setDependent = theCvar3Max;
241         e.op_setDependent = 0;
242         e.func_setDependent = func_null;
243         e.draw = setDependent_Draw;
244         setDependent_Check(e);
245 }
246 void setDependentWeird(entity e, float(entity) func)
247 {
248         e.draw_setDependent = e.draw;
249         e.func_setDependent = func;
250         e.draw = setDependent_Draw;
251         setDependent_Check(e);
252 }
253
254 void setZonedTooltip(entity e, string theTooltip, string theCvar)
255 {
256         if(theTooltip == "") // no tooltip, use cvar description then
257         {
258                 if(theCvar != "" && prvm_language == "en")
259                 {
260                         string t = cvar_description(theCvar);
261                         if(t != "" && t != "custom cvar")
262                                 theTooltip = t;
263                 }
264         }
265         else if(theTooltip == "-") // no cvar description as tooltip
266         {
267                 theTooltip = string_null;
268         }
269
270         if(e.tooltip)
271                 strunzone(e.tooltip);
272         e.tooltip = (theTooltip != "") ? strzone(theTooltip) : string_null;
273 }
274
275 void clearTooltip(entity e)
276 {
277         setZonedTooltip(e, string_null, string_null);
278 }
279
280 // URI SYSTEM ////////////////////////////////////////////////////////
281
282 float _Nex_ExtResponseSystem_Queried;
283 string _Nex_ExtResponseSystem_UpdateTo;
284 string _Nex_ExtResponseSystem_UpdateToURL;
285 string _Nex_ExtResponseSystem_Packs;
286 float _Nex_ExtResponseSystem_PacksStep;
287
288 /** engine callback */
289 void URI_Get_Callback(float id, float status, string data)
290 {
291         if(url_URI_Get_Callback(id, status, data))
292         {
293                 // handled
294         }
295         else if (id == URI_GET_DISCARD)
296         {
297                 // discard
298         }
299         else if (id >= URI_GET_CURL && id <= URI_GET_CURL_END)
300         {
301                 // sv_cmd curl
302                 Curl_URI_Get_Callback(id, status, data);
303         }
304         else if (id == URI_GET_UPDATENOTIFICATION)
305         {
306                 UpdateNotification_URI_Get_Callback(id, status, data);
307         }
308         else
309         {
310                 LOG_INFOF("Received HTTP request data for an invalid id %d.", id);
311         }
312 }
313
314 void DisableServerBackwardsCompatibility()
315 {
316         cvar_set("gameversion_min", ftos(100 * floor(cvar("gameversion") / 100)));
317 }
318
319 void UpdateNotification_URI_Get_Callback(float id, float status, string data)
320 {
321         float n;
322
323         if(_Nex_ExtResponseSystem_UpdateTo)
324         {
325                 LOG_TRACE("error: UpdateNotification_URI_Get_Callback has been called before");
326                 return;
327         }
328         if(status != 0)
329         {
330                 LOG_TRACEF("error receiving update notification: status is %d", status);
331                 return;
332         }
333         if(substring(data, 0, 1) == "<")
334         {
335                 LOG_TRACE("error: received HTML instead of an update notification");
336                 return;
337         }
338         if(strstrofs(data, "\r", 0) != -1)
339         {
340                 LOG_TRACE("error: received carriage returns from update notification server");
341                 return;
342         }
343
344         if(data == "")
345                 n = 0;
346         else
347                 n = tokenizebyseparator(data, "\n");
348
349         float i;
350         string s;
351
352         string un_version = "";
353         string un_download = "";
354         string un_url = "";
355         string un_bannedservers = "";
356         string un_emergency_pk3s = "";
357         string un_promoted = "";
358         string un_recommended = "";
359         string un_compatexpire = "";
360
361         for(i = 0; i < n; ++i)
362         {
363                 s = substring(argv(i), 2, -1);
364                 if(s == "") { continue; } // ignore empty lines
365
366                 switch(substring(argv(i), 0, 1))
367                 {
368                         case "V":
369                         {
370                                 un_version = s;
371                                 break;
372                         }
373                         case "C":
374                         {
375                                 un_compatexpire = s;
376                                 break;
377                         }
378                         case "D":
379                         {
380                                 un_download = s;
381                                 break;
382                         }
383                         case "U":
384                         {
385                                 un_url = s;
386                                 break;
387                         }
388                         case "B":
389                         {
390                                 APPEND_TO_STRING(un_bannedservers, " ", s);
391                                 break;
392                         }
393                         case "E":
394                         {
395                                 if(cvar("menu_updatecheck_getpacks"))
396                                         APPEND_TO_STRING(un_emergency_pk3s, " ", s);
397                                 break;
398                         }
399                         case "P":
400                         {
401                                 APPEND_TO_STRING(un_promoted, " ", s);
402                                 break;
403                         }
404                         case "R":
405                         {
406                                 APPEND_TO_STRING(un_recommended, " ", s);
407                                 break;
408                         }
409                 }
410         }
411
412         if(un_version != "")
413         {
414                 if(vercmp(cvar_string("g_xonoticversion"), un_version) < 0)
415                 {
416                         // update needed
417                         _Nex_ExtResponseSystem_UpdateTo = strzone(un_version);
418                         if(un_download) { LOG_INFOF(_("Update can be downloaded at:\n%s"), un_download); }
419                         if(un_url) { _Nex_ExtResponseSystem_UpdateToURL = strzone(un_url); }
420                         DisableServerBackwardsCompatibility();
421                 }
422                 else if(cvar_string("g_xonoticversion") == un_version)
423                 {
424                         if(un_compatexpire != "")
425                         {
426                                 string curdate = strftime(false, "%Y%m%d%H%M%S");
427                                 if (strcmp(curdate, un_compatexpire) >= 0)
428                                         DisableServerBackwardsCompatibility();
429                         }
430                 }
431         }
432
433         if(un_bannedservers != "")
434         {
435                 _Nex_ExtResponseSystem_BannedServers = strzone(un_bannedservers);
436                 _Nex_ExtResponseSystem_BannedServersNeedsRefresh = 1;
437         }
438
439         if(un_emergency_pk3s != "")
440         {
441                 _Nex_ExtResponseSystem_Packs = strzone(un_emergency_pk3s);
442                 _Nex_ExtResponseSystem_PacksStep = 1;
443         }
444
445         if(un_promoted != "")
446         {
447                 _Nex_ExtResponseSystem_PromotedServers = strzone(un_promoted);
448                 _Nex_ExtResponseSystem_PromotedServersNeedsRefresh = 1;
449         }
450
451         if(un_recommended != "")
452         {
453                 _Nex_ExtResponseSystem_RecommendedServers = strzone(un_recommended);
454                 _Nex_ExtResponseSystem_RecommendedServersNeedsRefresh = 1;
455         }
456 }
457
458 // END OF URI SYSTEM ////////////////////////////////////////////////////////
459
460 void updateCheck()
461 {
462         if(cvar("menu_updatecheck"))
463         {
464                 if(!_Nex_ExtResponseSystem_Queried)
465                 {
466                         _Nex_ExtResponseSystem_Queried = 1;
467                         float startcnt;
468                         string uri;
469
470                         cvar_set("cl_startcount", ftos(startcnt = cvar("cl_startcount") + 1));
471
472                         // for privacy, munge the start count a little
473                         startcnt = floor((floor(startcnt / 10) + random()) * 10);
474                         uri = sprintf("http://update.xonotic.org/checkupdate.txt?version=%s&cnt=%d", uri_escape(cvar_string("g_xonoticversion")), startcnt);
475                         uri_get(uri, URI_GET_UPDATENOTIFICATION);
476                 }
477         }
478
479         if(_Nex_ExtResponseSystem_PacksStep > 0)
480         {
481                 float n, i;
482                 float allgood;
483                 n = tokenize_console(_Nex_ExtResponseSystem_Packs);
484                 allgood = true;
485                 for(i = 0; i+1 < n; i += 2)
486                 {
487                         if(fexists(argv(i+1)))
488                                 continue;
489                         allgood = false;
490                         if(_Nex_ExtResponseSystem_PacksStep == 1) // first run
491                                 localcmd("\ncurl --pak \"", argv(i), "\"\n");
492                 }
493                 if(allgood)
494                 {
495                         if(_Nex_ExtResponseSystem_PacksStep == 2)
496                         {
497                                 if(!Menu_Active)
498                                         cvar_set("_menu_initialized", "0");
499                                         // HACK: cause m_hide call on next start
500                                 localcmd("\nmenu_restart\n");
501                         }
502                         _Nex_ExtResponseSystem_PacksStep = 0;
503                 }
504                 else
505                         _Nex_ExtResponseSystem_PacksStep = 2;
506         }
507
508 }
509
510 bool show_propermenu = false;
511
512 float preMenuInit()
513 {
514         vector sz;
515         vector boxA, boxB;
516
517         if(random() < 0.1)
518                 show_propermenu = true;
519
520         updateCheck();
521
522         MapInfo_Cache_Create();
523         MapInfo_Enumerate();
524         if(!_MapInfo_FilterGametype(MAPINFO_TYPE_ALL, 0, 0, 0, 1))
525         {
526                 draw_reset_cropped();
527
528                 sz = eX * 0.025 + eY * 0.025 * (draw_scale.x / draw_scale.y);
529                 draw_CenterText('0.5 0.5 0' - 1.25 * sz.y * eY, _("Autogenerating mapinfo for newly added maps..."), sz, '1 1 1', 1, 0);
530
531                 boxA = '0.05 0.5 0' + 0.25 * sz.y * eY;
532                 boxB = '0.95 0.5 0' + 1.25 * sz.y * eY;
533                 draw_Fill(boxA, boxB - boxA, '1 1 1', 1);
534
535                 boxA += sz * 0.1;
536                 boxB -= sz * 0.1;
537                 draw_Fill(boxA, boxB - boxA, '0.1 0.1 0.1', 1);
538
539                 boxB_x = boxA_x * (1 - MapInfo_progress) + boxB_x * MapInfo_progress;
540                 draw_Fill(boxA, boxB - boxA, '0 0 1', 1);
541
542                 return false;
543         }
544         return true;
545 }
546
547 string campaign_name_previous;
548 float campaign_won_previous;
549 #ifdef WATERMARK
550 string autocvar_menu_watermark = WATERMARK;
551 #else
552 string autocvar_menu_watermark = "";
553 #endif
554 void postMenuDraw()
555 {
556         if(autocvar_menu_watermark != "")
557         {
558                 draw_CenterText('0.5 0.1 0', sprintf(_("^1%s TEST BUILD"), autocvar_menu_watermark), globalToBoxSize('32 32 0', draw_scale), '1 1 1', 0.05, 1);
559         }
560 }
561 void DialogOpenButton_Click_withCoords(entity button, entity tab, vector theOrigin, vector theSize);
562 .entity winnerDialog;
563 void preMenuDraw()
564 {
565         vector fs, sz = '0 0 0', line, mid;
566
567         updateCheck();
568
569         if(_Nex_ExtResponseSystem_UpdateTo != "")
570         {
571                 // TODO rather turn this into a dialog
572                 fs = ((1/draw_scale.x) * eX + (1/draw_scale.y) * eY) * 12;
573                 line = eY * fs.y;
574                 string l1, l2;
575                 if(show_propermenu)
576                         l1 = sprintf("Jeff pay 4 new weapons for %s", _Nex_ExtResponseSystem_UpdateTo);
577                 else
578                         l1 = sprintf(_("Update to %s now!"), _Nex_ExtResponseSystem_UpdateTo);
579                 l2 = "http://www.xonotic.org/";
580                 if(_Nex_ExtResponseSystem_UpdateToURL)
581                         l2 = _Nex_ExtResponseSystem_UpdateToURL;
582
583                 sz_x = draw_TextWidth("    ", 0, fs) + max(
584                                 draw_TextWidth(l1, 0, fs),
585                                 draw_TextWidth(l2, 0, fs)
586                         );
587                 sz_y = 3 * fs.y;
588
589                 draw_alpha = bound(0, sin(time * 0.112 - 0.3) * 10, 1);
590                 mid = eX * (0.5 + 0.5 * (1 - sz.x) * cos(time * 0.071))
591                     + eY * (0.5 + 0.5 * (1 - sz.y) * sin(time * 0.071));
592
593                 draw_Fill(mid - 0.5 * sz, sz, '1 1 0', 1);
594                 draw_CenterText(mid - 1 * line, l1, fs, '1 0 0', 1, 0);
595                 draw_CenterText(mid - 0 * line, l2, fs, '0 0 1', 1, 0);
596         }
597         if (!campaign_name_previous)
598                 campaign_name_previous = strzone(strcat(campaign_name, "x")); // force unequal
599         if(campaign_name == campaign_name_previous)
600         {
601                 if(cvar(strcat("g_campaign", campaign_name, "_won")))
602                 {
603                         if(!campaign_won_previous)
604                         {
605                                 m_display();
606                                 DialogOpenButton_Click_withCoords(NULL, main.winnerDialog, '0 0 0', eX * conwidth + eY * conheight);
607                         }
608                         campaign_won_previous = 1;
609                 }
610                 else
611                         campaign_won_previous = 0;
612         }
613         else
614         {
615                 strunzone(campaign_name_previous);
616                 campaign_name_previous = strzone(campaign_name);
617                 campaign_won_previous = cvar(strcat("g_campaign", campaign_name, "_won"));
618         }
619 }
620
621 string resolvemod(string m)
622 {
623         if(m == "=")
624                 return getcurrentmod();
625         else
626                 return m;
627 }
628
629 float updateCompression()
630 {
631         float have_dds, have_jpg, have_tga;
632         float can_dds;
633         have_dds = (fexists("dds/particles/particlefont.dds"));
634         have_jpg = (fexists("particles/particlefont.jpg"));
635         have_tga = (fexists("particles/particlefont.tga"));
636         can_dds = GL_Have_TextureCompression();
637         if(have_dds && (have_jpg || have_tga))
638         {
639                 // both? Let's only use good quality precompressed files
640                 // but ONLY if we actually support it!
641                 if(can_dds)
642                 {
643                         // these builds are meant to have GOOD quality, so let's not compress non-skinframes
644                         cvar_set("gl_texturecompression", "0");
645                         return 1;
646
647                         //cvar_set("gl_texturecompression", cvar_string("r_texture_dds_load"));
648                         //return 2;
649                 }
650                 else
651                 {
652                         cvar_set("gl_texturecompression", "0");
653                         cvar_set("r_texture_dds_load", "0");
654                         return 0;
655                 }
656         }
657         else if(have_dds)
658         {
659                 // DDS only? We probably always want texture compression
660                 cvar_set("gl_texturecompression", "1");
661                 cvar_set("r_texture_dds_load", "1");
662                 if(!can_dds)
663                         LOG_INFO(_("^1ERROR: Texture compression is required but not supported.\n^1Expect visual problems."));
664                 return 0;
665         }
666         else
667         {
668                 // TGA only? Allow runtime compression
669                 if(can_dds)
670                 {
671                         cvar_set("gl_texturecompression", cvar_string("r_texture_dds_load"));
672                         return 2;
673                 }
674                 else
675                 {
676                         cvar_set("gl_texturecompression", "0");
677                         cvar_set("r_texture_dds_load", "0");
678                         return 0;
679                 }
680         }
681 }
682
683 // note: include only those that should be in the menu!
684 #define GAMETYPES \
685         GAMETYPE(MAPINFO_TYPE_DEATHMATCH) \
686         GAMETYPE(MAPINFO_TYPE_TEAM_DEATHMATCH) \
687         GAMETYPE(MAPINFO_TYPE_CTF) \
688         GAMETYPE(MAPINFO_TYPE_CA) \
689         GAMETYPE(MAPINFO_TYPE_FREEZETAG) \
690         GAMETYPE(MAPINFO_TYPE_KEEPAWAY) \
691         GAMETYPE(MAPINFO_TYPE_KEYHUNT) \
692         GAMETYPE(MAPINFO_TYPE_LMS) \
693         GAMETYPE(MAPINFO_TYPE_DOMINATION) \
694         GAMETYPE(MAPINFO_TYPE_NEXBALL) \
695         GAMETYPE(MAPINFO_TYPE_ONSLAUGHT) \
696         GAMETYPE(MAPINFO_TYPE_ASSAULT) \
697         /* GAMETYPE(MAPINFO_TYPE_INVASION) */ \
698         /**/
699
700 // hidden gametypes come last so indexing always works correctly
701 #define HIDDEN_GAMETYPES \
702         GAMETYPE(MAPINFO_TYPE_RACE) \
703         GAMETYPE(MAPINFO_TYPE_CTS) \
704         /**/
705
706 Gametype GameType_GetID(int cnt)
707 {
708         int i = 0;
709         #define GAMETYPE(it) { if (i++ == cnt) return it; }
710         GAMETYPES
711         HIDDEN_GAMETYPES
712         #undef GAMETYPE
713         return NULL;
714 }
715
716 int GameType_GetCount()
717 {
718         int i = 0;
719         #define GAMETYPE(id) ++i;
720         GAMETYPES
721         #undef GAMETYPE
722         #define GAMETYPE(it) if (cvar("developer")) ++i;
723         HIDDEN_GAMETYPES
724         #undef GAMETYPE
725         return i;
726 }
727
728 int GameType_GetTotalCount()
729 {
730         int i = 0;
731         #define GAMETYPE(id) ++i;
732         GAMETYPES
733         HIDDEN_GAMETYPES
734         #undef GAMETYPE
735         return i;
736 }
737
738 string GameType_GetName(int cnt)
739 {
740         Gametype i = GameType_GetID(cnt);
741         return i ? MapInfo_Type_ToText(i) : "";
742 }
743
744 string GameType_GetIcon(int cnt)
745 {
746         Gametype i = GameType_GetID(cnt);
747         return i ? strcat("gametype_", MapInfo_Type_ToString(i)) : "";
748 }
749
750 .void(entity) TR;
751 .void(entity, float, float, entity) TD;
752 .void(entity, float) TDempty;
753 .void(entity, float, float) gotoRC;
754 entity makeXonoticTextLabel(float theAlign, string theText);
755 entity makeXonoticTextSlider(string);
756 .void(entity, string, string) addValue;
757 .void(entity) configureXonoticTextSliderValues;
758 entity makeXonoticColorpickerString(string theCvar, string theDefaultCvar);
759 entity makeXonoticCheckBoxString(string, string, string, string);
760 entity makeXonoticCheckBox(float, string, string);
761 .bool sendCvars;
762
763 void dialog_hudpanel_main_checkbox(entity me, string panelname)
764 {
765         entity e;
766
767         me.TR(me);
768                 me.TDempty(me, 1.5);
769                 me.TD(me, 1, 2.5, e = makeXonoticCheckBox(0, strzone(strcat("hud_panel_", panelname)), _("Enable")));
770 }
771
772 void dialog_hudpanel_main_settings(entity me, string panelname)
773 {
774         float i;
775         entity e;
776
777         me.gotoRC(me, me.currentRow + 1.5, 0);
778                 me.TD(me, 1, 1.4, e = makeXonoticTextLabel(0, _("Background:")));
779                         me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg"))));
780                                 e.addValue(e, _("Default"), "");
781                                 e.addValue(e, _("Disable"), "0");
782                                 e.addValue(e, strzone(strcat("border_", panelname)), strzone(strcat("border_", panelname)));
783                                 e.configureXonoticTextSliderValues(e);
784         me.TR(me);
785                 me.TDempty(me, 0.2);
786                 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Color:")));
787                 me.TD(me, 2, 2.6, e = makeXonoticColorpickerString(strzone(strcat("hud_panel_", panelname, "_bg_color")), "hud_panel_bg_color"));
788                         setDependentStringNotEqual(e, strzone(strcat("hud_panel_", panelname, "_bg_color")), "");
789         me.TR(me);
790                 me.TDempty(me, 0.2);
791                 me.TD(me, 1, 1.0, e = makeXonoticCheckBoxString("", "1 1 1", strzone(strcat("hud_panel_", panelname, "_bg_color")), _("Use default")));
792         me.TR(me);
793                 me.TDempty(me, 0.2);
794                 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Border size:")));
795                         me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_border"))));
796                                 e.addValue(e, _("Default"), "");
797                                 e.addValue(e, _("Disable"), "0");
798                                 for(i = 1; i <= 10; ++i)
799                                         e.addValue(e, strzone(ftos_decimals(i * 2, 0)), strzone(ftos(i * 2)));
800                                 e.configureXonoticTextSliderValues(e);
801         me.TR(me);
802                 me.TDempty(me, 0.2);
803                 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Alpha:")));
804                         me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_alpha"))));
805                                 e.addValue(e, _("Default"), "");
806                                 for(i = 1; i <= 10; ++i)
807                                         e.addValue(e, strzone(ftos_decimals(i/10, 1)), strzone(ftos(i/10)));
808                                 e.configureXonoticTextSliderValues(e);
809         me.TR(me);
810                 me.TDempty(me, 0.2);
811                 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Team Color:")));
812                         me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_color_team"))));
813                                 e.addValue(e, _("Default"), "");
814                                 e.addValue(e, _("Disable"), "0");
815                                 for(i = 1; i <= 10; ++i)
816                                         e.addValue(e, strzone(ftos_decimals(i/10, 1)), strzone(ftos(i/10)));
817                                 e.configureXonoticTextSliderValues(e);
818         me.TR(me);
819                 me.TDempty(me, 0.4);
820                 me.TD(me, 1, 3.6, e = makeXonoticCheckBox(0, "hud_configure_teamcolorforced", _("Test team color in configure mode")));
821         me.TR(me);
822                 me.TDempty(me, 0.2);
823                 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Padding:")));
824                         me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_padding"))));
825                                 e.addValue(e, _("Default"), "");
826                                 for(i = 0; i <= 10; ++i)
827                                         e.addValue(e, strzone(ftos_decimals(i - 5, 0)), strzone(ftos(i - 5)));
828                                 e.configureXonoticTextSliderValues(e);
829 }
830
831 float getFadedAlpha(float currentAlpha, float startAlpha, float targetAlpha)
832 {
833         if(startAlpha < targetAlpha)
834                 currentAlpha = min(currentAlpha + frametime * 0.5, targetAlpha);
835         else
836                 currentAlpha = max(currentAlpha - frametime * 0.5, targetAlpha);
837         return currentAlpha;
838 }
839
840 void CheckSendCvars(entity me, string cvarnamestring)
841 {
842         if(me.sendCvars)
843         {
844                 LOG_INFOF("Sending cvar: %s -> %s", cvarnamestring, cvar_string(cvarnamestring));
845                 if(gamestatus & (GAME_CONNECTED | GAME_ISSERVER))
846                 {
847                         cmd(sprintf("\nsendcvar %s\n", cvarnamestring));
848                 }
849         }
850 }