Xonotic QuakeC
The free, fast arena FPS with crisp movement and a wide array of weapons
intermission.qc
Go to the documentation of this file.
1#include "mapvoting.qh"
2
3#include <common/mapinfo.qh>
4#include <common/util.qh>
5#include <server/bot/api.qh>
7#include <server/campaign.qh>
8#include <server/client.qh>
9#include <server/mapvoting.qh>
11#include <server/world.qh>
12
14{
15 stuffcmd(pl, sprintf("\nsettemp _nextmap %s\n", get_nextmap()));
16}
17
18void Set_NextMap(string mapname)
19{
20 if (mapname != "")
22 else
24
26}
27
32
33string GetMapname()
34{
35 return mapname;
36}
37
39
40// NOTE: this now expects the map list to be already tokenized and the count in Map_Count
42{
43 string map = GetMapname();
45
46 if(idx >= 0)
47 {
48 if(idx < Map_Count)
49 {
50 if(map == bufstr_get(maplist_buffer, idx))
51 {
52 return idx;
53 }
54 }
55 }
56
57 for(int pos = 0; pos < Map_Count; ++pos)
58 {
59 if(map == bufstr_get(maplist_buffer, pos))
60 return pos;
61 }
62
63 // resume normal maplist rotation if current map is not in g_maplist
64 return idx;
65}
66
67bool MapHasRightSize(string map)
68{
69 int minplayers = max(0, floor(autocvar_minplayers));
70 if (teamplay)
73 && (currentbots || autocvar_bot_number || player_count < minplayers))
74 {
75 string checkwp_msg = strcat("checkwp ", map);
76 if(!fexists(strcat("maps/", map, ".waypoints")))
77 {
78 LOG_TRACE(checkwp_msg, ": no waypoints");
79 return false;
80 }
81 LOG_TRACE(checkwp_msg, ": has waypoints");
82 }
83
85 return true;
86
87 // open map size restriction file
88 if(!MapReadSizes(map))
89 return true; // map has no size restrictions
90
91 string checksize_msg = strcat("MapHasRightSize ", map);
92 int player_limit = ((autocvar_g_maplist_sizes_count_maxplayers) ? GetPlayerLimit() : 0);
93 int pcount = ((player_limit > 0) ? min(player_count, player_limit) : player_count); // bind it to the player limit so that forced spectators don't influence the limits
94
96 pcount -= currentbots;
97 pcount -= rint(cvar("g_maplist_sizes_specparty") * pcount);
98
99 // ensure small maps can be selected when pcount is low
101 map_minplayers = 0;
102
103 if(pcount < map_minplayers)
104 {
105 LOG_TRACE(checksize_msg, ": not enough");
106 return false;
107 }
108 if(map_maxplayers && pcount > map_maxplayers)
109 {
110 LOG_TRACE(checksize_msg, ": too many");
111 return false;
112 }
113 LOG_TRACE(checksize_msg, ": right size");
114 return true;
115}
116
117string Map_Filename(string m)
118{
119 return strcat("maps/", m, ".bsp");
120}
121
122void Map_MarkAsRecent(string m)
123{
125}
126
127bool Map_IsRecent(string m)
128{
130 return false;
132}
133
134bool Map_Check(string m, int pass)
135{
136 if(pass <= 1)
137 {
138 if(Map_IsRecent(m))
139 return false;
140 }
141 if(MapInfo_CheckMap(m))
142 {
143 if(pass == 2)
144 return true;
145 // MapInfo_Map_flags was set by MapInfo_CheckMap()
147 return false;
148 if(MapHasRightSize(m))
149 return true;
150 return false;
151 }
152 else
153 {
154 string filename = Map_Filename(m);
155 LOG_DEBUG( "Couldn't select '", filename, "'..." );
156 }
157
158 return false;
159}
160
161void Map_Goto_SetStr(string nextmapname)
162{
163 if(getmapname_stored != "")
165 if(nextmapname == "")
167 else
168 getmapname_stored = strzone(nextmapname);
169}
170
171void Map_Goto_SetIndex(int position)
172{
173 Map_Current = position;
174 cvar_set("g_maplist_index", ftos(position));
175 Map_Goto_SetStr(bufstr_get(maplist_buffer, position));
176}
177
178void Map_Goto(float reinit)
179{
181}
182
183// return codes of map selectors:
184// -1 = temporary failure (that is, try some method that is guaranteed to succeed)
185// -2 = permanent failure
186int MaplistMethod_Iterate(void) // usual method
187{
188 int attempts = 42; // skip advanced checks if this many maps fail
189
190 LOG_TRACE("Trying MaplistMethod_Iterate");
191
192 for(int i = 1; i < Map_Count; ++i)
193 {
194 int mapindex = (i + Map_Current) % Map_Count;
195 if(Map_Check(bufstr_get(maplist_buffer, mapindex), 1))
196 return mapindex;
197
198 attempts -= 1;
199 if(attempts <= 0)
200 {
201 LOG_DEBUG("MaplistMethod_Iterate: Couldn't find a suitable map, falling back to next valid");
202 break;
203 }
204 }
205
206 // failing that, just accept the next map in the list
207 int mapindex = (1 + Map_Current) % Map_Count;
208 return mapindex;
209}
210
211int MaplistMethod_Repeat(void) // fallback method
212{
213 LOG_TRACE("Trying MaplistMethod_Repeat");
214
215 if(Map_Check(bufstr_get(maplist_buffer, Map_Current), 2))
216 return Map_Current;
217 return -2;
218}
219
220int MaplistMethod_Random(void) // random map selection
221{
222 int i, imax;
223
224 LOG_TRACE("Trying MaplistMethod_Random");
225
226 imax = 42;
227
228 for(i = 0; i <= imax; ++i)
229 {
230 int mapindex;
231 mapindex = (Map_Current + floor(random() * (Map_Count - 1) + 1)) % Map_Count; // any OTHER map
232 if(Map_Check(bufstr_get(maplist_buffer, mapindex), 1))
233 return mapindex;
234 }
235 return -1;
236}
237
238// NOTE: call Maplist_Close when you're done!
240{
241 bool have_maps = false;
242 if(autocvar_g_maplist != "")
243 {
244 // make sure there is at least one playable map in the list
245 bool needtrim = false;
247 {
248 if(!fexists(Map_Filename(it)))
249 {
250 needtrim = true;
251 if(have_maps)
252 break;
253 continue;
254 }
255 if(have_maps || !Map_Check(it, 2))
256 continue;
257 have_maps = true;
258 if(needtrim)
259 break;
260 });
261
262 // additionally trim any non-existent maps
263 if(needtrim)
264 {
265 int trimmedmaps = 0;
266 string newmaplist = "";
268 {
269 if(!fexists(Map_Filename(it)))
270 {
271 trimmedmaps += 1;
272 continue;
273 }
274 newmaplist = cons(newmaplist, it);
275 });
276 cvar_set("g_maplist", newmaplist);
277 LOG_DEBUGF("Maplist_Init: trimmed %d missing maps from the list", trimmedmaps);
278 }
279 }
280
281 if (!have_maps)
282 {
283 bprint( "Maplist contains no usable maps! Resetting it to default map list.\n" );
286 localcmd("\nmenu_cmd sync\n");
287 }
288
290
291 int _cnt = 0;
293 {
294 // NOTE: inlined maplist_shuffle function to avoid a second buffer, keep both in sync
296 {
297 int _j = floor(random() * (_cnt + 1));
298 if(_j != _cnt)
299 bufstr_set(maplist_buffer, _cnt, bufstr_get(maplist_buffer, _j));
300 bufstr_set(maplist_buffer, _j, it);
301 ++_cnt;
302 }
303 else
304 bufstr_set(maplist_buffer, i, it);
305 });
306
307 Map_Count = buf_getsize(maplist_buffer);
308
309 if(Map_Count == 0)
310 error("empty maplist, cannot select a new map");
311
313
314 return Map_Count;
315}
316
318{
319 buf_del(maplist_buffer);
320}
321
322// NOTE: call Maplist_Init() before making GetNextMap() call(s)
323string GetNextMap(void)
324{
325 int nextMap = -1;
326
327 if(nextMap == -1 && autocvar_g_maplist_selectrandom)
328 nextMap = MaplistMethod_Random();
329
330 if(nextMap == -1)
331 nextMap = MaplistMethod_Iterate();
332
333 if(nextMap == -1)
334 nextMap = MaplistMethod_Repeat();
335
336 if(nextMap >= 0)
337 {
338 Map_Goto_SetIndex(nextMap);
339 return getmapname_stored;
340 }
341
342 return "";
343}
344
345float DoNextMapOverride(float reinit)
346{
348 {
350 alreadychangedlevel = true;
351 return true;
352 }
354 {
356 {
357 localcmd("quit\n");
358 alreadychangedlevel = true;
359 return true;
360 }
361 }
363 {
365 alreadychangedlevel = true;
366 return true;
367 }
368 if (!reinit && autocvar_samelevel) // if samelevel is set, stay on same level
369 {
370 localcmd("restart\n");
371 alreadychangedlevel = true;
372 return true;
373 }
374 if(get_nextmap() != "")
375 {
376 string m;
378 if (m != get_nextmap())
379 Set_NextMap(m);
380
381 if(!m || gametypevote)
382 return false;
384 {
386 return false;
387 }
388
389 if(MapInfo_CheckMap(m))
390 {
392 Map_Goto(reinit);
393 alreadychangedlevel = true;
394 return true;
395 }
396 }
397 if(!reinit && autocvar_lastlevel)
398 {
400 localcmd("set lastlevel 0\ntogglemenu 1\n");
401 alreadychangedlevel = true;
402 return true;
403 }
404 return false;
405}
406
407void GotoNextMap(float reinit)
408{
409 //string nextmap;
410 //float n, nummaps;
411 //string s;
413 return;
414 alreadychangedlevel = true;
415
416 Maplist_Init();
417 string nextMap = GetNextMap();
419 if(nextMap == "")
420 error("Everything is broken - cannot find a next map. Please report this to the developers.");
421 Map_Goto(reinit);
422}
423
424string GotoMap(string m)
425{
427 if (!m)
428 return "The map you suggested is not available on this server.";
430 if (!MapInfo_CheckMap(m))
431 return "The map you suggested does not support the current gametype.";
432 if (m != get_nextmap())
433 Set_NextMap(m);
435 cvar_set("_endmatch", "1");
437 {
438 if(DoNextMapOverride(0))
439 return "Map switch initiated.";
440 else
441 return "Hm... no. For some reason I like THIS map more.";
442 }
443 else
444 return "Map switch will happen after scoreboard.";
445}
446
447
448/*
449============
450IntermissionThink
451
452When the player presses attack or jump, change to the next level
453============
454*/
457{
459
460 float server_screenshot = (autocvar_sv_autoscreenshot && CS_CVAR(this).cvar_cl_autoscreenshot);
461 float client_screenshot = (CS_CVAR(this).cvar_cl_autoscreenshot == 2);
462
463 if( (server_screenshot || client_screenshot)
464 && ((this.autoscreenshot > 0) && (time > this.autoscreenshot)) )
465 {
466 this.autoscreenshot = -1;
467 if(IS_REAL_CLIENT(this))
468 {
469 // in old clients "cl_autoscreenshot_screenshot_s %s %s;" takes the screenshot
470 // "cl_autoscreenshot_screenshot %s %s;" does nothing because the cl_autoscreenshot_screenshot alias
471 // doesn't exist; the cl_autoscreenshot_screenshot dummy cvar is created
472 // so that "cl_autoscreenshot_screenshot %s %s" doesn't print any warning in console
473
474 // in new clients "cl_autoscreenshot_screenshot %s %s;" takes the screenshot
475 // even if the cl_autoscreenshot_screenshot cvar is created, only the cl_autoscreenshot_screenshot alias
476 // is executed since only the alias is executed if a cvar with the same name exists
477 // cl_autoscreenshot_screenshot_s is not run as cl_autoscreenshot_screenshot alias clears it
478 // (it doesn't delete it so that "unalias cl_autoscreenshot_screenshot_s;" doesn't print any warning)
479
480 // this stuffcmd is needed only for Xonotic 0.8.6 or lower
481 // the s in cl_autoscreenshot_screenshot_s stands for server alias (alias name can't be longer than 32)
482 stuffcmd(this, sprintf("\n"
483 "alias cl_autoscreenshot_screenshot_s \"screenshot screenshots/autoscreenshot/%s-%s.jpg\";"
484 "set cl_autoscreenshot_screenshot 0;", GetMapname(), matchid));
485
486 // keep only this stuffcmd after the next release
487 stuffcmd(this, sprintf("\ncl_autoscreenshot_screenshot %s %s;"
488 "echo \"^5A screenshot has been taken at request of the server.\"\n", GetMapname(), matchid));
489
490 // this stuffcmd is needed only for Xonotic 0.8.6 or lower
491 stuffcmd(this, "\n"
492 "cl_autoscreenshot_screenshot_s %s %s;"
493 "unset cl_autoscreenshot_screenshot;"
494 "unalias cl_autoscreenshot_screenshot_s;");
495 }
496 return;
497 }
498
500 return;
501
504 return;
505
507}
508
510{
511 if(!e.autoscreenshot) // initial call
512 {
513 e.autoscreenshot = time + 0.1;
514 SetResourceExplicit(e, RES_HEALTH, -2342); // health in the first intermission phase
515 for (int slot = 0; slot < MAX_WEAPONSLOTS; ++slot)
516 {
517 .entity weaponentity = weaponentities[slot];
518 if(e.(weaponentity))
519 {
520 e.(weaponentity).effects = EF_NODRAW;
521 if (e.(weaponentity).weaponchild)
522 e.(weaponentity).weaponchild.effects = EF_NODRAW;
523 }
524 }
525 if(IS_REAL_CLIENT(e))
526 {
527 stuffcmd(e, "\nscr_printspeed 1000000\n");
531 });
533 {
534 stuffcmd(e, sprintf("\ncd loop %s\n", RandomSelection_chosen_string));
535 }
536 msg_entity = e;
538 }
539 }
540}
int currentbots
Definition api.qh:104
int player_count
Definition api.qh:103
var entity(vector mins, vector maxs,.entity tofield) findbox_tofield_OrFallback
bool SetResourceExplicit(entity e, Resource res_type, float amount)
Sets the resource amount of an entity without calling any hooks.
bool gametypevote
Definition mapvoting.qc:48
#define PHYS_INPUT_BUTTON_JUMP(s)
Definition player.qh:151
#define PHYS_INPUT_BUTTON_HOOK(s)
Definition player.qh:155
#define PHYS_INPUT_BUTTON_USE(s)
Definition player.qh:158
#define PHYS_INPUT_BUTTON_ATCK(s)
Definition player.qh:150
#define PHYS_INPUT_BUTTON_ATCK2(s)
Definition player.qh:152
int cvar_settemp_restore()
Definition util.qc:851
string mapname
float effects
float time
const float EF_NODRAW
int autocvar_bot_number
Definition cvars.qh:64
int autocvar_minplayers
Definition cvars.qh:68
int autocvar_minplayers_per_team
Definition cvars.qh:69
#define buf_create
#define pass(name, colormin, colormax)
ERASEABLE bool fexists(string f)
Definition file.qh:4
string Map_Filename(string m)
void Maplist_Close()
void Set_NextMap(string mapname)
string GetMapname()
void Map_Goto_SetIndex(int position)
int Map_Current
void FixIntermissionClient(entity e)
void IntermissionThink(entity this)
int Map_Count
bool MapHasRightSize(string map)
void Map_Goto_SetStr(string nextmapname)
float autoscreenshot
int MaplistMethod_Iterate(void)
bool Map_IsRecent(string m)
int GetMaplistPosition()
string GotoMap(string m)
int MaplistMethod_Random(void)
int Maplist_Init(void)
bool Map_Check(string m, int pass)
string GetGametype()
int MaplistMethod_Repeat(void)
void Map_Goto(float reinit)
string GetNextMap(void)
float DoNextMapOverride(float reinit)
void Send_NextMap_To_Player(entity pl)
void Map_MarkAsRecent(string m)
void GotoNextMap(float reinit)
string autocvar_sv_intermission_cdtrack
bool intermission_running
float intermission_exittime
string _nextmap
bool autocvar_samelevel
bool alreadychangedlevel
bool autocvar_lastlevel
#define get_nextmap()
int maplist_buffer
bool autocvar_sv_autoscreenshot
#define FOREACH_WORD(words, cond, body)
Definition iter.qh:33
#define LOG_TRACE(...)
Definition log.qh:76
#define LOG_DEBUG(...)
Definition log.qh:80
#define LOG_DEBUGF(...)
Definition log.qh:81
bool MapReadSizes(string map)
Definition mapinfo.qc:1409
float MapInfo_CheckMap(string s)
Definition mapinfo.qc:1510
int MapInfo_RequiredFlags()
Definition mapinfo.qc:1679
string MapInfo_Type_ToString(Gametype t)
Definition mapinfo.qc:656
void MapInfo_LoadMap(string s, float reinit)
Definition mapinfo.qc:1523
Gametype MapInfo_CurrentGametype()
Definition mapinfo.qc:1490
int MapInfo_ForbiddenFlags()
Definition mapinfo.qc:1664
string MapInfo_ListAllowedMaps(Gametype type, float pRequiredFlags, float pForbiddenFlags)
Definition mapinfo.qc:1542
float _MapInfo_GetTeamPlayBool(Gametype t)
Definition mapinfo.qc:539
int map_minplayers
Definition mapinfo.qh:190
Gametype MapInfo_LoadedGametype
Definition mapinfo.qh:220
int map_maxplayers
Definition mapinfo.qh:191
int MapInfo_Map_flags
Definition mapinfo.qh:15
const int MAPINFO_FLAG_DONOTWANT
Definition mapinfo.qh:164
bool autocvar_g_campaign
Definition menu.qc:747
void localcmd(string command,...)
void cvar_set(string name, string value)
float MSG_ONE
Definition menudefs.qc:56
float bound(float min, float value, float max)
float cvar(string name)
float random(void)
void bprint(string text,...)
float min(float f,...)
void strunzone(string s)
float rint(float f)
string ftos(float f)
void WriteByte(float data, float dest, float desto)
float floor(float f)
string strzone(string s)
float max(float f,...)
strcat(_("^F4Countdown stopped!"), "\n^BG", _("Teams are too unbalanced."))
#define error
Definition pre.qh:6
entity msg_entity
Definition progsdefs.qc:63
float SVC_INTERMISSION
Definition progsdefs.qc:340
#define stuffcmd(cl,...)
Definition progsdefs.qh:23
ERASEABLE void RandomSelection_Init()
Definition random.qc:4
string RandomSelection_chosen_string
Definition random.qh:7
#define RandomSelection_AddString(s, weight, priority)
Definition random.qh:16
#define AVAILABLE_TEAMS
void CampaignPostIntermission()
Definition campaign.qc:239
int GetPlayerLimit()
Definition client.qc:2127
string GameTypeVote_MapInfo_FixName(string m)
Definition mapvoting.qc:103
void MapVote_Start()
Definition mapvoting.qc:743
float autocvar_g_maplist_shuffle
Definition mapvoting.qh:9
bool autocvar_g_maplist_check_waypoints
Definition mapvoting.qh:4
bool autocvar_sv_vote_gametype
Definition mapvoting.qh:24
int autocvar_g_maplist_index
Definition mapvoting.qh:5
int autocvar_g_maplist_mostrecent_count
Definition mapvoting.qh:7
bool autocvar_g_maplist_sizes_count_maxplayers
Definition mapvoting.qh:21
bool autocvar_g_maplist_ignore_sizes
Definition mapvoting.qh:20
bool autocvar_g_maplist_sizes_count_bots
Definition mapvoting.qh:22
#define autocvar_g_maplist
Definition mapvoting.qh:3
float mapvote_initialized
Definition mapvoting.qh:46
string getmapname_stored
Definition mapvoting.qh:45
string autocvar_g_maplist_mostrecent
Definition mapvoting.qh:6
bool autocvar_g_maplist_selectrandom
Definition mapvoting.qh:8
#define CS_CVAR(this)
Definition state.qh:51
ERASEABLE string strwords(string s, int w)
Definition string.qh:361
#define strfree(this)
Definition string.qh:59
#define strhasword(s, w)
Definition string.qh:370
#define strcpy(this, s)
Definition string.qh:52
ERASEABLE string cons(string a, string b)
Definition string.qh:276
bool teamplay
Definition teams.qh:59
#define IS_REAL_CLIENT(v)
Definition utils.qh:17
#define FOREACH_CLIENT(cond, body)
Definition utils.qh:50
entity weaponchild
Definition all.qh:376
const int MAX_WEAPONSLOTS
Definition weapon.qh:16
entity weaponentities[MAX_WEAPONSLOTS]
Definition weapon.qh:17
string matchid
Definition world.qh:63
string autocvar_quit_and_redirect
Definition world.qh:13
bool autocvar_quit_when_empty
Definition world.qh:15
bool autocvar_sv_dedicated
Definition world.qh:41
string redirection_target
Definition world.qh:67