1 float VoteCheckNasty(string cmd)
3 if(strstrofs(cmd, ";", 0) >= 0)
5 if(strstrofs(cmd, "\n", 0) >= 0)
7 if(strstrofs(cmd, "\r", 0) >= 0)
9 if(strstrofs(cmd, "$", 0) >= 0)
14 entity GetKickVoteVictim(string vote, string cmd, entity caller)
21 tokens = tokenize(vote);
25 if(substring(argv(1), 0, 1) == "#")
27 ns = substring(argv(1), 1, 999);
40 GetKickVoteVictim_reason = "";
41 for(i = t; i < tokens; ++i)
42 GetKickVoteVictim_reason = strcat(GetKickVoteVictim_reason, argv(i), " ");
43 GetKickVoteVictim_reason = substring(GetKickVoteVictim_reason, 0, strlen(GetKickVoteVictim_reason) - 1);
46 if(ns == ftos(n)) if(n >= 1) if(n <= maxclients)
49 if(clienttype(e) == CLIENTTYPE_REAL)
51 GetKickVoteVictim_newcommand = strcat(argv(0), " # ", ns);
57 print_to(caller, strcat("Usage: ", cmd, " ", argv(0), " #playernumber (as in \"status\")\n"));
61 float GameCommand_Vote(string s, entity e) {
62 if(argv(0) == "help") {
63 print_to(e, " vote COMMANDS ARGUMENTS. See 'vote help' for more info.");
65 } else if(argv(0) == "vote") {
67 print_to(e, "^1You have to supply a vote command. See help for more info.");
68 } else if(argv(1) == "help") {
70 } else if(argv(1) == "status") {
72 print_to(e, strcat("^7Vote for ", votecalledvote_display, "^7 called by ^7", VoteNetname(votecaller), "^7."));
74 print_to(e, "^1No vote called.");
76 } else if(argv(1) == "call") {
77 if(!e || cvar("sv_vote_call")) {
78 if(tourneyInMatchStage
79 && cvar("g_tourney_disable_spec_vote")
80 && e.classname != "player") {
81 print_to(e, "^1Error: Only players can call a vote during the match-stage.");
83 else if(timeoutStatus) { //don't allow a vote call during a timeout
84 print_to(e, "^1Error: You can not call a vote while a timeout is active.");
87 print_to(e, "^1There is already a vote called.");
92 print_to(e, "^1Your vote is empty. See help for more info.");
94 && time < e.vote_next) {
95 print_to(e, strcat("^1You have to wait ^2", ftos(e.vote_next - time), "^1 seconds before you can again call a vote."));
96 } else if(VoteCheckNasty(vote)) {
97 print_to(e, "Syntax error in command. See help for more info.");
98 } else if(VoteAllowed(strcat1(argv(2)))) { // strcat seems to be necessary
99 // remap chmap to gotomap (forces intermission)
100 if(vote == "chmap" || vote == "gotomap") // won't work without arguments
102 if(substring(vote, 0, 6) == "chmap ")
103 vote = strcat("gotomap ", substring(vote, 6, strlen(vote) - 6));
104 if(substring(vote, 0, 8) == "gotomap ")
106 if(!(vote = ValidateMap(substring(vote, 8, strlen(vote) - 8), e)))
108 vote = strcat("gotomap ", vote);
111 // make kick and kickban votes a bit nicer (and reject them if formatted badly)
112 if(substring(vote, 0, 5) == "kick " || substring(vote, 0, 8) == "kickban ")
115 if(!(victim = GetKickVoteVictim(vote, "vcall", e)))
117 vote = GetKickVoteVictim_newcommand;
118 votecalledvote_display = strzone(strcat("^1", vote, " (^7", victim.netname, "^1): ", GetKickVoteVictim_reason));
122 votecalledvote_display = strzone(strcat("^1", vote));
124 votecalledvote = strzone(vote);
126 votecalledmaster = FALSE;
127 votefinished = time + cvar("sv_vote_timeout");
128 votecaller = e; // remember who called the vote
130 e.vote_vote = 1; // of course you vote yes
131 e.vote_next = time + cvar("sv_vote_wait");
133 bprint("\{1}^2* ^3", VoteNetname(votecaller), "^2 calls a vote for ", votecalledvote_display, "\n");
134 if(cvar("sv_eventlog"))
135 GameLogEcho(strcat(":vote:vcall:", ftos(votecaller.playerid), ":", votecalledvote_display), TRUE);
136 VoteCount(); // needed if you are the only one
138 print_to(e, "^1This vote is not ok. See help for more info.");
142 print_to(e, "^1Vote calling is NOT allowed.");
144 } else if(argv(1) == "stop") {
146 print_to(e, "^1No vote called.");
147 } else if(e == votecaller) { // the votecaller can stop a vote
149 } else if(!e) { // server admin / console can too
151 } else if(e.vote_master) { // masters can too
154 print_to(e, "^1You are not allowed to stop that Vote.");
156 } else if(argv(1) == "master") {
157 if(cvar("sv_vote_master")) {
159 print_to(e, "^1There is already a vote called.");
162 votecalledmaster = TRUE;
163 votecalledvote = strzone("XXX");
164 votecalledvote_display = strzone("^3master");
165 votefinished = time + cvar("sv_vote_timeout");
166 votecaller = e; // remember who called the vote
168 e.vote_vote = 1; // of course you vote yes
169 e.vote_next = time + cvar("sv_vote_wait");
171 bprint("\{1}^2* ^3", VoteNetname(votecaller), "^2 calls a vote to become ^3master^2.\n");
172 if(cvar("sv_eventlog"))
173 GameLogEcho(strcat(":vote:vcall:", ftos(votecaller.playerid), ":", votecalledvote_display), FALSE);
174 VoteCount(); // needed if you are the only one
177 print_to(e, "^1Vote to become master is NOT allowed.");
179 } else if(argv(1) == "do") {
180 if(!e || e.vote_master) {
181 local string dovote, dovote_display;
182 dovote = VoteParse();
184 print_to(e, "^1Your command was empty. See help for more info.");
185 } else if(VoteCheckNasty(dovote)) {
186 print_to(e, "Syntax error in command. See help for more info.");
187 } else if(VoteAllowed(strcat1(argv(2)))) { // strcat seems to be necessary
188 if(dovote == "chmap" || dovote == "gotomap") // won't work without arguments
190 if(substring(dovote, 0, 6) == "chmap ")
191 dovote = strcat("gotomap ", substring(dovote, 6, strlen(dovote) - 6));
192 if(substring(dovote, 0, 8) == "gotomap ")
194 if(!(dovote = ValidateMap(substring(dovote, 8, strlen(dovote) - 8), e)))
196 dovote = strcat("gotomap ", dovote);
199 dovote_display = dovote;
200 if(substring(dovote, 0, 5) == "kick " || substring(dovote, 0, 8) == "kickban ")
203 if(!(victim = GetKickVoteVictim(dovote, "vdo", e)))
205 dovote = GetKickVoteVictim_newcommand;
206 dovote_display = strcat("^1", dovote, " (^7", victim.netname, "^1): ", GetKickVoteVictim_reason);
208 bprint("\{1}^2* ^3", VoteNetname(e), "^2 used his ^3master^2 status to do \"^2", dovote_display, "^2\".\n");
209 if(cvar("sv_eventlog"))
210 GameLogEcho(strcat(":vote:vdo:", ftos(e.playerid), ":", dovote_display), FALSE);
211 localcmd(strcat(dovote, "\n"));
213 print_to(e, "^1This command is not ok. See help for more info.");
216 print_to(e, "^1You are NOT a master. You might need to login or vote to become master first. See help for more info.");
218 } else if(argv(1) == "login") {
219 local string masterpwd;
220 masterpwd = cvar_string("sv_vote_master_password");
221 if(masterpwd != "") {
223 granted = (masterpwd == argv(2));
225 e.vote_master = granted;
227 ServerConsoleEcho(strcat("Accepted master login from ", VoteNetname(e)), TRUE);
228 bprint("\{1}^2* ^3", VoteNetname(e), "^2 logged in as ^3master^2\n");
229 if(cvar("sv_eventlog"))
230 GameLogEcho(strcat(":vote:vlogin:", ftos(e.playerid)), FALSE);
233 ServerConsoleEcho(strcat("REJECTED master login from ", VoteNetname(e)), TRUE);
236 print_to(e, "^1Login to become master is NOT allowed.");
237 } else if(argv(1) == "yes") {
239 print_to(e, "^1No vote called.");
241 print_to(e, "^1You can't vote from the server console.");
242 } else if(e.vote_vote == 0
243 || cvar("sv_vote_change")) {
244 print_to(e, "^1You accepted the vote.");
246 centerprint_expire(e, CENTERPRIO_VOTE);
247 if(!cvar("sv_vote_singlecount")) {
251 print_to(e, "^1You have already voted.");
253 } else if(argv(1) == "no") {
255 print_to(e, "^1No vote called.");
257 print_to(e, "^1You can't vote from the server console.");
258 } else if(e.vote_vote == 0
259 || cvar("sv_vote_change")) {
260 print_to(e, "^1You rejected the vote.");
262 centerprint_expire(e, CENTERPRIO_VOTE);
263 if(!cvar("sv_vote_singlecount")) {
267 print_to(e, "^1You have already voted.");
269 } else if(argv(1) == "abstain" || argv(1) == "dontcare") {
271 print_to(e, "^1No vote called.");
273 print_to(e, "^1You can't vote from the server console.");
274 } else if(e.vote_vote == 0
275 || cvar("sv_vote_change")) {
276 print_to(e, "^1You abstained from your vote.");
278 centerprint_expire(e, CENTERPRIO_VOTE);
279 if(!cvar("sv_vote_singlecount")) {
283 print_to(e, "^1You have already voted.");
287 print_to(e, "^1Unknown vote command.");
294 void VoteHelp(entity e) {
295 local string vmasterdis;
296 if(!cvar("sv_vote_master")) {
297 vmasterdis = " ^1(disabled)";
300 local string vlogindis;
301 if("" == cvar_string("sv_vote_master_password")) {
302 vlogindis = " ^1(disabled)";
305 local string vcalldis;
306 if(!cvar("sv_vote_call")) {
307 vcalldis = " ^1(disabled)";
310 print_to(e, "^7You can use voting with \"^2cmd vote help^7\" \"^2cmd vote status^7\" \"^2cmd vote call ^3COMMAND ARGUMENTS^7\" \"^2cmd vote stop^7\" \"^2cmd vote master^7\" \"^2cmd vote login^7\" \"^2cmd vote do ^3COMMAND ARGUMENTS^7\" \"^2cmd vote yes^7\" \"^2cmd vote no^7\" \"^2cmd vote abstain^7\" \"^2cmd vote dontcare^7\".");
311 print_to(e, "^7Or if your version is up to date you can use these aliases \"^2vhelp^7\" \"^2vstatus^7\" \"^2vcall ^3COMMAND ARGUMENTS^7\" \"^2vstop^7\" \"^2vmaster^7\" \"^2vlogin^7\" \"^2vdo ^3COMMAND ARGUMENTS^7\" \"^2vyes^7\" \"^2vno^7\" \"^2abstain^7\" \"^2vdontcare^7\".");
312 print_to(e, "^7\"^2help^7\" shows this info.");
313 print_to(e, "^7\"^2status^7\" shows if there is a vote called and who called it.");
314 print_to(e, strcat("^7\"^2call^7\" is used to call a vote. See the list of allowed commands.", vcalldis, "^7"));
315 print_to(e, "^7\"^2stop^7\" can be used by the vote caller or an admin to stop a vote and maybe correct it.");
316 print_to(e, strcat("^7\"^2master^7\" call a vote to become master who can execute commands without a vote", vmasterdis, "^7"));
317 print_to(e, strcat("^7\"^2login^7\" login to become master who can execute commands without a vote.", vlogindis, "^7"));
318 print_to(e, "^7\"^2do^7\" executes a command if you are a master. See the list of allowed commands.");
319 print_to(e, "^7\"^2yes^7\", \"^2no^7\", \"^2abstain^7\" and \"^2dontcare^7\" to make your vote.");
320 print_to(e, "^7If enough of the players vote yes the vote is accepted.");
321 print_to(e, "^7If enough of the players vote no the vote is rejected.");
322 print_to(e, strcat("^7If neither the vote will timeout after ", cvar_string("sv_vote_timeout"), "^7 seconds."));
323 print_to(e, "^7You can call a vote for or execute these commands:");
324 print_to(e, strcat("^3", cvar_string("sv_vote_commands"), "^7 and maybe further ^3arguments^7"));
327 string VoteNetname(entity e)
332 if(cvar_string("sv_adminnick") != "") {
333 return cvar_string("sv_adminnick");
335 return cvar_string("hostname");
340 string ValidateMap(string m, entity e)
343 m = MapInfo_FixName(m);
346 print_to(e, "This map is not available on this server.");
350 if(!cvar("sv_vote_change_gametype"))
351 if(!IsSameGametype(m))
353 print_to(e, "This server does not allow changing the game type by map votes.");
357 if(!cvar("sv_vote_override_mostrecent"))
360 print_to(e, "This server does not allow for recent maps to be played again. Please be patient for some rounds.");
364 if(!MapInfo_CheckMap(m))
366 print_to(e, strcat("^1Invalid mapname, \"^3", m, "^1\" does not support the current game mode."));
370 if(!TryFile(strcat("maps/", m, ".mapcfg")))
372 print_to(e, strcat("^1Invalid mapname, \"^3", m, "^1\" does not exist on this server."));
382 if(votefinished > 0) // a vote was called
383 if(time > votefinished) // time is up
394 while(argv(index) != "") {
395 vote = strcat(vote, " ", argv(index));
399 // necessary for some of the string operations
400 vote = strzone(vote);
405 float VoteAllowed(string votecommand) {
406 tokenize(cvar_string("sv_vote_commands"));
409 while(argv(index) != "") {
410 if(votecommand == argv(index)) {
421 FOR_EACH_CLIENT(player)
423 player.vote_vote = 0;
424 centerprint_expire(player, CENTERPRIO_VOTE);
429 strunzone(votecalledvote);
430 strunzone(votecalledvote_display);
434 votecalledmaster = FALSE;
439 bprint("\{1}^2* ^3", VoteNetname(votecaller), "^2's vote for ^1", votecalledvote_display, "^2 was accepted\n");
443 votecaller.vote_master = 1;
446 //in g_tourney mode and if the vote is a timelimit-change, don't change it immediately but after restart
448 && substring(votecalledvote, 0, 10) == "timelimit ") {
449 if( stof(substring(votecalledvote, 10, strlen(votecalledvote) - 10)) > 0 ) {
450 timelimit_orig = stof(substring(votecalledvote, 10, strlen(votecalledvote) - 10));
451 bprint(strcat("The timelimit will be set to ", ftos(timelimit_orig), " minutes after the next restart!\n"));
453 else //calls like "timelimit -1" can pass immediately
454 localcmd(strcat(votecalledvote, "\n"));
457 localcmd(strcat(votecalledvote, "\n"));
460 votecaller.vote_next = 0; // people like your votes,
461 // no wait for next vote
467 bprint("\{1}^2* ^3", VoteNetname(votecaller), "^2's vote for ", votecalledvote_display, "^2 was rejected\n");
472 bprint("\{1}^2* ^3", VoteNetname(votecaller), "^2's vote for ", votecalledvote_display, "^2 timed out\n");
476 void VoteStop(entity stopper) {
477 bprint("\{1}^2* ^3", VoteNetname(stopper), "^2 stopped ^3", VoteNetname(votecaller), "^2's vote\n");
478 if(cvar("sv_eventlog"))
479 GameLogEcho(strcat(":vote:vstop:", ftos(stopper.playerid)), FALSE);
480 if(stopper == votecaller) {
481 // no wait for next vote so you can correct your vote
483 votecaller.vote_next = 0;
491 if(self.vote_vote == 0)
492 centerprint_atprio(self, CENTERPRIO_VOTE, strcat("^7^3", VoteNetname(votecaller), "^2 called a vote for ", votecalledvote_display, "\n\n^2You have not voted yet!\n^2HINT: By default, F1 is yes and F2 is no."));
495 void VoteSpam(float yescount, float nocount, float abstaincount, float notvoters, float mincount, string result)
500 s = strcat("\{1}^2* vote results: ^1", ftos(yescount), "^2:^1");
501 s = strcat(s, ftos(nocount), "^2 (^1");
502 s = strcat(s, ftos(mincount), "^2 needed), ^1");
503 s = strcat(s, ftos(abstaincount), "^2 didn't care, ^1");
504 s = strcat(s, ftos(notvoters), "^2 didn't vote\n");
508 s = strcat("\{1}^2* vote results: ^1", ftos(yescount), "^2:^1");
509 s = strcat(s, ftos(nocount), "^2, ^1");
510 s = strcat(s, ftos(abstaincount), "^2 didn't care, ^1");
511 s = strcat(s, ftos(notvoters), "^2 didn't have to vote\n");
514 if(cvar("sv_eventlog"))
516 s = strcat(":vote:v", result, ":", ftos(yescount));
517 s = strcat(s, ":", ftos(nocount));
518 s = strcat(s, ":", ftos(abstaincount));
519 s = strcat(s, ":", ftos(notvoters));
520 s = strcat(s, ":", ftos(mincount));
521 GameLogEcho(s, FALSE);
526 local float playercount;
528 local float yescount;
532 local float abstaincount;
535 //same for real players
536 local float realplayercount;
537 local float realplayeryescount;
538 local float realplayernocount;
539 local float realplayerabstaincount;
540 realplayercount = realplayernocount = realplayerabstaincount = realplayeryescount = 0;
542 FOR_EACH_REALCLIENT(player)
544 if(player.vote_vote == -1) {
546 } else if(player.vote_vote == 1) {
548 } else if(player.vote_vote == -2) {
552 //do the same for real players
553 if(player.classname == "player") {
554 if(player.vote_vote == -1) {
556 } else if(player.vote_vote == 1) {
557 realplayeryescount++;
558 } else if(player.vote_vote == -2) {
559 realplayerabstaincount++;
565 //in tournament mode, if we have at least one player then don't make the vote dependent on spectators (so specs don't have to press F1)
566 if(cvar("g_tourney"))
567 if(realplayercount > 0) {
568 yescount = realplayeryescount;
569 nocount = realplayernocount;
570 abstaincount = realplayerabstaincount;
571 playercount = realplayercount;
576 && playercount == 1) {
577 // if only one player is on the server becoming vote
578 // master is not allowed. This could be used for
579 // trolling or worse. 'self' is the user who has
580 // called the vote because this function is called
581 // by SV_ParseClientCommand. Maybe all voting should
582 // be disabled for a single player?
583 print_to(votecaller, "^1You are the only player on this server so you can not become vote master.");
585 votecaller.vote_next = 0;
590 votefactor = bound(0.5, cvar("sv_vote_majority_factor"), 0.999);
591 if(yescount > (playercount - abstaincount) * votefactor)
593 VoteSpam(yescount, nocount, abstaincount, playercount - yescount - nocount - abstaincount, -1, "yes");
596 else if(nocount >= (playercount - abstaincount) * (1 - votefactor)) // that means, yescount cannot reach minyes any more
598 VoteSpam(yescount, nocount, abstaincount, playercount - yescount - nocount - abstaincount, -1, "no");
601 else if(time > votefinished)
603 if(cvar("sv_vote_simple_majority"))
606 if(yescount > (yescount + nocount) * votefactor)
608 else if(yescount + nocount > 0)
612 VoteSpam(yescount, nocount, abstaincount, playercount - yescount - nocount - abstaincount, floor((yescount + nocount) * votefactor) + 1, result);
615 else if(result == "no")
622 VoteSpam(yescount, nocount, abstaincount, playercount - yescount - nocount - abstaincount, floor((playercount - abstaincount) * votefactor) + 1, "timeout");