From f681b6891f21e3e801b9149d1917913ca681542d Mon Sep 17 00:00:00 2001 From: Marko Lindqvist Date: Mon, 11 Oct 2021 23:09:48 +0300 Subject: [PATCH 34/47] gtk4: Port infradlg from gtk3.22-client See osdn #42982 Signed-off-by: Marko Lindqvist --- client/gui-gtk-4.0/Makefile.am | 1 + client/gui-gtk-4.0/infradlg.c | 170 +++++++++++++++++++++++++++++++++ client/gui-gtk-4.0/infradlg.h | 20 ++++ client/gui-gtk-4.0/mapctrl.c | 3 + client/gui-gtk-4.0/menu.c | 13 +++ data/gtk4_menus.xml | 3 + 6 files changed, 210 insertions(+) create mode 100644 client/gui-gtk-4.0/infradlg.h diff --git a/client/gui-gtk-4.0/Makefile.am b/client/gui-gtk-4.0/Makefile.am index 84fc657c2a..2ef9b4ad82 100644 --- a/client/gui-gtk-4.0/Makefile.am +++ b/client/gui-gtk-4.0/Makefile.am @@ -61,6 +61,7 @@ libgui_gtk3x_la_SOURCES = \ helpdlg.c \ helpdlg.h \ infradlg.c \ + infradlg.h \ inputdlg.c \ inputdlg.h \ inteldlg.c \ diff --git a/client/gui-gtk-4.0/infradlg.c b/client/gui-gtk-4.0/infradlg.c index 1f0ec80992..bc779d1259 100644 --- a/client/gui-gtk-4.0/infradlg.c +++ b/client/gui-gtk-4.0/infradlg.c @@ -15,12 +15,182 @@ #include #endif +#include + +/* utility */ +#include "fcintl.h" + +/* common */ +#include "extras.h" +#include "game.h" + /* client */ +#include "client_main.h" #include "dialogs_g.h" +#include "mapview_common.h" + +/* client/gui-gtk-4.0 */ +#include "gui_main.h" +#include "gui_stuff.h" + +#include "infradlg.h" + +static GtkWidget *infra_list_grid = NULL; +static GtkWidget *instruction_label = NULL; +static GtkWidget *points_label = NULL; +static int infra_rows = 0; + +struct infra_cb_data { + struct tile *ptile; + struct extra_type *pextra; +}; + +/************************************************************************//** + Is infra dialog currently open? +****************************************************************************/ +static bool infra_dialog_open(void) +{ + return infra_list_grid != NULL; +} + +/************************************************************************//** + Handle infra dialog closing. +****************************************************************************/ +static void infra_response_callback(GtkWidget *dlg, gint arg) +{ + infra_list_grid = NULL; + instruction_label = NULL; + points_label = NULL; + infra_rows = 0; + + client_infratile_set(NULL); + + gtk_widget_destroy(dlg); +} + +/************************************************************************//** + Handle user infra selection. +****************************************************************************/ +static void infra_selected_callback(GtkButton *but, gpointer userdata) +{ + struct infra_cb_data *cbdata = (struct infra_cb_data *)userdata; + + dsend_packet_player_place_infra(&client.conn, cbdata->ptile->index, + cbdata->pextra->id); +} + +/************************************************************************//** + Open infra placement dialog +****************************************************************************/ +void infra_dialog_popup(void) +{ + GtkWidget *dlg; + GtkWidget *main_box; + GtkWidget *sep; + + if (infra_dialog_open()) { + /* One infra dialog already open. */ + return; + } + + dlg = gtk_dialog_new_with_buttons(_("Place infrastructure"), NULL, 0, + _("Close"), GTK_RESPONSE_NO, + NULL); + + setup_dialog(dlg, toplevel); + gtk_dialog_set_default_response(GTK_DIALOG(dlg), GTK_RESPONSE_NO); + gtk_window_set_destroy_with_parent(GTK_WINDOW(dlg), TRUE); + + main_box = gtk_grid_new(); + gtk_orientable_set_orientation(GTK_ORIENTABLE(main_box), + GTK_ORIENTATION_VERTICAL); + + instruction_label = gtk_label_new(_("First click a tile.")); + gtk_container_add(GTK_CONTAINER(main_box), instruction_label); + + sep = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL); + gtk_container_add(GTK_CONTAINER(main_box), sep); + + points_label = gtk_label_new(_("- infrapoints")); + gtk_container_add(GTK_CONTAINER(main_box), points_label); + + sep = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL); + gtk_container_add(GTK_CONTAINER(main_box), sep); + + infra_list_grid = gtk_grid_new(); + gtk_container_add(GTK_CONTAINER(main_box), infra_list_grid); + + gtk_container_add(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dlg))), + main_box); + + g_signal_connect(dlg, "destroy", G_CALLBACK(infra_response_callback), NULL); + g_signal_connect(dlg, "response", G_CALLBACK(infra_response_callback), NULL); + + gtk_widget_show(gtk_dialog_get_content_area(GTK_DIALOG(dlg))); + gtk_widget_show(dlg); + + update_infra_dialog(); +} /************************************************************************//** Refresh infra dialog ****************************************************************************/ void update_infra_dialog(void) { + if (infra_dialog_open()) { + char buffer[100]; + + fc_snprintf(buffer, sizeof(buffer), _("%d infrapoints"), + client.conn.playing->economic.infra_points); + + gtk_label_set_text(GTK_LABEL(points_label), buffer); + } +} + +/************************************************************************//** + Are we in infra placement mode at the moment? +****************************************************************************/ +bool infra_placement_mode(void) +{ + return infra_list_grid != NULL; +} + +/************************************************************************//** + Set tile for the infra placement. +****************************************************************************/ +void infra_placement_set_tile(struct tile *ptile) +{ + while (infra_rows > 0) { + gtk_grid_remove_row(GTK_GRID(infra_list_grid), --infra_rows); + } + + if (!client_map_is_known_and_seen(ptile, client.conn.playing, V_MAIN)) { + return; + } + + client_infratile_set(ptile); + + extra_type_iterate(pextra) { + if (player_can_place_extra(pextra, client.conn.playing, ptile)) { + GtkWidget *but = gtk_button_new_with_label(extra_name_translation(pextra)); + struct infra_cb_data *cbdata = fc_malloc(sizeof(struct infra_cb_data)); + + cbdata->ptile = ptile; + cbdata->pextra = pextra; + + g_signal_connect(but, "clicked", + G_CALLBACK(infra_selected_callback), cbdata); + gtk_grid_attach(GTK_GRID(infra_list_grid), but, 1, infra_rows++, 1, 1); + } + } extra_type_iterate_end; + + if (infra_rows <= 0) { + gtk_label_set_text(GTK_LABEL(instruction_label), + _("No infra possible. Select another tile.")); + } else { + gtk_label_set_text(GTK_LABEL(instruction_label), + _("Select infra for the tile, or another tile.")); + } + + gtk_widget_show(infra_list_grid); } diff --git a/client/gui-gtk-4.0/infradlg.h b/client/gui-gtk-4.0/infradlg.h new file mode 100644 index 0000000000..8f413b07a5 --- /dev/null +++ b/client/gui-gtk-4.0/infradlg.h @@ -0,0 +1,20 @@ +/*********************************************************************** + Freeciv - Copyright (C) 1996-2005 - Freeciv Development Team + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2, or (at your option) + any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. +***********************************************************************/ +#ifndef FC__INFRADLG_H +#define FC__INFRADLG_H + +void infra_dialog_popup(void); +bool infra_placement_mode(void); +void infra_placement_set_tile(struct tile *ptile); + +#endif /* FC__INFRADLG_H */ diff --git a/client/gui-gtk-4.0/mapctrl.c b/client/gui-gtk-4.0/mapctrl.c index f9d9173548..36795d82eb 100644 --- a/client/gui-gtk-4.0/mapctrl.c +++ b/client/gui-gtk-4.0/mapctrl.c @@ -50,6 +50,7 @@ #include "editgui.h" #include "graphics.h" #include "gui_main.h" +#include "infradlg.h" #include "inputdlg.h" #include "mapview.h" #include "menu.h" @@ -229,6 +230,8 @@ gboolean butt_down_mapcanvas(GtkWidget *w, GdkEvent *ev, gpointer data) if (ptile) { toggle_tile_hilite(ptile); } + } else if (infra_placement_mode()) { + infra_placement_set_tile(ptile); } else { /* Plain LMB click. */ action_button_pressed(e_x, e_y, SELECT_POPUP); diff --git a/client/gui-gtk-4.0/menu.c b/client/gui-gtk-4.0/menu.c index 0a9c1de697..4929933e41 100644 --- a/client/gui-gtk-4.0/menu.c +++ b/client/gui-gtk-4.0/menu.c @@ -52,6 +52,7 @@ #include "gui_main.h" #include "gui_stuff.h" #include "helpdlg.h" +#include "infradlg.h" #include "luaconsole.h" #include "mapctrl.h" /* center_on_unit(). */ #include "messagedlg.h" @@ -208,6 +209,7 @@ static void select_same_type_tile_callback(GtkMenuItem *item, gpointer data); static void select_same_type_cont_callback(GtkMenuItem *item, gpointer data); static void select_same_type_callback(GtkMenuItem *item, gpointer data); static void select_dialog_callback(GtkMenuItem *item, gpointer data); +static void infra_dialog_callback(GtkMenuItem *item, gpointer data); static void unit_wait_callback(GtkMenuItem *item, gpointer data); static void unit_done_callback(GtkMenuItem *item, gpointer data); static void unit_goto_callback(GtkMenuItem *item, gpointer data); @@ -285,6 +287,8 @@ static struct menu_entry_info menu_entries[] = G_CALLBACK(find_city_callback), MGROUP_SAFE }, { "WORKLISTS", N_("Work_lists"), GDK_KEY_l, GDK_CONTROL_MASK, G_CALLBACK(worklists_callback), MGROUP_SAFE }, + { "INFRA_DLG", N_("Infra dialog"), GDK_KEY_i, GDK_CONTROL_MASK, + G_CALLBACK(infra_dialog_callback), MGROUP_SAFE }, { "CLIENT_LUA_SCRIPT", N_("Client _Lua Script"), 0, 0, G_CALLBACK(client_lua_script_callback), MGROUP_SAFE }, { "MAP_VIEW", N_("?noun:_View"), GDK_KEY_F1, 0, @@ -1361,6 +1365,14 @@ static void select_dialog_callback(GtkMenuItem *item, gpointer data) unit_select_dialog_popup(NULL); } +/************************************************************************//** + Open infra placement dialog. +****************************************************************************/ +static void infra_dialog_callback(GtkMenuItem *item, gpointer data) +{ + infra_dialog_popup(); +} + /************************************************************************//** Item "UNIT_WAIT" callback. ****************************************************************************/ @@ -2170,6 +2182,7 @@ void real_menus_update(void) menu_entry_set_sensitive("EDIT_MODE", can_conn_enable_editing(&client.conn)); editgui_refresh(); + menu_entry_set_sensitive("INFRA_DLG", terrain_control.infrapoints); { char road_buf[500]; diff --git a/data/gtk4_menus.xml b/data/gtk4_menus.xml index fd2232abee..01529d74c2 100644 --- a/data/gtk4_menus.xml +++ b/data/gtk4_menus.xml @@ -74,6 +74,9 @@ + + + -- 2.33.0