Stop sharing requirement_unit_state_ereq().
[freeciv.git] / common / events.c
blobed2ec4782ce57203db824a568ee138c038579da2
1 /**********************************************************************
2 Freeciv - Copyright (C) 1996 - A Kjeldberg, L Gregersen, P Unold
3 This program is free software; you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 2, or (at your option)
6 any later version.
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 GNU General Public License for more details.
12 ***********************************************************************/
14 #ifdef HAVE_CONFIG_H
15 #include <fc_config.h>
16 #endif
18 #include <stdio.h>
19 #include <stdlib.h>
21 /* utility */
22 #include "fcintl.h"
23 #include "log.h"
24 #include "mem.h"
25 #include "shared.h"
26 #include "support.h"
28 #include "events.h"
30 enum event_section_n {
31 E_S_ADVANCE,
32 E_S_BUILD,
33 E_S_CITY,
34 E_S_D_ME,
35 E_S_D_THEM,
36 E_S_GLOBAL,
37 E_S_HUT,
38 E_S_NATION,
39 E_S_TREATY,
40 E_S_UNIT,
41 E_S_VOTE,
42 E_S_WONDER,
43 E_S_XYZZY
47 * Information about all event sections, matching the enum above.
49 static const char *event_sections[] = {
50 /* TRANS: This and following strings are prefixes for event names, which
51 * replace %s. For instance, "Technology: Selected New Goal". */
52 N_("Technology: %s"),
53 N_("Improvement: %s"),
54 N_("City: %s"),
55 N_("Diplomat Action: %s"),
56 N_("Enemy Diplomat: %s"),
57 N_("Global: %s"),
58 N_("Hut: %s"),
59 N_("Nation: %s"),
60 N_("Treaty: %s"),
61 N_("Unit: %s"),
62 /* TRANS: "Vote" as a process */
63 N_("Vote: %s"),
64 N_("Wonder: %s"),
65 NULL
68 #define GEN_EV(event, section, descr) { #event, NULL, section, descr, NULL, event }
71 * Holds information about all event types. The entries don't have
72 * to be sorted.
73 * Every E_* event defined in common/events.h should have an entry here.
75 static struct {
76 const char *enum_name;
77 char *tag_name;
78 enum event_section_n esn;
79 char *descr_orig;
80 char *full_descr;
81 enum event_type event;
82 } events[] = {
83 /* TRANS: this and following strings are names for events which cause the
84 * server to generate messages. They are used in configuring how the client
85 * handles the different types of messages. Some of them will be displayed
86 * with prefixes, such as "Technology: Learned From Great Library". */
87 GEN_EV(E_TECH_GAIN, E_S_ADVANCE, N_("Acquired New Tech")),
88 GEN_EV(E_TECH_LEARNED, E_S_ADVANCE, N_("Learned New Tech")),
89 GEN_EV(E_TECH_GOAL, E_S_ADVANCE, N_("Selected New Goal")),
90 GEN_EV(E_TECH_LOST, E_S_ADVANCE, N_("Lost a Tech")),
91 GEN_EV(E_TECH_EMBASSY, E_S_ADVANCE, N_("Other Player Gained/Lost a Tech")),
92 GEN_EV(E_IMP_BUY, E_S_BUILD, N_("Bought")),
93 GEN_EV(E_IMP_BUILD, E_S_BUILD, N_("Built")),
94 GEN_EV(E_IMP_AUCTIONED, E_S_BUILD, N_("Forced to Sell")),
95 GEN_EV(E_IMP_AUTO, E_S_BUILD, N_("New Improvement Selected")),
96 GEN_EV(E_IMP_SOLD, E_S_BUILD, N_("Sold")),
97 GEN_EV(E_CITY_CANTBUILD, E_S_CITY, N_("Building Unavailable Item")),
98 GEN_EV(E_CITY_LOST, E_S_CITY, N_("Captured/Destroyed")),
99 GEN_EV(E_CITY_LOVE, E_S_CITY, N_("Celebrating")),
100 GEN_EV(E_CITY_DISORDER, E_S_CITY, N_("Civil Disorder")),
101 GEN_EV(E_CITY_FAMINE, E_S_CITY, N_("Famine")),
102 GEN_EV(E_CITY_FAMINE_FEARED, E_S_CITY, N_("Famine Feared")),
103 GEN_EV(E_CITY_GROWTH, E_S_CITY, N_("Growth")),
104 GEN_EV(E_CITY_MAY_SOON_GROW, E_S_CITY, N_("May Soon Grow")),
105 GEN_EV(E_CITY_AQUEDUCT, E_S_CITY, N_("Needs Aqueduct")),
106 GEN_EV(E_CITY_AQ_BUILDING, E_S_CITY, N_("Needs Aqueduct Being Built")),
107 GEN_EV(E_CITY_NORMAL, E_S_CITY, N_("Normal")),
108 GEN_EV(E_CITY_NUKED, E_S_CITY, N_("Nuked")),
109 GEN_EV(E_CITY_CMA_RELEASE, E_S_CITY, N_("Released from citizen governor")),
110 GEN_EV(E_CITY_GRAN_THROTTLE, E_S_CITY, N_("Suggest Growth Throttling")),
111 GEN_EV(E_CITY_TRANSFER, E_S_CITY, N_("Transfer")),
112 GEN_EV(E_CITY_BUILD, E_S_CITY, N_("Was Built")),
113 GEN_EV(E_CITY_PLAGUE, E_S_CITY, N_("Has Plague")),
114 GEN_EV(E_CITY_RADIUS_SQ, E_S_CITY, N_("City Map changed")),
115 GEN_EV(E_WORKLIST, E_S_CITY, N_("Worklist Events")),
116 GEN_EV(E_CITY_PRODUCTION_CHANGED, E_S_CITY, N_("Production changed")),
117 GEN_EV(E_DISASTER, E_S_CITY, N_("Disaster")),
118 GEN_EV(E_MY_DIPLOMAT_BRIBE, E_S_D_ME, N_("Bribe")),
119 GEN_EV(E_DIPLOMATIC_INCIDENT, E_S_D_ME, N_("Caused Incident")),
120 GEN_EV(E_MY_DIPLOMAT_ESCAPE, E_S_D_ME, N_("Escape")),
121 GEN_EV(E_MY_DIPLOMAT_EMBASSY, E_S_D_ME, N_("Embassy")),
122 GEN_EV(E_MY_DIPLOMAT_FAILED, E_S_D_ME, N_("Failed")),
123 GEN_EV(E_MY_DIPLOMAT_INCITE, E_S_D_ME, N_("Incite")),
124 GEN_EV(E_MY_DIPLOMAT_POISON, E_S_D_ME, N_("Poison")),
125 GEN_EV(E_MY_DIPLOMAT_SABOTAGE, E_S_D_ME, N_("Sabotage")),
126 GEN_EV(E_MY_DIPLOMAT_THEFT, E_S_D_ME, N_("Theft")),
127 GEN_EV(E_MY_SPY_STEAL_GOLD, E_S_D_ME, N_("Gold Theft")),
128 GEN_EV(E_MY_SPY_STEAL_MAP, E_S_D_ME, N_("Map Theft")),
129 GEN_EV(E_MY_SPY_NUKE, E_S_D_ME, N_("Suitcase Nuke")),
130 GEN_EV(E_ENEMY_DIPLOMAT_BRIBE, E_S_D_THEM, N_("Bribe")),
131 GEN_EV(E_ENEMY_DIPLOMAT_EMBASSY, E_S_D_THEM, N_("Embassy")),
132 GEN_EV(E_ENEMY_DIPLOMAT_FAILED, E_S_D_THEM, N_("Failed")),
133 GEN_EV(E_ENEMY_DIPLOMAT_INCITE, E_S_D_THEM, N_("Incite")),
134 GEN_EV(E_ENEMY_DIPLOMAT_POISON, E_S_D_THEM, N_("Poison")),
135 GEN_EV(E_ENEMY_DIPLOMAT_SABOTAGE, E_S_D_THEM, N_("Sabotage")),
136 GEN_EV(E_ENEMY_DIPLOMAT_THEFT, E_S_D_THEM, N_("Theft")),
137 GEN_EV(E_ENEMY_SPY_STEAL_GOLD, E_S_D_THEM, N_("Gold Theft")),
138 GEN_EV(E_ENEMY_SPY_STEAL_MAP, E_S_D_THEM, N_("Map Theft")),
139 GEN_EV(E_ENEMY_SPY_NUKE, E_S_D_THEM, N_("Suitcase Nuke")),
140 GEN_EV(E_GLOBAL_ECO, E_S_GLOBAL, N_("Eco-Disaster")),
141 GEN_EV(E_NUKE, E_S_GLOBAL, N_("Nuke Detonated")),
142 GEN_EV(E_HUT_BARB, E_S_HUT, N_("Barbarians in a Hut Roused")),
143 GEN_EV(E_HUT_CITY, E_S_HUT, N_("City Founded from Hut")),
144 GEN_EV(E_HUT_GOLD, E_S_HUT, N_("Gold Found in Hut")),
145 GEN_EV(E_HUT_BARB_KILLED, E_S_HUT, N_("Killed by Barbarians in a Hut")),
146 GEN_EV(E_HUT_MERC, E_S_HUT, N_("Mercenaries Found in Hut")),
147 GEN_EV(E_HUT_SETTLER, E_S_HUT, N_("Settler Found in Hut")),
148 GEN_EV(E_HUT_TECH, E_S_HUT, N_("Tech Found in Hut")),
149 GEN_EV(E_HUT_BARB_CITY_NEAR, E_S_HUT, N_("Unit Spared by Barbarians")),
150 GEN_EV(E_ACHIEVEMENT, E_S_NATION, N_("Achievements")),
151 GEN_EV(E_UPRISING, E_S_NATION, N_("Barbarian Uprising")),
152 GEN_EV(E_CIVIL_WAR, E_S_NATION, N_("Civil War")),
153 GEN_EV(E_ANARCHY, E_S_NATION, N_("Collapse to Anarchy")),
154 GEN_EV(E_FIRST_CONTACT, E_S_NATION, N_("First Contact")),
155 GEN_EV(E_NEW_GOVERNMENT, E_S_NATION, N_("Learned New Government")),
156 GEN_EV(E_LOW_ON_FUNDS, E_S_NATION, N_("Low Funds")),
157 GEN_EV(E_POLLUTION, E_S_NATION, N_("Pollution")),
158 GEN_EV(E_REVOLT_DONE, E_S_NATION, N_("Revolution Ended")),
159 GEN_EV(E_REVOLT_START, E_S_NATION, N_("Revolution Started")),
160 GEN_EV(E_SPACESHIP, E_S_NATION, N_("Spaceship Events")),
161 GEN_EV(E_TREATY_ALLIANCE, E_S_TREATY, N_("Alliance")),
162 GEN_EV(E_TREATY_BROKEN, E_S_TREATY, N_("Broken")),
163 GEN_EV(E_TREATY_CEASEFIRE, E_S_TREATY, N_("Cease-fire")),
164 GEN_EV(E_TREATY_EMBASSY, E_S_TREATY, N_("Embassy")),
165 GEN_EV(E_TREATY_PEACE, E_S_TREATY, N_("Peace")),
166 GEN_EV(E_TREATY_SHARED_VISION,E_S_TREATY, N_("Shared Vision")),
167 GEN_EV(E_UNIT_LOST_ATT, E_S_UNIT, N_("Attack Failed")),
168 GEN_EV(E_UNIT_WIN_ATT, E_S_UNIT, N_("Attack Succeeded")),
169 GEN_EV(E_UNIT_BUY, E_S_UNIT, N_("Bought")),
170 GEN_EV(E_UNIT_BUILT, E_S_UNIT, N_("Built")),
171 GEN_EV(E_UNIT_LOST_DEF, E_S_UNIT, N_("Defender Destroyed")),
172 GEN_EV(E_UNIT_WIN, E_S_UNIT, N_("Defender Survived")),
173 GEN_EV(E_UNIT_BECAME_VET, E_S_UNIT, N_("Promoted to Veteran")),
174 GEN_EV(E_UNIT_LOST_MISC, E_S_UNIT, N_("Lost outside battle")),
175 GEN_EV(E_UNIT_UPGRADED, E_S_UNIT, N_("Production Upgraded")),
176 GEN_EV(E_UNIT_RELOCATED, E_S_UNIT, N_("Relocated")),
177 GEN_EV(E_UNIT_ORDERS, E_S_UNIT, N_("Orders / goto events")),
178 GEN_EV(E_UNIT_BUILT_POP_COST, E_S_UNIT, N_("Built unit with population cost")),
179 GEN_EV(E_UNIT_WAS_EXPELLED, E_S_UNIT, N_("Was Expelled")),
180 GEN_EV(E_UNIT_DID_EXPEL, E_S_UNIT, N_("Did Expel")),
181 GEN_EV(E_UNIT_ACTION_FAILED, E_S_UNIT, N_("Action failed")),
182 /* TRANS: "vote" as a process */
183 GEN_EV(E_VOTE_NEW, E_S_VOTE, N_("New vote")),
184 /* TRANS: "Vote" as a process */
185 GEN_EV(E_VOTE_RESOLVED, E_S_VOTE, N_("Vote resolved")),
186 /* TRANS: "Vote" as a process */
187 GEN_EV(E_VOTE_ABORTED, E_S_VOTE, N_("Vote canceled")),
188 GEN_EV(E_WONDER_BUILD, E_S_WONDER, N_("Finished")),
189 GEN_EV(E_WONDER_OBSOLETE, E_S_WONDER, N_("Made Obsolete")),
190 GEN_EV(E_WONDER_STARTED, E_S_WONDER, N_("Started")),
191 GEN_EV(E_WONDER_STOPPED, E_S_WONDER, N_("Stopped")),
192 GEN_EV(E_WONDER_WILL_BE_BUILT,E_S_WONDER, N_("Will Finish Next Turn")),
193 GEN_EV(E_AI_DEBUG, E_S_XYZZY, N_("AI Debug messages")),
194 GEN_EV(E_BROADCAST_REPORT, E_S_XYZZY, N_("Broadcast Report")),
195 GEN_EV(E_CARAVAN_ACTION, E_S_XYZZY, N_("Caravan actions")),
196 GEN_EV(E_CHAT_ERROR, E_S_XYZZY, N_("Chat error messages")),
197 GEN_EV(E_CHAT_MSG, E_S_XYZZY, N_("Chat messages")),
198 GEN_EV(E_CONNECTION, E_S_XYZZY, N_("Connect/disconnect messages")),
199 GEN_EV(E_DIPLOMACY, E_S_XYZZY, N_("Diplomatic Message")),
200 GEN_EV(E_BAD_COMMAND, E_S_XYZZY, N_("Error message from bad command")),
201 GEN_EV(E_GAME_END, E_S_XYZZY, N_("Game Ended")),
202 GEN_EV(E_GAME_START, E_S_XYZZY, N_("Game Started")),
203 GEN_EV(E_NATION_SELECTED, E_S_XYZZY, N_("Nation Selected")),
204 GEN_EV(E_DESTROYED, E_S_XYZZY, N_("Player Destroyed")),
205 GEN_EV(E_REPORT, E_S_XYZZY, N_("Report")),
206 GEN_EV(E_LOG_FATAL, E_S_XYZZY, N_("Server Aborting")),
207 GEN_EV(E_LOG_ERROR, E_S_XYZZY, N_("Server Problems")),
208 GEN_EV(E_MESSAGE_WALL, E_S_XYZZY, N_("Message from server operator")),
209 GEN_EV(E_SETTING, E_S_XYZZY, N_("Server settings changed")),
210 GEN_EV(E_TURN_BELL, E_S_XYZZY, N_("Turn Bell")),
211 GEN_EV(E_SCRIPT, E_S_XYZZY, N_("Scenario/ruleset script message")),
212 /* TRANS: Event name for when the game year changes. */
213 GEN_EV(E_NEXT_YEAR, E_S_XYZZY, N_("Year Advance")),
214 GEN_EV(E_DEPRECATION_WARNING, E_S_XYZZY, N_("Deprecated Modpack syntax warnings")),
215 GEN_EV(E_SPONTANEOUS_EXTRA, E_S_XYZZY, N_("Extra Appears or Disappears")),
216 GEN_EV(E_UNIT_ILLEGAL_ACTION, E_S_UNIT, N_("Unit Illegal Action")),
217 GEN_EV(E_UNIT_ESCAPED, E_S_UNIT, N_("Unit escaped")),
218 GEN_EV(E_BEGINNER_HELP, E_S_XYZZY, N_("Help for beginners")),
219 /* The sound system also generates "e_game_quit", although there's no
220 * corresponding identifier E_GAME_QUIT. */
225 * Maps from enum event_type to indexes of events[]. Set by
226 * events_init.
228 static int event_to_index[E_COUNT];
230 enum event_type sorted_events[E_COUNT];
233 /**************************************************************************
234 Returns the translated description of the given event.
235 **************************************************************************/
236 const char *get_event_message_text(enum event_type event)
238 fc_assert_ret_val(event_type_is_valid(event), NULL);
240 if (events[event_to_index[event]].event == event) {
241 return events[event_to_index[event]].full_descr;
244 log_error("unknown event %d", event);
245 return "UNKNOWN EVENT"; /* FIXME: Should be marked for translation?
246 * we get non-translated in log message. */
249 /**************************************************************************
250 Comparison function for qsort; i1 and i2 are pointers to an event
251 (enum event_type).
252 **************************************************************************/
253 static int compar_event_message_texts(const void *i1, const void *i2)
255 const enum event_type *j1 = i1;
256 const enum event_type *j2 = i2;
258 return fc_strcasecmp(get_event_message_text(*j1),
259 get_event_message_text(*j2));
262 /****************************************************************************
263 Returns a string for the sound to be used for this message type.
264 ****************************************************************************/
265 const char *get_event_tag(enum event_type event)
267 fc_assert_ret_val(event_type_is_valid(event), NULL);
269 if (events[event_to_index[event]].event == event) {
270 return events[event_to_index[event]].tag_name;
272 log_error("unknown event %d", event);
273 return NULL;
276 /****************************************************************************
277 If is_city_event is FALSE this event doesn't effect a city even if
278 there is a city at the event location.
279 ****************************************************************************/
280 bool is_city_event(enum event_type event)
282 switch (event) {
283 case E_GLOBAL_ECO:
284 case E_CITY_LOST:
285 case E_UNIT_LOST_DEF: /* FIXME: Is this correct.
286 * I'd like to find now defendeseless city quickly! */
287 case E_UNIT_LOST_MISC:
288 case E_UNIT_WIN:
289 case E_ENEMY_DIPLOMAT_FAILED:
290 case E_ENEMY_DIPLOMAT_EMBASSY:
291 case E_ENEMY_DIPLOMAT_POISON:
292 case E_ENEMY_DIPLOMAT_BRIBE:
293 case E_ENEMY_DIPLOMAT_INCITE:
294 case E_ENEMY_DIPLOMAT_SABOTAGE:
295 case E_ENEMY_DIPLOMAT_THEFT:
296 case E_MY_DIPLOMAT_FAILED:
297 case E_MY_DIPLOMAT_EMBASSY:
298 case E_MY_DIPLOMAT_POISON:
299 case E_MY_DIPLOMAT_BRIBE:
300 case E_MY_DIPLOMAT_INCITE:
301 case E_MY_DIPLOMAT_SABOTAGE:
302 case E_MY_DIPLOMAT_THEFT:
303 case E_MY_DIPLOMAT_ESCAPE:
304 case E_UNIT_LOST_ATT:
305 case E_UNIT_WIN_ATT:
306 case E_UPRISING:
307 case E_UNIT_RELOCATED:
308 case E_UNIT_ILLEGAL_ACTION:
309 return FALSE;
311 default:
312 return TRUE;
316 /****************************************************************************
317 Initialize events.
318 Now also initialise sorted_events[].
319 ****************************************************************************/
320 void events_init(void)
322 int i;
324 for (i = 0; i < ARRAY_SIZE(event_to_index); i++) {
325 event_to_index[i] = 0;
328 for (i = 0; i < E_COUNT; i++) {
329 int j;
331 if (E_S_XYZZY > events[i].esn) {
332 const char *event_format = Q_(event_sections[events[i].esn]);
333 int l = 1 + strlen(event_format) + strlen(_(events[i].descr_orig));
335 events[i].full_descr = fc_malloc(l);
336 fc_snprintf(events[i].full_descr, l, event_format,
337 _(events[i].descr_orig));
338 } else {
339 /* No section part */
340 events[i].full_descr = _(events[i].descr_orig);
343 event_to_index[events[i].event] = i;
344 events[i].tag_name = fc_strdup(events[i].enum_name);
345 for (j = 0; j < strlen(events[i].tag_name); j++) {
346 events[i].tag_name[j] = fc_tolower(events[i].tag_name[j]);
348 log_debug("event[%d]=%d: name='%s' / '%s'\n"
349 "\tdescr_orig='%s'\n"
350 "\tdescr='%s'",
351 i, events[i].event, events[i].enum_name, events[i].tag_name,
352 events[i].descr_orig, events[i].full_descr);
355 for (i = 0; i <= event_type_max(); i++) {
356 /* Initialise sorted list of all (even possble missing) events. */
357 sorted_events[i] = i;
359 qsort(sorted_events, event_type_max() + 1, sizeof(*sorted_events),
360 compar_event_message_texts);
363 /****************************************************************************
364 Free events.
365 ****************************************************************************/
366 void events_free(void)
368 int i;
370 for (i = 0; i <= event_type_max(); i++) {
371 if (E_S_XYZZY > events[i].esn) {
372 /* We have allocated memory for this event */
373 free(events[i].full_descr);
374 events[i].full_descr = NULL;