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)
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 ***********************************************************************/
15 #include <fc_config.h>
20 #include <math.h> /* ceil */
24 #include "bitvector.h"
32 #include "clientutils.h"
34 #include "fc_types.h" /* LINE_BREAK */
36 #include "government.h"
39 #include "traderoutes.h"
43 #include "client_main.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
;
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
,
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
);
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) {
102 if (pplayer
->ai_controlled
) {
103 /* TRANS: "AI <player name>" */
104 fc_snprintf(buf
, buflen
, _("AI %s"), pplayer
->name
);
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) {
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
));
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
;
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)"),
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
));
176 extra_type_iterate(pextra
) {
177 if (pextra
->category
== ECAT_BONUS
&& tile_has_visible_extra(ptile
, pextra
)) {
179 astr_add(&str
, ",%s", extra_name_translation(pextra
));
181 astr_add_line(&str
, "%s", extra_name_translation(pextra
));
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
,
201 if (ds
->type
== DS_CEASEFIRE
) {
202 int turns
= ds
->turns_left
;
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)",
210 username
, nation
, turns
);
211 } else if (ds
->type
== DS_ARMISTICE
) {
212 int turns
= ds
->turns_left
;
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)",
220 username
, nation
, turns
);
224 /* TRANS: "Territory of <username>
225 * (<nation + team> | <diplomatic state>)" */
226 astr_add_line(&str
, _("Territory of %s (%s | %s)"),
228 diplo_nation_plural_adjectives
[type
]);
231 astr_add_line(&str
, _("Unclaimed territory"));
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
);
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)",
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)",
268 city_name_get(pcity
), username
, nation
, turns
);
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
);
281 astr_add(&str
, PL_(" | Occupied with %d unit.",
282 " | Occupied with %d units.", count
), count
);
284 astr_add(&str
, _(" | Not occupied."));
287 if (pcity
->client
.occupied
) {
288 astr_add(&str
, _(" | Occupied."));
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
));
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
)) {
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
)));
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(),
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)",
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)",
386 utype_name_translation(ptype
),
387 username
, nation
, turns
);
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
;
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;
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
;
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
);
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
));
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
;
466 /* just to be sure */
471 astr_add(&str
, (sq_dist
>= FAR_CITY_SQUARE_DIST
)
472 /* TRANS: on own line immediately following \n, ... <city> */
475 /* TRANS: on own line immediately following \n, ... <city> */
478 /* TRANS: on own line immediately following \n, ... <city> */
482 ? city_name_get(pcity
)
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
494 ****************************************************************************/
495 const char *unit_description(struct unit
*punit
)
498 struct player
*owner
= unit_owner(punit
);
499 struct player
*nationality
= unit_nationality(punit
);
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();
509 astr_add(&str
, "%s", utype_name_translation(ptype
));
512 const char *veteran_name
=
513 utype_veteran_name_translation(ptype
, punit
->veteran
);
515 astr_add(&str
, " (%s)", veteran_name
);
519 if (pplayer
== owner
) {
520 unit_upkeep_astr(punit
, &str
);
522 astr_add(&str
, "\n");
524 unit_activity_astr(punit
, &str
);
527 /* TRANS: on own line immediately following \n, ... <city> */
528 astr_add_line(&str
, _("from %s"), city_name_get(pcity
));
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
));
539 astr_add(&str
, "\n");
543 astr_add_line(&str
, "%s",
544 get_nearest_city_text(pcity_near
, pcity_near_dist
));
546 astr_add_line(&str
, "Unit ID: %d", punit
->id
);
549 return astr_str(&str
);
552 /****************************************************************************
553 Describe the airlift capacity of a city for the given units (from their
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
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
);
581 case AR_WRONG_UNITTYPE
:
584 case AR_BAD_SRC_CITY
:
585 case AR_BAD_DST_CITY
:
586 /* No chance of an airlift. */
587 this = AL_IMPOSSIBLE
;
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
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). */
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
);
607 /* City known not to be airlift-capable. */
608 this = AL_IMPOSSIBLE
;
611 && (game
.info
.airlifting_style
& AIRLIFTING_UNLIMITED_SRC
)) {
612 /* Unlimited capacity. */
615 /* Limited capacity (possibly zero right now). */
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
);
627 /* Unknown capacity. */
634 /* Now take the most optimistic view. */
635 best
= MAX(best
, this);
636 } unit_list_iterate_end
;
645 astr_set(&str
, "%d/%d", cur
, max
);
648 astr_set(&str
, _("Yes"));
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;
664 if (!client_has_player()) {
667 presearch
= research_get(client_player());
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
;
677 theirs
-= pplayer
->client
.tech_upkeep
;
679 } research_players_iterate_end
;
682 theirs
+= presearch
->client
.total_bulbs_prod
- ours
;
684 ours
-= client_player()->client
.tech_upkeep
;
695 return ours
+ theirs
;
698 /****************************************************************************
699 Returns the text to display in the science dialog.
700 ****************************************************************************/
701 const char *science_dialog_text(void)
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
;
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
718 return _("Progress: no research");
721 if (A_UNSET
== research
->researching
) {
722 astr_add(&str
, _("Progress: no research"));
724 int done
= research
->bulbs_researched
;
725 int total
= research
->client
.researching_cost
;
728 int turns
= MAX(1, ceil((double)total
) / perturn
);
730 astr_add(&str
, PL_("Progress: %d turn/advance",
731 "Progress: %d turns/advance",
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",
742 astr_add(&str
, _("Progress: none"));
745 astr_set(&ourbuf
, PL_("%d bulb/turn", "%d bulbs/turn", ours
), ours
);
747 /* Techpool version */
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
);
753 astr_clear(&theirbuf
);
755 astr_add(&str
, " (%s%s)", astr_str(&ourbuf
), astr_str(&theirbuf
));
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
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
;
788 if (research
->researching
== A_UNSET
) {
789 astr_add(&str
, _("%d/- (never)"), research
->bulbs_researched
);
794 int total
= research
->client
.researching_cost
;
795 int done
= research
->bulbs_researched
;
796 int perturn
= get_bulbs_per_turn(NULL
, NULL
, NULL
);
799 int turns
= ceil( (double)(total
- done
) / perturn
);
801 astr_add(&str
, PL_("%d/%d (%d turn)", "%d/%d (%d turns)", 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
);
811 astr_add(&str
, _("%d/%d (never)"), done
, total
);
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
);
831 int perturn
= get_bulbs_per_turn(NULL
, NULL
, NULL
);
832 static struct astring str
= ASTRING_INIT
;
833 struct astring buf1
= ASTRING_INIT
,
843 if (research_goal_tech_req(research
, goal
, research
->researching
)
844 || research
->researching
== goal
) {
845 bulbs_needed
-= research
->bulbs_researched
;
849 PL_("%d step", "%d steps", steps
), steps
);
851 PL_("%d bulb", "%d bulbs", bulbs_needed
), bulbs_needed
);
853 turns
= (bulbs_needed
+ perturn
- 1) / perturn
;
855 PL_("%d turn", "%d turns", turns
), turns
);
857 astr_set(&buf3
, _("never"));
859 astr_add_line(&str
, "(%s - %s - %s)",
860 astr_str(&buf1
), astr_str(&buf2
), astr_str(&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
;
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"));
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"));
907 astr_add_line(&str
, _("Moving: %s"),
908 team_name_translation(team_by_number(game
.info
.phase
)));
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
923 ****************************************************************************/
924 const char *get_info_label_text_popup(void)
926 static struct astring str
= ASTRING_INIT
;
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
,
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
) {
969 global_warming_scaled(&chance
, &rate
, 100);
970 astr_add_line(&str
, _("Global warming chance: %d%% (%+d%%/turn)"),
973 astr_add_line(&str
, _("Global warming deactivated."));
976 if (game
.info
.nuclear_winter
) {
978 nuclear_winter_scaled(&chance
, &rate
, 100);
979 astr_add_line(&str
, _("Nuclear winter chance: %d%% (%+d%%/turn)"),
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
;
1005 int count
= unit_list_size(punits
);
1008 astr_add(&str
, "%s", unit_name_translation(unit_list_get(punits
, 0)));
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
;
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
1040 /* Line 1. Goto or activity text. */
1041 if (count
> 0 && hover_state
!= HOVER_NONE
) {
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
);
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",
1061 astr_add_line(&str
, _("No units selected."));
1064 /* Lines 2, 3, 4, and possible 5 vary. */
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
,
1071 astr_add_line(&str
, "%s", tile_get_info_text(unit_tile(punit
), TRUE
,
1074 const char *infratext
= get_infrastructure_text(unit_tile(punit
)->extras
);
1076 if (*infratext
!= '\0') {
1077 astr_add_line(&str
, "%s", infratext
);
1079 astr_add_line(&str
, " ");
1083 astr_add_line(&str
, "%s", city_name_get(pcity
));
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
));
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
)) {
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
) {
1120 || types_count
[utype_index(top
[2])] < types_count
[utype_index(utype
)]) {
1124 || types_count
[utype_index(top
[1])] < types_count
[utype_index(top
[2])]) {
1129 || types_count
[utype_index(top
[0])] < types_count
[utype_index(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
])];
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
]));
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
);
1163 astr_add_line(&str
, " ");
1166 if (game
.info
.citizen_nationality
) {
1167 astr_add_line(&str
, " ");
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. */
1182 astr_add_line(&str
, "(Unit ID %d)", unit_list_get(punits
, 0)->id
);
1184 astr_add_line(&str
, " ");
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!"));
1202 } else if (unit_list_size(punits
) == 1) {
1203 return (UU_OK
== unit_upgrade_info(unit_list_front(punits
), buf
, bufsz
));
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
,
1215 int cost
= unit_upgrade_price(unit_owner(punit
),
1216 from_unittype
, to_unittype
);
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."));
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
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",
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
);
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!"));
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
)));
1273 /* TRANS: %s is a unit type */
1274 fc_snprintf(buf
, bufsz
, _("Disband %s?"),
1275 unit_name_translation(unit_list_front(punits
)));
1280 unit_list_iterate(punits
, punit
) {
1281 if (!unit_has_type_flag(punit
, UTYF_UNDISBANDABLE
)) {
1284 } unit_list_iterate_end
;
1286 fc_snprintf(buf
, bufsz
, _("None of these units may be disbanded."));
1289 /* TRANS: %d is never 0 or 1 */
1290 fc_snprintf(buf
, bufsz
, PL_("Disband %d unit?",
1291 "Disband %d units?", count
), count
);
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
;
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."));
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
;
1323 turns
= MAX(1, ceil((double) (total
- done
) / perturn
));
1324 } else if (perturn
< 0 ) {
1325 turns
= ceil((double) done
/ -perturn
);
1329 astr_set(&buf1
, _("No progress"));
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
));
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
;
1362 if (!game
.info
.global_warming
) {
1363 astr_add_line(&str
, _("Global warming deactivated."));
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%%"),
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
;
1386 if (!game
.info
.nuclear_winter
) {
1387 astr_add_line(&str
, _("Nuclear winter deactivated."));
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%%"),
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
;
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
;
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",
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))));
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
))
1471 return astr_str(&str
);
1474 /****************************************************************************
1475 Get the text showing the timeout. This is generally disaplyed on the info
1477 ****************************************************************************/
1478 const char *get_timeout_label_text(void)
1480 static struct astring str
= ASTRING_INIT
;
1484 if (is_waiting_turn_change() && game
.tinfo
.last_turn_change_time
>= 1.5) {
1485 double wt
= get_seconds_to_new_turn();
1488 astr_add(&str
, "%s", Q_("?timeout:wait"));
1490 astr_add(&str
, "%s: %s", Q_("?timeout:eta"), format_duration(wt
));
1493 if (current_turn_timeout() <= 0) {
1494 astr_add(&str
, "%s", Q_("?timeout:off"));
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
;
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);
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
;
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);
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
;
1571 if (pplayer
->score
.game
> 0
1572 || NULL
== client
.conn
.playing
1573 || pplayer
== client
.conn
.playing
) {
1574 astr_add(&str
, "%d", pplayer
->score
.game
);
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();
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
1607 astr_add_line(&str
, _("%s - %s"),
1608 ruler_title_for_player(pplayer
, buf
, sizeof(buf
)),
1611 /* TRANS: "Observer - 1985 AD" */
1612 astr_add_line(&str
, _("Observer - %s"),
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
);
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
;
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
;
1667 astr_add(&str
, PL_("%d enemy nationalist", "%d enemy nationalists", enemies
),
1673 astr_add(&str
, _("None."));
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
);
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
;
1723 if (basis
+step
<= 0) {
1724 /* Special case where penalty is disabled; see
1725 * player_content_citizens(). */
1727 PL_("Cities: %d total, but no penalty for empire size.",
1728 "Cities: %d total, but no penalty for empire size.",
1732 /* TRANS: %d is number of citizens */
1733 PL_("%d content per city.",
1734 "%d content per city.", content
),
1737 /* Can have up to and including 'basis' cities without penalty */
1738 int excess
= MAX(cities
- basis
, 0);
1745 penalty
= 1 + (excess
- 1) / step
;
1753 unhappy
= MIN(penalty
, content
);
1754 angry
= game
.info
.angrycitizen
? MAX(penalty
-content
, 0) : 0;
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 */
1763 /* Angry citizens can continue appearing indefinitely */
1772 /* TRANS: sentence fragment, will have text appended */
1773 PL_("Cities: %d total:",
1774 "Cities: %d total:", cities
),
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
);
1784 /* TRANS: Number of content [citizen(s)] ... */
1785 PL_("%d content before penalty.",
1786 "%d content before penalty.", content
),
1789 PL_("%d additional unhappy citizen.",
1790 "%d additional unhappy citizens.", unhappy
),
1794 PL_("%d angry citizen.",
1795 "%d angry citizens.", angry
),
1800 /* TRANS: appended to "Cities: %d total:"; preserve leading
1802 PL_(" not more than %d, so no empire size penalty.",
1803 " not more than %d, so no empire size penalty.", next
),
1806 /* TRANS: %d is number of citizens */
1807 PL_("%d content per city.",
1808 "%d content per city.", content
),
1811 if (next
>= cities
&& penalty
< content
) {
1813 PL_("With %d more city, another citizen will become "
1815 "With %d more cities, another citizen will become "
1819 } else if (next
>= cities
) {
1820 /* We maxed out the number of unhappy citizens, but they can get
1822 fc_assert(game
.info
.angrycitizen
);
1824 PL_("With %d more city, another citizen will become "
1826 "With %d more cities, another citizen will become "
1831 /* Either no Empire_Size_Step, or we maxed out on unhappy citizens
1832 * and ruleset doesn't allow angry ones. */
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
;
1853 int mleach
= get_city_bonus(pcity
, EFT_MARTIAL_LAW_EACH
);
1855 astr_add_line(&str
, "%s", _("Unlimited martial law in effect."));
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.",
1866 } else if (uhcfac
> 0) {
1868 _("Military units in the field may cause unhappiness. "));
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
;
1886 _("Luxury: %d total."),
1887 pcity
->prod
[O_LUXURY
]);
1888 return astr_str(&str
);