]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/common/notifications.qh
2987c3494889472f1c62af9407c3a91870190d50
[xonotic/xonotic-data.pk3dir.git] / qcsrc / common / notifications.qh
1 #ifndef NOTIFICATIONS_H
2 #define NOTIFICATIONS_H
3
4 #include <common/command/all.qh>
5
6 #include "constants.qh"
7 #include "teams.qh"
8 #include "util.qh"
9
10 /** main types/groups of notifications */
11 ENUMCLASS(MSG)
12         /** "Global" AND "personal" announcer messages */
13         CASE(MSG, ANNCE)
14         /** "Global" information messages */
15         CASE(MSG, INFO)
16         /** "Personal" centerprint messages */
17         CASE(MSG, CENTER)
18         /** Subcall MSG_INFO and/or MSG_CENTER notifications */
19         CASE(MSG, MULTI)
20         /** Choose which subcall wrapper to activate */
21         CASE(MSG, CHOICE)
22
23         /** Kill centerprint message */
24         CASE(MSG, CENTER_CPID)
25 ENUMCLASS_END(MSG)
26
27 string Get_Notif_TypeName(MSG net_type)
28 {
29         switch (net_type)
30         {
31                 case MSG_ANNCE: return "MSG_ANNCE";
32                 case MSG_INFO: return "MSG_INFO";
33                 case MSG_CENTER: return "MSG_CENTER";
34                 case MSG_CENTER_CPID: return "MSG_CENTER_CPID";
35                 case MSG_MULTI: return "MSG_MULTI";
36                 case MSG_CHOICE: return "MSG_CHOICE";
37         }
38         backtrace(sprintf("Get_Notif_TypeName(%d): Improper net type!\n", net_type));
39         return "";
40 }
41
42 typedef entity Notification;
43
44 // negative confirmations
45 /** allows various things to know when no information is added */
46 Notification NO_MSG;
47 STATIC_INIT(NO_MSG) { NO_MSG = new_pure(Notification); }
48 /** @deprecated */
49 const int NO_MSG_ = -12345;
50 /** allows Send_Notification to safely abort sending */
51 Notification NOTIF_ABORT;
52 STATIC_INIT(NOTIF_ABORT) { NOTIF_ABORT = new_pure(Notification); }
53
54 // used for notification system multi-team identifiers
55 #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)))
56 /** @deprecated use APP_TEAM_NUM */
57 #define APP_TEAM_ENT(ent, prefix) APP_TEAM_NUM(ent.team, prefix)
58
59 #define EIGHT_VARS_TO_VARARGS_VARLIST \
60         VARITEM(1, 0, s1) \
61         VARITEM(2, 0, XPD(s1, s2)) \
62         VARITEM(3, 0, XPD(s1, s2, s3)) \
63         VARITEM(4, 0, XPD(s1, s2, s3, s4)) \
64         VARITEM(0, 1, f1) \
65         VARITEM(1, 1, XPD(s1, f1)) \
66         VARITEM(2, 1, XPD(s1, s2, f1)) \
67         VARITEM(3, 1, XPD(s1, s2, s3, f1)) \
68         VARITEM(4, 1, XPD(s1, s2, s3, s4, f1)) \
69         VARITEM(0, 2, XPD(f1, f2)) \
70         VARITEM(1, 2, XPD(s1, f1, f2)) \
71         VARITEM(2, 2, XPD(s1, s2, f1, f2)) \
72         VARITEM(3, 2, XPD(s1, s2, s3, f1, f2)) \
73         VARITEM(4, 2, XPD(s1, s2, s3, s4, f1, f2)) \
74         VARITEM(0, 3, XPD(f1, f2, f3)) \
75         VARITEM(1, 3, XPD(s1, f1, f2, f3)) \
76         VARITEM(2, 3, XPD(s1, s2, f1, f2, f3)) \
77         VARITEM(3, 3, XPD(s1, s2, s3, f1, f2, f3)) \
78         VARITEM(4, 3, XPD(s1, s2, s3, s4, f1, f2, f3)) \
79         VARITEM(0, 4, XPD(f1, f2, f3, f4)) \
80         VARITEM(1, 4, XPD(s1, f1, f2, f3, f4)) \
81         VARITEM(2, 4, XPD(s1, s2, f1, f2, f3, f4)) \
82         VARITEM(3, 4, XPD(s1, s2, s3, f1, f2, f3, f4)) \
83         VARITEM(4, 4, XPD(s1, s2, s3, s4, f1, f2, f3, f4))
84
85 void Destroy_All_Notifications();
86 void Create_Notification_Entity(entity notif,
87         float var_default,
88         float var_cvar,
89         MSG typeId,
90         string namestring);
91 void Create_Notification_Entity_Annce(entity notif,
92                                                                                 float var_cvar,
93                                                                                 string namestring,
94                                                                                 /* MSG_ANNCE */
95                                                                                 float channel,
96                                                                                 string snd,
97                                                                                 float vol,
98                                                                                 float position);
99
100 void Create_Notification_Entity_InfoCenter(entity notif,
101                                                                                         float var_cvar,
102                                                                                         string namestring,
103                                                                                         int strnum,
104                                                                                         int flnum,
105                                                                                         /* MSG_INFO & MSG_CENTER */
106                                                                                         string args,
107                                                                                         string hudargs,
108                                                                                         string icon,
109                                                                                         float cpid,
110                                                                                         string durcnt,
111                                                                                         string normal,
112                                                                                         string gentle);
113
114 void Create_Notification_Entity_Multi(entity notif,
115                                                                                 float var_cvar,
116                                                                                 string namestring,
117                                                                                 /* MSG_MULTI */
118                                                                                 Notification anncename,
119                                                                                 Notification infoname,
120                                                                                 Notification centername);
121
122 void Create_Notification_Entity_Choice(entity notif,
123                                                                                 float var_cvar,
124                                                                                 string namestring,
125                                                                                 /* MSG_CHOICE */
126                                                                                 float challow_def,
127                                                                                 float challow_var,
128                                                                                 MSG chtype,
129                                                                                 Notification optiona,
130                                                                                 Notification optionb);
131
132 void Dump_Notifications(int fh, bool alsoprint);
133
134 GENERIC_COMMAND(dumpnotifs, "Dump all notifications into notifications_dump.txt")
135 {
136         switch (request)
137         {
138                 case CMD_REQUEST_COMMAND:
139                 {
140                         #ifndef MENUQC
141                         string filename = argv(1);
142                         bool alsoprint = false;
143                         if (filename == "")
144                         {
145                                 filename = "notifications_dump.cfg";
146                                 alsoprint = false;
147                         }
148                         else if (filename == "-")
149                         {
150                                 filename = "notifications_dump.cfg";
151                                 alsoprint = true;
152                         }
153                         int fh = fopen(filename, FILE_WRITE);
154                         if (fh >= 0)
155                         {
156                                 Dump_Notifications(fh, alsoprint);
157                                 LOG_INFOF("Dumping notifications... File located in ^2data/data/%s^7.\n", filename);
158                                 fclose(fh);
159                         }
160                         else
161                         {
162                                 LOG_INFOF("^1Error: ^7Could not open file '%s'!\n", filename);
163                         }
164                         #else
165                         LOG_INFO(_("Notification dump command only works with cl_cmd and sv_cmd.\n"));
166                         #endif
167                         return;
168                 }
169                 default:
170                 case CMD_REQUEST_USAGE:
171                 {
172                         LOG_INFO(strcat("\nUsage:^3 ", GetProgramCommandPrefix(), " dumpnotifs [filename]"));
173                         LOG_INFO("  Where 'filename' is the file to write (default is notifications_dump.cfg),\n");
174                         LOG_INFO("  if supplied with '-' output to console as well as default,\n");
175                         LOG_INFO("  if left blank, it will only write to default.\n");
176                         return;
177                 }
178         }
179 }
180
181 #ifdef NOTIFICATIONS_DEBUG
182 void Debug_Notification(string input)
183 {
184         switch (autocvar_notification_debug)
185         {
186                 case 1: { LOG_TRACE(input); break; }
187                 case 2: { LOG_INFO(input); break; }
188         }
189 }
190 #endif
191
192 void Local_Notification(MSG net_type, Notification net_name, ...count);
193 /** glue for networking, forwards to `Local_Notification` */
194 void Local_Notification_WOVA(
195         MSG net_type, Notification net_name,
196         float stringcount, float floatcount,
197         string s1, string s2, string s3, string s4,
198         float f1, float f2, float f3, float f4);
199
200 #ifdef CSQC
201 string prev_soundfile;
202 float prev_soundtime;
203 #endif
204
205 #ifdef SVQC
206 ENUMCLASS(NOTIF)
207         /** send to one client and their spectators */
208         CASE(NOTIF, ONE)
209         /** send ONLY to one client */
210         CASE(NOTIF, ONE_ONLY)
211         /** send only to X team and their spectators */
212         CASE(NOTIF, TEAM)
213         /** send only to X team and their spectators, except for Y person and their spectators */
214         CASE(NOTIF, TEAM_EXCEPT)
215         /** send to everyone */
216         CASE(NOTIF, ALL)
217         /** send to everyone except X person and their spectators */
218         CASE(NOTIF, ALL_EXCEPT)
219 ENUMCLASS_END(NOTIF)
220
221 string Get_Notif_BroadcastName(NOTIF broadcast)
222 {
223         switch (broadcast)
224         {
225                 case NOTIF_ONE: return "NOTIF_ONE";
226                 case NOTIF_ONE_ONLY: return "NOTIF_ONE_ONLY";
227                 case NOTIF_ALL_EXCEPT: return "NOTIF_ALL_EXCEPT";
228                 case NOTIF_ALL: return "NOTIF_ALL";
229                 case NOTIF_TEAM: return "NOTIF_TEAM";
230                 case NOTIF_TEAM_EXCEPT: return "NOTIF_TEAM_EXCEPT";
231         }
232         LOG_WARNINGF("Get_Notif_BroadcastName(%d): Improper broadcast!\n", broadcast);
233         return "";
234 }
235
236 void Kill_Notification(
237         NOTIF broadcast, entity client,
238         MSG net_type, float net_name);
239 void Send_Notification(
240         NOTIF broadcast, entity client,
241         MSG net_type, Notification net_name,
242         ...count);
243 void Send_Notification_WOVA(
244         NOTIF broadcast, entity client,
245         MSG net_type, Notification net_name,
246         float stringcount, float floatcount,
247         string s1, string s2, string s3, string s4,
248         float f1, float f2, float f3, float f4);
249 void Send_Notification_WOCOVA(
250         NOTIF broadcast, entity client,
251         MSG net_type, Notification net_name,
252         string s1, string s2, string s3, string s4,
253         float f1, float f2, float f3, float f4);
254 #endif
255
256 // ===========================
257 //  Special CVAR Declarations
258 // ===========================
259
260 // MAKE SURE THIS IS ALWAYS SYNCHRONIZED WITH THE DUMP
261 // NOTIFICATIONS FUNCTION IN THE .QC FILE!
262
263 #define NOTIF_ADD_AUTOCVAR(name,default) float autocvar_notification_##name = default;
264
265 float autocvar_notification_show_location = false;
266 string autocvar_notification_show_location_string = ""; //_(" at the %s");
267 float autocvar_notification_show_sprees = true;
268 float autocvar_notification_show_sprees_info = 3; // 0 = off, 1 = target only, 2 = attacker only, 3 = target and attacker
269 float autocvar_notification_show_sprees_info_newline = true;
270 float autocvar_notification_show_sprees_info_specialonly = true;
271 float autocvar_notification_errors_are_fatal = true;
272 float autocvar_notification_lifetime_runtime = 0.5;
273 float autocvar_notification_lifetime_mapload = 10;
274 float autocvar_notification_debug = false;
275
276 #ifdef SVQC
277 .float FRAG_VERBOSE;
278 void Notification_GetCvars();
279 float autocvar_notification_server_allows_location = 1; // 0 = no, 1 = yes
280 #else
281 float autocvar_notification_item_centerprinttime = 1.5;
282
283 // 0 = no, 1 = yes, 2 = forced on for all MSG_INFO notifs
284 // DISABLED IN CODE, BUT ENABLED IN CONFIG FOR COMPATIBILITY WITH OLD CLIENTS
285 float autocvar_notification_allow_chatboxprint = 0;
286
287 float autocvar_notification_show_sprees_center = true;
288 float autocvar_notification_show_sprees_center_specialonly = true;
289 #endif
290
291
292 // ============================
293 //  Notification Argument List
294 // ============================
295 /*
296  These arguments get replaced with the Local_Notification_sprintf
297  and other such functions found in notifications.qc to supply data
298  from networked notifications to their usage in sprintf... It
299  allows for more dynamic data to be inferred by the local
300  notification parser, so that the server does not have to network
301  anything too crazy on a per-client/per-situation basis.
302
303  Pay attention to the CSQC/SVQC relations, some of these are redefined
304  in slightly different ways for different programs, this is because the
305  server does a more conservative approach to the notifs than the client.
306
307  All arguments are swapped into strings, so be sure that your
308  sprintf usage matches with proper %s placement.
309
310  Argument descriptions:
311         s1-s4: string arguments to be literally swapped into sprintf
312         s2loc: s2 string of locations of deaths or other events
313         s3loc: s3 string of locations of deaths or other events
314         f1-f4: float arguments expanded into strings to be swapped into sprintf
315         f1p2dec: f1 float to string with 2 decimal places
316         f2p2dec: f2 float to string with 2 decimal places
317         f2primsec: f2 float primary or secondary selection for weapons
318         f3primsec: f3 float primary or secondary selection for weapons
319         f1secs: count_seconds of f1
320         f1points: point or points depending on f1
321         f1ord: count_ordinal of f1
322         f1time: process_time of f1
323         f1race_time: mmssss of f1
324         f2race_time: mmssss of f2
325         race_col: color of race time/position (i.e. good or bad)
326         race_diff: show time difference between f2 and f3
327         missing_teams: show which teams still need players
328         pass_key: find the keybind for "passing" or "dropping" in CTF game mode
329         frag_ping: show the ping of a player
330         frag_stats: show health/armor/ping of a player
331         frag_pos: show score status and position in the match of a player
332         spree_cen: centerprint notif for kill spree/how many kills they have
333         spree_inf: info notif for kill spree/how many kills they have
334         spree_end: placed at the end of murder messages to show ending of sprees
335         spree_lost: placed at the end of suicide messages to show losing of sprees
336         item_wepname: return full name of a weapon from weaponid
337         item_wepammo: ammo display for weapon from string
338         item_centime: amount of time to display weapon message in centerprint
339         item_buffname: return full name of a buff from buffid
340         death_team: show the full name of the team a player is switching from
341         minigame1_name: return human readable name of a minigame from its id(s1)
342         minigame1_d: return descriptor name of a minigame from its id(s1)
343 */
344
345 const float NOTIF_MAX_ARGS = 7;
346 const float NOTIF_MAX_HUDARGS = 2;
347 const float NOTIF_MAX_DURCNT = 2;
348
349 string arg_slot[NOTIF_MAX_ARGS];
350
351 const float ARG_CS_SV_HA = 1; // enabled on CSQC, SVQC, and Hudargs
352 const float ARG_CS_SV_DC = 2; // enabled on CSQC, SVQC, and durcnt centerprint
353 const float ARG_CS_SV = 3; // enabled on CSQC and SVQC
354 const float ARG_CS = 4; // unique result to CSQC
355 const float ARG_SV = 5; // unique result to SVQC
356 const float ARG_DC = 6; // unique result to durcnt/centerprint
357
358 // todo possible idea.... declare how many floats/strings each arg needs, and then dynamically increment the input
359 // this way, we don't need to have duplicates like i.e. s2loc and s3loc?
360
361 string BUFF_NAME(int i);
362
363 #define NOTIF_ARGUMENT_LIST \
364         ARG_CASE(ARG_CS_SV_HA,  "s1",            s1) \
365         ARG_CASE(ARG_CS_SV_HA,  "s2",            s2) \
366         ARG_CASE(ARG_CS_SV_HA,  "s3",            s3) \
367         ARG_CASE(ARG_CS_SV_HA,  "s4",            s4) \
368         ARG_CASE(ARG_CS_SV,     "s2loc",         ((autocvar_notification_show_location && (s2 != "")) ? sprintf(( ((tmp_s = autocvar_notification_show_location_string) != "") ? tmp_s : _(" (near %s)") ), s2) : "")) \
369         ARG_CASE(ARG_CS_SV,     "s3loc",         ((autocvar_notification_show_location && (s3 != "")) ? sprintf(( ((tmp_s = autocvar_notification_show_location_string) != "") ? tmp_s : _(" (near %s)") ), s3) : "")) \
370         ARG_CASE(ARG_CS_SV_DC,  "f1",            ftos(f1)) \
371         ARG_CASE(ARG_CS_SV_DC,  "f2",            ftos(f2)) \
372         ARG_CASE(ARG_CS_SV,     "f3",            ftos(f3)) \
373         ARG_CASE(ARG_CS_SV,     "f4",            ftos(f4)) \
374         ARG_CASE(ARG_CS_SV,     "f1p2dec",       ftos_decimals(f1/100, 2)) \
375         ARG_CASE(ARG_CS_SV,     "f2p2dec",       ftos_decimals(f2/100, 2)) \
376         ARG_CASE(ARG_CS,        "f2primsec",     (f2 ? _("secondary") : _("primary"))) \
377         ARG_CASE(ARG_CS,        "f3primsec",     (f3 ? _("secondary") : _("primary"))) \
378         ARG_CASE(ARG_CS,        "f1secs",        count_seconds(f1)) \
379         ARG_CASE(ARG_CS,        "f1points",      (f1 == 1 ? _("point") : _("points"))) \
380         ARG_CASE(ARG_CS_SV,     "f1ord",         count_ordinal(f1)) \
381         ARG_CASE(ARG_CS,        "f1time",        process_time(2, f1)) \
382         ARG_CASE(ARG_CS_SV_HA,  "f1race_time",   mmssss(f1)) \
383         ARG_CASE(ARG_CS_SV_HA,  "f2race_time",   mmssss(f2)) \
384         ARG_CASE(ARG_CS_SV_HA,  "f3race_time",   mmssss(f3)) \
385         ARG_CASE(ARG_CS_SV,     "race_col",      CCR(((f1 == 1) ? "^F1" : "^F2"))) \
386         ARG_CASE(ARG_CS_SV,     "race_diff",     ((f2 > f3) ? sprintf(CCR("^1[+%s]"), mmssss(f2 - f3)) : sprintf(CCR("^2[-%s]"), mmssss(f3 - f2)))) \
387         ARG_CASE(ARG_CS,        "missing_teams", notif_arg_missing_teams(f1)) \
388         ARG_CASE(ARG_CS,        "pass_key",      ((((tmp_s = getcommandkey("pass", "+use")) != "pass") && !(strstrofs(tmp_s, "not bound", 0) >= 0)) ? sprintf(CCR(_(" ^F1(Press %s)")), tmp_s) : "")) \
389         ARG_CASE(ARG_CS,        "frag_ping",     notif_arg_frag_ping(true, f2)) \
390         ARG_CASE(ARG_CS,        "frag_stats",    notif_arg_frag_stats(f2, f3, f4)) \
391         /*ARG_CASE(ARG_CS,      "frag_pos",      ((Should_Print_Score_Pos(f1)) ? sprintf("\n^BG%s", Read_Score_Pos(f1)) : ""))*/ \
392         ARG_CASE(ARG_CS,        "spree_cen",     (autocvar_notification_show_sprees ? notif_arg_spree_cen(f1) : "")) \
393         ARG_CASE(ARG_CS_SV,     "spree_inf",     (autocvar_notification_show_sprees ? notif_arg_spree_inf(1, input, s2, f2) : "")) \
394         ARG_CASE(ARG_CS_SV,     "spree_end",     (autocvar_notification_show_sprees ? notif_arg_spree_inf(-1, "", "", f1) : "")) \
395         ARG_CASE(ARG_CS_SV,     "spree_lost",    (autocvar_notification_show_sprees ? notif_arg_spree_inf(-2, "", "", f1) : "")) \
396         ARG_CASE(ARG_CS_SV,     "item_wepname",  Weapons_from(f1).m_name) \
397         ARG_CASE(ARG_CS_SV,     "item_buffname", BUFF_NAME(f1)) \
398         ARG_CASE(ARG_CS_SV,     "f3buffname",    BUFF_NAME(f3)) \
399         ARG_CASE(ARG_CS_SV,     "item_wepammo",  (s1 != "" ? sprintf(_(" with %s"), s1) : "")) \
400         ARG_CASE(ARG_DC,        "item_centime",  ftos(autocvar_notification_item_centerprinttime)) \
401         ARG_CASE(ARG_SV,        "death_team",    Team_ColoredFullName(f1)) \
402         ARG_CASE(ARG_CS,        "death_team",    Team_ColoredFullName(f1 - 1)) \
403         ARG_CASE(ARG_CS_SV_HA,  "minigame1_name",find(world,netname,s1).descriptor.message) \
404         ARG_CASE(ARG_CS_SV_HA,  "minigame1_d",   find(world,netname,s1).descriptor.netname)
405
406 #define NOTIF_HIT_MAX(count,funcname) MACRO_BEGIN { \
407         if(sel_num == count) { backtrace(sprintf("%s: Hit maximum arguments!\n", funcname)); break; } \
408 } MACRO_END
409 #define NOTIF_HIT_UNKNOWN(token,funcname) { backtrace(sprintf("%s: Hit unknown token in selected string! '%s'\n", funcname, selected)); break; }
410
411 #define KILL_SPREE_LIST \
412         SPREE_ITEM(3, 03, _("TRIPLE FRAG! "), _("%s^K1 made a TRIPLE FRAG! %s^BG"), _("%s^K1 made a TRIPLE SCORE! %s^BG")) \
413         SPREE_ITEM(5, 05, _("RAGE! "), _("%s^K1 unlocked RAGE! %s^BG"), _("%s^K1 made FIVE SCORES IN A ROW! %s^BG")) \
414         SPREE_ITEM(10, 10, _("MASSACRE! "), _("%s^K1 started a MASSACRE! %s^BG"), _("%s^K1 made TEN SCORES IN A ROW! %s^BG")) \
415         SPREE_ITEM(15, 15, _("MAYHEM! "), _("%s^K1 executed MAYHEM! %s^BG"), _("%s^K1 made FIFTEEN SCORES IN A ROW! %s^BG")) \
416         SPREE_ITEM(20, 20, _("BERSERKER! "), _("%s^K1 is a BERSERKER! %s^BG"), _("%s^K1 made TWENTY SCORES IN A ROW! %s^BG")) \
417         SPREE_ITEM(25, 25, _("CARNAGE! "), _("%s^K1 inflicts CARNAGE! %s^BG"), _("%s^K1 made TWENTY FIVE SCORES IN A ROW! %s^BG")) \
418         SPREE_ITEM(30, 30, _("ARMAGEDDON! "), _("%s^K1 unleashes ARMAGEDDON! %s^BG"), _("%s^K1 made THIRTY SCORES IN A ROW! %s^BG"))
419
420 #ifdef CSQC
421 string notif_arg_frag_ping(bool newline, float fping)
422 {
423         string s = newline ? "\n" : " ";
424         if (fping == NO_MSG_)
425                 return sprintf(CCR(_("%s(^F1Bot^BG)")), s);
426         else
427                 return sprintf(CCR(_("%s(Ping ^F1%d^BG)")), s, fping);
428 }
429
430 string notif_arg_frag_stats(float fhealth, float farmor, float fping)
431 {
432         string s = notif_arg_frag_ping(false, fping);
433         if (fhealth > 1)
434                 return sprintf(CCR(_("\n(Health ^1%d^BG / Armor ^2%d^BG)%s")), fhealth, farmor, s);
435         else
436                 return sprintf(CCR(_("\n(^F4Dead^BG)%s")), s);
437 }
438
439 string notif_arg_missing_teams(float f1)
440 {
441         return sprintf("%s%s%s%s",
442                 ((f1 & BIT(0)) ? sprintf("%s%s", Team_ColoredFullName(NUM_TEAM_1), (f1 & (BIT(1) | BIT(2) | BIT(3)) ? ", " : "")) : ""),
443                 ((f1 & BIT(1)) ? sprintf("%s%s", Team_ColoredFullName(NUM_TEAM_2), (f1 & (         BIT(2) | BIT(3)) ? ", " : "")) : ""),
444                 ((f1 & BIT(2)) ? sprintf("%s%s", Team_ColoredFullName(NUM_TEAM_3), (f1 & (                  BIT(3)) ? ", " : "")) : ""),
445                 ((f1 & BIT(3)) ?                 Team_ColoredFullName(NUM_TEAM_4)                                                 : "")
446         );
447 }
448
449 string notif_arg_spree_cen(float spree)
450 {
451         // 0 = off, 1 = target (but only for first victim) and attacker
452         if(autocvar_notification_show_sprees_center)
453         {
454                 if(spree > 1)
455                 {
456                         #define SPREE_ITEM(counta,countb,center,normal,gentle) \
457                                 case counta: { return normal_or_gentle(center, sprintf(_("%d score spree! "), spree)); }
458
459                         switch(spree)
460                         {
461                                 KILL_SPREE_LIST
462                                 default:
463                                 {
464                                         if (!autocvar_notification_show_sprees_center_specialonly)
465                                         {
466                                                 return
467                                                         sprintf(
468                                                                 normal_or_gentle(
469                                                                         _("%d frag spree! "),
470                                                                         _("%d score spree! ")
471                                                                 ),
472                                                                 spree);
473                                         }
474                                         else { return ""; } // don't show spree information if it isn't an achievement
475                                 }
476                         }
477
478                         #undef SPREE_ITEM
479                 }
480                 else if(spree == -1) // first blood
481                 {
482                         return normal_or_gentle(_("First blood! "), _("First score! "));
483                 }
484                 else if(spree == -2) // first victim
485                 {
486                         return normal_or_gentle(_("First victim! "), _("First casualty! "));
487                 }
488         }
489         return "";
490 }
491 #endif
492
493 string notif_arg_spree_inf(float type, string input, string player, float spree)
494 {
495         switch(type)
496         {
497                 case 1: // attacker kill spree
498                 {
499                         // 0 = off, 1 = target only, 2 = attacker only, 3 = target and attacker
500                         // this conditional (& 2) is true for 2 and 3
501                         if(autocvar_notification_show_sprees_info & 2)
502                         {
503                                 #ifdef CSQC
504                                 string spree_newline =
505                                         ( autocvar_notification_show_sprees_info_newline ?
506                                                 ((substring(input, 0, 1) == "\{3}") ? "\n\{3}" : "\n") : "" );
507                                 #else
508                                 string spree_newline =
509                                         (autocvar_notification_show_sprees_info_newline ? "\n" : "");
510                                 #endif
511
512                                 if(spree > 1)
513                                 {
514                                         #define SPREE_ITEM(counta,countb,center,normal,gentle) \
515                                                 case counta: { return sprintf(CCR(normal_or_gentle(normal, gentle)), player, spree_newline); }
516
517                                         switch(spree)
518                                         {
519                                                 KILL_SPREE_LIST
520                                                 default:
521                                                 {
522                                                         if (!autocvar_notification_show_sprees_info_specialonly)
523                                                         {
524                                                                 return
525                                                                         sprintf(
526                                                                                 CCR(normal_or_gentle(
527                                                                                         _("%s^K1 has %d frags in a row! %s^BG"),
528                                                                                         _("%s^K1 made %d scores in a row! %s^BG")
529                                                                                 )),
530                                                                                 player,
531                                                                                 spree,
532                                                                                 spree_newline
533                                                                         );
534                                                         }
535                                                         else { return ""; } // don't show spree information if it isn't an achievement
536                                                 }
537                                         }
538
539                                         #undef SPREE_ITEM
540                                 }
541                                 else if(spree == -1) // firstblood
542                                 {
543                                         return
544                                                 sprintf(
545                                                         CCR(normal_or_gentle(
546                                                                 _("%s^K1 drew first blood! %s^BG"),
547                                                                 _("%s^K1 got the first score! %s^BG")
548                                                         )),
549                                                         player,
550                                                         spree_newline
551                                                 );
552                                 }
553                         }
554                         break;
555                 }
556
557                 case -1: // kill spree ended
558                 {
559                         if((spree > 1) && (autocvar_notification_show_sprees_info & 1))
560                         {
561                                 return
562                                         sprintf(normal_or_gentle(
563                                                 _(", ending their %d frag spree"),
564                                                 _(", ending their %d score spree")
565                                                 ),
566                                                 spree
567                                         );
568                         }
569                         break;
570                 }
571
572                 case -2: // kill spree lost
573                 {
574                         if((spree > 1) && (autocvar_notification_show_sprees_info & 1))
575                         {
576                                 return
577                                         sprintf(normal_or_gentle(
578                                                 _(", losing their %d frag spree"),
579                                                 _(", losing their %d score spree")
580                                                 ),
581                                                 spree
582                                         );
583                         }
584                         break;
585                 }
586         }
587         return "";
588 }
589
590
591 // ====================================
592 //  Initialization/Create Declarations
593 // ====================================
594
595 enum {
596         NO_CPID
597 ,   CPID_ASSAULT_ROLE
598 ,   CPID_ROUND
599 ,   CPID_CAMPCHECK
600 ,   CPID_CTF_CAPSHIELD
601 ,   CPID_CTF_LOWPRIO
602 ,   CPID_CTF_PASS
603 ,   CPID_STALEMATE
604 ,   CPID_NADES
605 ,   CPID_IDLING
606 ,   CPID_ITEM
607 ,   CPID_PREVENT_JOIN
608 ,   CPID_KEEPAWAY
609 ,   CPID_KEEPAWAY_WARN
610 ,   CPID_KEYHUNT
611 ,   CPID_KEYHUNT_OTHER
612 ,   CPID_LMS
613 ,   CPID_MISSING_TEAMS
614 ,   CPID_MISSING_PLAYERS
615 ,   CPID_INSTAGIB_FINDAMMO
616 ,   CPID_MOTD
617 ,   CPID_NIX
618 ,   CPID_ONSLAUGHT
619 ,   CPID_ONS_CAPSHIELD
620 ,   CPID_OVERTIME
621 ,   CPID_POWERUP
622 ,   CPID_RACE_FINISHLAP
623 ,   CPID_TEAMCHANGE
624 ,   CPID_TIMEOUT
625 ,   CPID_VEHICLES
626 ,   CPID_VEHICLES_OTHER
627 // always last
628 ,   NOTIF_CPID_COUNT
629 };
630
631 // common notification entity values
632 .int nent_default;
633 .bool nent_enabled;
634 .MSG nent_type;
635 .string nent_name;
636 .int nent_stringcount;
637 .int nent_floatcount;
638
639 // MSG_ANNCE entity values
640 .int nent_channel;
641 .string nent_snd;
642 .float nent_vol;
643 .float nent_position;
644
645 // MSG_INFO and MSG_CENTER entity values
646 .string nent_args; // used by both
647 .string nent_hudargs; // used by info
648 .string nent_icon; // used by info
649 .float nent_cpid; // used by center
650 .string nent_durcnt; // used by center
651 .string nent_string; // used by both
652
653 // MSG_MULTI entity values
654 .entity nent_msgannce;
655 .entity nent_msginfo;
656 .entity nent_msgcenter;
657
658 // MSG_CHOICE entity values
659 .float nent_challow_def;
660 .float nent_challow_var;
661 .entity nent_optiona;
662 .entity nent_optionb;
663
664 // networked notification entity values
665 #ifdef SVQC
666 .NOTIF nent_broadcast;
667 #endif
668 .entity nent_client;
669 .MSG nent_net_type;
670 .float nent_net_name;
671 .string nent_strings[4];
672 .float nent_floats[4];
673
674 #define ACVNN(name) autocvar_notification_##name
675
676 REGISTRY(Notifications, BITS(11))
677 REGISTER_REGISTRY(Notifications)
678 REGISTRY_SORT(Notifications); STATIC_INIT(Notifications) { FOREACH(Notifications, true, LAMBDA(it.m_id = i)); }
679 REGISTRY_CHECK(Notifications)
680
681 const int NOTIF_CHOICE_MAX = 50;
682 int nent_choice_count = 0;
683 .int nent_choice_idx;
684 .int msg_choice_choices[NOTIF_CHOICE_MAX]; // set on each player containing MSG_CHOICE choices
685 // initialization error detection
686 bool notif_error;
687 bool notif_global_error;
688
689 STATIC_INIT_LATE(Notif_Choices) {
690         int c = 0;
691         FOREACH(Notifications, it.nent_type == MSG_CHOICE, { c++; });
692         if (c > NOTIF_CHOICE_MAX) {
693                 LOG_FATALF("Too many MSG_CHOICE notifications (%d)", c);
694         }
695 }
696
697 Notification Get_Notif_Ent(MSG net_type, int net_name)
698 {
699         if (net_type == MSG_CENTER_CPID) return NULL;
700         Notification it = _Notifications_from(net_name, NULL);
701         if (it.nent_type != net_type) {
702                 LOG_WARNINGF("Get_Notif_Ent(%s (%d), %s (%d)): Improper net type '%s'!\n",
703                         Get_Notif_TypeName(net_type), net_type,
704                         it.registered_id, net_name,
705                         Get_Notif_TypeName(it.nent_type)
706                 );
707                 return NULL;
708         }
709         return it;
710 }
711
712 #define MSG_ANNCE_NOTIF(name, default, sound, channel, volume, position) \
713         MSG_ANNCE_NOTIF_(ANNCE_##name, default, sound, channel, volume, position)
714 #define MSG_ANNCE_NOTIF_(name, default, sound, channel, volume, position) \
715         NOTIF_ADD_AUTOCVAR(name, default) \
716         REGISTER(Notifications, name, m_id, new_pure(msg_annce_notification)) { \
717                 Create_Notification_Entity      (this, default, ACVNN(name), MSG_ANNCE, strtoupper(#name)); \
718                 Create_Notification_Entity_Annce(this, ACVNN(name), strtoupper(#name), \
719                         channel,   /* channel  */ \
720                         sound,     /* snd      */ \
721                         volume,    /* vol      */ \
722                         position); /* position */ \
723         }
724
725 #define MSG_INFO_NOTIF(name, default, strnum, flnum, args, hudargs, icon, normal, gentle) \
726         MSG_INFO_NOTIF_(INFO_##name, default, strnum, flnum, args, hudargs, icon, normal, gentle)
727 #define MSG_INFO_NOTIF_(name, default, strnum, flnum, args, hudargs, icon, normal, gentle) \
728         NOTIF_ADD_AUTOCVAR(name, default) \
729         REGISTER(Notifications, name, m_id, new_pure(msg_info_notification)) { \
730                 Create_Notification_Entity           (this, default, ACVNN(name), MSG_INFO, strtoupper(#name)); \
731                 Create_Notification_Entity_InfoCenter(this, ACVNN(name), strtoupper(#name), strnum, flnum, \
732                         args,     /* args    */ \
733                         hudargs,  /* hudargs */ \
734                         icon,     /* icon    */ \
735                         NO_MSG_,  /* cpid    */ \
736                         "",       /* durcnt  */ \
737                         normal,   /* normal  */ \
738                         gentle);  /* gentle  */ \
739         }
740
741 .string nent_iconargs;
742 #define MULTIICON_INFO(name, default, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle) \
743         MULTIICON_INFO_(INFO_##name, default, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle)
744 #define MULTIICON_INFO_(name, default, strnum, flnum, args, hudargs, iconargs, icon, normal, gentle) \
745         NOTIF_ADD_AUTOCVAR(name, default) \
746         REGISTER(Notifications, name, m_id, new_pure(msg_info_notification)) { \
747                 Create_Notification_Entity           (this, default, ACVNN(name), MSG_INFO, strtoupper(#name)); \
748                 Create_Notification_Entity_InfoCenter(this, ACVNN(name), strtoupper(#name), strnum, flnum, \
749                         args,     /* args    */ \
750                         hudargs,  /* hudargs */ \
751                         icon,     /* icon    */ \
752                         NO_MSG_,  /* cpid    */ \
753                         "",       /* durcnt  */ \
754                         normal,   /* normal  */ \
755                         gentle);  /* gentle  */ \
756                 this.nent_iconargs = iconargs; \
757         }
758
759 #define MSG_CENTER_NOTIF(name, default, strnum, flnum, args, cpid, durcnt, normal, gentle) \
760         MSG_CENTER_NOTIF_(CENTER_##name, default, strnum, flnum, args, cpid, durcnt, normal, gentle)
761 #define MSG_CENTER_NOTIF_(name, default, strnum, flnum, args, cpid, durcnt, normal, gentle) \
762         NOTIF_ADD_AUTOCVAR(name, default) \
763         REGISTER(Notifications, name, m_id, new_pure(msg_center_notification)) { \
764                 Create_Notification_Entity           (this, default, ACVNN(name), MSG_CENTER, strtoupper(#name)); \
765                 Create_Notification_Entity_InfoCenter(this, ACVNN(name), strtoupper(#name), strnum, flnum, \
766                         args,    /* args    */ \
767                         "",      /* hudargs */ \
768                         "",      /* icon    */ \
769                         cpid,    /* cpid    */ \
770                         durcnt,  /* durcnt  */ \
771                         normal,  /* normal  */ \
772                         gentle); /* gentle  */ \
773         }
774
775 #define MSG_MULTI_NOTIF(name, default, anncename, infoname, centername) \
776         NOTIF_ADD_AUTOCVAR(name, default) \
777         REGISTER(Notifications, name, m_id, new_pure(msg_multi_notification)) { \
778                 Create_Notification_Entity      (this, default, ACVNN(name), MSG_MULTI, strtoupper(#name)); \
779                 Create_Notification_Entity_Multi(this, ACVNN(name), strtoupper(#name), \
780                         anncename,   /* anncename  */ \
781                         infoname,    /* infoname   */ \
782                         centername); /* centername */ \
783         }
784
785 #define MSG_CHOICE_NOTIF(name, default, challow, chtype, optiona, optionb) \
786         MSG_CHOICE_NOTIF_(CHOICE_##name, default, challow, chtype, optiona, optionb)
787 #define MSG_CHOICE_NOTIF_(name, default, challow, chtype, optiona, optionb) \
788         NOTIF_ADD_AUTOCVAR(name, default) \
789         NOTIF_ADD_AUTOCVAR(name##_ALLOWED, challow) \
790         REGISTER(Notifications, name, m_id, new_pure(msg_choice_notification)) { \
791                 this.nent_choice_idx = nent_choice_count++; \
792                 Create_Notification_Entity       (this, default, ACVNN(name), MSG_CHOICE, strtoupper(#name)); \
793                 Create_Notification_Entity_Choice(this, ACVNN(name), strtoupper(#name), \
794                         challow,                                 /* challow_def */ \
795                         autocvar_notification_##name##_ALLOWED,  /* challow_var */ \
796                         chtype,                                  /* chtype      */ \
797                         optiona,                                 /* optiona     */ \
798                         optionb);                                /* optionb     */ \
799         }
800
801 REGISTRY_BEGIN(Notifications)
802 {
803         notif_global_error = false;
804 }
805
806 REGISTRY_END(Notifications)
807 {
808         if (!notif_global_error) return;
809         // shit happened... stop the loading of the program now if this is unacceptable
810         if (autocvar_notification_errors_are_fatal)
811                 LOG_FATAL("Notification initialization failed! Read above and fix the errors!");
812         else
813                 LOG_SEVERE("Notification initialization failed! Read above and fix the errors!");
814 }
815
816 #include "notifications.inc"
817
818 #endif