3 #include <common/command/_mod.qh>
5 #include <common/constants.qh>
6 #include <common/teams.qh>
7 #include <common/util.qh>
8 #include <common/sounds/sound.qh>
9 #include <common/weapons/all.qh>
11 // Operator for bold notifications
12 #define BOLD_OPERATOR "^BOLD"
14 /** main types/groups of notifications */
16 /** "Global" AND "personal" announcer messages */
18 /** "Global" information messages */
20 /** "Personal" centerprint messages */
22 /** Subcall MSG_INFO and/or MSG_CENTER notifications */
24 /** Choose which subcall wrapper to activate */
26 /** Kill centerprint message @deprecated */
27 CASE(MSG, CENTER_KILL)
30 string Get_Notif_TypeName(MSG net_type)
34 case MSG_ANNCE: return "MSG_ANNCE";
35 case MSG_INFO: return "MSG_INFO";
36 case MSG_CENTER: return "MSG_CENTER";
37 case MSG_MULTI: return "MSG_MULTI";
38 case MSG_CHOICE: return "MSG_CHOICE";
39 case MSG_CENTER_KILL: return "MSG_CENTER_KILL";
41 LOG_WARNF("Get_Notif_TypeName(%d): Improper net type!", ORDINAL(net_type));
46 CASE(CPID, ASSAULT_ROLE)
49 CASE(CPID, CTF_CAPSHIELD)
50 CASE(CPID, CTF_LOWPRIO)
57 CASE(CPID, PREVENT_JOIN)
59 CASE(CPID, KEEPAWAY_WARN)
61 CASE(CPID, KEYHUNT_OTHER)
63 CASE(CPID, MISSING_TEAMS)
64 CASE(CPID, MISSING_PLAYERS)
65 CASE(CPID, INSTAGIB_FINDAMMO)
68 CASE(CPID, ONS_CAPSHIELD)
71 CASE(CPID, RACE_FINISHLAP)
73 CASE(CPID, TEAMCHANGE)
77 CASE(CPID, VEHICLES_OTHER)
82 USING(Notification, entity);
84 // used for notification system multi-team identifiers
85 #define APP_TEAM_NUM(num, prefix) ((num == NUM_TEAM_1) ? prefix##_RED : ((num == NUM_TEAM_2) ? prefix##_BLUE : ((num == NUM_TEAM_3) ? prefix##_YELLOW : prefix##_PINK)))
86 #define APP_NUM(num, prefix) ((num) ? APP_TEAM_NUM(num, prefix) : prefix##_NEUTRAL)
88 #define EIGHT_VARS_TO_VARARGS_VARLIST \
90 VARITEM(2, 0, XPD(s1, s2)) \
91 VARITEM(3, 0, XPD(s1, s2, s3)) \
92 VARITEM(4, 0, XPD(s1, s2, s3, s4)) \
94 VARITEM(1, 1, XPD(s1, f1)) \
95 VARITEM(2, 1, XPD(s1, s2, f1)) \
96 VARITEM(3, 1, XPD(s1, s2, s3, f1)) \
97 VARITEM(4, 1, XPD(s1, s2, s3, s4, f1)) \
98 VARITEM(0, 2, XPD(f1, f2)) \
99 VARITEM(1, 2, XPD(s1, f1, f2)) \
100 VARITEM(2, 2, XPD(s1, s2, f1, f2)) \
101 VARITEM(3, 2, XPD(s1, s2, s3, f1, f2)) \
102 VARITEM(4, 2, XPD(s1, s2, s3, s4, f1, f2)) \
103 VARITEM(0, 3, XPD(f1, f2, f3)) \
104 VARITEM(1, 3, XPD(s1, f1, f2, f3)) \
105 VARITEM(2, 3, XPD(s1, s2, f1, f2, f3)) \
106 VARITEM(3, 3, XPD(s1, s2, s3, f1, f2, f3)) \
107 VARITEM(4, 3, XPD(s1, s2, s3, s4, f1, f2, f3)) \
108 VARITEM(0, 4, XPD(f1, f2, f3, f4)) \
109 VARITEM(1, 4, XPD(s1, f1, f2, f3, f4)) \
110 VARITEM(2, 4, XPD(s1, s2, f1, f2, f3, f4)) \
111 VARITEM(3, 4, XPD(s1, s2, s3, f1, f2, f3, f4)) \
112 VARITEM(4, 4, XPD(s1, s2, s3, s4, f1, f2, f3, f4))
114 void Destroy_All_Notifications();
115 void Create_Notification_Entity(entity notif,
121 void Create_Notification_Entity_Annce(entity notif,
131 void Create_Notification_Entity_InfoCenter(entity notif,
136 /* MSG_INFO & MSG_CENTER */
145 void Create_Notification_Entity_Multi(entity notif,
149 Notification anncename,
150 Notification infoname,
151 Notification centername);
153 void Create_Notification_Entity_Choice(entity notif,
160 Notification optiona,
161 Notification optionb);
163 void Dump_Notifications(int fh, bool alsoprint);
165 #define DEFAULT_FILENAME "notifications_dump.cfg"
166 // NOTE: dumpeffectinfo, dumpnotifs, dumpturrets and dumpweapons use similar code
167 GENERIC_COMMAND(dumpnotifs, "Dump all notifications into " DEFAULT_FILENAME, false)
171 case CMD_REQUEST_COMMAND:
174 string filename = argv(1);
175 bool alsoprint = false;
178 filename = DEFAULT_FILENAME;
181 else if (filename == "-")
183 filename = DEFAULT_FILENAME;
186 int fh = fopen(filename, FILE_WRITE);
189 Dump_Notifications(fh, alsoprint);
190 LOG_INFOF("Dumping notifications... File located in ^2data/data/%s^7.", filename);
195 LOG_INFOF("^1Error: ^7Could not open file '%s'!", filename);
198 LOG_INFO("Notification dump command only works with cl_cmd and sv_cmd.");
203 case CMD_REQUEST_USAGE:
205 LOG_HELP("Usage:^3 ", GetProgramCommandPrefix(), " dumpnotifs [<filename>]");
206 LOG_HELPF(" Where <filename> is the file to write (default is %s),", DEFAULT_FILENAME);
207 LOG_HELP(" if supplied with '-' output to console as well as default,");
208 LOG_HELP(" if left blank, it will only write to default.");
213 #undef DEFAULT_FILENAME
215 #ifdef NOTIFICATIONS_DEBUG
216 bool autocvar_notification_debug = false;
217 void Debug_Notification(string input)
219 switch (autocvar_notification_debug)
221 case 1: { LOG_TRACE(input); break; }
222 case 2: { LOG_INFO(input); break; }
227 void Local_Notification(MSG net_type, Notification net_name, ...count);
228 /** glue for networking, forwards to `Local_Notification` */
229 void Local_Notification_WOVA(
230 MSG net_type, Notification net_name,
231 float stringcount, float floatcount,
232 string s1, string s2, string s3, string s4,
233 float f1, float f2, float f3, float f4);
236 string prev_soundfile;
237 float prev_soundtime;
241 IntrusiveList g_notifications;
242 STATIC_INIT(g_notifications) { g_notifications = IL_NEW(); }
247 /** send to one client and their spectators */
249 /** send ONLY to one client */
250 CASE(NOTIF, ONE_ONLY)
251 /** send only to X team and their spectators */
253 /** send only to X team and their spectators, except for Y person and their spectators */
254 CASE(NOTIF, TEAM_EXCEPT)
255 /** send to everyone */
257 /** send to everyone except X person and their spectators */
258 CASE(NOTIF, ALL_EXCEPT)
261 string Get_Notif_BroadcastName(NOTIF broadcast)
265 case NOTIF_ONE: return "NOTIF_ONE";
266 case NOTIF_ONE_ONLY: return "NOTIF_ONE_ONLY";
267 case NOTIF_ALL_EXCEPT: return "NOTIF_ALL_EXCEPT";
268 case NOTIF_ALL: return "NOTIF_ALL";
269 case NOTIF_TEAM: return "NOTIF_TEAM";
270 case NOTIF_TEAM_EXCEPT: return "NOTIF_TEAM_EXCEPT";
272 LOG_WARNF("Get_Notif_BroadcastName(%d): Improper broadcast!", broadcast);
276 void Kill_Notification(
277 NOTIF broadcast, entity client,
278 MSG net_type, CPID net_name);
279 void Send_Notification(
280 NOTIF broadcast, entity client,
281 MSG net_type, Notification net_name,
283 void Send_Notification_WOVA(
284 NOTIF broadcast, entity client,
285 MSG net_type, Notification net_name,
286 float stringcount, float floatcount,
287 string s1, string s2, string s3, string s4,
288 float f1, float f2, float f3, float f4);
289 void Send_Notification_WOCOVA(
290 NOTIF broadcast, entity client,
291 MSG net_type, Notification net_name,
292 string s1, string s2, string s3, string s4,
293 float f1, float f2, float f3, float f4);
296 // ===========================
297 // Special CVAR Declarations
298 // ===========================
300 // MAKE SURE THIS IS ALWAYS SYNCHRONIZED WITH THE DUMP
301 // NOTIFICATIONS FUNCTION IN THE .QC FILE!
303 #define NOTIF_ADD_AUTOCVAR(name,defaultvalue) float autocvar_notification_##name = defaultvalue;
305 float autocvar_notification_show_location = false;
306 string autocvar_notification_show_location_string = ""; //_(" at the %s");
307 float autocvar_notification_show_sprees = true;
308 float autocvar_notification_show_sprees_info = 3; // 0 = off, 1 = target only, 2 = attacker only, 3 = target and attacker
309 float autocvar_notification_show_sprees_info_newline = true;
310 float autocvar_notification_show_sprees_info_specialonly = true;
311 float autocvar_notification_errors_are_fatal = true;
313 float autocvar_notification_lifetime_runtime = 0.5;
314 float autocvar_notification_lifetime_mapload = 10;
318 void Notification_GetCvars(entity this, entity store);
319 float autocvar_notification_server_allows_location = 1; // 0 = no, 1 = yes
321 float autocvar_notification_item_centerprinttime = 1.5;
323 // 0 = no, 1 = yes, 2 = forced on for all MSG_INFO notifs
324 // DISABLED IN CODE, BUT ENABLED IN CONFIG FOR COMPATIBILITY WITH OLD CLIENTS
325 float autocvar_notification_allow_chatboxprint = 0;
327 float autocvar_notification_show_sprees_center = true;
328 float autocvar_notification_show_sprees_center_specialonly = true;
332 // ============================
333 // Notification Argument List
334 // ============================
336 These arguments get replaced with the Local_Notification_sprintf
337 and other such functions found in all.qc to supply data
338 from networked notifications to their usage in sprintf... It
339 allows for more dynamic data to be inferred by the local
340 notification parser, so that the server does not have to network
341 anything too crazy on a per-client/per-situation basis.
343 Pay attention to the CSQC/SVQC relations, some of these are redefined
344 in slightly different ways for different programs, this is because the
345 server does a more conservative approach to the notifs than the client.
347 All arguments are swapped into strings, so be sure that your
348 sprintf usage matches with proper %s placement.
350 Argument descriptions:
351 s1-s4: string arguments to be literally swapped into sprintf
352 s2loc: s2 string of locations of deaths or other events
353 s3loc: s3 string of locations of deaths or other events
354 f1-f4: float arguments expanded into strings to be swapped into sprintf
355 f1p2dec: f1 float to string with 2 decimal places
356 f2p2dec: f2 float to string with 2 decimal places
357 f2primsec: f2 float primary or secondary selection for weapons
358 f3primsec: f3 float primary or secondary selection for weapons
359 f1secs: count_seconds of f1
360 f1points: point or points depending on f1
361 f1ord: count_ordinal of f1
362 f1time: process_time of f1
363 f1race_time: TIME_ENCODED_TOSTRING of f1
364 f2race_time: TIME_ENCODED_TOSTRING of f2
365 race_col: color of race time/position (i.e. good or bad)
366 race_diff: show time difference between f2 and f3
367 missing_teams: show which teams still need players
368 pass_key: find the keybind for "passing" or "dropping" in CTF game mode
369 nade_key: find the keybind for nade throwing
370 frag_ping: show the ping of a player
371 frag_stats: show health/armor/ping of a player
372 frag_pos: show score status and position in the match of a player
373 spree_cen: centerprint notif for kill spree/how many kills they have
374 spree_inf: info notif for kill spree/how many kills they have
375 spree_end: placed at the end of murder messages to show ending of sprees
376 spree_lost: placed at the end of suicide messages to show losing of sprees
377 item_wepname: return full name of a weapon from weaponid
378 item_wepammo: ammo display for weapon from f1 and f2
379 item_centime: amount of time to display weapon message in centerprint
380 item_buffname: return full name of a buff from buffid
381 death_team: show the full name of the team a player is switching from
382 minigame1_name: return human readable name of a minigame from its id(s1)
383 minigame1_d: return descriptor name of a minigame from its id(s1)
386 const float NOTIF_MAX_ARGS = 7;
387 const float NOTIF_MAX_HUDARGS = 2;
388 const float NOTIF_MAX_DURCNT = 2;
391 const int NOTIF_QUEUE_MAX = 10;
392 entity notif_queue_entity[NOTIF_QUEUE_MAX];
393 MSG notif_queue_type[NOTIF_QUEUE_MAX];
394 float notif_queue_time[NOTIF_QUEUE_MAX];
396 float notif_queue_next_time;
397 int notif_queue_length;
399 void Local_Notification_Queue_Process();
402 string arg_slot[NOTIF_MAX_ARGS];
404 const float ARG_CS_SV_HA = 1; // enabled on CSQC, SVQC, and Hudargs
405 const float ARG_CS_SV_DC = 2; // enabled on CSQC, SVQC, and durcnt centerprint
406 const float ARG_CS_SV = 3; // enabled on CSQC and SVQC
407 const float ARG_CS = 4; // unique result to CSQC
408 const float ARG_SV = 5; // unique result to SVQC
409 const float ARG_DC = 6; // unique result to durcnt/centerprint
411 // todo possible idea.... declare how many floats/strings each arg needs, and then dynamically increment the input
412 // this way, we don't need to have duplicates like i.e. s2loc and s3loc?
414 string BUFF_NAME(int i);
416 #define NOTIF_ARGUMENT_LIST \
417 ARG_CASE(ARG_CS_SV_HA, "s1", s1) \
418 ARG_CASE(ARG_CS_SV_HA, "s2", s2) \
419 ARG_CASE(ARG_CS_SV_HA, "s3", s3) \
420 ARG_CASE(ARG_CS_SV_HA, "s4", s4) \
421 ARG_CASE(ARG_CS_SV, "s2loc", ((autocvar_notification_show_location && (s2 != "")) ? sprintf(( ((tmp_s = autocvar_notification_show_location_string) != "") ? tmp_s : _(" (near %s)") ), s2) : "")) \
422 ARG_CASE(ARG_CS_SV, "s3loc", ((autocvar_notification_show_location && (s3 != "")) ? sprintf(( ((tmp_s = autocvar_notification_show_location_string) != "") ? tmp_s : _(" (near %s)") ), s3) : "")) \
423 ARG_CASE(ARG_CS_SV_DC, "f1", ftos(f1)) \
424 ARG_CASE(ARG_CS_SV_DC, "f2", ftos(f2)) \
425 ARG_CASE(ARG_CS_SV, "f3", ftos(f3)) \
426 ARG_CASE(ARG_CS_SV, "f4", ftos(f4)) \
427 ARG_CASE(ARG_CS_SV, "f1dtime", ftos_decimals(TIME_DECODE(f1), 2)) \
428 ARG_CASE(ARG_CS_SV, "f2dtime", ftos_decimals(TIME_DECODE(f2), 2)) \
429 ARG_CASE(ARG_CS, "f2primsec", (f2 ? _("secondary") : _("primary"))) \
430 ARG_CASE(ARG_CS, "f3primsec", (f3 ? _("secondary") : _("primary"))) \
431 ARG_CASE(ARG_CS, "f1secs", count_seconds(f1)) \
432 ARG_CASE(ARG_CS, "f1points", (f1 == 1 ? _("point") : _("points"))) \
433 ARG_CASE(ARG_CS_SV, "f1ord", count_ordinal(f1)) \
434 ARG_CASE(ARG_CS_SV, "f1time", process_time(2, f1)) \
435 ARG_CASE(ARG_CS_SV_HA, "f1race_time", TIME_ENCODED_TOSTRING(f1, true)) \
436 ARG_CASE(ARG_CS_SV_HA, "f2race_time", TIME_ENCODED_TOSTRING(f2, true)) \
437 ARG_CASE(ARG_CS_SV_HA, "f3race_time", TIME_ENCODED_TOSTRING(f3, true)) \
438 ARG_CASE(ARG_CS_SV, "race_col", CCR(((f1 == 1) ? "^F1" : "^F2"))) \
439 ARG_CASE(ARG_CS_SV, "race_diff", ((f2 > f3) ? sprintf(CCR("^1[+%s]"), TIME_ENCODED_TOSTRING(f2 - f3, true)) : sprintf(CCR("^2[-%s]"), TIME_ENCODED_TOSTRING(f3 - f2, true)))) \
440 ARG_CASE(ARG_CS, "missing_teams", notif_arg_missing_teams(f1)) \
441 ARG_CASE(ARG_CS, "pass_key", getcommandkey(_("drop flag"), "+use")) \
442 ARG_CASE(ARG_CS, "nade_key", getcommandkey(_("throw nade"), "dropweapon")) \
443 ARG_CASE(ARG_CS, "join_key", getcommandkey(_("jump"), "+jump")) \
444 ARG_CASE(ARG_CS, "frag_ping", notif_arg_frag_ping(true, f2)) \
445 ARG_CASE(ARG_CS, "frag_stats", notif_arg_frag_stats(f2, f3, f4)) \
446 /*ARG_CASE(ARG_CS, "frag_pos", ((Should_Print_Score_Pos(f1)) ? sprintf("\n^BG%s", Read_Score_Pos(f1)) : ""))*/ \
447 ARG_CASE(ARG_CS, "spree_cen", (autocvar_notification_show_sprees ? notif_arg_spree_cen(f1) : "")) \
448 ARG_CASE(ARG_CS_SV, "spree_inf", (autocvar_notification_show_sprees ? notif_arg_spree_inf(1, input, s2, f2) : "")) \
449 ARG_CASE(ARG_CS_SV, "spree_end", (autocvar_notification_show_sprees ? notif_arg_spree_inf(-1, "", "", f1) : "")) \
450 ARG_CASE(ARG_CS_SV, "spree_lost", (autocvar_notification_show_sprees ? notif_arg_spree_inf(-2, "", "", f1) : "")) \
451 ARG_CASE(ARG_CS_SV, "item_wepname", REGISTRY_GET(Weapons, f1).m_name) \
452 ARG_CASE(ARG_CS_SV, "item_buffname", BUFF_NAME(f1)) \
453 ARG_CASE(ARG_CS_SV, "f3buffname", BUFF_NAME(f3)) \
454 ARG_CASE(ARG_CS_SV, "item_wepammo", (f2 > 0 ? notif_arg_item_wepammo(f1, f2) : "")) \
455 ARG_CASE(ARG_DC, "item_centime", ftos(autocvar_notification_item_centerprinttime)) \
456 ARG_CASE(ARG_SV, "death_team", Team_ColoredFullName(f1)) \
457 ARG_CASE(ARG_CS, "death_team", Team_ColoredFullName(f1 - 1)) \
458 ARG_CASE(ARG_CS_SV_HA, "minigame1_name",find(NULL,netname,s1).descriptor.message) \
459 ARG_CASE(ARG_CS_SV_HA, "minigame1_d", find(NULL,netname,s1).descriptor.netname)
461 #define NOTIF_HIT_MAX(count,funcname) MACRO_BEGIN \
462 if(sel_num == count) { backtrace(sprintf("%s: Hit maximum arguments!\n", funcname)); break; } \
465 #define NOTIF_HIT_UNKNOWN(token,funcname) { backtrace(sprintf("%s: Hit unknown token in selected string! '%s'\n", funcname, selected)); break; }
467 #define KILL_SPREE_LIST \
468 SPREE_ITEM(3, 03, _("TRIPLE FRAG! "), _("%s^K1 made a TRIPLE FRAG! %s^BG"), _("%s^K1 made a TRIPLE SCORE! %s^BG")) \
469 SPREE_ITEM(5, 05, _("RAGE! "), _("%s^K1 unlocked RAGE! %s^BG"), _("%s^K1 made FIVE SCORES IN A ROW! %s^BG")) \
470 SPREE_ITEM(10, 10, _("MASSACRE! "), _("%s^K1 started a MASSACRE! %s^BG"), _("%s^K1 made TEN SCORES IN A ROW! %s^BG")) \
471 SPREE_ITEM(15, 15, _("MAYHEM! "), _("%s^K1 executed MAYHEM! %s^BG"), _("%s^K1 made FIFTEEN SCORES IN A ROW! %s^BG")) \
472 SPREE_ITEM(20, 20, _("BERSERKER! "), _("%s^K1 is a BERSERKER! %s^BG"), _("%s^K1 made TWENTY SCORES IN A ROW! %s^BG")) \
473 SPREE_ITEM(25, 25, _("CARNAGE! "), _("%s^K1 inflicts CARNAGE! %s^BG"), _("%s^K1 made TWENTY FIVE SCORES IN A ROW! %s^BG")) \
474 SPREE_ITEM(30, 30, _("ARMAGEDDON! "), _("%s^K1 unleashes ARMAGEDDON! %s^BG"), _("%s^K1 made THIRTY SCORES IN A ROW! %s^BG"))
477 string notif_arg_frag_ping(bool newline, float fping)
479 string s = newline ? "\n" : " ";
481 return sprintf(CCR(_("%s(^F1Bot^BG)")), s);
483 return sprintf(CCR(_("%s(Ping ^F1%d^BG)")), s, fping);
486 string notif_arg_frag_stats(float fhealth, float farmor, float fping)
488 string s = notif_arg_frag_ping(false, fping);
490 return sprintf(CCR(_("\n(Health ^1%d^BG / Armor ^2%d^BG)%s")), fhealth, farmor, s);
492 return sprintf(CCR(_("\n(^F4Dead^BG)%s")), s);
495 string notif_arg_missing_teams(float f1)
498 ((f1 & BIT(0)) ? strcat(Team_ColoredFullName(NUM_TEAM_1), (f1 >> 1) ? ", " : "") : ""),
499 ((f1 & BIT(1)) ? strcat(Team_ColoredFullName(NUM_TEAM_2), (f1 >> 2) ? ", " : "") : ""),
500 ((f1 & BIT(2)) ? strcat(Team_ColoredFullName(NUM_TEAM_3), (f1 >> 3) ? ", " : "") : ""),
501 ((f1 & BIT(3)) ? Team_ColoredFullName(NUM_TEAM_4) : "")
505 string notif_arg_spree_cen(float spree)
507 // 0 = off, 1 = target (but only for first victim) and attacker
508 if(autocvar_notification_show_sprees_center)
512 #define SPREE_ITEM(counta,countb,center,normal,gentle) \
513 case counta: { return normal_or_gentle(center, sprintf(_("%d score spree! "), spree)); }
520 if (!autocvar_notification_show_sprees_center_specialonly)
525 _("%d frag spree! "),
526 _("%d score spree! ")
530 else { return ""; } // don't show spree information if it isn't an achievement
536 else if(spree == -1) // first blood
538 return normal_or_gentle(_("First blood! "), _("First score! "));
540 else if(spree == -2) // first victim
542 return normal_or_gentle(_("First victim! "), _("First casualty! "));
549 string notif_arg_spree_inf(float type, string input, string player, float spree)
553 case 1: // attacker kill spree
555 // 0 = off, 1 = target only, 2 = attacker only, 3 = target and attacker
556 // this conditional (& 2) is true for 2 and 3
557 if(autocvar_notification_show_sprees_info & 2)
560 string spree_newline =
561 ( autocvar_notification_show_sprees_info_newline ?
562 ((substring(input, 0, 1) == "\{3}") ? "\n\{3}" : "\n") : "" );
564 string spree_newline =
565 (autocvar_notification_show_sprees_info_newline ? "\n" : "");
570 #define SPREE_ITEM(counta,countb,center,normal,gentle) \
571 case counta: { return sprintf(CCR(normal_or_gentle(normal, gentle)), player, spree_newline); }
578 if (!autocvar_notification_show_sprees_info_specialonly)
582 CCR(normal_or_gentle(
583 _("%s^K1 has %d frags in a row! %s^BG"),
584 _("%s^K1 made %d scores in a row! %s^BG")
591 else { return ""; } // don't show spree information if it isn't an achievement
597 else if(spree == -1) // firstblood
601 CCR(normal_or_gentle(
602 _("%s^K1 drew first blood! %s^BG"),
603 _("%s^K1 got the first score! %s^BG")
613 case -1: // kill spree ended
615 if((spree > 1) && (autocvar_notification_show_sprees_info & 1))
618 sprintf(normal_or_gentle(
619 _(", ending their %d frag spree"),
620 _(", ending their %d score spree")
628 case -2: // kill spree lost
630 if((spree > 1) && (autocvar_notification_show_sprees_info & 1))
633 sprintf(normal_or_gentle(
634 _(", losing their %d frag spree"),
635 _(", losing their %d score spree")
646 string notif_arg_item_wepammo(float f1, float f2)
648 string ammoitems = "";
649 Weapon wep = REGISTRY_GET(Weapons, f1);
650 // TODO: registry handles
651 switch (wep.ammo_type)
653 case RES_SHELLS: ammoitems = ITEM_Shells.m_name; break;
654 case RES_BULLETS: ammoitems = ITEM_Bullets.m_name; break;
655 case RES_ROCKETS: ammoitems = ITEM_Rockets.m_name; break;
656 case RES_CELLS: ammoitems = ITEM_Cells.m_name; break;
657 case RES_PLASMA: ammoitems = ITEM_Plasma.m_name; break;
658 case RES_FUEL: ammoitems = ITEM_JetpackFuel.m_name; break;
659 default: return ""; // doesn't use ammo
661 return sprintf(_(" with %d %s"), f2, strtolower(ammoitems));
665 // ====================================
666 // Initialization/Create Declarations
667 // ====================================
669 // common notification entity values
674 .int nent_stringcount;
675 .int nent_floatcount;
678 // MSG_ANNCE entity values
682 .float nent_position;
683 .float nent_queuetime;
685 // MSG_INFO and MSG_CENTER entity values
686 .string nent_args; // used by both
687 .string nent_hudargs; // used by info
688 .string nent_icon; // used by info
689 .CPID nent_cpid; // used by center
690 .string nent_durcnt; // used by center
691 .string nent_string; // used by both
693 // MSG_MULTI entity values
694 .entity nent_msgannce;
695 .entity nent_msginfo;
696 .entity nent_msgcenter;
698 // MSG_CHOICE entity values
699 .float nent_challow_def;
700 .float nent_challow_var;
701 .entity nent_optiona;
702 .entity nent_optionb;
704 // networked notification entity values
706 .NOTIF nent_broadcast;
710 .float nent_net_name;
711 .string nent_strings[4];
712 .float nent_floats[4];
714 #define ACVNN(name) autocvar_notification_##name
716 REGISTRY(Notifications, BITS(11))
717 REGISTER_REGISTRY(Notifications)
718 REGISTRY_SORT(Notifications)
720 REGISTRY_DEFINE_GET(Notifications, NULL)
721 STATIC_INIT(Notifications) { FOREACH(Notifications, true, it.m_id = i); }
722 REGISTRY_CHECK(Notifications)
724 const int NOTIF_CHOICE_MAX = 20;
725 // NOTE: a team choice is actually made of 4 choices (one per team) with the same nent_choice_idx
726 // thus they are counted as 1 in nent_choice_count
727 int nent_choice_count = 0;
728 .int nent_choice_idx;
729 .int msg_choice_choices[NOTIF_CHOICE_MAX]; // set on each player containing MSG_CHOICE choices
730 // initialization error detection
732 bool notif_global_error;
734 STATIC_INIT_LATE(Notif_Choices)
736 if (nent_choice_count > NOTIF_CHOICE_MAX)
737 LOG_FATALF("Too many MSG_CHOICE notifications (%d), hit NOTIF_CHOICE_MAX (%d) limit",
738 nent_choice_count, NOTIF_CHOICE_MAX);
741 string Get_Notif_CvarName(Notification notif)
743 if(!notif.nent_teamnum)
744 return notif.nent_name;
745 return substring(notif.nent_name, 0, -strlen(Static_Team_ColorName(notif.nent_teamnum)) - 2);
748 Notification Get_Notif_Ent(MSG net_type, int net_name)
750 Notification it = REGISTRY_GET(Notifications, net_name);
751 if (it.nent_type != net_type) {
752 LOG_WARNF("Get_Notif_Ent(%s (%d), %s (%d)): Improper net type '%s'!",
753 Get_Notif_TypeName(net_type), net_type,
754 it.registered_id, net_name,
755 Get_Notif_TypeName(it.nent_type)
762 #define MSG_ANNCE_NOTIF_TEAM(teamnum, name, cvarname, defaultvalue, sound, channel, volume, position, queuetime) \
763 MSG_ANNCE_NOTIF_(teamnum, ANNCE_##name, ANNCE_##cvarname, defaultvalue, sound, channel, volume, position, queuetime)
765 #define MSG_ANNCE_NOTIF(name, defaultvalue, sound, channel, volume, position, queuetime) \
766 NOTIF_ADD_AUTOCVAR(ANNCE_##name, defaultvalue) \
767 MSG_ANNCE_NOTIF_(0, ANNCE_##name, ANNCE_##name, defaultvalue, sound, channel, volume, position, queuetime)
769 #define MSG_ANNCE_NOTIF_(teamnum, name, cvarname, defaultvalue, sound, channel, volume, position, queuetime) \
770 REGISTER(Notifications, name, m_id, new_pure(msg_annce_notification)) { \
771 Create_Notification_Entity (this, defaultvalue, ACVNN(cvarname), MSG_ANNCE, strtoupper(#name), teamnum); \
772 Create_Notification_Entity_Annce(this, ACVNN(cvarname), strtoupper(#name), \
773 channel, /* channel */ \
776 position, /* position */ \
777 queuetime); /* queuetime */ \
780 #define MSG_INFO_NOTIF_TEAM(teamnum, name, cvarname, defaultvalue, strnum, flnum, args, hudargs, icon, normal, gentle) \
781 MSG_INFO_NOTIF_(teamnum, INFO_##name, INFO_##cvarname, defaultvalue, strnum, flnum, args, hudargs, icon, normal, gentle)
783 #define MSG_INFO_NOTIF(name, defaultvalue, strnum, flnum, args, hudargs, icon, normal, gentle) \
784 NOTIF_ADD_AUTOCVAR(INFO_##name, defaultvalue) \
785 MSG_INFO_NOTIF_(0, INFO_##name, INFO_##name, defaultvalue, strnum, flnum, args, hudargs, icon, normal, gentle)
787 #define MSG_INFO_NOTIF_(teamnum, name, cvarname, defaultvalue, strnum, flnum, args, hudargs, icon, normal, gentle) \
788 REGISTER(Notifications, name, m_id, new_pure(msg_info_notification)) { \
789 Create_Notification_Entity (this, defaultvalue, ACVNN(cvarname), MSG_INFO, strtoupper(#name), teamnum); \
790 Create_Notification_Entity_InfoCenter(this, ACVNN(cvarname), strtoupper(#name), strnum, flnum, \
792 hudargs, /* hudargs */ \
794 CPID_Null,/* cpid */ \
796 normal, /* normal */ \
797 gentle); /* gentle */ \
800 .string nent_iconargs;
801 #define MULTIICON_INFO(name, defaultvalue, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle) \
802 MULTIICON_INFO_(INFO_##name, defaultvalue, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle)
803 #define MULTIICON_INFO_(name, defaultvalue, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle) \
804 NOTIF_ADD_AUTOCVAR(name, defaultvalue) \
805 REGISTER(Notifications, name, m_id, new_pure(msg_info_notification)) { \
806 Create_Notification_Entity (this, defaultvalue, ACVNN(name), MSG_INFO, strtoupper(#name), 0); \
807 Create_Notification_Entity_InfoCenter(this, ACVNN(name), strtoupper(#name), strnum, flnum, \
809 hudargs, /* hudargs */ \
811 CPID_Null,/* cpid */ \
813 normal, /* normal */ \
814 gentle); /* gentle */ \
815 this.nent_iconargs = iconargs; \
818 #define MSG_CENTER_NOTIF_TEAM(teamnum, name, cvarname, defaultvalue, strnum, flnum, args, cpid, durcnt, normal, gentle) \
819 MSG_CENTER_NOTIF_(teamnum, CENTER_##name, CENTER_##cvarname, defaultvalue, strnum, flnum, args, cpid, durcnt, normal, gentle)
821 #define MSG_CENTER_NOTIF(name, defaultvalue, strnum, flnum, args, cpid, durcnt, normal, gentle) \
822 NOTIF_ADD_AUTOCVAR(CENTER_##name, defaultvalue) \
823 MSG_CENTER_NOTIF_(0, CENTER_##name, CENTER_##name, defaultvalue, strnum, flnum, args, cpid, durcnt, normal, gentle)
825 #define MSG_CENTER_NOTIF_(teamnum, name, cvarname, defaultvalue, strnum, flnum, args, cpid, durcnt, normal, gentle) \
826 REGISTER(Notifications, name, m_id, new_pure(msg_center_notification)) { \
827 Create_Notification_Entity (this, defaultvalue, ACVNN(cvarname), MSG_CENTER, strtoupper(#name), teamnum); \
828 Create_Notification_Entity_InfoCenter(this, ACVNN(cvarname), strtoupper(#name), strnum, flnum, \
833 durcnt, /* durcnt */ \
834 normal, /* normal */ \
835 gentle); /* gentle */ \
838 #define MSG_MULTI_NOTIF(name, defaultvalue, anncename, infoname, centername) \
839 NOTIF_ADD_AUTOCVAR(name, defaultvalue) \
840 REGISTER(Notifications, name, m_id, new_pure(msg_multi_notification)) { \
841 Create_Notification_Entity (this, defaultvalue, ACVNN(name), MSG_MULTI, strtoupper(#name), 0); \
842 Create_Notification_Entity_Multi(this, ACVNN(name), strtoupper(#name), \
843 anncename, /* anncename */ \
844 infoname, /* infoname */ \
845 centername); /* centername */ \
848 #define MSG_CHOICE_NOTIF_TEAM(teamnum, name, cvarname, defaultvalue, challow, chtype, optiona, optionb) \
849 MSG_CHOICE_NOTIF_(teamnum, CHOICE_##name, CHOICE_##cvarname, defaultvalue, challow, chtype, optiona, optionb)
851 #define MSG_CHOICE_NOTIF(name, defaultvalue, challow, chtype, optiona, optionb) \
852 NOTIF_ADD_AUTOCVAR(CHOICE_##name, defaultvalue) \
853 NOTIF_ADD_AUTOCVAR(CHOICE_##name##_ALLOWED, challow) \
854 MSG_CHOICE_NOTIF_(0, CHOICE_##name, CHOICE_##name, defaultvalue, challow, chtype, optiona, optionb)
856 #define MSG_CHOICE_NOTIF_(teamnum, name, cvarname, defaultvalue, challow, chtype, optiona, optionb) \
857 REGISTER(Notifications, name, m_id, new_pure(msg_choice_notification)) { \
858 this.nent_choice_idx = nent_choice_count; \
859 if (!teamnum || teamnum == NUM_TEAM_4) \
860 nent_choice_count++; \
861 Create_Notification_Entity (this, defaultvalue, ACVNN(cvarname), MSG_CHOICE, strtoupper(#name), teamnum); \
862 Create_Notification_Entity_Choice(this, ACVNN(cvarname), strtoupper(#name), \
863 challow, /* challow_def */ \
864 autocvar_notification_##cvarname##_ALLOWED, /* challow_var */ \
865 chtype, /* chtype */ \
866 optiona, /* optiona */ \
867 optionb); /* optionb */ \
870 REGISTRY_BEGIN(Notifications)
872 notif_global_error = false;
875 REGISTRY_END(Notifications)
877 if (!notif_global_error) return;
878 // shit happened... stop the loading of the program now if this is unacceptable
879 if (autocvar_notification_errors_are_fatal)
880 LOG_FATAL("Notification initialization failed! Read above and fix the errors!");
882 LOG_SEVERE("Notification initialization failed! Read above and fix the errors!");
887 void ReplicateVars(bool would_destroy)
890 FOREACH(Notifications, it.nent_type == MSG_CHOICE && (!it.nent_teamnum || it.nent_teamnum == NUM_TEAM_1), {
891 string cvarname = strcat("notification_", Get_Notif_CvarName(it));
892 // NOTE: REPLICATE_SIMPLE can return;
893 REPLICATE_SIMPLE(it.cvar_value, cvarname);