When mixer is not available, recommend SDL2_mixer instead of SDL1.2 mixer
[freeciv.git] / client / text.c
blob7eba42c729fe9ecd7aeb79159f2f8f9f8c60913a
1 /***********************************************************************
2 Freeciv - Copyright (C) 2002 - The Freeciv Project
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 <stdarg.h>
19 #include <string.h>
20 #include <math.h> /* ceil */
22 /* utility */
23 #include "astring.h"
24 #include "bitvector.h"
25 #include "fcintl.h"
26 #include "log.h"
27 #include "support.h"
29 /* common */
30 #include "calendar.h"
31 #include "citizens.h"
32 #include "clientutils.h"
33 #include "combat.h"
34 #include "fc_types.h" /* LINE_BREAK */
35 #include "game.h"
36 #include "government.h"
37 #include "map.h"
38 #include "research.h"
39 #include "traderoutes.h"
40 #include "unitlist.h"
42 /* client */
43 #include "client_main.h"
44 #include "climap.h"
45 #include "climisc.h"
46 #include "control.h"
47 #include "goto.h"
49 #include "text.h"
52 static int get_bulbs_per_turn(int *pours, bool *pteam, int *ptheirs);
54 /****************************************************************************
55 Return a (static) string with a tile's food/prod/trade
56 ****************************************************************************/
57 const char *get_tile_output_text(const struct tile *ptile)
59 static struct astring str = ASTRING_INIT;
60 int i;
61 char output_text[O_LAST][16];
63 for (i = 0; i < O_LAST; i++) {
64 int before_penalty = 0;
65 int x = city_tile_output(NULL, ptile, FALSE, i);
67 if (NULL != client.conn.playing) {
68 before_penalty = get_player_output_bonus(client.conn.playing,
69 get_output_type(i),
70 EFT_OUTPUT_PENALTY_TILE);
73 if (before_penalty > 0 && x > before_penalty) {
74 fc_snprintf(output_text[i], sizeof(output_text[i]), "%d(-1)", x);
75 } else {
76 fc_snprintf(output_text[i], sizeof(output_text[i]), "%d", x);
80 astr_set(&str, "%s/%s/%s", output_text[O_FOOD],
81 output_text[O_SHIELD], output_text[O_TRADE]);
83 return astr_str(&str);
86 /****************************************************************************
87 For AIs, fill the buffer with their player name prefixed with "AI". For
88 humans, just fill it with their username.
89 ****************************************************************************/
90 static inline void get_full_username(char *buf, int buflen,
91 const struct player *pplayer)
93 if (!buf || buflen < 1) {
94 return;
97 if (!pplayer) {
98 buf[0] = '\0';
99 return;
102 if (pplayer->ai_controlled) {
103 /* TRANS: "AI <player name>" */
104 fc_snprintf(buf, buflen, _("AI %s"), pplayer->name);
105 } else {
106 fc_strlcpy(buf, pplayer->username, buflen);
110 /****************************************************************************
111 Fill the buffer with the player's nation name (in adjective form) and
112 optionally add the player's team name.
113 ****************************************************************************/
114 static inline void get_full_nation(char *buf, int buflen,
115 const struct player *pplayer)
117 if (!buf || buflen < 1) {
118 return;
121 if (!pplayer) {
122 buf[0] = '\0';
123 return;
126 if (pplayer->team) {
127 /* TRANS: "<nation adjective>, team <team name>" */
128 fc_snprintf(buf, buflen, _("%s, team %s"),
129 nation_adjective_for_player(pplayer),
130 team_name_translation(pplayer->team));
131 } else {
132 fc_strlcpy(buf, nation_adjective_for_player(pplayer), buflen);
136 /****************************************************************************
137 Text to popup on a middle-click in the mapview.
138 ****************************************************************************/
139 const char *popup_info_text(struct tile *ptile)
141 const char *activity_text;
142 struct city *pcity = tile_city(ptile);
143 struct unit *punit = find_visible_unit(ptile);
144 const char *diplo_nation_plural_adjectives[DS_LAST] =
145 {"" /* unused, DS_ARMISTICE */, Q_("?nation:Hostile"),
146 "" /* unused, DS_CEASEFIRE */,
147 Q_("?nation:Peaceful"), Q_("?nation:Friendly"),
148 Q_("?nation:Mysterious"), Q_("?nation:Friendly(team)")};
149 const char *diplo_city_adjectives[DS_LAST] =
150 {"" /* unused, DS_ARMISTICE */, Q_("?city:Hostile"),
151 "" /* unused, DS_CEASEFIRE */,
152 Q_("?city:Peaceful"), Q_("?city:Friendly"), Q_("?city:Mysterious"),
153 Q_("?city:Friendly(team)")};
154 static struct astring str = ASTRING_INIT;
155 char username[MAX_LEN_NAME + 32];
156 char nation[2 * MAX_LEN_NAME + 32];
157 int tile_x, tile_y, nat_x, nat_y;
158 bool first;
160 astr_clear(&str);
161 index_to_map_pos(&tile_x, &tile_y, tile_index(ptile));
162 astr_add_line(&str, _("Location: (%d, %d) [%d]"),
163 tile_x, tile_y, tile_continent(ptile));
164 index_to_native_pos(&nat_x, &nat_y, tile_index(ptile));
165 astr_add_line(&str, _("Native coordinates: (%d, %d)"),
166 nat_x, nat_y);
168 if (client_tile_get_known(ptile) == TILE_UNKNOWN) {
169 astr_add(&str, _("Unknown"));
170 return astr_str(&str);
172 astr_add_line(&str, _("Terrain: %s"), tile_get_info_text(ptile, TRUE, 0));
173 astr_add_line(&str, _("Food/Prod/Trade: %s"),
174 get_tile_output_text(ptile));
175 first = TRUE;
176 extra_type_iterate(pextra) {
177 if (pextra->category == ECAT_BONUS && tile_has_visible_extra(ptile, pextra)) {
178 if (!first) {
179 astr_add(&str, ",%s", extra_name_translation(pextra));
180 } else {
181 astr_add_line(&str, "%s", extra_name_translation(pextra));
182 first = FALSE;
185 } extra_type_iterate_end;
186 if (BORDERS_DISABLED != game.info.borders && !pcity) {
187 struct player *owner = tile_owner(ptile);
189 get_full_username(username, sizeof(username), owner);
190 get_full_nation(nation, sizeof(nation), owner);
192 if (NULL != client.conn.playing && owner == client.conn.playing) {
193 astr_add_line(&str, _("Our territory"));
194 } else if (NULL != owner && NULL == client.conn.playing) {
195 /* TRANS: "Territory of <username> (<nation + team>)" */
196 astr_add_line(&str, _("Territory of %s (%s)"), username, nation);
197 } else if (NULL != owner) {
198 struct player_diplstate *ds = player_diplstate_get(client.conn.playing,
199 owner);
201 if (ds->type == DS_CEASEFIRE) {
202 int turns = ds->turns_left;
204 astr_add_line(&str,
205 /* TRANS: "Territory of <username> (<nation + team>)
206 * (<number> turn cease-fire)" */
207 PL_("Territory of %s (%s) (%d turn cease-fire)",
208 "Territory of %s (%s) (%d turn cease-fire)",
209 turns),
210 username, nation, turns);
211 } else if (ds->type == DS_ARMISTICE) {
212 int turns = ds->turns_left;
214 astr_add_line(&str,
215 /* TRANS: "Territory of <username> (<nation + team>)
216 * (<number> turn armistice)" */
217 PL_("Territory of %s (%s) (%d turn armistice)",
218 "Territory of %s (%s) (%d turn armistice)",
219 turns),
220 username, nation, turns);
221 } else {
222 int type = ds->type;
224 /* TRANS: "Territory of <username>
225 * (<nation + team> | <diplomatic state>)" */
226 astr_add_line(&str, _("Territory of %s (%s | %s)"),
227 username, nation,
228 diplo_nation_plural_adjectives[type]);
230 } else {
231 astr_add_line(&str, _("Unclaimed territory"));
234 if (pcity) {
235 /* Look at city owner, not tile owner (the two should be the same, if
236 * borders are in use). */
237 struct player *owner = city_owner(pcity);
238 const char *improvements[improvement_count()];
239 int has_improvements = 0;
241 get_full_username(username, sizeof(username), owner);
242 get_full_nation(nation, sizeof(nation), owner);
244 if (NULL == client.conn.playing || owner == client.conn.playing) {
245 /* TRANS: "City: <city name> | <username> (<nation + team>)" */
246 astr_add_line(&str, _("City: %s | %s (%s)"),
247 city_name_get(pcity), username, nation);
248 } else {
249 struct player_diplstate *ds
250 = player_diplstate_get(client_player(), owner);
251 if (ds->type == DS_CEASEFIRE) {
252 int turns = ds->turns_left;
254 /* TRANS: "City: <city name> | <username>
255 * (<nation + team>, <number> turn cease-fire)" */
256 astr_add_line(&str, PL_("City: %s | %s (%s, %d turn cease-fire)",
257 "City: %s | %s (%s, %d turn cease-fire)",
258 turns),
259 city_name_get(pcity), username, nation, turns);
260 } else if (ds->type == DS_ARMISTICE) {
261 int turns = ds->turns_left;
263 /* TRANS: "City: <city name> | <username>
264 * (<nation + team>, <number> turn armistice)" */
265 astr_add_line(&str, PL_("City: %s | %s (%s, %d turn armistice)",
266 "City: %s | %s (%s, %d turn armistice)",
267 turns),
268 city_name_get(pcity), username, nation, turns);
269 } else {
270 /* TRANS: "City: <city name> | <username>
271 * (<nation + team>, <diplomatic state>)" */
272 astr_add_line(&str, _("City: %s | %s (%s, %s)"),
273 city_name_get(pcity), username, nation,
274 diplo_city_adjectives[ds->type]);
277 if (can_player_see_units_in_city(client_player(), pcity)) {
278 int count = unit_list_size(ptile->units);
280 if (count > 0) {
281 astr_add(&str, PL_(" | Occupied with %d unit.",
282 " | Occupied with %d units.", count), count);
283 } else {
284 astr_add(&str, _(" | Not occupied."));
286 } else {
287 if (pcity->client.occupied) {
288 astr_add(&str, _(" | Occupied."));
289 } else {
290 astr_add(&str, _(" | Not occupied."));
293 improvement_iterate(pimprove) {
294 if (is_improvement_visible(pimprove)
295 && city_has_building(pcity, pimprove)) {
296 improvements[has_improvements++] =
297 improvement_name_translation(pimprove);
299 } improvement_iterate_end;
301 if (0 < has_improvements) {
302 struct astring list = ASTRING_INIT;
304 astr_build_and_list(&list, improvements, has_improvements);
305 /* TRANS: %s is a list of "and"-separated improvements. */
306 astr_add_line(&str, _(" with %s."), astr_str(&list));
307 astr_free(&list);
310 unit_list_iterate(get_units_in_focus(), pfocus_unit) {
311 struct city *hcity = game_city_by_number(pfocus_unit->homecity);
313 if (utype_can_do_action(unit_type_get(pfocus_unit), ACTION_TRADE_ROUTE)
314 && can_cities_trade(hcity, pcity)
315 && can_establish_trade_route(hcity, pcity)) {
316 /* TRANS: "Trade from Warsaw: 5" */
317 astr_add_line(&str, _("Trade from %s: %d"),
318 city_name_get(hcity),
319 trade_between_cities(hcity, pcity));
321 } unit_list_iterate_end;
324 const char *infratext = get_infrastructure_text(ptile->extras);
326 if (*infratext != '\0') {
327 astr_add_line(&str, _("Infrastructure: %s"), infratext);
330 activity_text = concat_tile_activity_text(ptile);
331 if (strlen(activity_text) > 0) {
332 astr_add_line(&str, _("Activity: %s"), activity_text);
334 if (punit && !pcity) {
335 struct player *owner = unit_owner(punit);
336 struct unit_type *ptype = unit_type_get(punit);
338 get_full_username(username, sizeof(username), owner);
339 get_full_nation(nation, sizeof(nation), owner);
341 if (!client_player() || owner == client_player()) {
342 struct city *hcity = player_city_by_number(owner, punit->homecity);
344 /* TRANS: "Unit: <unit type> | <username> (<nation + team>)" */
345 astr_add_line(&str, _("Unit: %s | %s (%s)"),
346 utype_name_translation(ptype), username, nation);
348 if (game.info.citizen_nationality
349 && unit_nationality(punit) != unit_owner(punit)) {
350 if (hcity != NULL) {
351 /* TRANS: on own line immediately following \n, "from <city> |
352 * <nationality> people" */
353 astr_add_line(&str, _("from %s | %s people"), city_name_get(hcity),
354 nation_adjective_for_player(unit_nationality(punit)));
355 } else {
356 /* TRANS: Nationality of the people comprising a unit, if
357 * different from owner. */
358 astr_add_line(&str, _("%s people"),
359 nation_adjective_for_player(unit_nationality(punit)));
361 } else if (hcity != NULL) {
362 /* TRANS: on own line immediately following \n, ... <city> */
363 astr_add_line(&str, _("from %s"), city_name_get(hcity));
365 } else if (NULL != owner) {
366 struct player_diplstate *ds = player_diplstate_get(client_player(),
367 owner);
368 if (ds->type == DS_CEASEFIRE) {
369 int turns = ds->turns_left;
371 /* TRANS: "Unit: <unit type> | <username> (<nation + team>,
372 * <number> turn cease-fire)" */
373 astr_add_line(&str, PL_("Unit: %s | %s (%s, %d turn cease-fire)",
374 "Unit: %s | %s (%s, %d turn cease-fire)",
375 turns),
376 utype_name_translation(ptype),
377 username, nation, turns);
378 } else if (ds->type == DS_ARMISTICE) {
379 int turns = ds->turns_left;
381 /* TRANS: "Unit: <unit type> | <username> (<nation + team>,
382 * <number> turn armistice)" */
383 astr_add_line(&str, PL_("Unit: %s | %s (%s, %d turn armistice)",
384 "Unit: %s | %s (%s, %d turn armistice)",
385 turns),
386 utype_name_translation(ptype),
387 username, nation, turns);
388 } else {
389 /* TRANS: "Unit: <unit type> | <username> (<nation + team>,
390 * <diplomatic state>)" */
391 astr_add_line(&str, _("Unit: %s | %s (%s, %s)"),
392 utype_name_translation(ptype), username, nation,
393 diplo_city_adjectives[ds->type]);
397 unit_list_iterate(get_units_in_focus(), pfocus_unit) {
398 int att_chance = FC_INFINITY, def_chance = FC_INFINITY;
399 bool found = FALSE;
401 unit_list_iterate(ptile->units, tile_unit) {
402 if (unit_owner(tile_unit) != unit_owner(pfocus_unit)) {
403 int att = unit_win_chance(pfocus_unit, tile_unit) * 100;
404 int def = (1.0 - unit_win_chance(tile_unit, pfocus_unit)) * 100;
406 found = TRUE;
408 /* Presumably the best attacker and defender will be used. */
409 att_chance = MIN(att, att_chance);
410 def_chance = MIN(def, def_chance);
412 } unit_list_iterate_end;
414 if (found) {
415 /* TRANS: "Chance to win: A:95% D:46%" */
416 astr_add_line(&str, _("Chance to win: A:%d%% D:%d%%"),
417 att_chance, def_chance);
419 } unit_list_iterate_end;
421 /* TRANS: A is attack power, D is defense power, FP is firepower,
422 * HP is hitpoints (current and max). */
423 astr_add_line(&str, _("A:%d D:%d FP:%d HP:%d/%d"),
424 ptype->attack_strength, ptype->defense_strength,
425 ptype->firepower, punit->hp, ptype->hp);
427 const char *veteran_name =
428 utype_veteran_name_translation(ptype, punit->veteran);
429 if (veteran_name) {
430 astr_add(&str, " (%s)", veteran_name);
434 if (unit_owner(punit) == client_player()
435 || client_is_global_observer()) {
436 /* Show bribe cost for own units. */
437 astr_add_line(&str, _("Probable bribe cost: %d"),
438 unit_bribe_cost(punit, NULL));
439 } else {
440 /* We can only give an (lower) boundary for units of other players. */
441 astr_add_line(&str, _("Estimated bribe cost: > %d"),
442 unit_bribe_cost(punit, client_player()));
445 if ((NULL == client.conn.playing || owner == client.conn.playing)
446 && unit_list_size(ptile->units) >= 2) {
447 /* TRANS: "5 more" units on this tile */
448 astr_add(&str, _(" (%d more)"), unit_list_size(ptile->units) - 1);
452 astr_break_lines(&str, LINE_BREAK);
453 return astr_str(&str);
456 #define FAR_CITY_SQUARE_DIST (2*(6*6))
457 /****************************************************************************
458 Returns the text describing the city and its distance.
459 ****************************************************************************/
460 const char *get_nearest_city_text(struct city *pcity, int sq_dist)
462 static struct astring str = ASTRING_INIT;
464 astr_clear(&str);
466 /* just to be sure */
467 if (!pcity) {
468 sq_dist = -1;
471 astr_add(&str, (sq_dist >= FAR_CITY_SQUARE_DIST)
472 /* TRANS: on own line immediately following \n, ... <city> */
473 ? _("far from %s")
474 : (sq_dist > 0)
475 /* TRANS: on own line immediately following \n, ... <city> */
476 ? _("near %s")
477 : (sq_dist == 0)
478 /* TRANS: on own line immediately following \n, ... <city> */
479 ? _("in %s")
480 : "%s",
481 pcity
482 ? city_name_get(pcity)
483 : "");
485 return astr_str(&str);
488 /****************************************************************************
489 Returns the unit description.
490 Used in e.g. city report tooltips.
492 FIXME: This function is not re-entrant because it returns a pointer to
493 static data.
494 ****************************************************************************/
495 const char *unit_description(struct unit *punit)
497 int pcity_near_dist;
498 struct player *owner = unit_owner(punit);
499 struct player *nationality = unit_nationality(punit);
500 struct city *pcity =
501 player_city_by_number(owner, punit->homecity);
502 struct city *pcity_near = get_nearest_city(punit, &pcity_near_dist);
503 struct unit_type *ptype = unit_type_get(punit);
504 static struct astring str = ASTRING_INIT;
505 const struct player *pplayer = client_player();
507 astr_clear(&str);
509 astr_add(&str, "%s", utype_name_translation(ptype));
512 const char *veteran_name =
513 utype_veteran_name_translation(ptype, punit->veteran);
514 if (veteran_name) {
515 astr_add(&str, " (%s)", veteran_name);
519 if (pplayer == owner) {
520 unit_upkeep_astr(punit, &str);
521 } else {
522 astr_add(&str, "\n");
524 unit_activity_astr(punit, &str);
526 if (pcity) {
527 /* TRANS: on own line immediately following \n, ... <city> */
528 astr_add_line(&str, _("from %s"), city_name_get(pcity));
529 } else {
530 astr_add(&str, "\n");
532 if (game.info.citizen_nationality) {
533 if (nationality != NULL && owner != nationality) {
534 /* TRANS: Nationality of the people comprising a unit, if
535 * different from owner. */
536 astr_add_line(&str, _("%s people"),
537 nation_adjective_for_player(nationality));
538 } else {
539 astr_add(&str, "\n");
543 astr_add_line(&str, "%s",
544 get_nearest_city_text(pcity_near, pcity_near_dist));
545 #ifdef DEBUG
546 astr_add_line(&str, "Unit ID: %d", punit->id);
547 #endif
549 return astr_str(&str);
552 /****************************************************************************
553 Describe the airlift capacity of a city for the given units (from their
554 current positions).
555 If pdest is non-NULL, describe its capacity as a destination, otherwise
556 describe the capacity of the city the unit's currently in (if any) as a
557 source. (If the units in the list are in different cities, this will
558 probably not give a useful result in this case.)
559 If not all of the listed units can be airlifted, return the description
560 for those that can.
561 Returns NULL if an airlift is not possible for any of the units.
562 ****************************************************************************/
563 const char *get_airlift_text(const struct unit_list *punits,
564 const struct city *pdest)
566 static struct astring str = ASTRING_INIT;
567 bool src = (pdest == NULL);
568 enum texttype { AL_IMPOSSIBLE, AL_UNKNOWN, AL_FINITE, AL_INFINITE }
569 best = AL_IMPOSSIBLE;
570 int cur = 0, max = 0;
572 unit_list_iterate(punits, punit) {
573 enum texttype this = AL_IMPOSSIBLE;
574 enum unit_airlift_result result;
576 /* NULL will tell us about the capability of airlifting from source */
577 result = test_unit_can_airlift_to(client_player(), punit, pdest);
579 switch(result) {
580 case AR_NO_MOVES:
581 case AR_WRONG_UNITTYPE:
582 case AR_OCCUPIED:
583 case AR_NOT_IN_CITY:
584 case AR_BAD_SRC_CITY:
585 case AR_BAD_DST_CITY:
586 /* No chance of an airlift. */
587 this = AL_IMPOSSIBLE;
588 break;
589 case AR_OK:
590 case AR_OK_SRC_UNKNOWN:
591 case AR_OK_DST_UNKNOWN:
592 case AR_SRC_NO_FLIGHTS:
593 case AR_DST_NO_FLIGHTS:
594 /* May or may not be able to airlift now, but there's a chance we could
595 * later */
597 const struct city *pcity = src ? tile_city(unit_tile(punit)) : pdest;
598 fc_assert_ret_val(pcity != NULL, fc_strdup("-"));
599 if (!src && (game.info.airlifting_style & AIRLIFTING_UNLIMITED_DEST)) {
600 /* No restrictions on destination (and we can infer this even for
601 * other players' cities). */
602 this = AL_INFINITE;
603 } else if (client_player() == city_owner(pcity)) {
604 /* A city we know about. */
605 int this_cur = pcity->airlift, this_max = city_airlift_max(pcity);
606 if (this_max <= 0) {
607 /* City known not to be airlift-capable. */
608 this = AL_IMPOSSIBLE;
609 } else {
610 if (src
611 && (game.info.airlifting_style & AIRLIFTING_UNLIMITED_SRC)) {
612 /* Unlimited capacity. */
613 this = AL_INFINITE;
614 } else {
615 /* Limited capacity (possibly zero right now). */
616 this = AL_FINITE;
617 /* Store the numbers. This whole setup assumes that numeric
618 * capacity isn't unit-dependent. */
619 if (best == AL_FINITE) {
620 fc_assert(cur == this_cur && max == this_max);
622 cur = this_cur;
623 max = this_max;
626 } else {
627 /* Unknown capacity. */
628 this = AL_UNKNOWN;
631 break;
634 /* Now take the most optimistic view. */
635 best = MAX(best, this);
636 } unit_list_iterate_end;
638 switch(best) {
639 case AL_IMPOSSIBLE:
640 return NULL;
641 case AL_UNKNOWN:
642 astr_set(&str, "?");
643 break;
644 case AL_FINITE:
645 astr_set(&str, "%d/%d", cur, max);
646 break;
647 case AL_INFINITE:
648 astr_set(&str, _("Yes"));
649 break;
652 return astr_str(&str);
655 /****************************************************************************
656 Return total expected bulbs.
657 ****************************************************************************/
658 static int get_bulbs_per_turn(int *pours, bool *pteam, int *ptheirs)
660 const struct research *presearch;
661 int ours = 0, theirs = 0;
662 bool team = FALSE;
664 if (!client_has_player()) {
665 return 0;
667 presearch = research_get(client_player());
669 /* Sum up science */
670 research_players_iterate(presearch, pplayer) {
671 if (pplayer == client_player()) {
672 city_list_iterate(pplayer->cities, pcity) {
673 ours += pcity->surplus[O_SCIENCE];
674 } city_list_iterate_end;
675 } else {
676 team = TRUE;
677 theirs -= pplayer->client.tech_upkeep;
679 } research_players_iterate_end;
681 if (team) {
682 theirs += presearch->client.total_bulbs_prod - ours;
684 ours -= client_player()->client.tech_upkeep;
686 if (pours) {
687 *pours = ours;
689 if (pteam) {
690 *pteam = team;
692 if (ptheirs) {
693 *ptheirs = theirs;
695 return ours + theirs;
698 /****************************************************************************
699 Returns the text to display in the science dialog.
700 ****************************************************************************/
701 const char *science_dialog_text(void)
703 bool team;
704 int ours, theirs, perturn, upkeep;
705 static struct astring str = ASTRING_INIT;
706 struct astring ourbuf = ASTRING_INIT, theirbuf = ASTRING_INIT;
707 struct research *research;
709 astr_clear(&str);
711 perturn = get_bulbs_per_turn(&ours, &team, &theirs);
713 research = research_get(client_player());
714 upkeep = client_player()->client.tech_upkeep;
716 if (NULL == client.conn.playing || (ours == 0 && theirs == 0
717 && upkeep == 0)) {
718 return _("Progress: no research");
721 if (A_UNSET == research->researching) {
722 astr_add(&str, _("Progress: no research"));
723 } else {
724 int done = research->bulbs_researched;
725 int total = research->client.researching_cost;
727 if (perturn > 0) {
728 int turns = MAX(1, ceil((double)total) / perturn);
730 astr_add(&str, PL_("Progress: %d turn/advance",
731 "Progress: %d turns/advance",
732 turns), turns);
733 } else if (perturn < 0 ) {
734 /* negative number of bulbs per turn due to tech upkeep */
735 int turns = ceil((double) done / -perturn);
737 astr_add(&str, PL_("Progress: %d turn/advance loss",
738 "Progress: %d turns/advance loss",
739 turns), turns);
740 } else {
741 /* no research */
742 astr_add(&str, _("Progress: none"));
745 astr_set(&ourbuf, PL_("%d bulb/turn", "%d bulbs/turn", ours), ours);
746 if (team) {
747 /* Techpool version */
748 astr_set(&theirbuf,
749 /* TRANS: This is appended to "%d bulb/turn" text */
750 PL_(", %d bulb/turn from team",
751 ", %d bulbs/turn from team", theirs), theirs);
752 } else {
753 astr_clear(&theirbuf);
755 astr_add(&str, " (%s%s)", astr_str(&ourbuf), astr_str(&theirbuf));
756 astr_free(&ourbuf);
757 astr_free(&theirbuf);
759 if (game.info.tech_upkeep_style != TECH_UPKEEP_NONE) {
760 /* perturn is defined as: (bulbs produced) - upkeep */
761 astr_add_line(&str, _("Bulbs produced per turn: %d"), perturn + upkeep);
762 /* TRANS: keep leading space; appended to "Bulbs produced per turn: %d" */
763 astr_add(&str, _(" (needed for technology upkeep: %d)"), upkeep);
766 return astr_str(&str);
769 /****************************************************************************
770 Get the short science-target text. This is usually shown directly in
771 the progress bar.
773 5/28 - 3 turns
775 The "percent" value, if given, will be set to the completion percentage
776 of the research target (actually it's a [0,1] scale not a percent).
777 ****************************************************************************/
778 const char *get_science_target_text(double *percent)
780 struct research *research = research_get(client_player());
781 static struct astring str = ASTRING_INIT;
783 if (!research) {
784 return "-";
787 astr_clear(&str);
788 if (research->researching == A_UNSET) {
789 astr_add(&str, _("%d/- (never)"), research->bulbs_researched);
790 if (percent) {
791 *percent = 0.0;
793 } else {
794 int total = research->client.researching_cost;
795 int done = research->bulbs_researched;
796 int perturn = get_bulbs_per_turn(NULL, NULL, NULL);
798 if (perturn > 0) {
799 int turns = ceil( (double)(total - done) / perturn );
801 astr_add(&str, PL_("%d/%d (%d turn)", "%d/%d (%d turns)", turns),
802 done, total, turns);
803 } else if (perturn < 0 ) {
804 /* negative number of bulbs per turn due to tech upkeep */
805 int turns = ceil( (double)done / -perturn );
807 astr_add(&str, PL_("%d/%d (%d turn)", "%d/%d (%d turns)", turns),
808 done, perturn, turns);
809 } else {
810 /* no research */
811 astr_add(&str, _("%d/%d (never)"), done, total);
813 if (percent) {
814 *percent = (double)done / (double)total;
815 *percent = CLIP(0.0, *percent, 1.0);
819 return astr_str(&str);
822 /****************************************************************************
823 Set the science-goal-label text as if we're researching the given goal.
824 ****************************************************************************/
825 const char *get_science_goal_text(Tech_type_id goal)
827 const struct research *research = research_get(client_player());
828 int steps = research_goal_unknown_techs(research, goal);
829 int bulbs_needed = research_goal_bulbs_required(research, goal);
830 int turns;
831 int perturn = get_bulbs_per_turn(NULL, NULL, NULL);
832 static struct astring str = ASTRING_INIT;
833 struct astring buf1 = ASTRING_INIT,
834 buf2 = ASTRING_INIT,
835 buf3 = ASTRING_INIT;
837 if (!research) {
838 return "-";
841 astr_clear(&str);
843 if (research_goal_tech_req(research, goal, research->researching)
844 || research->researching == goal) {
845 bulbs_needed -= research->bulbs_researched;
848 astr_set(&buf1,
849 PL_("%d step", "%d steps", steps), steps);
850 astr_set(&buf2,
851 PL_("%d bulb", "%d bulbs", bulbs_needed), bulbs_needed);
852 if (perturn > 0) {
853 turns = (bulbs_needed + perturn - 1) / perturn;
854 astr_set(&buf3,
855 PL_("%d turn", "%d turns", turns), turns);
856 } else {
857 astr_set(&buf3, _("never"));
859 astr_add_line(&str, "(%s - %s - %s)",
860 astr_str(&buf1), astr_str(&buf2), astr_str(&buf3));
861 astr_free(&buf1);
862 astr_free(&buf2);
863 astr_free(&buf3);
865 return astr_str(&str);
868 /****************************************************************************
869 Return the text for the label on the info panel. (This is traditionally
870 shown to the left of the mapview.)
872 Clicking on this text should bring up the get_info_label_text_popup text.
873 ****************************************************************************/
874 const char *get_info_label_text(bool moreinfo)
876 static struct astring str = ASTRING_INIT;
878 astr_clear(&str);
880 if (NULL != client.conn.playing) {
881 astr_add_line(&str, _("Population: %s"),
882 population_to_text(civ_population(client.conn.playing)));
884 astr_add_line(&str, _("Year: %s (T%d)"),
885 calendar_text(), game.info.turn);
887 if (NULL != client.conn.playing) {
888 astr_add_line(&str, _("Gold: %d (%+d)"),
889 client.conn.playing->economic.gold,
890 player_get_expected_income(client.conn.playing));
891 astr_add_line(&str, _("Tax: %d Lux: %d Sci: %d"),
892 client.conn.playing->economic.tax,
893 client.conn.playing->economic.luxury,
894 client.conn.playing->economic.science);
896 if (game.info.phase_mode == PMT_PLAYERS_ALTERNATE) {
897 if (game.info.phase < 0 || game.info.phase >= player_count()) {
898 astr_add_line(&str, _("Moving: Nobody"));
899 } else {
900 astr_add_line(&str, _("Moving: %s"),
901 player_name(player_by_number(game.info.phase)));
903 } else if (game.info.phase_mode == PMT_TEAMS_ALTERNATE) {
904 if (game.info.phase < 0 || game.info.phase >= team_count()) {
905 astr_add_line(&str, _("Moving: Nobody"));
906 } else {
907 astr_add_line(&str, _("Moving: %s"),
908 team_name_translation(team_by_number(game.info.phase)));
912 if (moreinfo) {
913 astr_add_line(&str, _("(Click for more info)"));
916 return astr_str(&str);
919 /****************************************************************************
920 Return the text for the popup label on the info panel. (This is
921 traditionally done as a popup whenever the regular info text is clicked
922 on.)
923 ****************************************************************************/
924 const char *get_info_label_text_popup(void)
926 static struct astring str = ASTRING_INIT;
928 astr_clear(&str);
930 if (NULL != client.conn.playing) {
931 astr_add_line(&str, _("%s People"),
932 population_to_text(civ_population(client.conn.playing)));
934 astr_add_line(&str, _("Year: %s"), calendar_text());
935 astr_add_line(&str, _("Turn: %d"), game.info.turn);
937 if (NULL != client.conn.playing) {
938 const struct research *presearch = research_get(client_player());
939 int perturn = get_bulbs_per_turn(NULL, NULL, NULL);
940 int upkeep = client_player()->client.tech_upkeep;
942 astr_add_line(&str, _("Gold: %d"),
943 client.conn.playing->economic.gold);
944 astr_add_line(&str, _("Net Income: %d"),
945 player_get_expected_income(client.conn.playing));
946 /* TRANS: Gold, luxury, and science rates are in percentage values. */
947 astr_add_line(&str, _("Tax rates: Gold:%d%% Luxury:%d%% Science:%d%%"),
948 client.conn.playing->economic.tax,
949 client.conn.playing->economic.luxury,
950 client.conn.playing->economic.science);
951 astr_add_line(&str, _("Researching %s: %s"),
952 research_advance_name_translation(presearch,
953 presearch->researching),
954 get_science_target_text(NULL));
955 /* perturn is defined as: (bulbs produced) - upkeep */
956 if (game.info.tech_upkeep_style != TECH_UPKEEP_NONE) {
957 astr_add_line(&str, _("Bulbs per turn: %d - %d = %d"), perturn + upkeep,
958 upkeep, perturn);
959 } else {
960 fc_assert(upkeep == 0);
961 astr_add_line(&str, _("Bulbs per turn: %d"), perturn);
965 /* See also get_global_warming_tooltip and get_nuclear_winter_tooltip. */
967 if (game.info.global_warming) {
968 int chance, rate;
969 global_warming_scaled(&chance, &rate, 100);
970 astr_add_line(&str, _("Global warming chance: %d%% (%+d%%/turn)"),
971 chance, rate);
972 } else {
973 astr_add_line(&str, _("Global warming deactivated."));
976 if (game.info.nuclear_winter) {
977 int chance, rate;
978 nuclear_winter_scaled(&chance, &rate, 100);
979 astr_add_line(&str, _("Nuclear winter chance: %d%% (%+d%%/turn)"),
980 chance, rate);
981 } else {
982 astr_add_line(&str, _("Nuclear winter deactivated."));
985 if (NULL != client.conn.playing) {
986 astr_add_line(&str, _("Government: %s"),
987 government_name_for_player(client.conn.playing));
990 return astr_str(&str);
993 /****************************************************************************
994 Return the title text for the unit info shown in the info panel.
996 FIXME: this should be renamed.
997 ****************************************************************************/
998 const char *get_unit_info_label_text1(struct unit_list *punits)
1000 static struct astring str = ASTRING_INIT;
1002 astr_clear(&str);
1004 if (punits) {
1005 int count = unit_list_size(punits);
1007 if (count == 1) {
1008 astr_add(&str, "%s", unit_name_translation(unit_list_get(punits, 0)));
1009 } else {
1010 astr_add(&str, PL_("%d unit", "%d units", count), count);
1013 return astr_str(&str);
1016 /****************************************************************************
1017 Return the text body for the unit info shown in the info panel.
1019 FIXME: this should be renamed.
1020 ****************************************************************************/
1021 const char *get_unit_info_label_text2(struct unit_list *punits, int linebreaks)
1023 static struct astring str = ASTRING_INIT;
1024 int count;
1026 astr_clear(&str);
1028 if (!punits) {
1029 return "";
1032 count = unit_list_size(punits);
1034 /* This text should always have the same number of lines if
1035 * 'linebreaks' has no flags at all. Otherwise the GUI widgets may be
1036 * confused and try to resize themselves. If caller asks for
1037 * conditional 'linebreaks', it should take care of these problems
1038 * itself. */
1040 /* Line 1. Goto or activity text. */
1041 if (count > 0 && hover_state != HOVER_NONE) {
1042 int min, max;
1044 if (!goto_get_turns(&min, &max)) {
1045 /* TRANS: Impossible to reach goto target tile */
1046 astr_add_line(&str, "%s", Q_("?goto:Unreachable"));
1047 } else if (min == max) {
1048 astr_add_line(&str, _("Turns to target: %d"), max);
1049 } else {
1050 astr_add_line(&str, _("Turns to target: %d to %d"), min, max);
1052 } else if (count == 1) {
1053 astr_add_line(&str, "%s",
1054 unit_activity_text(unit_list_get(punits, 0)));
1055 } else if (count > 1) {
1056 astr_add_line(&str, PL_("%d unit selected",
1057 "%d units selected",
1058 count),
1059 count);
1060 } else {
1061 astr_add_line(&str, _("No units selected."));
1064 /* Lines 2, 3, 4, and possible 5 vary. */
1065 if (count == 1) {
1066 struct unit *punit = unit_list_get(punits, 0);
1067 struct player *owner = unit_owner(punit);
1068 struct city *pcity = player_city_by_number(owner,
1069 punit->homecity);
1071 astr_add_line(&str, "%s", tile_get_info_text(unit_tile(punit), TRUE,
1072 linebreaks));
1074 const char *infratext = get_infrastructure_text(unit_tile(punit)->extras);
1076 if (*infratext != '\0') {
1077 astr_add_line(&str, "%s", infratext);
1078 } else {
1079 astr_add_line(&str, " ");
1082 if (pcity) {
1083 astr_add_line(&str, "%s", city_name_get(pcity));
1084 } else {
1085 astr_add_line(&str, " ");
1088 if (game.info.citizen_nationality) {
1089 struct player *nationality = unit_nationality(punit);
1091 /* Line 5, nationality text */
1092 if (nationality != NULL && owner != nationality) {
1093 /* TRANS: Nationality of the people comprising a unit, if
1094 * different from owner. */
1095 astr_add_line(&str, _("%s people"),
1096 nation_adjective_for_player(nationality));
1097 } else {
1098 astr_add_line(&str, " ");
1102 } else if (count > 1) {
1103 int mil = 0, nonmil = 0;
1104 int types_count[U_LAST], i;
1105 struct unit_type *top[3];
1107 memset(types_count, 0, sizeof(types_count));
1108 unit_list_iterate(punits, punit) {
1109 if (unit_has_type_flag(punit, UTYF_CIVILIAN)) {
1110 nonmil++;
1111 } else {
1112 mil++;
1114 types_count[utype_index(unit_type_get(punit))]++;
1115 } unit_list_iterate_end;
1117 top[0] = top[1] = top[2] = NULL;
1118 unit_type_iterate(utype) {
1119 if (!top[2]
1120 || types_count[utype_index(top[2])] < types_count[utype_index(utype)]) {
1121 top[2] = utype;
1123 if (!top[1]
1124 || types_count[utype_index(top[1])] < types_count[utype_index(top[2])]) {
1125 top[2] = top[1];
1126 top[1] = utype;
1128 if (!top[0]
1129 || types_count[utype_index(top[0])] < types_count[utype_index(utype)]) {
1130 top[1] = top[0];
1131 top[0] = utype;
1135 } unit_type_iterate_end;
1137 for (i = 0; i < 2; i++) {
1138 if (top[i] && types_count[utype_index(top[i])] > 0) {
1139 if (utype_has_flag(top[i], UTYF_CIVILIAN)) {
1140 nonmil -= types_count[utype_index(top[i])];
1141 } else {
1142 mil -= types_count[utype_index(top[i])];
1144 astr_add_line(&str, "%d: %s",
1145 types_count[utype_index(top[i])],
1146 utype_name_translation(top[i]));
1147 } else {
1148 astr_add_line(&str, " ");
1152 if (top[2] && types_count[utype_index(top[2])] > 0
1153 && types_count[utype_index(top[2])] == nonmil + mil) {
1154 astr_add_line(&str, "%d: %s", types_count[utype_index(top[2])],
1155 utype_name_translation(top[2]));
1156 } else if (nonmil > 0 && mil > 0) {
1157 astr_add_line(&str, _("Others: %d civil; %d military"), nonmil, mil);
1158 } else if (nonmil > 0) {
1159 astr_add_line(&str, _("Others: %d civilian"), nonmil);
1160 } else if (mil > 0) {
1161 astr_add_line(&str, _("Others: %d military"), mil);
1162 } else {
1163 astr_add_line(&str, " ");
1166 if (game.info.citizen_nationality) {
1167 astr_add_line(&str, " ");
1169 } else {
1170 astr_add_line(&str, " ");
1171 astr_add_line(&str, " ");
1172 astr_add_line(&str, " ");
1174 if (game.info.citizen_nationality) {
1175 astr_add_line(&str, " ");
1179 /* Line 5/6. Debug text. */
1180 #ifdef DEBUG
1181 if (count == 1) {
1182 astr_add_line(&str, "(Unit ID %d)", unit_list_get(punits, 0)->id);
1183 } else {
1184 astr_add_line(&str, " ");
1186 #endif /* DEBUG */
1188 return astr_str(&str);
1191 /****************************************************************************
1192 Return text about upgrading these unit lists.
1194 Returns TRUE iff any units can be upgraded.
1195 ****************************************************************************/
1196 bool get_units_upgrade_info(char *buf, size_t bufsz,
1197 struct unit_list *punits)
1199 if (unit_list_size(punits) == 0) {
1200 fc_snprintf(buf, bufsz, _("No units to upgrade!"));
1201 return FALSE;
1202 } else if (unit_list_size(punits) == 1) {
1203 return (UU_OK == unit_upgrade_info(unit_list_front(punits), buf, bufsz));
1204 } else {
1205 int upgrade_cost = 0;
1206 int num_upgraded = 0;
1207 int min_upgrade_cost = FC_INFINITY;
1209 unit_list_iterate(punits, punit) {
1210 if (unit_owner(punit) == client_player()
1211 && UU_OK == unit_upgrade_test(punit, FALSE)) {
1212 struct unit_type *from_unittype = unit_type_get(punit);
1213 struct unit_type *to_unittype = can_upgrade_unittype(client.conn.playing,
1214 from_unittype);
1215 int cost = unit_upgrade_price(unit_owner(punit),
1216 from_unittype, to_unittype);
1218 num_upgraded++;
1219 upgrade_cost += cost;
1220 min_upgrade_cost = MIN(min_upgrade_cost, cost);
1222 } unit_list_iterate_end;
1223 if (num_upgraded == 0) {
1224 fc_snprintf(buf, bufsz, _("None of these units may be upgraded."));
1225 return FALSE;
1226 } else {
1227 /* This may trigger sometimes if you don't have enough money for
1228 * a full upgrade. If you have enough to upgrade at least one, it
1229 * will do it. */
1230 /* Construct prompt in several parts to allow separate pluralisation
1231 * by localizations */
1232 char tbuf[MAX_LEN_MSG], ubuf[MAX_LEN_MSG];
1233 fc_snprintf(tbuf, ARRAY_SIZE(tbuf), PL_("Treasury contains %d gold.",
1234 "Treasury contains %d gold.",
1235 client_player()->economic.gold),
1236 client_player()->economic.gold);
1237 /* TRANS: this whole string is a sentence fragment that is only ever
1238 * used by including it in another string (search comments for this
1239 * string to find it) */
1240 fc_snprintf(ubuf, ARRAY_SIZE(ubuf), PL_("Upgrade %d unit",
1241 "Upgrade %d units",
1242 num_upgraded),
1243 num_upgraded);
1244 /* TRANS: This is complicated. The first %s is a pre-pluralised
1245 * sentence fragment "Upgrade %d unit(s)"; the second is pre-pluralised
1246 * "Treasury contains %d gold." So the whole thing reads
1247 * "Upgrade 13 units for 1000 gold?\nTreasury contains 2000 gold." */
1248 fc_snprintf(buf, bufsz, PL_("%s for %d gold?\n%s",
1249 "%s for %d gold?\n%s", upgrade_cost),
1250 ubuf, upgrade_cost, tbuf);
1251 return TRUE;
1256 /****************************************************************************
1257 Return text about disbanding these units.
1259 Returns TRUE iff any units can be disbanded.
1260 ****************************************************************************/
1261 bool get_units_disband_info(char *buf, size_t bufsz,
1262 struct unit_list *punits)
1264 if (unit_list_size(punits) == 0) {
1265 fc_snprintf(buf, bufsz, _("No units to disband!"));
1266 return FALSE;
1267 } else if (unit_list_size(punits) == 1) {
1268 if (unit_has_type_flag(unit_list_front(punits), UTYF_UNDISBANDABLE)) {
1269 fc_snprintf(buf, bufsz, _("%s refuses to disband!"),
1270 unit_name_translation(unit_list_front(punits)));
1271 return FALSE;
1272 } else {
1273 /* TRANS: %s is a unit type */
1274 fc_snprintf(buf, bufsz, _("Disband %s?"),
1275 unit_name_translation(unit_list_front(punits)));
1276 return TRUE;
1278 } else {
1279 int count = 0;
1280 unit_list_iterate(punits, punit) {
1281 if (!unit_has_type_flag(punit, UTYF_UNDISBANDABLE)) {
1282 count++;
1284 } unit_list_iterate_end;
1285 if (count == 0) {
1286 fc_snprintf(buf, bufsz, _("None of these units may be disbanded."));
1287 return FALSE;
1288 } else {
1289 /* TRANS: %d is never 0 or 1 */
1290 fc_snprintf(buf, bufsz, PL_("Disband %d unit?",
1291 "Disband %d units?", count), count);
1292 return TRUE;
1297 /****************************************************************************
1298 Get a tooltip text for the info panel research indicator. See
1299 client_research_sprite().
1300 ****************************************************************************/
1301 const char *get_bulb_tooltip(void)
1303 static struct astring str = ASTRING_INIT;
1305 astr_clear(&str);
1307 astr_add_line(&str, _("Shows your progress in "
1308 "researching the current technology."));
1310 if (NULL != client.conn.playing) {
1311 struct research *research = research_get(client_player());
1313 if (research->researching == A_UNSET) {
1314 astr_add_line(&str, _("no research target."));
1315 } else {
1316 int turns = 0;
1317 int perturn = get_bulbs_per_turn(NULL, NULL, NULL);
1318 int done = research->bulbs_researched;
1319 int total = research->client.researching_cost;
1320 struct astring buf1 = ASTRING_INIT, buf2 = ASTRING_INIT;
1322 if (perturn > 0) {
1323 turns = MAX(1, ceil((double) (total - done) / perturn));
1324 } else if (perturn < 0 ) {
1325 turns = ceil((double) done / -perturn);
1328 if (turns == 0) {
1329 astr_set(&buf1, _("No progress"));
1330 } else {
1331 astr_set(&buf1, PL_("%d turn", "%d turns", turns), turns);
1334 /* TRANS: <perturn> bulbs/turn */
1335 astr_set(&buf2, PL_("%d bulb/turn", "%d bulbs/turn", perturn), perturn);
1337 /* TRANS: <tech>: <amount>/<total bulbs> */
1338 astr_add_line(&str, _("%s: %d/%d (%s, %s)."),
1339 research_advance_name_translation(research,
1340 research->researching),
1341 research->bulbs_researched,
1342 research->client.researching_cost,
1343 astr_str(&buf1), astr_str(&buf2));
1345 astr_free(&buf1);
1346 astr_free(&buf2);
1349 return astr_str(&str);
1352 /****************************************************************************
1353 Get a tooltip text for the info panel global warning indicator. See also
1354 client_warming_sprite().
1355 ****************************************************************************/
1356 const char *get_global_warming_tooltip(void)
1358 static struct astring str = ASTRING_INIT;
1360 astr_clear(&str);
1362 if (!game.info.global_warming) {
1363 astr_add_line(&str, _("Global warming deactivated."));
1364 } else {
1365 int chance, rate;
1366 global_warming_scaled(&chance, &rate, 100);
1367 astr_add_line(&str, _("Shows the progress of global warming:"));
1368 astr_add_line(&str, _("Pollution rate: %d%%"), rate);
1369 astr_add_line(&str, _("Chance of catastrophic warming each turn: %d%%"),
1370 chance);
1373 return astr_str(&str);
1376 /****************************************************************************
1377 Get a tooltip text for the info panel nuclear winter indicator. See also
1378 client_cooling_sprite().
1379 ****************************************************************************/
1380 const char *get_nuclear_winter_tooltip(void)
1382 static struct astring str = ASTRING_INIT;
1384 astr_clear(&str);
1386 if (!game.info.nuclear_winter) {
1387 astr_add_line(&str, _("Nuclear winter deactivated."));
1388 } else {
1389 int chance, rate;
1390 nuclear_winter_scaled(&chance, &rate, 100);
1391 astr_add_line(&str, _("Shows the progress of nuclear winter:"));
1392 astr_add_line(&str, _("Fallout rate: %d%%"), rate);
1393 astr_add_line(&str, _("Chance of catastrophic winter each turn: %d%%"),
1394 chance);
1397 return astr_str(&str);
1400 /****************************************************************************
1401 Get a tooltip text for the info panel government indicator. See also
1402 government_by_number(...)->sprite.
1403 ****************************************************************************/
1404 const char *get_government_tooltip(void)
1406 static struct astring str = ASTRING_INIT;
1408 astr_clear(&str);
1410 astr_add_line(&str, _("Shows your current government:"));
1412 if (NULL != client.conn.playing) {
1413 astr_add_line(&str, "%s",
1414 government_name_for_player(client.conn.playing));
1416 return astr_str(&str);
1419 /****************************************************************************
1420 Returns a description of the given spaceship. If there is no spaceship
1421 (pship is NULL) then text with dummy values is returned.
1422 ****************************************************************************/
1423 const char *get_spaceship_descr(struct player_spaceship *pship)
1425 struct player_spaceship ship;
1426 static struct astring str = ASTRING_INIT;
1428 astr_clear(&str);
1430 if (!pship) {
1431 pship = &ship;
1432 memset(&ship, 0, sizeof(ship));
1435 /* TRANS: spaceship text; should have constant width. */
1436 astr_add_line(&str, _("Population: %5d"), pship->population);
1438 /* TRANS: spaceship text; should have constant width. */
1439 astr_add_line(&str, _("Support: %5d %%"),
1440 (int) (pship->support_rate * 100.0));
1442 /* TRANS: spaceship text; should have constant width. */
1443 astr_add_line(&str, _("Energy: %5d %%"),
1444 (int) (pship->energy_rate * 100.0));
1446 /* TRANS: spaceship text; should have constant width. */
1447 astr_add_line(&str, PL_("Mass: %5d ton",
1448 "Mass: %5d tons",
1449 pship->mass), pship->mass);
1451 if (pship->propulsion > 0) {
1452 /* TRANS: spaceship text; should have constant width. */
1453 astr_add_line(&str, _("Travel time: %5.1f years"),
1454 (float) (0.1 * ((int) (pship->travel_time * 10.0))));
1455 } else {
1456 /* TRANS: spaceship text; should have constant width. */
1457 astr_add_line(&str, "%s", _("Travel time: N/A "));
1460 /* TRANS: spaceship text; should have constant width. */
1461 astr_add_line(&str, _("Success prob.: %5d %%"),
1462 (int) (pship->success_rate * 100.0));
1464 /* TRANS: spaceship text; should have constant width. */
1465 astr_add_line(&str, _("Year of arrival: %8s"),
1466 (pship->state == SSHIP_LAUNCHED)
1467 ? textyear((int) (pship->launch_year +
1468 (int) pship->travel_time))
1469 : "- ");
1471 return astr_str(&str);
1474 /****************************************************************************
1475 Get the text showing the timeout. This is generally disaplyed on the info
1476 panel.
1477 ****************************************************************************/
1478 const char *get_timeout_label_text(void)
1480 static struct astring str = ASTRING_INIT;
1482 astr_clear(&str);
1484 if (is_waiting_turn_change() && game.tinfo.last_turn_change_time >= 1.5) {
1485 double wt = get_seconds_to_new_turn();
1487 if (wt < 0.01) {
1488 astr_add(&str, "%s", Q_("?timeout:wait"));
1489 } else {
1490 astr_add(&str, "%s: %s", Q_("?timeout:eta"), format_duration(wt));
1492 } else {
1493 if (current_turn_timeout() <= 0) {
1494 astr_add(&str, "%s", Q_("?timeout:off"));
1495 } else {
1496 astr_add(&str, "%s", format_duration(get_seconds_to_turndone()));
1500 return astr_str(&str);
1503 /****************************************************************************
1504 Format a duration, in seconds, so it comes up in minutes or hours if
1505 that would be more meaningful.
1507 (7 characters, maximum. Enough for, e.g., "99h 59m".)
1508 ****************************************************************************/
1509 const char *format_duration(int duration)
1511 static struct astring str = ASTRING_INIT;
1513 astr_clear(&str);
1515 if (duration < 0) {
1516 duration = 0;
1518 if (duration < 60) {
1519 astr_add(&str, Q_("?seconds:%02ds"), duration);
1520 } else if (duration < 3600) { /* < 60 minutes */
1521 astr_add(&str, Q_("?mins/secs:%02dm %02ds"), duration / 60, duration % 60);
1522 } else if (duration < 360000) { /* < 100 hours */
1523 astr_add(&str, Q_("?hrs/mns:%02dh %02dm"), duration / 3600, (duration / 60) % 60);
1524 } else if (duration < 8640000) { /* < 100 days */
1525 astr_add(&str, Q_("?dys/hrs:%02dd %02dh"), duration / 86400,
1526 (duration / 3600) % 24);
1527 } else {
1528 astr_add(&str, "%s", Q_("?duration:overflow"));
1531 return astr_str(&str);
1534 /****************************************************************************
1535 Return text giving the ping time for the player. This is generally used
1536 used in the playerdlg. This should only be used in playerdlg_common.c.
1537 ****************************************************************************/
1538 const char *get_ping_time_text(const struct player *pplayer)
1540 static struct astring str = ASTRING_INIT;
1542 astr_clear(&str);
1544 conn_list_iterate(pplayer->connections, pconn) {
1545 if (!pconn->observer
1546 /* Certainly not needed, but safer. */
1547 && 0 == strcmp(pconn->username, pplayer->username)) {
1548 if (pconn->ping_time != -1) {
1549 double ping_time_in_ms = 1000 * pconn->ping_time;
1551 astr_add(&str, _("%6d.%02d ms"), (int) ping_time_in_ms,
1552 ((int) (ping_time_in_ms * 100.0)) % 100);
1554 break;
1556 } conn_list_iterate_end;
1558 return astr_str(&str);
1561 /****************************************************************************
1562 Return text giving the score of the player. This should only be used
1563 in playerdlg_common.c.
1564 ****************************************************************************/
1565 const char *get_score_text(const struct player *pplayer)
1567 static struct astring str = ASTRING_INIT;
1569 astr_clear(&str);
1571 if (pplayer->score.game > 0
1572 || NULL == client.conn.playing
1573 || pplayer == client.conn.playing) {
1574 astr_add(&str, "%d", pplayer->score.game);
1575 } else {
1576 astr_add(&str, "?");
1579 return astr_str(&str);
1582 /****************************************************************************
1583 Get the title for a "report". This may include the city, economy,
1584 military, trade, player, etc., reports. Some clients may generate the
1585 text themselves to get a better GUI layout.
1586 ****************************************************************************/
1587 const char *get_report_title(const char *report_name)
1589 static struct astring str = ASTRING_INIT;
1590 const struct player *pplayer = client_player();
1592 astr_clear(&str);
1594 astr_add_line(&str, "%s", report_name);
1596 if (pplayer != NULL) {
1597 char buf[4 * MAX_LEN_NAME];
1599 /* TRANS: <nation adjective> <government name>.
1600 * E.g. "Polish Republic". */
1601 astr_add_line(&str, Q_("?nationgovernment:%s %s"),
1602 nation_adjective_for_player(pplayer),
1603 government_name_for_player(pplayer));
1605 /* TRANS: Just appending 2 strings, using the correct localized
1606 * syntax. */
1607 astr_add_line(&str, _("%s - %s"),
1608 ruler_title_for_player(pplayer, buf, sizeof(buf)),
1609 calendar_text());
1610 } else {
1611 /* TRANS: "Observer - 1985 AD" */
1612 astr_add_line(&str, _("Observer - %s"),
1613 calendar_text());
1615 return astr_str(&str);
1618 /****************************************************************************
1619 Describing buildings that affect happiness.
1620 ****************************************************************************/
1621 const char *text_happiness_buildings(const struct city *pcity)
1623 struct effect_list *plist = effect_list_new();
1624 static struct astring str = ASTRING_INIT;
1626 get_city_bonus_effects(plist, pcity, NULL, EFT_MAKE_CONTENT);
1627 if (0 < effect_list_size(plist)) {
1628 struct astring effects = ASTRING_INIT;
1630 get_effect_list_req_text(plist, &effects);
1631 astr_set(&str, _("Buildings: %s."), astr_str(&effects));
1632 astr_free(&effects);
1633 } else {
1634 astr_set(&str, _("Buildings: None."));
1636 effect_list_destroy(plist);
1638 /* Add line breaks after 80 characters. */
1639 astr_break_lines(&str, 80);
1641 return astr_str(&str);
1644 /****************************************************************************
1645 Describing nationality effects that affect happiness.
1646 ****************************************************************************/
1647 const char *text_happiness_nationality(const struct city *pcity)
1649 static struct astring str = ASTRING_INIT;
1650 int enemies = 0;
1652 astr_clear(&str);
1654 astr_add_line(&str, _("Nationality: "));
1656 if (game.info.citizen_nationality) {
1657 if (get_city_bonus(pcity, EFT_ENEMY_CITIZEN_UNHAPPY_PCT) > 0) {
1658 struct player *owner = city_owner(pcity);
1660 citizens_foreign_iterate(pcity, pslot, nationality) {
1661 if (pplayers_at_war(owner, player_slot_get_player(pslot))) {
1662 enemies += nationality;
1664 } citizens_foreign_iterate_end;
1666 if (enemies > 0) {
1667 astr_add(&str, PL_("%d enemy nationalist", "%d enemy nationalists", enemies),
1668 enemies);
1672 if (enemies == 0) {
1673 astr_add(&str, _("None."));
1675 } else {
1676 astr_add(&str, _("Disabled."));
1679 return astr_str(&str);
1682 /****************************************************************************
1683 Describing wonders that affect happiness.
1684 ****************************************************************************/
1685 const char *text_happiness_wonders(const struct city *pcity)
1687 struct effect_list *plist = effect_list_new();
1688 static struct astring str = ASTRING_INIT;
1690 get_city_bonus_effects(plist, pcity, NULL, EFT_MAKE_HAPPY);
1691 get_city_bonus_effects(plist, pcity, NULL, EFT_FORCE_CONTENT);
1692 get_city_bonus_effects(plist, pcity, NULL, EFT_NO_UNHAPPY);
1693 if (0 < effect_list_size(plist)) {
1694 struct astring effects = ASTRING_INIT;
1696 get_effect_list_req_text(plist, &effects);
1697 astr_set(&str, _("Wonders: %s."), astr_str(&effects));
1698 astr_free(&effects);
1699 } else {
1700 astr_set(&str, _("Wonders: None."));
1703 /* Add line breaks after 80 characters. */
1704 astr_break_lines(&str, 80);
1706 return astr_str(&str);
1709 /****************************************************************************
1710 Describing city factors that affect happiness.
1711 ****************************************************************************/
1712 const char *text_happiness_cities(const struct city *pcity)
1714 struct player *pplayer = city_owner(pcity);
1715 int cities = city_list_size(pplayer->cities);
1716 int content = get_player_bonus(pplayer, EFT_CITY_UNHAPPY_SIZE);
1717 int basis = get_player_bonus(pplayer, EFT_EMPIRE_SIZE_BASE);
1718 int step = get_player_bonus(pplayer, EFT_EMPIRE_SIZE_STEP);
1719 static struct astring str = ASTRING_INIT;
1721 astr_clear(&str);
1723 if (basis+step <= 0) {
1724 /* Special case where penalty is disabled; see
1725 * player_content_citizens(). */
1726 astr_add_line(&str,
1727 PL_("Cities: %d total, but no penalty for empire size.",
1728 "Cities: %d total, but no penalty for empire size.",
1729 cities),
1730 cities);
1731 astr_add_line(&str,
1732 /* TRANS: %d is number of citizens */
1733 PL_("%d content per city.",
1734 "%d content per city.", content),
1735 content);
1736 } else {
1737 /* Can have up to and including 'basis' cities without penalty */
1738 int excess = MAX(cities - basis, 0);
1739 int penalty;
1740 int unhappy, angry;
1741 int last, next;
1743 if (excess > 0) {
1744 if (step > 0) {
1745 penalty = 1 + (excess - 1) / step;
1746 } else {
1747 penalty = 1;
1749 } else {
1750 penalty = 0;
1753 unhappy = MIN(penalty, content);
1754 angry = game.info.angrycitizen ? MAX(penalty-content, 0) : 0;
1755 if (penalty >= 1) {
1756 /* 'last' is when last actual malcontent appeared, will saturate
1757 * if no angry citizens */
1758 last = basis + (unhappy+angry-1)*step;
1759 if (!game.info.angrycitizen && unhappy == content) {
1760 /* Maxed out unhappy citizens, so no more penalties */
1761 next = 0;
1762 } else {
1763 /* Angry citizens can continue appearing indefinitely */
1764 next = last + step;
1766 } else {
1767 last = 0;
1768 next = basis;
1771 astr_add_line(&str,
1772 /* TRANS: sentence fragment, will have text appended */
1773 PL_("Cities: %d total:",
1774 "Cities: %d total:", cities),
1775 cities);
1776 if (excess > 0) {
1777 astr_add(&str,
1778 /* TRANS: appended to "Cities: %d total:"; preserve leading
1779 * space. Pluralized in "nearest threshold of %d cities". */
1780 PL_(" %d over nearest threshold of %d city.",
1781 " %d over nearest threshold of %d cities.", last),
1782 cities - last, last);
1783 astr_add_line(&str,
1784 /* TRANS: Number of content [citizen(s)] ... */
1785 PL_("%d content before penalty.",
1786 "%d content before penalty.", content),
1787 content);
1788 astr_add_line(&str,
1789 PL_("%d additional unhappy citizen.",
1790 "%d additional unhappy citizens.", unhappy),
1791 unhappy);
1792 if (angry > 0) {
1793 astr_add_line(&str,
1794 PL_("%d angry citizen.",
1795 "%d angry citizens.", angry),
1796 angry);
1798 } else {
1799 astr_add(&str,
1800 /* TRANS: appended to "Cities: %d total:"; preserve leading
1801 * space. */
1802 PL_(" not more than %d, so no empire size penalty.",
1803 " not more than %d, so no empire size penalty.", next),
1804 next);
1805 astr_add_line(&str,
1806 /* TRANS: %d is number of citizens */
1807 PL_("%d content per city.",
1808 "%d content per city.", content),
1809 content);
1811 if (next >= cities && penalty < content) {
1812 astr_add_line(&str,
1813 PL_("With %d more city, another citizen will become "
1814 "unhappy.",
1815 "With %d more cities, another citizen will become "
1816 "unhappy.",
1817 next + 1 - cities),
1818 next + 1 - cities);
1819 } else if (next >= cities) {
1820 /* We maxed out the number of unhappy citizens, but they can get
1821 * angry instead. */
1822 fc_assert(game.info.angrycitizen);
1823 astr_add_line(&str,
1824 PL_("With %d more city, another citizen will become "
1825 "angry.",
1826 "With %d more cities, another citizen will become "
1827 "angry.",
1828 next + 1 - cities),
1829 next + 1 - cities);
1830 } else {
1831 /* Either no Empire_Size_Step, or we maxed out on unhappy citizens
1832 * and ruleset doesn't allow angry ones. */
1833 astr_add_line(&str,
1834 _("More cities will not cause more unhappy citizens."));
1838 return astr_str(&str);
1841 /****************************************************************************
1842 Describing units that affect happiness.
1843 ****************************************************************************/
1844 const char *text_happiness_units(const struct city *pcity)
1846 int mlmax = get_city_bonus(pcity, EFT_MARTIAL_LAW_MAX);
1847 int uhcfac = get_city_bonus(pcity, EFT_UNHAPPY_FACTOR);
1848 static struct astring str = ASTRING_INIT;
1850 astr_clear(&str);
1852 if (mlmax > 0) {
1853 int mleach = get_city_bonus(pcity, EFT_MARTIAL_LAW_EACH);
1854 if (mlmax == 100) {
1855 astr_add_line(&str, "%s", _("Unlimited martial law in effect."));
1856 } else {
1857 astr_add_line(&str, PL_("%d military unit may impose martial law.",
1858 "Up to %d military units may impose martial "
1859 "law.", mlmax), mlmax);
1861 astr_add_line(&str, PL_("Each military unit makes %d "
1862 "unhappy citizen content.",
1863 "Each military unit makes %d "
1864 "unhappy citizens content.",
1865 mleach), mleach);
1866 } else if (uhcfac > 0) {
1867 astr_add_line(&str,
1868 _("Military units in the field may cause unhappiness. "));
1869 } else {
1870 astr_add_line(&str,
1871 _("Military units have no happiness effect. "));
1873 return astr_str(&str);
1876 /****************************************************************************
1877 Describing luxuries that affect happiness.
1878 ****************************************************************************/
1879 const char *text_happiness_luxuries(const struct city *pcity)
1881 static struct astring str = ASTRING_INIT;
1883 astr_clear(&str);
1885 astr_add_line(&str,
1886 _("Luxury: %d total."),
1887 pcity->prod[O_LUXURY]);
1888 return astr_str(&str);