1 float GL_CheckExtension(string ext)
3 return (strstrofs(strcat(" ", cvar_string("gl_info_extensions"), " "), strcat(" ", ext, " "), 0) >= 0);
6 float GL_Have_TextureCompression()
8 return (GL_CheckExtension("GL_EXT_texture_compression_s3tc") && GL_CheckExtension("GL_ARB_texture_compression"));
13 tooltipdb = db_load(language_filename("tooltips.db"));
21 string getZonedTooltipForIdentifier(string s)
28 t = db_get(tooltipdb, s);
34 if(prvm_language == "en")
36 t = cvar_description(s);
37 if(t != "" && t != "custom cvar")
40 dprint("WARNING: no tooltip set for ", s, "\n");
44 void forAllDescendants(entity root, void(entity, entity) funcPre, void(entity, entity) funcPost, entity pass)
46 depthfirst(root, parent, firstChild, nextSibling, funcPre, funcPost, pass);
50 void SUB_Null_ee(entity e1, entity e2)
53 void saveCvarsOf(entity ignore, entity e)
58 void loadCvarsOf(entity ignore, entity e)
63 void saveAllCvars(entity root)
65 forAllDescendants(root, saveCvarsOf, SUB_Null_ee, NULL);
67 void loadAllCvars(entity root)
69 forAllDescendants(root, loadCvarsOf, SUB_Null_ee, NULL);
72 .string cvarNames_Multi;
73 .void(entity me) saveCvars_Multi;
74 string getCvarsMulti(entity me)
76 if (me.cvarNames_Multi)
77 return me.cvarNames_Multi;
80 void saveCvarsMulti(entity me)
85 me.saveCvars_Multi(me);
86 s = cvar_string(me.cvarName);
88 n = tokenize_console(me.cvarNames_Multi);
89 for(i = 0; i < n; ++i)
92 void makeMulti(entity e, string otherCvars)
94 e.cvarNames_Multi = otherCvars;
95 e.saveCvars_Multi = e.saveCvars;
96 e.saveCvars = saveCvarsMulti;
99 .void(entity me) saveCvars_Callback;
100 .entity saveCvars_Callback_ent;
101 .void(entity me, entity cb) saveCvars_Callback_func;
102 void saveCvarsCallback(entity me)
104 me.saveCvars_Callback(me);
105 me.saveCvars_Callback_func(me.saveCvars_Callback_ent, me);
107 void makeCallback(entity e, entity cbent, void(entity, entity) cbfunc)
109 e.saveCvars_Callback = e.saveCvars;
110 e.saveCvars = saveCvarsCallback;
111 e.saveCvars_Callback_ent = cbent;
112 e.saveCvars_Callback_func = cbfunc;
115 .void(entity) draw_setDependent;
116 .string cvar_setDependent;
117 .float cvarMin_setDependent;
118 .float cvarMax_setDependent;
119 .string cvar2_setDependent;
120 .float cvar2Min_setDependent;
121 .float cvar2Max_setDependent;
122 .string cvar3_setDependent;
123 .float cvar3Min_setDependent;
124 .float cvar3Max_setDependent;
125 .float op_setDependent;
126 .string cvarString_setDependent;
127 .string cvarValue_setDependent;
128 .float(entity) func_setDependent;
129 void setDependent_Check(entity e)
133 if(e.func_setDependent)
135 e.disabled = !(e.func_setDependent(e));
137 else if(e.cvarString_setDependent)
139 s = cvar_string(e.cvarString_setDependent);
140 e.disabled = (cvar_string(e.cvarString_setDependent) == e.cvarValue_setDependent);
144 if(e.cvar_setDependent)
146 f = cvar(e.cvar_setDependent);
147 if(e.cvarMin_setDependent <= e.cvarMax_setDependent)
148 e.disabled = ((f < e.cvarMin_setDependent) || (f > e.cvarMax_setDependent));
150 e.disabled = ((f >= e.cvarMax_setDependent) && (f <= e.cvarMin_setDependent));
152 if(e.cvar2_setDependent)
154 f = cvar(e.cvar2_setDependent);
155 if(e.cvar2Min_setDependent <= e.cvar2Max_setDependent)
156 e.disabled = (e.disabled + ((f < e.cvar2Min_setDependent) || (f > e.cvar2Max_setDependent)) > e.op_setDependent);
158 e.disabled = (e.disabled + ((f >= e.cvar2Max_setDependent) && (f <= e.cvar2Min_setDependent)) > e.op_setDependent);
160 if(e.cvar3_setDependent)
162 f = cvar(e.cvar3_setDependent);
163 if(e.cvar3Min_setDependent <= e.cvar3Max_setDependent)
164 e.disabled = (e.disabled + ((f < e.cvar3Min_setDependent) || (f > e.cvar3Max_setDependent)) > e.op_setDependent);
166 e.disabled = (e.disabled + ((f >= e.cvar3Max_setDependent) && (f <= e.cvar3Min_setDependent)) > e.op_setDependent);
170 void setDependent_Draw(entity e)
172 setDependent_Check(e);
173 e.draw_setDependent(e);
175 void setDependent(entity e, string theCvarName, float theCvarMin, float theCvarMax)
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);
187 void setDependentStringNotEqual(entity e, string theCvarName, string theCvarValue)
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);
199 void setDependentAND(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max)
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);
214 void setDependentOR(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max)
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);
229 void setDependentAND3(entity e, string theCvarName, float theCvarMin, float theCvarMax, string theCvar2Name, float theCvar2Min, float theCvar2Max, string theCvar3Name, float theCvar3Min, float theCvar3Max)
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);
246 void setDependentWeird(entity e, float(entity) func)
248 e.draw_setDependent = e.draw;
249 e.func_setDependent = func;
250 e.draw = setDependent_Draw;
251 setDependent_Check(e);
254 // URI SYSTEM ////////////////////////////////////////////////////////
256 float _Nex_ExtResponseSystem_Queried;
257 string _Nex_ExtResponseSystem_UpdateTo;
258 string _Nex_ExtResponseSystem_UpdateToURL;
259 string _Nex_ExtResponseSystem_Packs;
260 float _Nex_ExtResponseSystem_PacksStep;
262 void URI_Get_Callback(float id, float status, string data)
264 if(url_URI_Get_Callback(id, status, data))
268 else if (id == URI_GET_DISCARD)
272 else if (id >= URI_GET_CURL && id <= URI_GET_CURL_END)
275 Curl_URI_Get_Callback(id, status, data);
277 else if (id == URI_GET_UPDATENOTIFICATION)
279 UpdateNotification_URI_Get_Callback(id, status, data);
283 print(sprintf(_("Received HTTP request data for an invalid id %d.\n"), id));
287 void UpdateNotification_URI_Get_Callback(float id, float status, string data)
291 if(_Nex_ExtResponseSystem_UpdateTo)
293 dprint("error: UpdateNotification_URI_Get_Callback has been called before\n");
298 print(sprintf(_("error receiving update notification: status is %d\n"), status));
301 if(substring(data, 0, 1) == "<")
303 print(_("error: received HTML instead of an update notification\n"));
306 if(strstrofs(data, "\r", 0) != -1)
308 print(_("error: received carriage returns from update notification server\n"));
315 n = tokenizebyseparator(data, "\n");
319 _Nex_ExtResponseSystem_UpdateTo = argv(0);
321 if(vercmp(cvar_string("g_xonoticversion"), _Nex_ExtResponseSystem_UpdateTo) >= 0)
323 _Nex_ExtResponseSystem_UpdateTo = ""; // no update needed
329 print(sprintf(_("Update can be downloaded at:\n%s\n"), argv(1)));
331 _Nex_ExtResponseSystem_UpdateToURL = strzone(argv(2));
334 _Nex_ExtResponseSystem_UpdateTo = strzone(_Nex_ExtResponseSystem_UpdateTo);
338 _Nex_ExtResponseSystem_BannedServers = strzone(argv(3));
339 _Nex_ExtResponseSystem_BannedServersNeedsRefresh = 1;
344 if(cvar("menu_updatecheck_getpacks"))
346 _Nex_ExtResponseSystem_Packs = strzone(argv(4));
347 _Nex_ExtResponseSystem_PacksStep = 1;
353 // END OF URI SYSTEM ////////////////////////////////////////////////////////
360 MapInfo_Cache_Create();
362 if(!MapInfo_FilterGametype(MAPINFO_TYPE_ALL, 0, 0, 0, 1))
364 draw_reset_cropped();
366 sz = eX * 0.025 + eY * 0.025 * (draw_scale_x / draw_scale_y);
367 draw_CenterText('0.5 0.5 0' - 1.25 * sz_y * eY, _("Autogenerating mapinfo for newly added maps..."), sz, '1 1 1', 1, 0);
369 boxA = '0.05 0.5 0' + 0.25 * sz_y * eY;
370 boxB = '0.95 0.5 0' + 1.25 * sz_y * eY;
371 draw_Fill(boxA, boxB - boxA, '1 1 1', 1);
375 draw_Fill(boxA, boxB - boxA, '0.1 0.1 0.1', 1);
377 boxB_x = boxA_x * (1 - MapInfo_progress) + boxB_x * MapInfo_progress;
378 draw_Fill(boxA, boxB - boxA, '0 0 1', 1);
385 string campaign_name_previous;
386 float campaign_won_previous;
388 var string autocvar_menu_watermark = WATERMARK();
390 var string autocvar_menu_watermark = "";
394 if(autocvar_menu_watermark != "")
396 vector fs = '48 48 0';
397 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);
402 vector fs, sz, line, mid;
404 if(cvar("menu_updatecheck"))
406 if(!_Nex_ExtResponseSystem_Queried)
408 _Nex_ExtResponseSystem_Queried = 1;
412 cvar_set("cl_startcount", ftos(startcnt = cvar("cl_startcount") + 1));
414 // for privacy, munge the start count a little
415 startcnt = floor((floor(startcnt / 10) + random()) * 10);
416 uri = sprintf("http://www.xonotic.org/dl/checkupdate.txt?version=%s&cnt=%d", uri_escape(cvar_string("g_xonoticversion")), startcnt);
419 float cvar_handle, popcon_handle;
422 cvar_handle = buf_create();
423 buf_cvarlist(cvar_handle, "", "");
424 n = buf_getsize(cvar_handle);
425 popcon_handle = buf_create();
426 for(i= 0, j = 0; i < n; ++i)
428 k = bufstr_get(cvar_handle, i);
429 if(!(cvar_type(k) & CVAR_TYPEFLAG_SAVED))
431 s = sprintf("%s=%d", uri_escape(k), cvar_string(k) != cvar_defstring(k));
432 bufstr_set(popcon_handle, j, s);
435 buf_del(cvar_handle);
437 uri, URI_GET_UPDATENOTIFICATION,
438 "application/x-www-form-urlencoded",
442 buf_del(popcon_handle);
444 uri_get(uri, URI_GET_UPDATENOTIFICATION);
449 if(_Nex_ExtResponseSystem_PacksStep > 0)
453 n = tokenize_console(_Nex_ExtResponseSystem_Packs);
455 for(i = 0; i+1 < n; i += 2)
457 if(fexists(argv(i+1)))
460 if(_Nex_ExtResponseSystem_PacksStep == 1) // first run
461 localcmd("\ncurl --pak \"", argv(i), "\"\n");
465 if(_Nex_ExtResponseSystem_PacksStep == 2)
468 cvar_set("_menu_initialized", "0");
469 // HACK: cause m_hide call on next start
470 localcmd("\nmenu_restart\n");
472 _Nex_ExtResponseSystem_PacksStep = 0;
475 _Nex_ExtResponseSystem_PacksStep = 2;
478 if(_Nex_ExtResponseSystem_UpdateTo != "")
480 // TODO rather turn this into a dialog
481 fs = ((1/draw_scale_x) * eX + (1/draw_scale_y) * eY) * 12;
484 l1 = sprintf(_("Update to %s now!"), _Nex_ExtResponseSystem_UpdateTo);
485 l2 = "http://www.xonotic.org/";
486 if(_Nex_ExtResponseSystem_UpdateToURL)
487 l2 = _Nex_ExtResponseSystem_UpdateToURL;
489 sz_x = draw_TextWidth(" ", 0, fs) + max(
490 draw_TextWidth(l1, 0, fs),
491 draw_TextWidth(l2, 0, fs)
495 draw_alpha = bound(0, sin(time * 0.112 - 0.3) * 10, 1);
496 mid = eX * (0.5 + 0.5 * (1 - sz_x) * cos(time * 0.071))
497 + eY * (0.5 + 0.5 * (1 - sz_y) * sin(time * 0.071));
499 draw_Fill(mid - 0.5 * sz, sz, '1 1 0', 1);
500 draw_CenterText(mid - 1 * line, l1, fs, '1 0 0', 1, 0);
501 draw_CenterText(mid - 0 * line, l2, fs, '0 0 1', 1, 0);
503 if not(campaign_name_previous)
504 campaign_name_previous = strzone(strcat(campaign_name, "x")); // force unequal
505 if(campaign_name == campaign_name_previous)
507 if(cvar(strcat("g_campaign", campaign_name, "_won")))
509 if(!campaign_won_previous)
512 DialogOpenButton_Click_withCoords(NULL, main.winnerDialog, '0 0 0', eX * conwidth + eY * conheight);
514 campaign_won_previous = 1;
517 campaign_won_previous = 0;
521 strunzone(campaign_name_previous);
522 campaign_name_previous = strzone(campaign_name);
523 campaign_won_previous = cvar(strcat("g_campaign", campaign_name, "_won"));
527 string resolvemod(string m)
530 return getcurrentmod();
535 float updateCompression()
537 float have_dds, have_jpg, have_tga;
539 have_dds = (fexists("dds/particles/particlefont.dds"));
540 have_jpg = (fexists("particles/particlefont.jpg"));
541 have_tga = (fexists("particles/particlefont.tga"));
542 can_dds = GL_Have_TextureCompression();
543 if(have_dds && (have_jpg || have_tga))
545 // both? Let's only use good quality precompressed files
546 // but ONLY if we actually support it!
549 // these builds are meant to have GOOD quality, so let's not compress non-skinframes
550 cvar_set("gl_texturecompression", "0");
553 //cvar_set("gl_texturecompression", cvar_string("r_texture_dds_load"));
558 cvar_set("gl_texturecompression", "0");
559 cvar_set("r_texture_dds_load", "0");
565 // DDS only? We probably always want texture compression
566 cvar_set("gl_texturecompression", "1");
567 cvar_set("r_texture_dds_load", "1");
569 print(_("^1ERROR: Texture compression is required but not supported.\n^1Expect visual problems.\n"));
574 // TGA only? Allow runtime compression
577 cvar_set("gl_texturecompression", cvar_string("r_texture_dds_load"));
582 cvar_set("gl_texturecompression", "0");
583 cvar_set("r_texture_dds_load", "0");
589 // note: include only those that should be in the menu!
591 GAMETYPE(MAPINFO_TYPE_ARENA) \
592 GAMETYPE(MAPINFO_TYPE_ASSAULT) \
593 GAMETYPE(MAPINFO_TYPE_CTF) \
594 GAMETYPE(MAPINFO_TYPE_CA) \
595 GAMETYPE(MAPINFO_TYPE_DEATHMATCH) \
596 GAMETYPE(MAPINFO_TYPE_DOMINATION) \
597 GAMETYPE(MAPINFO_TYPE_FREEZETAG) \
598 GAMETYPE(MAPINFO_TYPE_KEEPAWAY) \
599 GAMETYPE(MAPINFO_TYPE_KEYHUNT) \
600 GAMETYPE(MAPINFO_TYPE_LMS) \
601 GAMETYPE(MAPINFO_TYPE_NEXBALL) \
602 GAMETYPE(MAPINFO_TYPE_ONSLAUGHT) \
603 GAMETYPE(MAPINFO_TYPE_RACE) \
604 GAMETYPE(MAPINFO_TYPE_CTS) \
605 GAMETYPE(MAPINFO_TYPE_RUNEMATCH) \
606 GAMETYPE(MAPINFO_TYPE_TEAM_DEATHMATCH) \
609 float GameType_GetID(float cnt)
614 #define GAMETYPE(id) if(i++ == cnt) return id;
621 float GameType_GetCount()
626 #define GAMETYPE(id) ++i;
633 string GameType_GetName(float cnt)
635 float i = GameType_GetID(cnt);
638 return MapInfo_Type_ToText(i);
643 string GameType_GetIcon(float cnt)
645 float i = GameType_GetID(cnt);
648 return strcat("gametype_", MapInfo_Type_ToString(i));
653 /*string GameType_GetTeams(float cnt) // poor implementation, later something else could be done that's better?
655 float i = GameType_GetID(cnt);
656 string s = _MapInfo_GetDefaultEx(i);
660 if(strstrofs(s, "teams", 0) >= 0)
661 return _("teamplay");
663 return _("free for all");
666 return _("tuba for all");
669 void dialog_hudpanel_common_notoggle(entity me, string panelname)
675 me.TD(me, 1, 1.4, e = makeXonoticTextLabel(0, _("Background:")));
676 me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg"))));
677 e.addValue(e, _("Default"), "");
678 e.addValue(e, _("Disable"), "0");
679 e.addValue(e, strzone(strcat("border_", panelname)), strzone(strcat("border_", panelname)));
680 e.configureXonoticTextSliderValues(e);
683 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Color:")));
684 me.TD(me, 2, 2.6, e = makeXonoticColorpickerString(strzone(strcat("hud_panel_", panelname, "_bg_color")), "hud_panel_bg_color"));
685 setDependentStringNotEqual(e, strzone(strcat("hud_panel_", panelname, "_bg_color")), "");
688 me.TD(me, 1, 1.0, e = makeXonoticCheckBoxString("", "1 1 1", strzone(strcat("hud_panel_", panelname, "_bg_color")), _("Use default")));
691 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Border size:")));
692 me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_border"))));
693 e.addValue(e, _("Default"), "");
694 e.addValue(e, _("Disable"), "0");
695 for(i = 1; i <= 10; ++i)
696 e.addValue(e, strzone(ftos_decimals(i * 2, 0)), strzone(ftos(i * 2)));
697 e.configureXonoticTextSliderValues(e);
700 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Alpha:")));
701 me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_alpha"))));
702 e.addValue(e, _("Default"), "");
703 for(i = 1; i <= 10; ++i)
704 e.addValue(e, strzone(ftos_decimals(i/10, 1)), strzone(ftos(i/10)));
705 e.configureXonoticTextSliderValues(e);
708 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Team Color:")));
709 me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_color_team"))));
710 e.addValue(e, _("Default"), "");
711 e.addValue(e, _("Disable"), "0");
712 for(i = 1; i <= 10; ++i)
713 e.addValue(e, strzone(ftos_decimals(i/10, 1)), strzone(ftos(i/10)));
714 e.configureXonoticTextSliderValues(e);
717 me.TD(me, 1, 3.6, e = makeXonoticCheckBox(0, "hud_configure_teamcolorforced", _("Test team color in configure mode")));
720 me.TD(me, 1, 1.2, e = makeXonoticTextLabel(0, _("Padding:")));
721 me.TD(me, 1, 2.6, e = makeXonoticTextSlider(strzone(strcat("hud_panel_", panelname, "_bg_padding"))));
722 e.addValue(e, _("Default"), "");
723 for(i = 0; i <= 10; ++i)
724 e.addValue(e, strzone(ftos_decimals(i - 5, 0)), strzone(ftos(i - 5)));
725 e.configureXonoticTextSliderValues(e);