Freeciv-3.1
Loading...
Searching...
No Matches
helpdlg.c
Go to the documentation of this file.
1/***********************************************************************
2 Freeciv - Copyright (C) 1996 - A Kjeldberg, L Gregersen, P Unold
3 This program is free software; you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 2, or (at your option)
6 any later version.
7
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***********************************************************************/
13
14#ifdef HAVE_CONFIG_H
15#include <fc_config.h>
16#endif
17
18#include <stdio.h>
19#include <stdlib.h>
20#include <string.h>
21#include <math.h> /* sqrt */
22
23#include <gtk/gtk.h>
24
25/* utility */
26#include "fcintl.h"
27#include "mem.h"
28#include "shared.h"
29#include "support.h"
30
31/* common */
32#include "city.h"
33#include "game.h"
34#include "government.h"
35#include "movement.h"
36#include "specialist.h"
37#include "tech.h"
38#include "unit.h"
39#include "map.h"
40#include "research.h"
41#include "version.h"
42
43/* client */
44#include "client_main.h"
45#include "climisc.h"
46#include "helpdata.h"
47#include "options.h"
48#include "tilespec.h"
49
50/* client/gui-gtk-3.22 */
51#include "colors.h"
52#include "graphics.h"
53#include "gui_main.h"
54#include "gui_stuff.h"
55
56#include "helpdlg.h"
57
58#define TECH_TREE_DEPTH 20
59
60/*
61 * Globals.
62 */
63static GtkWidget *help_dialog_shell;
64static GtkWidget *help_view_sw;
65
66static GtkWidget *help_view;
67
68static GtkWidget *help_frame;
69static int help_frame_width = 520, help_frame_height = 350;
70static GtkTextBuffer *help_text;
71static GtkWidget *help_text_sw;
72static GtkWidget *help_vbox;
73static GtkWidget *help_tile;
74static GtkWidget *help_box;
75static GtkWidget *help_itable;
76static GtkWidget *help_wtable;
77static GtkWidget *help_utable;
78static GtkWidget *help_ttable;
79static GtkWidget *help_etable;
80static GtkWidget *help_tree;
81static GtkTreeStore *tstore;
82
83static GtkWidget *help_tree_sw;
84static GtkWidget *help_tree_expand;
85static GtkWidget *help_tree_collapse;
86static GtkWidget *help_tree_buttons_hbox;
87static GtkWidget *help_ilabel[6];
88static GtkWidget *help_wlabel[6];
89static GtkWidget *help_ulabel[5][5];
90static GtkWidget *help_tlabel[2][5];
91static GtkWidget *help_elabel[6];
92
93static bool help_advances[A_LAST];
94
95static GPtrArray *help_history;
97
99{
101 union {
102 enum help_page_type page;
103 const struct requirement *req;
104 } u;
105};
106
107static const char *help_ilabel_name[6] =
108{ N_("Base Cost:"), NULL, N_("Upkeep:"), NULL, N_("Requirement:"), NULL };
110
111static const char *help_wlabel_name[6] =
112{ N_("Base Cost:"), NULL, N_("Requirement:"), NULL, N_("Obsolete by:"), NULL };
114
116
117static const char *help_ulabel_name[5][5] =
118{
119 { N_("Cost:"), NULL, NULL, N_("Attack:"), NULL },
120 { N_("Defense:"), NULL, NULL, N_("Move:"), NULL },
121 { N_("Firepower:"), NULL, NULL, N_("Hitpoints:"), NULL },
122 { N_("Basic Upkeep:"), NULL, NULL, N_("Vision:"), NULL },
123 { N_("Requirement:"), NULL, NULL, N_("Obsolete by:"), NULL }
124};
125
126static const char *help_tlabel_name[2][5] =
127{
128 { N_("Move/Defense:"), NULL, NULL, N_("Food/Res/Trade:"), NULL },
129 { N_("Resources:"), NULL, NULL, NULL, NULL }
130};
131
132static const char *help_elabel_name[6] =
133/* TRANS: Label for build cost for extras in help. Will be followed by
134 * something like "3 MP" (where MP = Movement Points) */
135{ N_("Build:"), NULL,
136/* TRANS: Extra conflicts in help. Will be followed by a list of extras
137 * that can't be built on the same tile as this one. */
138 N_("Conflicts with:"), NULL,
139/* TRANS: Extra bonus in help. Will be followed by food/production/trade
140 * stats like "0/0/+1", "0/+50%/0" */
141 N_("Bonus (F/P/T):"), NULL };
142
143#define REQ_LABEL_NONE _("?tech:None")
144#define REQ_LABEL_NEVER _("(Never)")
145
146static void create_help_dialog(void);
147static void help_update_dialog(const struct help_item *pitem);
148
149static void select_help_item_string(const char *item,
150 enum help_page_type htype);
151static void help_command_update(void);
152static void help_command_callback(GtkWidget *w, gint response_id);
153
154/**********************************************************************/
169
170/**********************************************************************/
173static void set_title_topic(char *topic)
174{
175 if (strcmp(topic, _(HELP_ABOUT_ITEM)) == 0) {
176 gtk_frame_set_label(GTK_FRAME(help_frame), freeciv_name_version());
177 } else {
178 gtk_frame_set_label(GTK_FRAME(help_frame), topic);
179 }
180 return;
181}
182
183/**********************************************************************/
187{
188 if (help_dialog_shell) {
189 gtk_widget_destroy(help_dialog_shell);
190 }
191}
192
193/**********************************************************************/
196void popup_help_dialog_typed(const char *item, enum help_page_type htype)
197{
198 if (!help_dialog_shell) {
201 GTK_WINDOW(help_dialog_shell), 10, 10);
202 }
203 gtk_window_present(GTK_WINDOW(help_dialog_shell));
204
206}
207
208/**********************************************************************/
216
217/**********************************************************************/
226static void create_tech_tree(int tech, int levels, GtkTreeIter *parent)
227{
228 const struct research *presearch;
229 int bg;
230 int turns_to_tech;
231 bool original;
232 GtkTreeIter l;
233 GValue value = { 0, };
234 enum tech_state state;
235
236 if (advance_required(tech, AR_ONE) == A_LAST
237 && advance_required(tech, AR_TWO) == A_LAST) {
238 bg = COLOR_REQTREE_UNKNOWN;
239
240 gtk_tree_store_append(tstore, &l, parent);
241 help_advances[tech] = TRUE;
242
243 g_value_init(&value, G_TYPE_STRING);
244 g_value_set_static_string(&value, _("Removed"));
245 gtk_tree_store_set_value(tstore, &l, 0, &value);
246 g_value_unset(&value);
247
248 gtk_tree_store_set(tstore, &l,
249 1, -1,
250 2, tech,
251 3, &get_color(tileset, bg)->color
252 -1);
253 return;
254 }
255
256 presearch = research_get(client_player());
257
258 state = research_invention_state(presearch, tech);
259 if (tech_state_is_valid(state)) {
260 switch (state) {
261 case TECH_UNKNOWN:
262 bg = COLOR_REQTREE_UNKNOWN;
263 break;
264 case TECH_KNOWN:
265 bg = COLOR_REQTREE_KNOWN;
266 break;
267 case TECH_PREREQS_KNOWN:
268 bg = COLOR_REQTREE_PREREQS_KNOWN;
269 break;
270 }
271 } else {
272 bg = COLOR_REQTREE_BACKGROUND;
273 }
274 turns_to_tech = research_goal_unknown_techs(presearch, tech);
275
276 /* l is the original in the tree. */
277 original = !help_advances[tech];
278
279 gtk_tree_store_append(tstore, &l, parent);
280 help_advances[tech] = TRUE;
281
282 g_value_init(&value, G_TYPE_STRING);
283 g_value_set_static_string(&value,
285 tech));
286 gtk_tree_store_set_value(tstore, &l, 0, &value);
287 g_value_unset(&value);
288
289 gtk_tree_store_set(tstore, &l,
290 1, turns_to_tech,
291 2, tech,
292 3, &get_color(tileset, bg)->color,
293 -1);
294
295 if (--levels <= 0) {
296 return;
297 }
298
299 if (original) {
300 /* only add children to orginals */
301 if (advance_required(tech, AR_ONE) != A_NONE)
302 create_tech_tree(advance_required(tech, AR_ONE), levels, &l);
303 if (advance_required(tech, AR_TWO) != A_NONE)
304 create_tech_tree(advance_required(tech, AR_TWO), levels, &l);
305 }
306 return;
307}
308
309/**********************************************************************/
313 GtkTreePath *path,
314 GtkTreeViewColumn *col,
315 gpointer data)
316{
317 GtkTreeIter it;
318 gint tech;
319
320 gtk_tree_model_get_iter(GTK_TREE_MODEL(tstore), &it, path);
321 gtk_tree_model_get(GTK_TREE_MODEL(tstore), &it, 2, &tech, -1);
323 HELP_TECH);
324}
325
326/**********************************************************************/
329static void help_tech_tree_expand_callback(GtkWidget *w, gpointer data)
330{
331 gtk_tree_view_expand_all(GTK_TREE_VIEW(data));
332}
333
334/**********************************************************************/
337static void help_tech_tree_collapse_callback(GtkWidget *w, gpointer data)
338{
339 gtk_tree_view_collapse_all(GTK_TREE_VIEW(data));
340}
341
342/**********************************************************************/
345static void help_hyperlink_callback(GtkWidget *w)
346{
347 const char *s;
348 struct help_page_selection *select;
350
351 s = gtk_label_get_text(GTK_LABEL(w));
352 select = (struct help_page_selection *)(g_object_get_data(G_OBJECT(w), "page_type"));
353
354 switch (select->type) {
355 case HPAGE_SRC_ENUM:
356 type = select->u.page;
357 break;
358 case HPAGE_SRC_REQ:
360 break;
361 }
362
363 if (type == HELP_LAST) {
364 return;
365 }
366
367 /* FIXME: May be able to skip, or may need to modify, advances[A_NONE]
368 below, depending on which i18n is done elsewhere.
369 */
370 if (strcmp(s, REQ_LABEL_NEVER) != 0
374 }
375}
376
377/**********************************************************************/
380static GtkWidget *help_hyperlink_new(GtkWidget *label,
381 struct help_page_selection *select)
382{
383 GtkWidget *button;
384
385 button = gtk_button_new();
386 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
387 gtk_widget_set_halign(label, GTK_ALIGN_CENTER);
388 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
389 gtk_widget_set_name(label, "help_link");
390 gtk_container_add(GTK_CONTAINER(button), label);
391 gtk_widget_show(button);
392 g_signal_connect_swapped(button, "clicked",
393 G_CALLBACK(help_hyperlink_callback), label);
394 g_object_set_data(G_OBJECT(label), "page_type", select);
395
396 return button;
397}
398
399/**********************************************************************/
402static GtkWidget *help_hyperlink_new_page(GtkWidget *label,
403 enum help_page_type page)
404{
405 return help_hyperlink_new(label, &(page_selections[page]));
406}
407
408/**********************************************************************/
411static GtkWidget *help_slink_new(const gchar *txt,
412 struct help_page_selection *select)
413{
414 GtkWidget *button, *label;
415
416 label = gtk_label_new(txt);
417 gtk_widget_set_halign(label, GTK_ALIGN_CENTER);
418 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
419 button = help_hyperlink_new(label, select);
420
421 return button;
422}
423
424/**********************************************************************/
427static GtkWidget *help_slink_new_page(const gchar *txt,
428 enum help_page_type page)
429{
430 return help_slink_new(txt, &(page_selections[page]));
431}
432
433/**********************************************************************/
436static void help_box_hide(void)
437{
438 gtk_widget_hide(help_box);
439
440 gtk_widget_hide(help_tile);
441
442 gtk_widget_hide(help_itable);
443 gtk_widget_hide(help_wtable);
444 gtk_widget_hide(help_utable);
445 gtk_widget_hide(help_ttable);
446 gtk_widget_hide(help_etable);
447
448 gtk_widget_hide(help_tile); /* FIXME: twice? */
449
450 gtk_widget_hide(help_vbox);
451 gtk_widget_hide(help_text_sw);
452
453 gtk_widget_hide(help_tree_sw);
454 gtk_widget_hide(help_tree_buttons_hbox);
455}
456
457/**********************************************************************/
460static void help_destroy_callback(GtkWidget *w, gpointer data)
461{
462 g_ptr_array_free(help_history, TRUE);
463 help_dialog_shell = NULL;
464}
465
466/**********************************************************************/
469static void activated_topic(GtkTreeView *view, gpointer data)
470{
471 GtkTreePath *path;
472 GtkTreeViewColumn *col;
473 GtkTreeModel *model;
474 GtkTreeIter it;
475 struct help_item *pitem;
476
477 model = gtk_tree_view_get_model(view);
478
479 gtk_tree_view_get_cursor(view, &path, &col);
480 gtk_tree_model_get_iter(model, &it, path);
481 gtk_tree_path_free(path);
482
483 if (!path) {
484 return;
485 }
486
487 gtk_tree_model_get(model, &it, 1, &pitem, -1);
488
489 if (help_history_pos >= 0
490 && g_ptr_array_index(help_history, help_history_pos) == (gpointer) pitem) {
491 return;
492 }
493
494 help_update_dialog(pitem);
495
496 /* add to history. */
497 if (help_history_pos < help_history->len - 1) {
498 g_ptr_array_set_size(help_history, help_history_pos + 1);
499 }
501
502 g_ptr_array_add(help_history, (gpointer)pitem);
504}
505
506/**********************************************************************/
509static void create_help_dialog(void)
510{
511 GtkWidget *hbox;
512 GtkWidget *button;
513 GtkWidget *text;
514 int i, j;
515 GtkCellRenderer *rend;
516 GtkTreeViewColumn *col;
517 GArray *array;
518 GtkTreeStore *store;
519 GtkTreeSelection *selection;
520
521 help_history = g_ptr_array_new();
522 help_history_pos = -1;
523
524 help_dialog_shell = gtk_dialog_new_with_buttons(_("Freeciv Help Browser"),
525 NULL,
526 0,
527 _("_Back"),
528 1,
529 _("_Forward"),
530 2,
531 _("_Close"),
532 GTK_RESPONSE_CLOSE,
533 NULL);
535 gtk_dialog_set_default_response(GTK_DIALOG(help_dialog_shell),
536 GTK_RESPONSE_CLOSE);
537 gtk_widget_set_name(help_dialog_shell, "Freeciv");
538
539 g_signal_connect(help_dialog_shell, "response",
540 G_CALLBACK(help_command_callback), NULL);
541 g_signal_connect(help_dialog_shell, "destroy",
542 G_CALLBACK(help_destroy_callback), NULL);
543
544 hbox = gtk_grid_new();
545 gtk_grid_set_column_spacing(GTK_GRID(hbox), 5);
546 gtk_container_add(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(help_dialog_shell))), hbox);
547 gtk_widget_show(hbox);
548
549 /* build tree store. */
550 store = gtk_tree_store_new(2, G_TYPE_STRING, G_TYPE_POINTER);
551
552 array = g_array_new(FALSE, FALSE, sizeof(GtkTreeIter));
553 help_items_iterate(pitem) {
554 GtkTreeIter *it, *parent;
555 const char *s;
556 int depth;
557
558 for (s = pitem->topic; *s == ' '; s++) {
559 /* nothing */
560 }
561 depth = s - pitem->topic;
562
563 array = g_array_set_size(array, depth+1);
564
565 if (depth > 0) {
566 parent = &g_array_index(array, GtkTreeIter, depth-1);
567 } else {
568 parent = NULL;
569 }
570
571 it = &g_array_index(array, GtkTreeIter, depth);
572 gtk_tree_store_append(store, it, parent);
573
574 gtk_tree_store_set(store, it, 0, pitem->topic, 1, pitem, -1);
576
577
578 /* create tree view. */
579 help_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
580 g_object_unref(store);
581 gtk_tree_view_columns_autosize(GTK_TREE_VIEW(help_view));
582 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(help_view), FALSE);
583
584 g_signal_connect(help_view, "cursor-changed",
585 G_CALLBACK(activated_topic), NULL);
586
587 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(help_view));
588 gtk_tree_selection_set_mode(selection, GTK_SELECTION_BROWSE);
589
590 rend = gtk_cell_renderer_text_new();
591 col = gtk_tree_view_column_new_with_attributes(NULL, rend, "text", 0, NULL);
592 gtk_tree_view_append_column(GTK_TREE_VIEW(help_view), col);
593
594 help_view_sw = gtk_scrolled_window_new(NULL, NULL);
595 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(help_view_sw),
596 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
597 gtk_widget_set_size_request(help_view_sw, 190, -1);
598 gtk_container_add(GTK_CONTAINER(help_view_sw), help_view);
599 gtk_widget_show(help_view);
600 gtk_container_add(GTK_CONTAINER(hbox), help_view_sw);
601 gtk_widget_show(help_view_sw);
602
603 help_frame = gtk_frame_new("");
604 gtk_container_add(GTK_CONTAINER(hbox), help_frame);
605 gtk_widget_set_size_request(help_frame, help_frame_width, help_frame_height);
606 gtk_widget_show(help_frame);
607
608 help_box = gtk_grid_new();
609 gtk_grid_set_row_spacing(GTK_GRID(help_box), 5);
610 gtk_orientable_set_orientation(GTK_ORIENTABLE(help_box),
611 GTK_ORIENTATION_VERTICAL);
612 gtk_container_add(GTK_CONTAINER(help_frame), help_box);
613
614 help_tile = gtk_image_new();
615 gtk_container_add(GTK_CONTAINER(help_box), help_tile);
616
617 help_itable = gtk_grid_new();
618 gtk_container_add(GTK_CONTAINER(help_box), help_itable);
619
620 for (i = 0; i < 6; i++) {
621 help_ilabel[i] =
622 gtk_label_new(help_ilabel_name[i] != NULL ? _(help_ilabel_name[i]) : "");
623 gtk_widget_set_hexpand(help_ilabel[i], TRUE);
624
625 if (i == 5) {
627 gtk_grid_attach(GTK_GRID(help_itable), button, i, 0, 1, 1);
628 } else {
629 gtk_grid_attach(GTK_GRID(help_itable), help_ilabel[i], i, 0, 1, 1);
630 gtk_widget_set_name(help_ilabel[i], "help_label");
631 }
632
633 gtk_widget_show(help_ilabel[i]);
634 }
635
636 help_wtable = gtk_grid_new();
637 gtk_container_add(GTK_CONTAINER(help_box), help_wtable);
638
639 for (i = 0; i < 6; i++) {
640 help_wlabel[i] =
641 gtk_label_new(help_wlabel_name[i] ? _(help_wlabel_name[i]) : "");
642 gtk_widget_set_hexpand(help_wlabel[i], TRUE);
643
644 if (i == 3 || i == 5) {
646 gtk_grid_attach(GTK_GRID(help_wtable), button, i, 0, 1, 1);
647 } else {
648 gtk_grid_attach(GTK_GRID(help_wtable), help_wlabel[i], i, 0, 1, 1);
649 gtk_widget_set_name(help_wlabel[i], "help_label");
650 }
651
652 gtk_widget_show(help_wlabel[i]);
653 }
654
655 help_utable = gtk_grid_new();
656 gtk_container_add(GTK_CONTAINER(help_box), help_utable);
657
658 for (i = 0; i < 5; i++) {
659 for (j = 0; j < 5; j++) {
660 help_ulabel[j][i] =
661 gtk_label_new(help_ulabel_name[j][i] ? _(help_ulabel_name[j][i]) : "");
662 gtk_widget_set_hexpand(help_ulabel[j][i], TRUE);
663
664 if (j == 4 && (i == 1 || i == 4)) {
665 if (i == 1) {
667 } else {
669 }
670
671 gtk_grid_attach(GTK_GRID(help_utable), button, i, j, 1, 1);
672 } else {
673 gtk_grid_attach(GTK_GRID(help_utable), help_ulabel[j][i],
674 i, j, 1, 1);
675 gtk_widget_set_name(help_ulabel[j][i], "help_label");
676 }
677 gtk_widget_show(help_ulabel[j][i]);
678 }
679 }
680
681 help_ttable = gtk_grid_new();
682 gtk_container_add(GTK_CONTAINER(help_box), help_ttable);
683
684 for (j = 0; j < 2; j++) {
685 for (i = 0; i < 5; i++) {
686 help_tlabel[j][i] =
687 gtk_label_new(help_tlabel_name[j][i] ? _(help_tlabel_name[j][i]) : "");
688 gtk_widget_set_hexpand(help_tlabel[j][i], TRUE);
689 gtk_widget_set_name(help_tlabel[j][i], "help_label");
690
691 /* Ugly (but these numbers are hardcoded in help_update_terrain() too) */
692 if (j == 1 && i == 1) {
693 /* Extra wide cell for terrain specials */
694 gtk_grid_attach(GTK_GRID(help_ttable), help_tlabel[j][i],
695 i, j, 4, 1);
696 gtk_widget_show(help_tlabel[j][i]);
697 break; /* skip rest of row */
698 } else {
699 gtk_grid_attach(GTK_GRID(help_ttable), help_tlabel[j][i],
700 i, j, 1, 1);
701 gtk_widget_show(help_tlabel[j][i]);
702 }
703 }
704 }
705
706 help_etable = gtk_grid_new();
707 gtk_container_add(GTK_CONTAINER(help_box), help_etable);
708
709 for (i = 0; i < 6; i++) {
710 help_elabel[i] =
711 gtk_label_new(help_elabel_name[i] ? _(help_elabel_name[i]) : "");
712 gtk_widget_set_hexpand(help_elabel[i], TRUE);
713 gtk_grid_attach(GTK_GRID(help_etable), help_elabel[i], i % 4, i / 4, 1, 1);
714 gtk_widget_set_name(help_elabel[i], "help_label");
715 gtk_widget_show(help_elabel[i]);
716 }
717
718 help_vbox = gtk_grid_new();
719 gtk_grid_set_row_spacing(GTK_GRID(help_vbox), 1);
720 gtk_orientable_set_orientation(GTK_ORIENTABLE(help_vbox),
721 GTK_ORIENTATION_VERTICAL);
722 gtk_container_set_border_width(GTK_CONTAINER(help_vbox), 5);
723 gtk_container_add(GTK_CONTAINER(help_box), help_vbox);
724
725 text = gtk_text_view_new();
726 gtk_widget_set_hexpand(text, TRUE);
727 gtk_widget_set_vexpand(text, TRUE);
728 gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(text), FALSE);
729 gtk_text_view_set_editable(GTK_TEXT_VIEW(text), FALSE);
730 gtk_container_set_border_width(GTK_CONTAINER(text), 5);
731 gtk_text_view_set_wrap_mode(GTK_TEXT_VIEW(text), GTK_WRAP_WORD);
732 gtk_widget_set_name(text, "help_text");
733 help_text = gtk_text_view_get_buffer(GTK_TEXT_VIEW(text));
734 gtk_widget_show(text);
735
736 help_text_sw = gtk_scrolled_window_new(NULL, NULL);
737 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(help_text_sw),
738 GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
739 gtk_container_add(GTK_CONTAINER(help_text_sw), text);
740 gtk_container_add(GTK_CONTAINER(help_box), help_text_sw);
741
742 /* build tech store. */
743 tstore = gtk_tree_store_new(4,
744 G_TYPE_STRING, /* tech name */
745 G_TYPE_INT, /* turns to tech */
746 G_TYPE_INT, /* tech id */
747 GDK_TYPE_RGBA); /* color */
748 help_tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(tstore));
749 gtk_widget_set_hexpand(help_tree, TRUE);
750 gtk_widget_set_vexpand(help_tree, TRUE);
751 g_object_unref(tstore);
752 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(help_tree), FALSE);
753
754 g_signal_connect(help_tree, "row_activated",
755 G_CALLBACK(help_tech_tree_activated_callback), NULL);
756
757
758 col = gtk_tree_view_column_new();
759
760 rend = gtk_cell_renderer_text_new();
761 g_object_set(rend, "weight", PANGO_WEIGHT_BOLD, NULL);
762 gtk_tree_view_column_pack_start(col, rend, TRUE);
763 gtk_tree_view_column_set_attributes(col, rend,
764 "text", 0,
765 "background-rgba", 3,
766 NULL);
767 rend = gtk_cell_renderer_text_new();
768 g_object_set(rend, "weight", PANGO_WEIGHT_BOLD, "xalign", 1.0, NULL);
769 gtk_tree_view_column_pack_start(col, rend, FALSE);
770 gtk_tree_view_column_set_attributes(col, rend,
771 "text", 1,
772 "background-rgba", 3,
773 NULL);
774
775 gtk_tree_view_append_column(GTK_TREE_VIEW(help_tree), col);
776
777
778 help_tree_sw = gtk_scrolled_window_new(NULL, NULL);
779 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(help_tree_sw),
780 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
781 gtk_container_add(GTK_CONTAINER(help_tree_sw), help_tree);
782 gtk_widget_show(help_tree);
783 gtk_container_add(GTK_CONTAINER(help_box), help_tree_sw);
784
786 gtk_button_new_with_label(_("Expand All"));
787 help_tree_collapse = gtk_button_new_with_label(_("Collapse All"));
788
789 g_signal_connect(help_tree_expand, "clicked",
791 g_signal_connect(help_tree_collapse, "clicked",
793
794 help_tree_buttons_hbox = gtk_button_box_new(GTK_ORIENTATION_HORIZONTAL);
795 gtk_container_add(GTK_CONTAINER(help_tree_buttons_hbox), help_tree_expand);
796 gtk_container_add(GTK_CONTAINER(help_tree_buttons_hbox), help_tree_collapse);
797 gtk_container_add(GTK_CONTAINER(help_box), help_tree_buttons_hbox);
798 gtk_widget_show_all(help_tree_buttons_hbox);
799}
800
801/**********************************************************************/
804static void set_help_tile_from_sprite(struct sprite *spr)
805{
806 if (spr == NULL) {
807 return;
808 }
809
810 gtk_image_set_from_surface(GTK_IMAGE(help_tile), spr->surface);
811 gtk_widget_show(help_tile);
812}
813
814/**********************************************************************/
817static void set_help_tile_from_terrain(struct terrain *pterr)
818{
820 cairo_t *cr;
821 int i;
822
823 canvas.surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
826
827 cr = cairo_create(canvas.surface);
828 cairo_set_operator(cr, CAIRO_OPERATOR_CLEAR);
829 cairo_paint(cr);
830 cairo_destroy(cr);
831
832 for (i = 0; i < 3; i++) {
833 struct drawn_sprite sprs[80];
835 i, pterr);
836
837 put_drawn_sprites(&canvas, 1.0, 0, 0, count, sprs, FALSE);
838 }
839
840 gtk_image_set_from_surface(GTK_IMAGE(help_tile), canvas.surface);
841 gtk_widget_show(help_tile);
842 cairo_surface_destroy(canvas.surface);
843}
844
845/**********************************************************************/
848static void set_help_tile_from_extra(const struct extra_type *pextra)
849{
851 cairo_t *cr;
852 struct drawn_sprite sprs[80];
853 int count;
854
855 canvas.surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
858
859 cr = cairo_create(canvas.surface);
860 cairo_set_operator(cr, CAIRO_OPERATOR_CLEAR);
861 cairo_paint(cr);
862 cairo_destroy(cr);
863
864 count = fill_basic_extra_sprite_array(tileset, sprs, pextra);
865
866 put_drawn_sprites(&canvas, 1.0, 0, 0, count, sprs, FALSE);
867
868 gtk_image_set_from_surface(GTK_IMAGE(help_tile), canvas.surface);
869 gtk_widget_show(help_tile);
870 cairo_surface_destroy(canvas.surface);
871}
872
873/**********************************************************************/
876static void help_update_improvement(const struct help_item *pitem,
877 char *title)
878{
879 char buf[8192];
881
882 if (imp && !is_great_wonder(imp)) {
884 char req_buf[512];
885
886 sprintf(buf, "%d", impr_base_build_shield_cost(imp));
887 gtk_label_set_text(GTK_LABEL(help_ilabel[1]), buf);
888 sprintf(buf, "%d", imp->upkeep);
889 gtk_label_set_text(GTK_LABEL(help_ilabel[3]), buf);
890
891 /* FIXME: this should show ranges, negated reqs, and all the
892 * MAX_NUM_REQS reqs.
893 * Currently it's limited to 1 req but this code is partially prepared
894 * to be extended. Remember MAX_NUM_REQS is a compile-time
895 * definition. */
896 help_impr_req.u.req = NULL;
897 requirement_vector_iterate(&imp->reqs, preq) {
898 if (!preq->present) {
899 continue;
900 }
901 req = universal_name_translation(&preq->source, req_buf, sizeof(req_buf));
902 help_impr_req.u.req = preq;
903 break;
905 gtk_label_set_text(GTK_LABEL(help_ilabel[5]), req);
906/* create_tech_tree(help_improvement_tree, 0, imp->tech_req, 3);*/
907 } else {
908 gtk_label_set_text(GTK_LABEL(help_ilabel[1]), "0");
909 gtk_label_set_text(GTK_LABEL(help_ilabel[3]), "0");
910 gtk_label_set_text(GTK_LABEL(help_ilabel[5]), REQ_LABEL_NEVER);
911/* create_tech_tree(help_improvement_tree, 0, advance_count(), 3);*/
912 }
913
915
916 gtk_widget_show(help_itable);
917
918 helptext_building(buf, sizeof(buf), client.conn.playing, pitem->text, imp);
919 gtk_text_buffer_set_text(help_text, buf, -1);
920 gtk_widget_show(help_text_sw);
921}
922
923/**********************************************************************/
926static void help_update_wonder(const struct help_item *pitem,
927 char *title)
928{
929 char buf[8192];
931
932 if (imp && is_great_wonder(imp)) {
933 int i;
934 char req_buf[512];
935
936 sprintf(buf, "%d", impr_base_build_shield_cost(imp));
937 gtk_label_set_text(GTK_LABEL(help_wlabel[1]), buf);
938
939 /* FIXME: this should show ranges, negated reqs, and all the
940 * MAX_NUM_REQS reqs.
941 * Currently it's limited to 1 req but this code is partially prepared
942 * to be extended. Remember MAX_NUM_REQS is a compile-time
943 * definition. */
944 i = 0;
945 help_wndr_req.u.req = NULL;
946 requirement_vector_iterate(&imp->reqs, preq) {
947 if (!preq->present) {
948 continue;
949 }
950 gtk_label_set_text(GTK_LABEL(help_wlabel[3 + i]),
951 universal_name_translation(&preq->source,
952 req_buf, sizeof(req_buf)));
953 help_wndr_req.u.req = preq;
954 i++;
955 break;
957
958 gtk_label_set_text(GTK_LABEL(help_wlabel[5]), REQ_LABEL_NEVER);
960 if (pobs->source.kind == VUT_ADVANCE && pobs->present) {
961 gtk_label_set_text(GTK_LABEL(help_wlabel[5]),
963 (pobs->source.value.advance));
964
965 break;
966 }
968
969/* create_tech_tree(help_improvement_tree, 0, imp->tech_req, 3);*/
970 } else {
971 /* can't find wonder */
972 gtk_label_set_text(GTK_LABEL(help_wlabel[1]), "0");
973 gtk_label_set_text(GTK_LABEL(help_wlabel[3]), REQ_LABEL_NEVER);
974 gtk_label_set_text(GTK_LABEL(help_wlabel[5]), skip_intl_qualifier_prefix(REQ_LABEL_NONE));
975/* create_tech_tree(help_improvement_tree, 0, advance_count(), 3); */
976 }
977
979
980 gtk_widget_show(help_wtable);
981
982 helptext_building(buf, sizeof(buf), client.conn.playing, pitem->text, imp);
983 gtk_text_buffer_set_text(help_text, buf, -1);
984 gtk_widget_show(help_text_sw);
985}
986
987/**********************************************************************/
990static void help_update_unit_type(const struct help_item *pitem,
991 char *title)
992{
993 char buf[8192];
995
996 if (utype) {
997 sprintf(buf, "%d", utype_build_shield_cost_base(utype));
998 gtk_label_set_text(GTK_LABEL(help_ulabel[0][1]), buf);
999 sprintf(buf, "%d", utype->attack_strength);
1000 gtk_label_set_text(GTK_LABEL(help_ulabel[0][4]), buf);
1001 sprintf(buf, "%d", utype->defense_strength);
1002 gtk_label_set_text(GTK_LABEL(help_ulabel[1][1]), buf);
1003 sprintf(buf, "%s", move_points_text(utype->move_rate, TRUE));
1004 gtk_label_set_text(GTK_LABEL(help_ulabel[1][4]), buf);
1005 sprintf(buf, "%d", utype->firepower);
1006 gtk_label_set_text(GTK_LABEL(help_ulabel[2][1]), buf);
1007 sprintf(buf, "%d", utype->hp);
1008 gtk_label_set_text(GTK_LABEL(help_ulabel[2][4]), buf);
1009 gtk_label_set_text(GTK_LABEL(help_ulabel[3][1]),
1011 sprintf(buf, "%d", (int)sqrt((double)utype->vision_radius_sq));
1012 gtk_label_set_text(GTK_LABEL(help_ulabel[3][4]), buf);
1013 if (A_NEVER == utype->require_advance) {
1014 gtk_label_set_text(GTK_LABEL(help_ulabel[4][1]), REQ_LABEL_NEVER);
1015 } else {
1016 gtk_label_set_text(GTK_LABEL(help_ulabel[4][1]),
1018 }
1019/* create_tech_tree(help_improvement_tree, 0, advance_number(utype->require_advance), 3);*/
1020 if (U_NOT_OBSOLETED == utype->obsoleted_by) {
1021 gtk_label_set_text(GTK_LABEL(help_ulabel[4][4]), skip_intl_qualifier_prefix(REQ_LABEL_NONE));
1022 } else {
1023 gtk_label_set_text(GTK_LABEL(help_ulabel[4][4]),
1025 }
1026
1027 helptext_unit(buf, sizeof(buf), client.conn.playing, pitem->text, utype);
1028
1029 gtk_text_buffer_set_text(help_text, buf, -1);
1030 gtk_widget_show(help_text_sw);
1031
1033 direction8_invalid()));
1034 } else {
1035 gtk_label_set_text(GTK_LABEL(help_ulabel[0][1]), "0");
1036 gtk_label_set_text(GTK_LABEL(help_ulabel[0][4]), "0");
1037 gtk_label_set_text(GTK_LABEL(help_ulabel[1][1]), "0");
1038 gtk_label_set_text(GTK_LABEL(help_ulabel[1][4]), "0");
1039 gtk_label_set_text(GTK_LABEL(help_ulabel[2][1]), "0");
1040 gtk_label_set_text(GTK_LABEL(help_ulabel[2][4]), "0");
1041 gtk_label_set_text(GTK_LABEL(help_ulabel[3][1]), "0");
1042 gtk_label_set_text(GTK_LABEL(help_ulabel[3][4]), "0");
1043
1044 gtk_label_set_text(GTK_LABEL(help_ulabel[4][1]), REQ_LABEL_NEVER);
1045/* create_tech_tree(help_improvement_tree, 0, A_LAST, 3);*/
1046 gtk_label_set_text(GTK_LABEL(help_ulabel[4][4]), skip_intl_qualifier_prefix(REQ_LABEL_NONE));
1047
1048 gtk_text_buffer_set_text(help_text, buf, -1);
1049 gtk_widget_show(help_text_sw);
1050 }
1051 gtk_widget_show(help_utable);
1052}
1053
1054/**********************************************************************/
1057static char *fc_chomp(char *str, size_t len)
1058{
1059 gchar *i;
1060
1061 if (!str || !*str) {
1062 return str;
1063 }
1064
1065 i = str + len;
1066 for (i = g_utf8_find_prev_char(str, i);
1067 (i && g_unichar_isspace(g_utf8_get_char(i)));
1068 i = g_utf8_find_prev_char(str, i)) {
1069 *i = '\0';
1070 }
1071 return str;
1072}
1073
1074/**********************************************************************/
1077static void help_update_tech(const struct help_item *pitem, char *title)
1078{
1079 int i, j;
1080 GtkWidget *w, *hbox;
1081 char buf[8192];
1082 struct advance *padvance = advance_by_translated_name(title);
1083
1084 if (padvance && !is_future_tech(i = advance_number(padvance))) {
1085 GtkTextBuffer *txt;
1086 size_t len;
1087
1088 gtk_container_foreach(GTK_CONTAINER(help_vbox), (GtkCallback)gtk_widget_destroy, NULL);
1089
1090 for (j = 0; j < ARRAY_SIZE(help_advances); j++) {
1091 help_advances[j] = FALSE;
1092 }
1093 gtk_tree_store_clear(tstore);
1095 gtk_widget_show(help_tree_sw);
1096 gtk_widget_show(help_tree_buttons_hbox);
1097
1098 helptext_advance(buf, sizeof(buf), client.conn.playing, pitem->text, i);
1099 len = strlen(buf);
1100 fc_chomp(buf, len);
1101
1103
1104 w = gtk_text_view_new();
1105 gtk_widget_set_hexpand(w, TRUE);
1106 gtk_widget_set_vexpand(w, TRUE);
1107 gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(w), FALSE);
1108 gtk_text_view_set_wrap_mode(GTK_TEXT_VIEW(w), GTK_WRAP_WORD);
1109 gtk_widget_set_name(w, "help_text");
1110 gtk_container_set_border_width(GTK_CONTAINER(w), 5);
1111 gtk_text_view_set_editable(GTK_TEXT_VIEW(w), FALSE);
1112 gtk_container_add(GTK_CONTAINER(help_vbox), w);
1113 gtk_widget_show(w);
1114
1115 txt = gtk_text_view_get_buffer(GTK_TEXT_VIEW(w));
1116 if (txt) {
1117 gtk_text_buffer_set_text(txt, buf, -1);
1118 }
1119
1120 w = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);
1121 g_object_set(w, "margin", 5, NULL);
1122 gtk_widget_set_hexpand(w, TRUE);
1123 gtk_widget_set_vexpand(w, TRUE);
1124 gtk_container_add(GTK_CONTAINER(help_vbox), w);
1125 gtk_widget_show(w);
1126
1127 governments_iterate(pgov) {
1128 /* FIXME: need a more general mechanism for this, since this
1129 * helptext needs to be shown in all possible req source types. */
1130 requirement_vector_iterate(&pgov->reqs, preq) {
1131 if (VUT_ADVANCE == preq->source.kind
1132 && preq->source.value.advance == padvance) {
1133 hbox = gtk_grid_new();
1134 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1135 w = gtk_label_new(_("Allows"));
1136 gtk_container_add(GTK_CONTAINER(hbox), w);
1139 gtk_container_add(GTK_CONTAINER(hbox), w);
1140 gtk_widget_show_all(hbox);
1141 }
1144
1145 improvement_iterate(pimprove) {
1146 if (valid_improvement(pimprove)) {
1147 requirement_vector_iterate(&pimprove->reqs, preq) {
1148 if (VUT_ADVANCE == preq->source.kind
1149 && preq->source.value.advance == padvance) {
1150 hbox = gtk_grid_new();
1151 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1152 w = gtk_label_new(_("Allows"));
1153 gtk_container_add(GTK_CONTAINER(hbox), w);
1155 is_great_wonder(pimprove)
1156 ? HELP_WONDER
1158 gtk_container_add(GTK_CONTAINER(hbox), w);
1159 gtk_widget_show_all(hbox);
1160 }
1162 requirement_vector_iterate(&pimprove->obsolete_by, pobs) {
1163 if (pobs->source.kind == VUT_ADVANCE
1164 && pobs->source.value.advance == padvance
1165 && pobs->present) {
1166 hbox = gtk_grid_new();
1167 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1168 w = gtk_label_new(_("Obsoletes"));
1169 gtk_container_add(GTK_CONTAINER(hbox), w);
1171 is_great_wonder(pimprove)
1172 ? HELP_WONDER
1174 gtk_container_add(GTK_CONTAINER(hbox), w);
1175 gtk_widget_show_all(hbox);
1176 }
1178 }
1180
1181 unit_type_iterate(punittype) {
1182 if (padvance != punittype->require_advance) {
1183 continue;
1184 }
1185 hbox = gtk_grid_new();
1186 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1187 w = gtk_label_new(_("Allows"));
1188 gtk_container_add(GTK_CONTAINER(hbox), w);
1190 gtk_container_add(GTK_CONTAINER(hbox), w);
1191 gtk_widget_show_all(hbox);
1193
1194 advance_iterate(A_NONE, ptest) {
1195 if (padvance == advance_requires(ptest, AR_ONE)) {
1197 hbox = gtk_grid_new();
1198 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1199 w = gtk_label_new(_("Allows"));
1200 gtk_container_add(GTK_CONTAINER(hbox), w);
1202 gtk_container_add(GTK_CONTAINER(hbox), w);
1203 gtk_widget_show_all(hbox);
1204 } else {
1205 hbox = gtk_grid_new();
1206 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1207 w = gtk_label_new(_("Allows"));
1208 gtk_container_add(GTK_CONTAINER(hbox), w);
1210 gtk_container_add(GTK_CONTAINER(hbox), w);
1211 w = gtk_label_new(_("with"));
1212 gtk_container_add(GTK_CONTAINER(hbox), w);
1214 AR_TWO)),
1215 HELP_TECH);
1216 gtk_container_add(GTK_CONTAINER(hbox), w);
1217 w = gtk_label_new(Q_("?techhelp:"));
1218 gtk_container_add(GTK_CONTAINER(hbox), w);
1219 gtk_widget_show_all(hbox);
1220 }
1221 }
1222 if (padvance == advance_requires(ptest, AR_TWO)) {
1223 hbox = gtk_grid_new();
1224 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1225 w = gtk_label_new(_("Allows"));
1226 gtk_container_add(GTK_CONTAINER(hbox), w);
1228 gtk_container_add(GTK_CONTAINER(hbox), w);
1229 w = gtk_label_new(_("with"));
1230 gtk_container_add(GTK_CONTAINER(hbox), w);
1232 HELP_TECH);
1233 gtk_container_add(GTK_CONTAINER(hbox), w);
1234 w = gtk_label_new(Q_("?techhelp:"));
1235 gtk_container_add(GTK_CONTAINER(hbox), w);
1236 gtk_widget_show_all(hbox);
1237 }
1239 gtk_widget_show(help_vbox);
1240 }
1241}
1242
1243/**********************************************************************/
1246static void add_act_help_for_terrain(const char *act_label,
1247 const char *result_link_label,
1248 enum help_page_type result_link_type,
1249 const char *descr_label)
1250{
1251 GtkWidget *w;
1252 GtkWidget *hbox;
1253
1254 hbox = gtk_grid_new();
1255 gtk_container_add(GTK_CONTAINER(help_vbox), hbox);
1256 w = gtk_label_new(act_label);
1257 gtk_container_add(GTK_CONTAINER(hbox), w);
1258 w = help_slink_new_page(result_link_label, result_link_type);
1259 gtk_container_add(GTK_CONTAINER(hbox), w);
1260 w = gtk_label_new(descr_label);
1261 gtk_container_add(GTK_CONTAINER(hbox), w);
1262
1263 gtk_widget_show_all(hbox);
1264}
1265
1266/**********************************************************************/
1269static void help_extras_of_act_for_terrain(struct terrain *pterr,
1270 enum unit_activity act,
1271 char *label)
1272{
1273 enum extra_cause cause = activity_to_extra_cause(act);
1274
1275 extra_type_by_cause_iterate(cause, pextra) {
1276 if (pextra->buildable
1277 && requirement_fulfilled_by_terrain(pterr, &(pextra->reqs))) {
1280 helptext_extra_for_terrain_str(pextra, pterr,
1281 act));
1282 }
1284}
1285
1286/**********************************************************************/
1289static void help_update_terrain(const struct help_item *pitem,
1290 char *title)
1291{
1292 char buf[8192];
1293 struct terrain *pterrain = terrain_by_translated_name(title);
1294
1295 if (pterrain) {
1296 struct universal for_terr = { .kind = VUT_TERRAIN, .value = { .terrain = pterrain }};
1297
1299
1300 {
1301 /* 25 => "1.25"; 50 => "1.5"; 100 => "2.0" */
1302 int defbonus = pterrain->defense_bonus + 100;
1303 int frac = defbonus % 100;
1304
1305 if ((frac % 10) == 0) {
1306 frac /= 10;
1307 }
1308 sprintf(buf, "%d/%d.%d",
1309 pterrain->movement_cost, defbonus / 100, frac);
1310 }
1311 gtk_label_set_text(GTK_LABEL(help_tlabel[0][1]), buf);
1312
1313 sprintf(buf, "%d/%d/%d",
1314 pterrain->output[O_FOOD],
1315 pterrain->output[O_SHIELD],
1316 pterrain->output[O_TRADE]);
1317 gtk_label_set_text(GTK_LABEL(help_tlabel[0][4]), buf);
1318
1319 buf[0] = '\0';
1320 if (*(pterrain->resources)) {
1321 struct extra_type **r;
1322
1323 for (r = pterrain->resources; *r; r++) {
1324 /* TRANS: " Whales (2/1/2)," */
1325 sprintf (buf + strlen (buf), " %s (%d/%d/%d),",
1327 pterrain->output[O_FOOD] + (*r)->data.resource->output[O_FOOD],
1328 pterrain->output[O_SHIELD] + (*r)->data.resource->output[O_SHIELD],
1329 pterrain->output[O_TRADE] + (*r)->data.resource->output[O_TRADE]);
1330 }
1331 buf[strlen (buf) - 1] = '.';
1332 } else {
1333 /* TRANS: "Resources: (none)" */
1334 sprintf (buf + strlen (buf), _("(none)"));
1335 }
1336 gtk_label_set_text(GTK_LABEL(help_tlabel[1][1]), buf);
1337
1338 gtk_container_foreach(GTK_CONTAINER(help_vbox), (GtkCallback)gtk_widget_destroy, NULL);
1339
1340 if (pterrain->cultivate_result != T_NONE
1341 && action_id_univs_not_blocking(ACTION_CULTIVATE,
1342 NULL, &for_terr)) {
1343 fc_snprintf(buf, sizeof(buf),
1344 PL_("%d turn", "%d turns", pterrain->cultivate_time),
1345 pterrain->cultivate_time);
1346 add_act_help_for_terrain(_("Cultivate Rslt/Time"),
1348 HELP_TERRAIN, buf);
1349 }
1350
1351 if (pterrain->plant_result != T_NONE
1352 && action_id_univs_not_blocking(ACTION_PLANT, NULL, &for_terr)) {
1353 fc_snprintf(buf, sizeof(buf),
1354 PL_("%d turn", "%d turns", pterrain->plant_time),
1355 pterrain->plant_time);
1356 add_act_help_for_terrain(_("Plant Rslt/Time"),
1358 HELP_TERRAIN, buf);
1359 }
1360
1361 if (pterrain->transform_result != T_NONE
1362 && action_id_univs_not_blocking(ACTION_TRANSFORM_TERRAIN,
1363 NULL, &for_terr)) {
1364 fc_snprintf(buf, sizeof(buf),
1365 PL_("%d turn", "%d turns", pterrain->transform_time),
1366 pterrain->transform_time);
1367 add_act_help_for_terrain(_("Trans. Rslt/Time"),
1369 HELP_TERRAIN, buf);
1370 }
1371
1372 if (action_id_univs_not_blocking(ACTION_IRRIGATE, NULL, &for_terr)) {
1373 help_extras_of_act_for_terrain(pterrain, ACTIVITY_IRRIGATE, _("Build as irrigation"));
1374 }
1375 if (action_id_univs_not_blocking(ACTION_MINE, NULL, &for_terr)) {
1376 help_extras_of_act_for_terrain(pterrain, ACTIVITY_MINE, _("Build as mine"));
1377 }
1378 if (pterrain->road_time != 0
1379 && action_id_univs_not_blocking(ACTION_ROAD, NULL, &for_terr)) {
1380 help_extras_of_act_for_terrain(pterrain, ACTIVITY_GEN_ROAD, _("Build as road"));
1381 }
1382 if (pterrain->base_time != 0
1383 && action_id_univs_not_blocking(ACTION_BASE, NULL, &for_terr)) {
1384 help_extras_of_act_for_terrain(pterrain, ACTIVITY_BASE, _("Build as base"));
1385 }
1386 gtk_widget_show(help_vbox);
1387 }
1388
1389 helptext_terrain(buf, sizeof(buf), client.conn.playing, pitem->text, pterrain);
1390
1391 gtk_text_buffer_set_text(help_text, buf, -1);
1392 gtk_widget_show(help_text_sw);
1393
1394 gtk_widget_show(help_ttable);
1395}
1396
1397/**********************************************************************/
1400static void help_update_extra(const struct help_item *pitem, char *title)
1401{
1402 char buf[8192];
1404
1405 buf[0] = '\0';
1406 if (pextra == NULL) {
1407 strcat(buf, pitem->text);
1408 } else {
1409 struct road_type *proad = extra_road_get(pextra);
1410 bool is_resource = is_extra_caused_by(pextra, EC_RESOURCE);
1411
1413
1414 /* Cost to build */
1415 if (pextra->buildable) {
1416 if (pextra->build_time != 0) {
1417 /* TRANS: "MP" = movement points */
1418 sprintf(buf, _("%d MP"), pextra->build_time);
1419 } else {
1420 /* TRANS: Build time depends on terrain. */
1421 sprintf(buf, _("Terrain specific"));
1422 }
1423 } else {
1424 sprintf(buf, "-");
1425 }
1426 gtk_label_set_text(GTK_LABEL(help_elabel[1]), buf);
1427 /* Conflicting extras */
1428 buf[0] = '\0';
1429 if (is_resource) {
1430 /* TRANS: (Resource extra) Conflicts with: */
1431 strcat(buf, _("Other Resources"));
1432 }
1433 extra_type_iterate(pextra2) {
1434 if (!can_extras_coexist(pextra, pextra2)
1435 && (!is_resource || !is_extra_caused_by(pextra2, EC_RESOURCE))) {
1436 if (buf[0] != '\0') {
1437 strcat(buf, "/");
1438 }
1439 strcat(buf, extra_name_translation(pextra2));
1440 }
1442 /* TRANS: "Conflicts with: (none)" (extras) */
1443 gtk_label_set_text(GTK_LABEL(help_elabel[3]), buf[0] ? buf : _("(none)"));
1444
1445 /* Bonus */
1446 if (proad != NULL) {
1447 const char *bonus = NULL;
1448
1450 if (proad->tile_incr[o] > 0) {
1451 /* TRANS: Road bonus depends on terrain. */
1452 bonus = _("Terrain specific");
1453 break;
1454 }
1456 if (!bonus) {
1457 bonus = helptext_road_bonus_str(NULL, proad);
1458 }
1459 if (!bonus) {
1460 /* TRANS: No output bonus from a road */
1461 bonus = Q_("?bonus:None");
1462 }
1463 gtk_label_set_text(GTK_LABEL(help_elabel[5]), bonus);
1464 gtk_widget_set_visible(GTK_WIDGET(help_elabel[4]), TRUE);
1465 gtk_widget_set_visible(GTK_WIDGET(help_elabel[5]), TRUE);
1466 } else {
1467 gtk_widget_set_visible(GTK_WIDGET(help_elabel[4]), FALSE);
1468 gtk_widget_set_visible(GTK_WIDGET(help_elabel[5]), FALSE);
1469 }
1470
1471 helptext_extra(buf, sizeof(buf), client.conn.playing, pitem->text, pextra);
1472 }
1473 gtk_widget_show(help_etable);
1474
1475 gtk_text_buffer_set_text(help_text, buf, -1);
1476 gtk_widget_show(help_text_sw);
1477}
1478
1479/**********************************************************************/
1482static void help_update_goods(const struct help_item *pitem,
1483 char *title)
1484{
1485 char buf[8192];
1486 struct goods_type *pgood = goods_by_translated_name(title);
1487
1488 if (!pgood) {
1489 strcat(buf, pitem->text);
1490 } else {
1491 helptext_goods(buf, sizeof(buf), client.conn.playing, pitem->text,
1492 pgood);
1493 }
1494
1495 gtk_text_buffer_set_text(help_text, buf, -1);
1496 gtk_widget_show(help_text_sw);
1497}
1498
1499/**********************************************************************/
1502static void help_update_specialist(const struct help_item *pitem,
1503 char *title)
1504{
1505 char buf[8192];
1507
1508 if (!pspec) {
1509 strcat(buf, pitem->text);
1510 } else {
1511 helptext_specialist(buf, sizeof(buf), client.conn.playing, pitem->text,
1512 pspec);
1513 }
1514
1515 gtk_text_buffer_set_text(help_text, buf, -1);
1516 gtk_widget_show(help_text_sw);
1517}
1518
1519/**********************************************************************/
1522static void help_update_government(const struct help_item *pitem,
1523 char *title)
1524{
1525 char buf[8192];
1527
1528 if (!gov) {
1529 strcat(buf, pitem->text);
1530 } else {
1531 helptext_government(buf, sizeof(buf), client.conn.playing, pitem->text, gov);
1532 }
1533
1534 gtk_text_buffer_set_text(help_text, buf, -1);
1535 gtk_widget_show(help_text_sw);
1536}
1537
1538/**********************************************************************/
1541static void help_update_nation(const struct help_item *pitem, char *title,
1542 struct nation_type *pnation)
1543{
1544 char buf[4096];
1545
1546 if (!pnation) {
1547 strcat(buf, pitem->text);
1548 } else {
1549 helptext_nation(buf, sizeof(buf), pnation, pitem->text);
1550
1552 }
1553
1554 gtk_text_buffer_set_text(help_text, buf, -1);
1555 gtk_widget_show(help_text_sw);
1556}
1557
1558/**********************************************************************/
1561static void help_update_dialog(const struct help_item *pitem)
1562{
1563 char *top;
1564
1565 /* figure out what kind of item is required for pitem ingo */
1566
1567 for (top = pitem->topic; *top == ' '; top++) {
1568 /* nothing */
1569 }
1570
1571 help_box_hide();
1572 gtk_text_buffer_set_text(help_text, "", -1);
1573
1574 switch (pitem->type) {
1575 case HELP_IMPROVEMENT:
1576 help_update_improvement(pitem, top);
1577 break;
1578 case HELP_WONDER:
1579 help_update_wonder(pitem, top);
1580 break;
1581 case HELP_UNIT:
1582 help_update_unit_type(pitem, top);
1583 break;
1584 case HELP_TECH:
1585 help_update_tech(pitem, top);
1586 break;
1587 case HELP_TERRAIN:
1588 help_update_terrain(pitem, top);
1589 break;
1590 case HELP_EXTRA:
1591 help_update_extra(pitem, top);
1592 break;
1593 case HELP_GOODS:
1594 help_update_goods(pitem, top);
1595 break;
1596 case HELP_SPECIALIST:
1597 help_update_specialist(pitem, top);
1598 break;
1599 case HELP_GOVERNMENT:
1600 help_update_government(pitem, top);
1601 break;
1602 case HELP_NATIONS:
1604 break;
1605 case HELP_TEXT:
1606 default:
1607 /* it was a pure text item */
1608 gtk_text_buffer_set_text(help_text, pitem->text, -1);
1609 gtk_widget_show(help_text_sw);
1610 break;
1611 }
1612 set_title_topic(pitem->topic);
1613
1614 gtk_widget_show(help_box);
1615}
1616
1617/**********************************************************************/
1620static void help_item_zoom(GtkTreePath *path)
1621{
1622 GtkTreeModel *model;
1623 GtkTreeIter it, child, item;
1624 GtkTreeSelection *selection;
1625
1626 model = gtk_tree_view_get_model(GTK_TREE_VIEW(help_view));
1627 gtk_tree_model_get_iter(model, &item, path);
1628
1629 for (child=item; gtk_tree_model_iter_parent(model, &it, &child); child=it) {
1630 GtkTreePath *it_path;
1631
1632 it_path = gtk_tree_model_get_path(model, &it);
1633 gtk_tree_view_expand_row(GTK_TREE_VIEW(help_view), it_path, TRUE);
1634 gtk_tree_path_free(it_path);
1635 }
1636
1637 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(help_view));
1638 gtk_tree_selection_select_iter(selection, &item);
1639 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(help_view), path, NULL,
1640 TRUE, 0.0, 0.0);
1641}
1642
1643/**********************************************************************/
1646static GtkTreePath *help_item_path(const struct help_item *pitem)
1647{
1648 GtkTreePath *path;
1649 bool next;
1650
1651 path = gtk_tree_path_new_first();
1652 next = FALSE;
1653 help_items_iterate(pitem2) {
1654 const char *s;
1655 int depth;
1656
1657 for (s = pitem2->topic; *s == ' '; s++) {
1658 /* nothing */
1659 }
1660 depth = s - pitem2->topic + 1;
1661
1662 while (depth < gtk_tree_path_get_depth(path)) {
1663 gtk_tree_path_up(path);
1664 gtk_tree_path_next(path);
1665 next = FALSE;
1666 }
1667 while (depth > gtk_tree_path_get_depth(path)) {
1668 gtk_tree_path_down(path);
1669 next = FALSE;
1670 }
1671
1672 if (next) {
1673 gtk_tree_path_next(path);
1674 }
1675
1676 if (pitem == pitem2)
1677 break;
1678
1679 next = TRUE;
1681
1682 return path;
1683}
1684
1685/**********************************************************************/
1688static void select_help_item_string(const char *item, enum help_page_type htype)
1689{
1690 const struct help_item *pitem;
1691 int idx;
1692 GtkTreePath *path;
1693 GtkTreeViewColumn *col;
1694
1695 if (!(pitem = get_help_item_spec(item, htype, &idx))) {
1696 return;
1697 }
1698
1699 path = help_item_path(pitem);
1700 help_item_zoom(path);
1701
1702 col = gtk_tree_view_get_column(GTK_TREE_VIEW(help_view), 0);
1703 gtk_tree_view_set_cursor(GTK_TREE_VIEW(help_view), path, col, FALSE);
1704 gtk_tree_path_free(path);
1705}
1706
1707/**********************************************************************/
1710static void help_command_update(void)
1711{
1712 GtkDialog *dialog = GTK_DIALOG(help_dialog_shell);
1713
1714 if (help_history_pos < 0) {
1715 gtk_dialog_set_response_sensitive(dialog, 1, FALSE);
1716 gtk_dialog_set_response_sensitive(dialog, 2, FALSE);
1717 } else {
1718 gtk_dialog_set_response_sensitive(dialog, 1, TRUE);
1719 gtk_dialog_set_response_sensitive(dialog, 2, TRUE);
1720
1721 if (help_history_pos == 0) {
1722 gtk_dialog_set_response_sensitive(dialog, 1, FALSE);
1723 }
1724 if (help_history_pos >= help_history->len - 1) {
1725 gtk_dialog_set_response_sensitive(dialog, 2, FALSE);
1726 }
1727 }
1728}
1729
1730/**********************************************************************/
1733static void help_command_callback(GtkWidget *w, gint response_id)
1734{
1735 GtkTreePath *path;
1736 const struct help_item *pitem;
1737
1738 if (response_id == 1) {
1739 if (help_history_pos > 0) {
1741
1742 pitem = g_ptr_array_index(help_history, help_history_pos);
1743 path = help_item_path(pitem);
1744 help_item_zoom(path);
1745 help_update_dialog(pitem);
1747 }
1748 } else if (response_id == 2) {
1749 if (help_history_pos < help_history->len - 1) {
1751
1752 pitem = g_ptr_array_index(help_history, help_history_pos);
1753 path = help_item_path(pitem);
1754 help_item_zoom(path);
1755 help_update_dialog(pitem);
1757 }
1758 } else {
1759 /* Save size of the dialog. */
1760 help_frame_width = gtk_widget_get_allocated_width(help_frame);
1761 help_frame_height = gtk_widget_get_allocated_height(help_frame);
1762 gtk_widget_destroy(help_dialog_shell);
1763 }
1764}
#define action_id_univs_not_blocking(act_id, act_uni, tgt_uni)
Definition actions.h:964
#define str
Definition astring.c:76
#define output_type_iterate(output)
Definition city.h:821
#define output_type_iterate_end
Definition city.h:827
struct civclient client
#define client_player()
struct color * get_color(const struct tileset *t, enum color_std stdcolor)
struct extra_type * extra_type_by_translated_name(const char *name)
Definition extras.c:227
enum extra_cause activity_to_extra_cause(enum unit_activity act)
Definition extras.c:1028
bool can_extras_coexist(const struct extra_type *pextra1, const struct extra_type *pextra2)
Definition extras.c:952
const char * extra_name_translation(const struct extra_type *pextra)
Definition extras.c:186
#define extra_type_iterate(_p)
Definition extras.h:291
#define extra_type_iterate_end
Definition extras.h:297
#define is_extra_caused_by(e, c)
Definition extras.h:196
#define extra_road_get(_e_)
Definition extras.h:185
#define extra_type_by_cause_iterate_end
Definition extras.h:315
#define extra_type_by_cause_iterate(_cause, _extra)
Definition extras.h:309
@ O_SHIELD
Definition fc_types.h:91
@ O_FOOD
Definition fc_types.h:91
@ O_TRADE
Definition fc_types.h:91
const char * skip_intl_qualifier_prefix(const char *str)
Definition fcintl.c:48
#define Q_(String)
Definition fcintl.h:70
#define PL_(String1, String2, n)
Definition fcintl.h:71
#define _(String)
Definition fcintl.h:67
#define N_(String)
Definition fcintl.h:69
const char * government_name_translation(const struct government *pgovern)
Definition government.c:142
struct government * government_by_translated_name(const char *name)
Definition government.c:39
#define governments_iterate(NAME_pgov)
Definition government.h:121
#define governments_iterate_end
Definition government.h:124
#define FC_STATIC_CANVAS_INIT
Definition canvas.h:27
GtkWidget * toplevel
Definition gui_main.c:124
void set_relative_window_position(GtkWindow *ref, GtkWindow *w, int px, int py)
Definition gui_stuff.c:58
void setup_dialog(GtkWidget *shell, GtkWidget *parent)
Definition gui_stuff.c:281
static GtkWidget * help_tree_sw
Definition helpdlg.c:82
static void set_title_topic(char *topic)
Definition helpdlg.c:172
static void activated_topic(GtkTreeView *view, gpointer data)
Definition helpdlg.c:467
static GtkWidget * help_tree
Definition helpdlg.c:79
static GtkWidget * help_tree_expand
Definition helpdlg.c:83
static GtkWidget * help_ulabel[5][5]
Definition helpdlg.c:88
static void help_update_specialist(const struct help_item *pitem, char *title)
Definition helpdlg.c:1498
static void help_hyperlink_callback(GtkWidget *w)
Definition helpdlg.c:343
static GtkWidget * help_elabel[6]
Definition helpdlg.c:90
static GtkWidget * help_etable
Definition helpdlg.c:78
static void help_destroy_callback(GtkWidget *w, gpointer data)
Definition helpdlg.c:458
static void set_help_tile_from_extra(const struct extra_type *pextra)
Definition helpdlg.c:848
static GtkWidget * help_frame
Definition helpdlg.c:68
static void help_tech_tree_activated_callback(GtkTreeView *view, GtkTreePath *path, GtkTreeViewColumn *col, gpointer data)
Definition helpdlg.c:310
static void help_update_tech(const struct help_item *pitem, char *title)
Definition helpdlg.c:1076
static void help_item_zoom(GtkTreePath *path)
Definition helpdlg.c:1616
static GtkWidget * help_slink_new_page(const gchar *txt, enum help_page_type page)
Definition helpdlg.c:425
static GtkWidget * help_tlabel[2][5]
Definition helpdlg.c:89
static GtkTreePath * help_item_path(const struct help_item *pitem)
Definition helpdlg.c:1642
static GtkWidget * help_tree_collapse
Definition helpdlg.c:84
static void help_update_nation(const struct help_item *pitem, char *title, struct nation_type *pnation)
Definition helpdlg.c:1537
static void create_help_dialog(void)
Definition helpdlg.c:508
static GtkTextBuffer * help_text
Definition helpdlg.c:69
static void help_update_wonder(const struct help_item *pitem, char *title)
Definition helpdlg.c:926
static GtkWidget * help_hyperlink_new_page(GtkWidget *label, enum help_page_type page)
Definition helpdlg.c:400
static GtkWidget * help_dialog_shell
Definition helpdlg.c:63
static GtkWidget * help_text_sw
Definition helpdlg.c:70
static GPtrArray * help_history
Definition helpdlg.c:94
static const char * help_elabel_name[6]
Definition helpdlg.c:131
static void create_tech_tree(int tech, int levels, GtkTreeIter *parent)
Definition helpdlg.c:225
static void help_update_extra(const struct help_item *pitem, char *title)
Definition helpdlg.c:1399
static GtkWidget * help_ilabel[6]
Definition helpdlg.c:86
static void set_help_tile_from_sprite(struct sprite *spr)
Definition helpdlg.c:804
static const char * help_wlabel_name[6]
Definition helpdlg.c:110
static const char * help_ulabel_name[5][5]
Definition helpdlg.c:116
static void help_tech_tree_collapse_callback(GtkWidget *w, gpointer data)
Definition helpdlg.c:335
static void set_help_tile_from_terrain(struct terrain *pterr)
Definition helpdlg.c:817
static GtkWidget * help_view
Definition helpdlg.c:66
static void help_update_goods(const struct help_item *pitem, char *title)
Definition helpdlg.c:1478
void help_system_init(void)
Definition helpdlg.c:156
static void help_box_hide(void)
Definition helpdlg.c:434
static void help_update_unit_type(const struct help_item *pitem, char *title)
Definition helpdlg.c:990
void popdown_help_dialog(void)
Definition helpdlg.c:185
static void help_update_terrain(const struct help_item *pitem, char *title)
Definition helpdlg.c:1288
static void help_update_dialog(const struct help_item *pitem)
Definition helpdlg.c:1557
static char * fc_chomp(char *str, size_t len)
Definition helpdlg.c:1057
void popup_help_dialog_string(const char *item)
Definition helpdlg.c:211
static int help_history_pos
Definition helpdlg.c:95
static struct help_page_selection help_wndr_req
Definition helpdlg.c:112
static GtkWidget * help_wlabel[6]
Definition helpdlg.c:87
static GtkWidget * help_box
Definition helpdlg.c:73
static void help_extras_of_act_for_terrain(struct terrain *pterr, enum unit_activity act, char *label)
Definition helpdlg.c:1268
void popup_help_dialog_typed(const char *item, enum help_page_type htype)
Definition helpdlg.c:195
#define REQ_LABEL_NEVER
Definition helpdlg.c:143
static void help_command_update(void)
Definition helpdlg.c:1706
static struct help_page_selection page_selections[HELP_LAST]
Definition helpdlg.c:114
#define TECH_TREE_DEPTH
Definition helpdlg.c:58
static const char * help_tlabel_name[2][5]
Definition helpdlg.c:125
static struct help_page_selection help_impr_req
Definition helpdlg.c:108
#define REQ_LABEL_NONE
Definition helpdlg.c:142
static GtkWidget * help_wtable
Definition helpdlg.c:75
static GtkWidget * help_slink_new(const gchar *txt, struct help_page_selection *select)
Definition helpdlg.c:409
static GtkWidget * help_vbox
Definition helpdlg.c:71
static void help_tech_tree_expand_callback(GtkWidget *w, gpointer data)
Definition helpdlg.c:327
static GtkWidget * help_tile
Definition helpdlg.c:72
static GtkWidget * help_hyperlink_new(GtkWidget *label, struct help_page_selection *select)
Definition helpdlg.c:378
static void help_update_government(const struct help_item *pitem, char *title)
Definition helpdlg.c:1518
static GtkWidget * help_ttable
Definition helpdlg.c:77
static void help_update_improvement(const struct help_item *pitem, char *title)
Definition helpdlg.c:876
static const char * help_ilabel_name[6]
Definition helpdlg.c:106
static GtkWidget * help_tree_buttons_hbox
Definition helpdlg.c:85
static GtkWidget * help_view_sw
Definition helpdlg.c:64
static void select_help_item_string(const char *item, enum help_page_type htype)
Definition helpdlg.c:1684
static bool help_advances[A_LAST]
Definition helpdlg.c:92
static GtkWidget * help_utable
Definition helpdlg.c:76
static GtkWidget * help_itable
Definition helpdlg.c:74
static GtkTreeStore * tstore
Definition helpdlg.c:80
static void add_act_help_for_terrain(const char *act_label, const char *result_link_label, enum help_page_type result_link_type, const char *descr_label)
Definition helpdlg.c:1245
static void help_command_callback(GtkWidget *w, gint response_id)
Definition helpdlg.c:1729
const char * title
Definition repodlgs.c:1313
GType type
Definition repodlgs.c:1312
static int help_frame_width
Definition helpdlg.c:69
static int help_frame_height
Definition helpdlg.c:69
void helptext_government(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, struct government *gov)
Definition helpdata.c:4235
void helptext_advance(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, int i)
Definition helpdata.c:3175
enum help_page_type help_type_by_requirement(const struct requirement *req)
Definition helpdata.c:5097
void helptext_extra(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, struct extra_type *pextra)
Definition helpdata.c:3679
void helptext_goods(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, struct goods_type *pgood)
Definition helpdata.c:4154
const char * helptext_road_bonus_str(const struct terrain *pterrain, const struct road_type *proad)
Definition helpdata.c:3515
char * helptext_unit_upkeep_str(const struct unit_type *utype)
Definition helpdata.c:4894
const char * helptext_extra_for_terrain_str(struct extra_type *pextra, struct terrain *pterrain, enum unit_activity act)
Definition helpdata.c:3645
void helptext_specialist(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, struct specialist *pspec)
Definition helpdata.c:4200
const struct help_item * get_help_item_spec(const char *name, enum help_page_type htype, int *pos)
Definition helpdata.c:1281
char * helptext_building(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, const struct impr_type *pimprove)
Definition helpdata.c:1374
void helptext_terrain(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, struct terrain *pterrain)
Definition helpdata.c:3398
char * helptext_unit(char *buf, size_t bufsz, struct player *pplayer, const char *user_text, const struct unit_type *utype)
Definition helpdata.c:1823
void helptext_nation(char *buf, size_t bufsz, struct nation_type *pnation, const char *user_text)
Definition helpdata.c:4931
#define help_items_iterate(pitem)
Definition helpdata.h:70
#define help_items_iterate_end
Definition helpdata.h:74
#define HELP_ABOUT_ITEM
help_page_type
Definition helpdlg_g.h:20
@ HELP_ANY
Definition helpdlg_g.h:20
@ HELP_TERRAIN
Definition helpdlg_g.h:21
@ HELP_EXTRA
Definition helpdlg_g.h:21
@ HELP_NATIONS
Definition helpdlg_g.h:23
@ HELP_LAST
Definition helpdlg_g.h:24
@ HELP_IMPROVEMENT
Definition helpdlg_g.h:20
@ HELP_UNIT
Definition helpdlg_g.h:20
@ HELP_SPECIALIST
Definition helpdlg_g.h:22
@ HELP_GOVERNMENT
Definition helpdlg_g.h:22
@ HELP_GOODS
Definition helpdlg_g.h:22
@ HELP_WONDER
Definition helpdlg_g.h:21
@ HELP_TECH
Definition helpdlg_g.h:21
@ HELP_TEXT
Definition helpdlg_g.h:20
const struct impr_type * valid_improvement(const struct impr_type *pimprove)
int impr_base_build_shield_cost(const struct impr_type *pimprove)
bool is_great_wonder(const struct impr_type *pimprove)
struct impr_type * improvement_by_translated_name(const char *name)
const char * improvement_name_translation(const struct impr_type *pimprove)
#define improvement_iterate_end
#define improvement_iterate(_p)
void put_drawn_sprites(struct canvas *pcanvas, float zoom, int canvas_x, int canvas_y, int count, struct drawn_sprite *pdrawn, bool fog)
const char * move_points_text(int mp, bool reduce)
Definition movement.c:1005
struct nation_type * nation_by_translated_plural(const char *name)
Definition nation.c:105
int len
Definition packhand.c:125
const char * universal_name_translation(const struct universal *psource, char *buf, size_t bufsz)
#define requirement_fulfilled_by_terrain(_ter_, _rqs_)
#define requirement_vector_iterate_end
#define requirement_vector_iterate(req_vec, preq)
int research_goal_unknown_techs(const struct research *presearch, Tech_type_id goal)
Definition research.c:747
const char * research_advance_name_translation(const struct research *presearch, Tech_type_id tech)
Definition research.c:271
struct research * research_get(const struct player *pplayer)
Definition research.c:126
enum tech_state research_invention_state(const struct research *presearch, Tech_type_id tech)
Definition research.c:616
#define ARRAY_SIZE(x)
Definition shared.h:85
struct specialist * specialist_by_translated_name(const char *name)
Definition specialist.c:130
cairo_surface_t * surface
Definition canvas.h:22
struct connection conn
Definition client_main.h:96
Definition colors.h:20
struct player * playing
Definition connection.h:156
bool buildable
Definition extras.h:112
int build_time
Definition extras.h:114
char * topic
Definition helpdata.h:26
enum help_page_type type
Definition helpdata.h:27
char * text
Definition helpdata.h:26
enum help_page_selection::@152 type
enum help_page_type page
Definition helpdlg.c:101
const struct requirement * req
Definition helpdlg.c:102
union help_page_selection::@153 u
struct requirement_vector obsolete_by
Definition improvement.h:76
struct requirement_vector reqs
Definition improvement.h:75
Definition climisc.h:82
int tile_incr[O_LAST]
Definition road.h:81
cairo_surface_t * surface
Definition sprite.h:23
struct terrain * cultivate_result
Definition terrain.h:203
struct extra_type ** resources
Definition terrain.h:197
int road_time
Definition terrain.h:201
int plant_time
Definition terrain.h:207
struct terrain * plant_result
Definition terrain.h:206
int defense_bonus
Definition terrain.h:193
int cultivate_time
Definition terrain.h:204
int movement_cost
Definition terrain.h:192
int output[O_LAST]
Definition terrain.h:195
int transform_time
Definition terrain.h:218
struct terrain * transform_result
Definition terrain.h:217
int base_time
Definition terrain.h:200
int defense_strength
Definition unittype.h:496
int firepower
Definition unittype.h:506
const struct unit_type * obsoleted_by
Definition unittype.h:510
int vision_radius_sq
Definition unittype.h:503
int move_rate
Definition unittype.h:497
struct advance * require_advance
Definition unittype.h:500
int attack_strength
Definition unittype.h:495
enum universals_n kind
Definition fc_types.h:758
int fc_snprintf(char *str, size_t n, const char *format,...)
Definition support.c:969
#define TRUE
Definition support.h:46
#define FALSE
Definition support.h:47
struct advance * advance_by_number(const Tech_type_id atype)
Definition tech.c:107
bool is_future_tech(Tech_type_id tech)
Definition tech.c:281
struct advance * advance_by_translated_name(const char *name)
Definition tech.c:185
struct advance * advance_requires(const struct advance *padvance, enum tech_req require)
Definition tech.c:136
const char * advance_name_translation(const struct advance *padvance)
Definition tech.c:290
Tech_type_id advance_required(const Tech_type_id tech, enum tech_req require)
Definition tech.c:121
Tech_type_id advance_number(const struct advance *padvance)
Definition tech.c:98
#define A_NEVER
Definition tech.h:51
@ AR_TWO
Definition tech.h:112
@ AR_ONE
Definition tech.h:111
#define advance_iterate(_start, _p)
Definition tech.h:264
#define A_NONE
Definition tech.h:43
#define advance_iterate_end
Definition tech.h:270
#define A_LAST
Definition tech.h:45
struct terrain * terrain_by_translated_name(const char *name)
Definition terrain.c:190
const char * terrain_name_translation(const struct terrain *pterrain)
Definition terrain.c:226
#define T_NONE
Definition terrain.h:56
struct sprite * get_building_sprite(const struct tileset *t, const struct impr_type *pimprove)
Definition tilespec.c:6495
struct sprite * get_unittype_sprite(const struct tileset *t, const struct unit_type *punittype, enum direction8 facing)
Definition tilespec.c:6517
int fill_basic_extra_sprite_array(const struct tileset *t, struct drawn_sprite *sprs, const struct extra_type *pextra)
Definition tilespec.c:6898
int fill_basic_terrain_layer_sprite_array(struct tileset *t, struct drawn_sprite *sprs, int layer, struct terrain *pterrain)
Definition tilespec.c:6865
int tileset_tile_height(const struct tileset *t)
Definition tilespec.c:728
struct sprite * get_nation_flag_sprite(const struct tileset *t, const struct nation_type *pnation)
Definition tilespec.c:6468
int tileset_tile_width(const struct tileset *t)
Definition tilespec.c:716
struct sprite * get_tech_sprite(const struct tileset *t, Tech_type_id tech)
Definition tilespec.c:6486
struct goods_type * goods_by_translated_name(const char *name)
int utype_build_shield_cost_base(const struct unit_type *punittype)
Definition unittype.c:1520
struct unit_type * unit_type_by_translated_name(const char *name)
Definition unittype.c:1804
const char * utype_name_translation(const struct unit_type *punittype)
Definition unittype.c:1612
#define unit_type_iterate(_p)
Definition unittype.h:841
#define unit_type_iterate_end
Definition unittype.h:848
#define U_NOT_OBSOLETED
Definition unittype.h:509
const char * freeciv_name_version(void)
Definition version.c:35