Freeciv-3.1
Loading...
Searching...
No Matches
editgui.c
Go to the documentation of this file.
1/***********************************************************************
2 Freeciv - Copyright (C) 2005 - The Freeciv Project
3 This program is free software; you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 2, or (at your option)
6 any later version.
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 <math.h>
19#include <stdio.h>
20#include <stdlib.h>
21#include <string.h>
22
23#include <gtk/gtk.h>
24#include <gdk/gdkkeysyms.h>
25
26/* utility */
27#include "fcintl.h"
28#include "log.h"
29#include "shared.h"
30#include "support.h"
31
32/* common */
33#include "connection.h"
34#include "game.h"
35#include "government.h"
36#include "packets.h"
37
38/* client */
39#include "chatline_common.h"
40#include "client_main.h"
41#include "dialogs_g.h"
42#include "editor.h"
43#include "mapview_common.h"
44#include "tilespec.h"
45
46/* client/gui-gtk-3.22 */
47#include "canvas.h"
48#include "editgui.h"
49#include "editprop.h"
50#include "gui_main.h"
51#include "gui_stuff.h"
52#include "plrdlg.h"
53#include "sprite.h"
54
55
63
71
79
84
86 struct editbar *editbar_parent;
87
88 GtkWidget *dialog;
89
90 GtkListStore *store;
91 GtkWidget *view;
92};
93
94static struct tool_value_selector *
95create_tool_value_selector(struct editbar *eb_parent,
96 enum editor_tool_type ett);
97static void editinfobox_refresh(struct editinfobox *ei);
98static void editbar_player_pov_combobox_changed(GtkComboBox *combo,
99 gpointer user_data);
100static void editbar_mode_button_toggled(GtkToggleButton *tb,
101 gpointer userdata);
102static void editbar_tool_button_toggled(GtkToggleButton *tb,
103 gpointer userdata);
104static void try_to_set_editor_tool(enum editor_tool_type ett);
105
106static struct editbar *editor_toolbar;
108
109/************************************************************************/
113static void refresh_all_buttons(struct editbar *eb)
114{
115 enum editor_tool_type ett;
116 enum editor_tool_mode etm;
117 GtkWidget *tb = NULL;
118 int i;
119
120 if (eb == NULL) {
121 return;
122 }
123
124 ett = editor_get_tool();
125 etm = editor_tool_get_mode(ett);
126
127 for (i = 0; i < NUM_EDITOR_TOOL_MODES; i++) {
128 tb = eb->mode_buttons[i];
129 if (tb == NULL) {
130 continue;
131 }
132 disable_gobject_callback(G_OBJECT(tb),
133 G_CALLBACK(editbar_mode_button_toggled));
134 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(tb), i == etm);
135 enable_gobject_callback(G_OBJECT(tb),
136 G_CALLBACK(editbar_mode_button_toggled));
137 gtk_widget_set_sensitive(tb, editor_tool_has_mode(ett, i));
138 }
139
140 if (ett < NUM_EDITOR_TOOL_TYPES
141 && eb->tool_buttons[ett] != NULL) {
142 tb = eb->tool_buttons[ett];
143 disable_gobject_callback(G_OBJECT(tb),
144 G_CALLBACK(editbar_tool_button_toggled));
145 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(tb), TRUE);
146 enable_gobject_callback(G_OBJECT(tb),
147 G_CALLBACK(editbar_tool_button_toggled));
148 }
149}
150
151/************************************************************************/
154static void editbar_mode_button_toggled(GtkToggleButton *tb,
155 gpointer userdata)
156{
157 gboolean active;
158 enum editor_tool_mode etm;
159 enum editor_tool_type ett;
160
161 etm = GPOINTER_TO_INT(userdata);
162 if (!(etm < NUM_EDITOR_TOOL_MODES)) {
163 return;
164 }
165
166 active = gtk_toggle_button_get_active(tb);
167 ett = editor_get_tool();
168
169 editor_tool_set_mode(ett, active ? etm : ETM_PAINT);
171}
172
173/************************************************************************/
178{
179 if (!(ett < NUM_EDITOR_TOOL_TYPES)) {
180 return;
181 }
182
183 if (!editor_tool_is_usable(ett)) {
184 GtkWidget *dialog;
185 dialog = gtk_message_dialog_new(GTK_WINDOW(toplevel),
186 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
187 GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "%s",
188 _("The current ruleset does not define any "
189 "objects corresponding to this editor tool."));
190 gtk_window_set_title(GTK_WINDOW(dialog), editor_tool_get_name(ett));
191 gtk_dialog_run(GTK_DIALOG(dialog));
192 gtk_widget_destroy(dialog);
193 } else {
194 editor_set_tool(ett);
195 }
196}
197
198/************************************************************************/
201static void editbar_tool_button_toggled(GtkToggleButton *tb,
202 gpointer userdata)
203{
204 gboolean active;
205 enum editor_tool_type ett;
206
207 active = gtk_toggle_button_get_active(tb);
208 ett = GPOINTER_TO_INT(userdata);
209
210 if (active) {
213 }
214}
215
216/************************************************************************/
224{
225 GtkListStore *store;
226 GdkPixbuf *flag;
227 GtkTreeIter iter;
228 GtkWidget *combo;
229 int index = -1, i;
230
231 if (eb == NULL || eb->player_pov_store == NULL) {
232 return;
233 }
234
235 store = eb->player_pov_store;
236 gtk_list_store_clear(store);
237
238 gtk_list_store_append(store, &iter);
239 gtk_list_store_set(store, &iter, PPV_COL_NAME, _("Global Observer"),
240 PPV_COL_PLAYER_NO, -1, -1);
241
242 i = 1;
243 players_iterate(pplayer) {
244 gtk_list_store_append(store, &iter);
245 gtk_list_store_set(store, &iter, PPV_COL_NAME, player_name(pplayer),
246 PPV_COL_PLAYER_NO, player_number(pplayer), -1);
247
248 if (pplayer->nation != NO_NATION_SELECTED) {
249 flag = get_flag(pplayer->nation);
250
251 if (flag != NULL) {
252 gtk_list_store_set(store, &iter, PPV_COL_FLAG, flag, -1);
253 g_object_unref(flag);
254 }
255 }
256 if (pplayer == client_player()) {
257 index = i;
258 }
259 i++;
261
263 index = 0;
264 }
265
266 combo = eb->player_pov_combobox;
267 gtk_combo_box_set_active(GTK_COMBO_BOX(combo), index);
268}
269
270/************************************************************************/
277static void editbar_player_pov_combobox_changed(GtkComboBox *combo,
278 gpointer user_data)
279{
280 struct editbar *eb = user_data;
281 int id;
282 struct player *pplayer;
283 GtkTreeIter iter;
284 GtkTreeModel *model;
285
286 if (eb == NULL || eb->widget == NULL
287 || !gtk_widget_get_visible(eb->widget)) {
288 return;
289 }
290
291 if (!gtk_combo_box_get_active_iter(combo, &iter)) {
292 return;
293 }
294
295 model = gtk_combo_box_get_model(combo);
296 gtk_tree_model_get(model, &iter, PPV_COL_PLAYER_NO, &id, -1);
297
298 if ((client_is_global_observer() && id == -1)
299 || client_player_number() == id) {
300 return;
301 }
302
303 /* Ugh... hard-coded server command strings. :( */
304 if (id == -1) {
305 send_chat("/observe");
306 return;
307 }
308
309 pplayer = player_by_number(id);
310 if (pplayer != NULL) {
311 send_chat_printf("/take \"%s\"", pplayer->name);
312 }
313}
314
315/************************************************************************/
320{
321 gint res;
322 GtkWidget *dialog;
323 GtkTreeSelection *sel;
324 GtkTreeIter iter;
325 GtkTreeModel *model;
326 int id;
327
328 if (tvs == NULL) {
329 return -1;
330 }
331
332 dialog = tvs->dialog;
333 res = gtk_dialog_run(GTK_DIALOG(dialog));
334 gtk_widget_hide(dialog);
335
336 if (res != GTK_RESPONSE_ACCEPT) {
337 return -1;
338 }
339
340 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tvs->view));
341 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) {
342 return -1;
343 }
344
345 gtk_tree_model_get(model, &iter, TVS_COL_ID, &id, -1);
346
347 return id;
348}
349
350/************************************************************************/
357{
358 struct editbar *eb;
359 struct tool_value_selector *tvs;
360 int res = -1;
361
362 eb = editgui_get_editbar();
363 if (eb == NULL || !(ett < NUM_EDITOR_TOOL_TYPES)) {
364 return FALSE;
365 }
366
367 if (!editor_tool_has_value(ett)) {
368 return FALSE;
369 }
370
371 tvs = eb->tool_selectors[ett];
372 if (!tvs) {
373 return FALSE;
374 }
375
376 res = tool_value_selector_run(tvs);
377
378 if (res >= 0) {
379 editor_tool_set_value(ett, res);
381 }
382
383 return TRUE;
384}
385
386/************************************************************************/
389static gboolean editbar_tool_button_mouse_click(GtkWidget *w,
390 GdkEventButton *ev,
391 gpointer userdata)
392{
393 int ett;
394
395 if (ev->button != 3) {
396 return FALSE; /* Propagate event further. */
397 }
398
399 ett = GPOINTER_TO_INT(userdata);
400 return editgui_run_tool_selection(ett);
401}
402
403/************************************************************************/
407static void editbar_add_tool_button(struct editbar *eb,
408 enum editor_tool_type ett)
409{
410 GdkPixbuf *pixbuf;
411 GtkWidget *image, *button, *hbox;
412 GtkRadioButton *parent = NULL;
413 struct sprite *sprite;
414 int i;
415
416 if (!eb || !(ett < NUM_EDITOR_TOOL_TYPES)) {
417 return;
418 }
419
420 for (i = 0; i < NUM_EDITOR_TOOL_TYPES; i++) {
421 if (eb->tool_buttons[i] != NULL) {
422 parent = GTK_RADIO_BUTTON(eb->tool_buttons[i]);
423 break;
424 }
425 }
426
427 if (parent == NULL) {
428 button = gtk_radio_button_new(NULL);
429 } else {
430 button = gtk_radio_button_new_from_widget(parent);
431 }
432
434 fc_assert_ret(sprite != NULL);
435 pixbuf = sprite_get_pixbuf(sprite);
436 image = gtk_image_new_from_pixbuf(pixbuf);
437 g_object_unref(G_OBJECT(pixbuf));
438
439 gtk_container_add(GTK_CONTAINER(button), image);
440 gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(button), FALSE);
441 gtk_widget_set_tooltip_text(button, editor_tool_get_tooltip(ett));
442 gtk_size_group_add_widget(eb->size_group, button);
443 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
444 gtk_widget_set_focus_on_click(button, FALSE);
445
446 g_signal_connect(button, "toggled",
447 G_CALLBACK(editbar_tool_button_toggled), GINT_TO_POINTER(ett));
448 g_signal_connect(button, "button_press_event",
449 G_CALLBACK(editbar_tool_button_mouse_click), GINT_TO_POINTER(ett));
450
451 hbox = eb->widget;
452 gtk_container_add(GTK_CONTAINER(hbox), button);
453 eb->tool_buttons[ett] = button;
454
455 if (editor_tool_has_value(ett)) {
456 eb->tool_selectors[ett] = create_tool_value_selector(eb, ett);
457 }
458}
459
460/************************************************************************/
464 gpointer userdata)
465{
466 struct property_editor *pe;
467
472}
473
474/************************************************************************/
478static void editbar_add_mode_button(struct editbar *eb,
479 enum editor_tool_mode etm)
480{
481 GdkPixbuf *pixbuf;
482 GtkWidget *image, *button, *hbox;
483 struct sprite *sprite;
484 const char *tooltip;
485
486 if (!eb || !(etm < NUM_EDITOR_TOOL_MODES)) {
487 return;
488 }
489
490 button = gtk_toggle_button_new();
491
493 fc_assert_ret(sprite != NULL);
494 pixbuf = sprite_get_pixbuf(sprite);
495 image = gtk_image_new_from_pixbuf(pixbuf);
496 g_object_unref(G_OBJECT(pixbuf));
497
498 gtk_container_add(GTK_CONTAINER(button), image);
499 gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(button), FALSE);
501 if (tooltip != NULL) {
502 gtk_widget_set_tooltip_text(button, tooltip);
503 }
504 gtk_size_group_add_widget(eb->size_group, button);
505 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
506 gtk_widget_set_focus_on_click(button, FALSE);
507
508 g_signal_connect(button, "toggled",
509 G_CALLBACK(editbar_mode_button_toggled), GINT_TO_POINTER(etm));
510
511 hbox = eb->widget;
512 gtk_container_add(GTK_CONTAINER(hbox), button);
513 eb->mode_buttons[etm] = button;
514}
515
516/************************************************************************/
519static struct editbar *editbar_create(void)
520{
521 struct editbar *eb;
522 GtkWidget *hbox, *button, *combo, *image, *separator, *vbox;
523 GtkListStore *store;
524 GtkCellRenderer *cell;
525 GdkPixbuf *pixbuf;
526 const struct editor_sprites *sprites;
527
528 eb = fc_calloc(1, sizeof(struct editbar));
529
530 hbox = gtk_grid_new();
531 gtk_grid_set_column_spacing(GTK_GRID(hbox), 4);
532 eb->widget = hbox;
533 eb->size_group = gtk_size_group_new(GTK_SIZE_GROUP_BOTH);
534
535 sprites = get_editor_sprites(tileset);
536
538
539 separator = gtk_separator_new(GTK_ORIENTATION_VERTICAL);
540 gtk_container_add(GTK_CONTAINER(hbox), separator);
541
552
553 separator = gtk_separator_new(GTK_ORIENTATION_VERTICAL);
554 gtk_container_add(GTK_CONTAINER(hbox), separator);
555
556 /* Player POV indicator. */
557 vbox = gtk_grid_new();
558 gtk_orientable_set_orientation(GTK_ORIENTABLE(vbox),
559 GTK_ORIENTATION_VERTICAL);
560 gtk_container_add(GTK_CONTAINER(hbox), vbox);
561
562 store = gtk_list_store_new(PPV_NUM_COLS,
563 GDK_TYPE_PIXBUF,
564 G_TYPE_STRING,
565 G_TYPE_INT);
566 eb->player_pov_store = store;
567
568 combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(store));
569
570 cell = gtk_cell_renderer_pixbuf_new();
571 gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), cell, FALSE);
572 gtk_cell_layout_add_attribute(GTK_CELL_LAYOUT(combo),
573 cell, "pixbuf", PPV_COL_FLAG);
574
575 cell = gtk_cell_renderer_text_new();
576 gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), cell, TRUE);
577 gtk_cell_layout_add_attribute(GTK_CELL_LAYOUT(combo),
578 cell, "text", PPV_COL_NAME);
579
580 gtk_widget_set_size_request(combo, 140, -1);
581 g_signal_connect(combo, "changed",
583
584 gtk_widget_set_tooltip_text(combo,
585 _("Switch player point-of-view. Use this to edit "
586 "from the perspective of different players, or "
587 "even as a global observer."));
588 gtk_container_add(GTK_CONTAINER(vbox), combo);
589 eb->player_pov_combobox = combo;
590
591 /* Property editor button. */
592 button = gtk_button_new();
593 pixbuf = sprite_get_pixbuf(sprites->properties);
594 image = gtk_image_new_from_pixbuf(pixbuf);
595 g_object_unref(G_OBJECT(pixbuf));
596 gtk_container_add(GTK_CONTAINER(button), image);
597 gtk_widget_set_tooltip_text(button, _("Show the property editor."));
598 gtk_size_group_add_widget(eb->size_group, button);
599 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
600 gtk_widget_set_focus_on_click(button, FALSE);
601 g_signal_connect(button, "clicked",
603 gtk_container_add(GTK_CONTAINER(hbox), button);
604
605 return eb;
606}
607
608/************************************************************************/
613 enum editor_tool_type ett)
614{
615 GtkTreeSelection *sel;
616 GtkTreeIter iter;
617 GtkTreeModel *model;
618 int value, store_value;
619 struct tool_value_selector *tvs;
620
621 if (!editor_is_active() || !eb || !editor_tool_has_value(ett)) {
622 return;
623 }
624
625 tvs = eb->tool_selectors[ett];
626
627 if (!tvs) {
628 return;
629 }
630
631 value = editor_tool_get_value(ett);
632 model = GTK_TREE_MODEL(tvs->store);
633 if (!gtk_tree_model_get_iter_first(model, &iter)) {
634 return;
635 }
636
637 do {
638 gtk_tree_model_get(model, &iter, TVS_COL_ID, &store_value, -1);
639 if (value == store_value) {
640 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tvs->view));
641 gtk_tree_selection_select_iter(sel, &iter);
642 break;
643 }
644 } while (gtk_tree_model_iter_next(model, &iter));
645}
646
647/************************************************************************/
652{
653 int ett;
654
655 if (!eb) {
656 return;
657 }
658
659 for (ett = 0; ett < NUM_EDITOR_TOOL_TYPES; ett++) {
660 if (editor_tool_has_value(ett)) {
662 }
663 }
664}
665
666/************************************************************************/
669static void editbar_refresh(struct editbar *eb)
670{
671 if (eb == NULL || eb->widget == NULL) {
672 return;
673 }
674
675 if (!editor_is_active()) {
676 gtk_widget_hide(eb->widget);
677 return;
678 }
679
683
684 gtk_widget_show_all(eb->widget);
685}
686
687/************************************************************************/
696static GdkPixbuf *create_terrain_pixbuf(struct terrain *pterrain)
697{
698 int w, h, i;
699 GdkPixbuf *pixbuf;
701 cairo_t *cr;
702
705
706 canvas.surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, w, h);
707
708 cr = cairo_create(canvas.surface);
709 cairo_set_operator(cr, CAIRO_OPERATOR_CLEAR);
710 cairo_paint(cr);
711 cairo_destroy(cr);
712
713 for (i = 0; i < 3; i++) {
714 struct drawn_sprite sprs[80];
716 i, pterrain);
717
718 put_drawn_sprites(&canvas, 1.0, 0, 0, count, sprs, FALSE);
719 }
720
721 pixbuf = surface_get_pixbuf(canvas.surface, w, h);
722 cairo_surface_destroy(canvas.surface);
723
724 return pixbuf;
725}
726
727/************************************************************************/
730static void clear_tool_store(GtkListStore *store)
731{
732 GtkTreeIter iter;
733 GtkTreeModel *model = GTK_TREE_MODEL(store);
734
735 if (gtk_tree_model_get_iter_first(model, &iter)) {
736 do {
737 GdkPixbuf *pixbuf;
738
739 gtk_tree_model_get(model, &iter, TVS_COL_IMAGE, &pixbuf, -1);
740 if (pixbuf != NULL) {
741 g_object_unref(pixbuf);
742 }
743 } while (gtk_tree_model_iter_next(model, &iter));
744 }
745
746 gtk_list_store_clear(store);
747}
748
749/************************************************************************/
761
762/************************************************************************/
765static void editbar_reload_tileset(struct editbar *eb)
766{
767 GtkTreeIter iter;
768 GtkListStore *store;
769 GdkPixbuf *pixbuf;
770 struct sprite *sprite;
771 struct tool_value_selector *tvs;
772
773 if (eb == NULL || tileset == NULL) {
774 return;
775 }
776
778
779 /* Reload terrains. */
780
781 tvs = eb->tool_selectors[ETT_TERRAIN];
782 store = tvs->store;
783
784 terrain_type_iterate(pterrain) {
785 gtk_list_store_append(store, &iter);
786 gtk_list_store_set(store, &iter,
787 TVS_COL_ID, terrain_number(pterrain),
789 -1);
790 pixbuf = create_terrain_pixbuf(pterrain);
791 if (pixbuf != NULL) {
792 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
793 g_object_unref(pixbuf);
794 }
796
797
798 /* Reload terrain resources. */
799
801 store = tvs->store;
802
803 extra_type_by_cause_iterate(EC_RESOURCE, pextra) {
804 gtk_list_store_append(store, &iter);
805 gtk_list_store_set(store, &iter,
806 TVS_COL_ID, extra_index(pextra),
808 -1);
809 pixbuf = create_extra_pixbuf(pextra);
810 if (pixbuf != NULL) {
811 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
812 g_object_unref(pixbuf);
813 }
815
816 /* Reload terrain specials. */
817
819 store = tvs->store;
820
822 gtk_list_store_append(store, &iter);
823 gtk_list_store_set(store, &iter,
824 TVS_COL_ID, extra_index(pextra),
826 -1);
827 pixbuf = create_extra_pixbuf(pextra);
828 if (pixbuf != NULL) {
829 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
830 g_object_unref(pixbuf);
831 }
833
834 /* Reload roads. */
835
836 tvs = eb->tool_selectors[ETT_ROAD];
837 store = tvs->store;
838
839 extra_type_by_cause_iterate(EC_ROAD, pextra) {
840 gtk_list_store_append(store, &iter);
841 gtk_list_store_set(store, &iter,
842 TVS_COL_ID, extra_index(pextra),
844 -1);
845 pixbuf = create_extra_pixbuf(pextra);
846 if (pixbuf != NULL) {
847 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
848 g_object_unref(pixbuf);
849 }
851
852 /* Reload military bases. */
853
855 store = tvs->store;
856
857 extra_type_by_cause_iterate(EC_BASE, pextra) {
858 gtk_list_store_append(store, &iter);
859 gtk_list_store_set(store, &iter,
860 TVS_COL_ID, extra_index(pextra),
862 -1);
863 pixbuf = create_extra_pixbuf(pextra);
864 if (pixbuf != NULL) {
865 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
866 g_object_unref(pixbuf);
867 }
869
870
871 /* Reload unit types. */
872
873 tvs = eb->tool_selectors[ETT_UNIT];
874 store = tvs->store;
875
876 unit_type_iterate(putype) {
877 gtk_list_store_append(store, &iter);
878 gtk_list_store_set(store, &iter,
879 TVS_COL_ID, utype_number(putype),
881 -1);
882 sprite = get_unittype_sprite(tileset, putype, direction8_invalid());
883 if (sprite == NULL) {
884 continue;
885 }
886 pixbuf = sprite_get_pixbuf(sprite);
887 if (pixbuf != NULL) {
888 gtk_list_store_set(store, &iter, TVS_COL_IMAGE, pixbuf, -1);
889 g_object_unref(G_OBJECT(pixbuf));
890 }
892}
893
894/************************************************************************/
897static int convert_modifiers(int gdk_event_state)
898{
899 int modifiers = EKM_NONE;
900
901 if (gdk_event_state & GDK_SHIFT_MASK) {
902 modifiers |= EKM_SHIFT;
903 }
904 if (gdk_event_state & GDK_CONTROL_MASK) {
905 modifiers |= EKM_CTRL;
906 }
907 if (gdk_event_state & GDK_MOD1_MASK) {
908 modifiers |= EKM_ALT;
909 }
910
911 return modifiers;
912}
913
914/************************************************************************/
917static int convert_mouse_button(int gdk_mouse_button)
918{
919 switch (gdk_mouse_button) {
920 case 1:
921 return MOUSE_BUTTON_LEFT;
922 break;
923 case 2:
924 return MOUSE_BUTTON_MIDDLE;
925 break;
926 case 3:
927 return MOUSE_BUTTON_RIGHT;
928 break;
929 default:
930 break;
931 }
932
933 return MOUSE_BUTTON_OTHER;
934}
935
936/************************************************************************/
939gboolean handle_edit_mouse_button_press(GdkEventButton *ev)
940{
941 if (ev->type != GDK_BUTTON_PRESS) {
942 return TRUE;
943 }
944
945 editor_mouse_button_press(ev->x, ev->y,
946 convert_mouse_button(ev->button),
947 convert_modifiers(ev->state));
948
949 return TRUE;
950}
951
952/************************************************************************/
955gboolean handle_edit_mouse_button_release(GdkEventButton *ev)
956{
957 if (ev->type != GDK_BUTTON_RELEASE) {
958 return TRUE;
959 }
960
961 editor_mouse_button_release(ev->x, ev->y,
962 convert_mouse_button(ev->button),
963 convert_modifiers(ev->state));
964 return TRUE;
965}
966
967/************************************************************************/
970gboolean handle_edit_mouse_move(GdkEventMotion *ev)
971{
972 editor_mouse_move(ev->x, ev->y, convert_modifiers(ev->state));
973 return TRUE;
974}
975
976/************************************************************************/
980 GtkTreePath *path,
981 GtkTreeViewColumn *col,
982 gpointer user_data)
983{
984 struct tool_value_selector *tvs = user_data;
985
986 gtk_dialog_response(GTK_DIALOG(tvs->dialog), GTK_RESPONSE_ACCEPT);
987}
988
989/************************************************************************/
992static struct tool_value_selector *
994 enum editor_tool_type ett)
995{
996 struct tool_value_selector *tvs;
997 GtkWidget *vbox, *view, *scrollwin;
998 GtkCellRenderer *cell;
999 GtkListStore *store;
1000 GtkTreeViewColumn *col;
1001 GtkTreeSelection *sel;
1002
1003 tvs = fc_calloc(1, sizeof(struct tool_value_selector));
1004
1005 tvs->editbar_parent = eb;
1006
1007 tvs->dialog = gtk_dialog_new_with_buttons(_("Select Tool Value"),
1008 GTK_WINDOW(toplevel), GTK_DIALOG_MODAL,
1009 _("_OK"), GTK_RESPONSE_ACCEPT,
1010 _("_Cancel"), GTK_RESPONSE_REJECT,
1011 NULL);
1012 vbox = gtk_dialog_get_content_area(GTK_DIALOG(tvs->dialog));
1013
1014 store = gtk_list_store_new(TVS_NUM_COLS,
1015 GDK_TYPE_PIXBUF,
1016 G_TYPE_INT,
1017 G_TYPE_STRING);
1018 tvs->store = store;
1019
1020 scrollwin = gtk_scrolled_window_new(NULL, NULL);
1021 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrollwin),
1022 GTK_SHADOW_ETCHED_IN);
1023 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollwin),
1024 GTK_POLICY_NEVER,
1025 GTK_POLICY_AUTOMATIC);
1026 gtk_scrolled_window_set_min_content_height(GTK_SCROLLED_WINDOW(scrollwin),
1028 gtk_box_pack_start(GTK_BOX(vbox), scrollwin, TRUE, TRUE, 0);
1029
1030 view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(tvs->store));
1031 gtk_widget_set_size_request(view, -1, 10 * tileset_tile_height(tileset));
1032 gtk_tree_view_set_search_column(GTK_TREE_VIEW(view), TVS_COL_NAME);
1033 g_signal_connect(view, "row-activated",
1035
1036 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(view));
1037 gtk_tree_selection_set_mode(sel, GTK_SELECTION_SINGLE);
1038
1039 cell = gtk_cell_renderer_pixbuf_new();
1040 col = gtk_tree_view_column_new_with_attributes(editor_tool_get_name(ett),
1041 cell, "pixbuf",
1042 TVS_COL_IMAGE, NULL);
1043 gtk_tree_view_column_set_resizable(col, FALSE);
1044 gtk_tree_view_column_set_sizing(col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1045 gtk_tree_view_column_set_reorderable(col, FALSE);
1046 gtk_tree_view_column_set_clickable(col, FALSE);
1047 gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
1048
1049 cell = gtk_cell_renderer_text_new();
1050 col = gtk_tree_view_column_new_with_attributes("ID", cell,
1051 "text", TVS_COL_ID, NULL);
1052 gtk_tree_view_column_set_resizable(col, FALSE);
1053 gtk_tree_view_column_set_sizing(col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1054 gtk_tree_view_column_set_reorderable(col, FALSE);
1055 gtk_tree_view_column_set_clickable(col, FALSE);
1056 gtk_tree_view_column_set_sort_column_id(col, TVS_COL_ID);
1057 gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
1058
1059 cell = gtk_cell_renderer_text_new();
1060 col = gtk_tree_view_column_new_with_attributes("Name", cell,
1061 "text", TVS_COL_NAME, NULL);
1062 gtk_tree_view_column_set_resizable(col, FALSE);
1063 gtk_tree_view_column_set_sizing(col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1064 gtk_tree_view_column_set_reorderable(col, FALSE);
1065 gtk_tree_view_column_set_clickable(col, FALSE);
1066 gtk_tree_view_column_set_sort_column_id(col, TVS_COL_NAME);
1067 gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
1068
1069 gtk_container_add(GTK_CONTAINER(scrollwin), view);
1070 tvs->view = view;
1071
1072 /* Show everything but the window itself. */
1073 gtk_widget_show_all(vbox);
1074
1075 return tvs;
1076}
1077
1078/************************************************************************/
1081static gboolean editinfobox_handle_tool_image_button_press(GtkWidget *evbox,
1082 GdkEventButton *ev,
1083 gpointer data)
1084{
1086 return TRUE;
1087}
1088
1089/************************************************************************/
1092static gboolean editinfobox_handle_mode_image_button_press(GtkWidget *evbox,
1093 GdkEventButton *ev,
1094 gpointer data)
1095{
1098
1099 return TRUE;
1100}
1101
1102/************************************************************************/
1105static void editinfobox_spin_button_value_changed(GtkSpinButton *spinbutton,
1106 gpointer userdata)
1107{
1108 struct editinfobox *ei;
1109 int which, value;
1110 enum editor_tool_type ett;
1111
1113
1114 if (!ei) {
1115 return;
1116 }
1117
1118 value = gtk_spin_button_get_value_as_int(spinbutton);
1119 which = GPOINTER_TO_INT(userdata);
1120 ett = editor_get_tool();
1121
1122 switch (which) {
1123 case SPIN_BUTTON_SIZE:
1124 editor_tool_set_size(ett, value);
1125 break;
1126 case SPIN_BUTTON_COUNT:
1127 editor_tool_set_count(ett, value);
1128 break;
1129 default:
1130 return;
1131 break;
1132 }
1133
1135}
1136
1137/************************************************************************/
1141static void editinfobox_tool_applied_player_changed(GtkComboBox *combo,
1142 gpointer userdata)
1143{
1144 struct editinfobox *ei = userdata;
1145 GtkTreeIter iter;
1146 GtkTreeModel *model;
1147 int player_no = -1;
1148
1149 if (ei == NULL) {
1150 return;
1151 }
1152
1153 if (!gtk_combo_box_get_active_iter(combo, &iter)) {
1154 return;
1155 }
1156
1157 model = gtk_combo_box_get_model(combo);
1158 gtk_tree_model_get(model, &iter, TAP_COL_PLAYER_NO, &player_no, -1);
1159
1161}
1162
1163/************************************************************************/
1167{
1168 GtkWidget *label, *vbox, *frame, *hbox, *vbox2, *image, *evbox;
1169 GtkWidget *spin, *combo;
1170 GtkListStore *store;
1171 GtkCellRenderer *cell;
1172 struct editinfobox *ei;
1173 char buf[128];
1174
1175 ei = fc_calloc(1, sizeof(*ei));
1176
1177 frame = gtk_frame_new(_("Editor Tool"));
1178 gtk_container_set_border_width(GTK_CONTAINER(frame), 4);
1179 ei->widget = frame;
1180
1181 vbox = gtk_grid_new();
1182 gtk_orientable_set_orientation(GTK_ORIENTABLE(vbox),
1183 GTK_ORIENTATION_VERTICAL);
1184 gtk_grid_set_row_spacing(GTK_GRID(vbox), 8);
1185 gtk_container_set_border_width(GTK_CONTAINER(vbox), 4);
1186 gtk_container_add(GTK_CONTAINER(frame), vbox);
1187
1188 /* tool section */
1189 hbox = gtk_grid_new();
1190 gtk_grid_set_column_spacing(GTK_GRID(hbox), 8);
1191 gtk_container_add(GTK_CONTAINER(vbox), hbox);
1192
1193 evbox = gtk_event_box_new();
1194 gtk_widget_set_tooltip_text(evbox, _("Click to change value if applicable."));
1195 g_signal_connect(evbox, "button_press_event",
1197 gtk_container_add(GTK_CONTAINER(hbox), evbox);
1198
1199 image = gtk_image_new();
1200 gtk_container_add(GTK_CONTAINER(evbox), image);
1201 ei->tool_image = image;
1202
1203 vbox2 = gtk_grid_new();
1204 gtk_orientable_set_orientation(GTK_ORIENTABLE(vbox2),
1205 GTK_ORIENTATION_VERTICAL);
1206 gtk_grid_set_row_spacing(GTK_GRID(vbox2), 4);
1207 gtk_container_add(GTK_CONTAINER(hbox), vbox2);
1208
1209 label = gtk_label_new(NULL);
1210 gtk_widget_set_halign(label, GTK_ALIGN_START);
1211 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
1212 gtk_container_add(GTK_CONTAINER(vbox2), label);
1213 ei->tool_label = label;
1214
1215 label = gtk_label_new(NULL);
1216 gtk_widget_set_halign(label, GTK_ALIGN_START);
1217 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
1218 gtk_container_add(GTK_CONTAINER(vbox2), label);
1219 ei->tool_value_label = label;
1220
1221 /* mode section */
1222 hbox = gtk_grid_new();
1223 gtk_grid_set_column_spacing(GTK_GRID(hbox), 8);
1224 gtk_container_add(GTK_CONTAINER(vbox), hbox);
1225
1226 evbox = gtk_event_box_new();
1227 gtk_widget_set_tooltip_text(evbox, _("Click to change tool mode."));
1228 g_signal_connect(evbox, "button_press_event",
1230 gtk_container_add(GTK_CONTAINER(hbox), evbox);
1231
1232 image = gtk_image_new();
1233 gtk_container_add(GTK_CONTAINER(evbox), image);
1234 ei->mode_image = image;
1235
1236 vbox2 = gtk_grid_new();
1237 gtk_orientable_set_orientation(GTK_ORIENTABLE(vbox2),
1238 GTK_ORIENTATION_VERTICAL);
1239 gtk_grid_set_row_spacing(GTK_GRID(vbox2), 4);
1240 gtk_container_add(GTK_CONTAINER(hbox), vbox2);
1241
1242 label = gtk_label_new(NULL);
1243 fc_snprintf(buf, sizeof(buf), "<span weight=\"bold\">%s</span>",
1244 _("Mode"));
1245 gtk_label_set_markup(GTK_LABEL(label), buf);
1246 gtk_widget_set_halign(label, GTK_ALIGN_START);
1247 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
1248 gtk_container_add(GTK_CONTAINER(vbox2), label);
1249
1250 label = gtk_label_new(NULL);
1251 gtk_widget_set_halign(label, GTK_ALIGN_START);
1252 gtk_widget_set_valign(label, GTK_ALIGN_CENTER);
1253 gtk_container_add(GTK_CONTAINER(vbox2), label);
1254 ei->mode_label = label;
1255
1256 /* spinner section */
1257 hbox = gtk_grid_new();
1258 gtk_grid_set_column_spacing(GTK_GRID(hbox), 8);
1259 gtk_container_add(GTK_CONTAINER(vbox), hbox);
1260 ei->size_hbox = hbox;
1261 spin = gtk_spin_button_new_with_range(1, 255, 1);
1262 gtk_widget_set_tooltip_text(spin,
1263 _("Use this to change the \"size\" parameter for the tool. "
1264 "This parameter controls for example the half-width "
1265 "of the square of tiles that will be affected by the "
1266 "tool, or the size of a created city."));
1267 g_signal_connect(spin, "value-changed",
1269 GINT_TO_POINTER(SPIN_BUTTON_SIZE));
1270 gtk_container_add(GTK_CONTAINER(hbox), spin);
1271 ei->size_spin_button = spin;
1272 label = gtk_label_new(_("Size"));
1273 gtk_container_add(GTK_CONTAINER(hbox), label);
1274
1275 hbox = gtk_grid_new();
1276 gtk_grid_set_column_spacing(GTK_GRID(hbox), 8);
1277 gtk_container_add(GTK_CONTAINER(vbox), hbox);
1278 ei->count_hbox = hbox;
1279 spin = gtk_spin_button_new_with_range(1, 255, 1);
1280 gtk_widget_set_tooltip_text(spin,
1281 _("Use this to change the tool's \"count\" parameter. "
1282 "This controls for example how many units are placed "
1283 "at once with the unit tool."));
1284 g_signal_connect(spin, "value-changed",
1286 GINT_TO_POINTER(SPIN_BUTTON_COUNT));
1287 gtk_container_add(GTK_CONTAINER(hbox), spin);
1288 ei->count_spin_button = spin;
1289 label = gtk_label_new(_("Count"));
1290 gtk_container_add(GTK_CONTAINER(hbox), label);
1291
1292 /* combo section */
1293 store = gtk_list_store_new(TAP_NUM_COLS,
1294 GDK_TYPE_PIXBUF,
1295 G_TYPE_STRING,
1296 G_TYPE_INT);
1297 ei->tool_applied_player_store = store;
1298 combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(store));
1299
1300 cell = gtk_cell_renderer_pixbuf_new();
1301 gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), cell, FALSE);
1302 gtk_cell_layout_add_attribute(GTK_CELL_LAYOUT(combo),
1303 cell, "pixbuf", TAP_COL_FLAG);
1304
1305 cell = gtk_cell_renderer_text_new();
1306 gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), cell, TRUE);
1307 gtk_cell_layout_add_attribute(GTK_CELL_LAYOUT(combo),
1308 cell, "text", TAP_COL_NAME);
1309
1310 gtk_widget_set_size_request(combo, 132, -1);
1311 g_signal_connect(combo, "changed",
1313
1314 gtk_widget_set_tooltip_text(combo,
1315 _("Use this to change the \"applied player\" tool parameter. "
1316 "This controls for example under which player units and cities "
1317 "are created."));
1318 gtk_container_add(GTK_CONTAINER(vbox), combo);
1319 ei->tool_applied_player_combobox = combo;
1320
1321 /* We add a ref to the editinfobox widget so that it is
1322 * not destroyed when replaced by the unit info box when
1323 * we leave edit mode. See editinfobox_refresh(). */
1324 g_object_ref(ei->widget);
1325
1326 /* The edit info box starts with no parent, so we have to
1327 * show its internal widgets manually. */
1328 gtk_widget_show_all(ei->widget);
1329
1330 return ei;
1331}
1332
1333/************************************************************************/
1338{
1339 enum editor_tool_type ett;
1340 GtkListStore *store;
1341 GtkWidget *combo;
1342 GtkTreeIter iter;
1343 GdkPixbuf *flag;
1344 int apno, index, i;
1345
1346 if (!ei) {
1347 return;
1348 }
1349
1350 ett = editor_get_tool();
1351 combo = ei->tool_applied_player_combobox;
1352 store = ei->tool_applied_player_store;
1353
1354 if (!combo || !store) {
1355 return;
1356 }
1357
1359 gtk_widget_hide(combo);
1360 return;
1361 }
1362
1363 gtk_list_store_clear(store);
1364
1366 index = -1;
1367 i = 0;
1368
1369 players_iterate(pplayer) {
1370 gtk_list_store_append(store, &iter);
1371 gtk_list_store_set(store, &iter,
1373 TAP_COL_NAME, player_name(pplayer), -1);
1374
1375 if (pplayer->nation != NO_NATION_SELECTED) {
1376 flag = get_flag(pplayer->nation);
1377
1378 if (flag != NULL) {
1379 gtk_list_store_set(store, &iter, TAP_COL_FLAG, flag, -1);
1380 g_object_unref(flag);
1381 }
1382 }
1383 if (apno == player_number(pplayer)) {
1384 index = i;
1385 }
1386 i++;
1388
1389 if (index == -1) {
1390 if (player_count() > 0) {
1391 index = 0;
1392 }
1394 }
1395 gtk_combo_box_set_active(GTK_COMBO_BOX(combo), index);
1396 gtk_widget_show(combo);
1397}
1398
1399/************************************************************************/
1405static GdkPixbuf *get_tool_value_pixbuf(enum editor_tool_type ett,
1406 int value)
1407{
1408 struct sprite *sprite = NULL;
1409 GdkPixbuf *pixbuf = NULL;
1410 struct terrain *pterrain;
1411 struct unit_type *putype;
1412 const struct editor_sprites *sprites;
1413
1414 sprites = get_editor_sprites(tileset);
1415 if (!sprites) {
1416 return NULL;
1417 }
1418
1419 switch (ett) {
1420 case ETT_TERRAIN:
1421 pterrain = terrain_by_number(value);
1422 if (pterrain) {
1423 pixbuf = create_terrain_pixbuf(pterrain);
1424 }
1425 break;
1428 case ETT_ROAD:
1429 case ETT_MILITARY_BASE:
1430 if (value >= 0) {
1431 pixbuf = create_extra_pixbuf(extra_by_number(value));
1432 }
1433 break;
1434 case ETT_UNIT:
1435 putype = utype_by_number(value);
1436 if (putype) {
1437 sprite = get_unittype_sprite(tileset, putype, direction8_invalid());
1438 }
1439 break;
1440 case ETT_CITY:
1441 sprite = sprites->city;
1442 break;
1443 case ETT_VISION:
1444 sprite = sprites->vision;
1445 break;
1446 case ETT_STARTPOS:
1447 sprite = sprites->startpos;
1448 break;
1449 case ETT_COPYPASTE:
1450 sprite = sprites->copypaste;
1451 break;
1452 default:
1453 break;
1454 }
1455
1456 if (sprite) {
1457 pixbuf = sprite_get_pixbuf(sprite);
1458 /*
1459 if (pixbuf) {
1460 g_object_ref(pixbuf);
1461 }
1462 */
1463 sprite = NULL;
1464 }
1465
1466 return pixbuf;
1467}
1468
1469/************************************************************************/
1475static GdkPixbuf *get_tool_mode_pixbuf(enum editor_tool_mode etm)
1476{
1477 struct sprite *sprite = NULL;
1478 GdkPixbuf *pixbuf = NULL;
1479
1481
1482 if (sprite) {
1483 pixbuf = sprite_get_pixbuf(sprite);
1484 /*
1485 if (pixbuf) {
1486 g_object_ref(pixbuf);
1487 }
1488 */
1489 }
1490
1491 return pixbuf;
1492}
1493
1494/************************************************************************/
1499static void replace_widget(GtkWidget *old, GtkWidget *new)
1500{
1501 GtkWidget *parent;
1502
1503 parent = gtk_widget_get_parent(old);
1504 if (!parent) {
1505 return;
1506 }
1507
1508 gtk_widget_hide(old);
1509 gtk_container_remove(GTK_CONTAINER(parent), old);
1510 gtk_container_add(GTK_CONTAINER(parent), new);
1511 gtk_widget_show_all(new);
1512}
1513
1514/************************************************************************/
1517static void editinfobox_refresh(struct editinfobox *ei)
1518{
1519 GdkPixbuf *pixbuf = NULL;
1520 GtkLabel *label;
1521 enum editor_tool_type ett;
1522 enum editor_tool_mode etm;
1523 int value;
1524 char buf[256];
1525
1526 if (ei == NULL) {
1527 return;
1528 }
1529
1530 if (!editor_is_active()) {
1532 return;
1533 }
1534
1535 ett = editor_get_tool();
1536 etm = editor_tool_get_mode(ett);
1537 value = editor_tool_get_value(ett);
1538
1539 label = GTK_LABEL(ei->mode_label);
1540 gtk_label_set_text(label, editor_tool_get_mode_name(ett, etm));
1541
1542 pixbuf = get_tool_mode_pixbuf(etm);
1543 gtk_image_set_from_pixbuf(GTK_IMAGE(ei->mode_image), pixbuf);
1544 if (pixbuf) {
1545 g_object_unref(pixbuf);
1546 pixbuf = NULL;
1547 }
1548
1549 fc_snprintf(buf, sizeof(buf), "<span weight=\"bold\">%s</span>",
1551 gtk_label_set_markup(GTK_LABEL(ei->tool_label), buf);
1552
1553 if (etm == ETM_COPY || etm == ETM_PASTE) {
1554 struct edit_buffer *ebuf;
1555 struct sprite *spr;
1556 char status[256];
1557
1558 ebuf = editor_get_copy_buffer();
1559 edit_buffer_get_status_string(ebuf, status, sizeof(status));
1560 gtk_label_set_text(GTK_LABEL(ei->tool_value_label), status);
1561
1562 spr = editor_tool_get_sprite(ett);
1563 pixbuf = spr ? sprite_get_pixbuf(spr) : NULL;
1564 gtk_image_set_from_pixbuf(GTK_IMAGE(ei->tool_image), pixbuf);
1565 if (pixbuf) {
1566 g_object_unref(G_OBJECT(pixbuf));
1567 pixbuf = NULL;
1568 }
1569 } else {
1570 pixbuf = get_tool_value_pixbuf(ett, value);
1571 gtk_image_set_from_pixbuf(GTK_IMAGE(ei->tool_image), pixbuf);
1572 if (pixbuf) {
1573 g_object_unref(pixbuf);
1574 pixbuf = NULL;
1575 }
1576 gtk_label_set_text(GTK_LABEL(ei->tool_value_label),
1577 editor_tool_get_value_name(ett, value));
1578 }
1579
1581
1582 if (editor_tool_has_size(ett)) {
1583 gtk_spin_button_set_value(GTK_SPIN_BUTTON(ei->size_spin_button),
1585 gtk_widget_show(ei->size_hbox);
1586 } else {
1587 gtk_widget_hide(ei->size_hbox);
1588 }
1589
1590 if (editor_tool_has_count(ett)) {
1591 gtk_spin_button_set_value(GTK_SPIN_BUTTON(ei->count_spin_button),
1593 gtk_widget_show(ei->count_hbox);
1594 } else {
1595 gtk_widget_hide(ei->count_hbox);
1596 }
1597
1599}
1600
1601/************************************************************************/
1604static gboolean handle_edit_key_press_with_ctrl(GdkEventKey *ev)
1605{
1606 return FALSE; /* Don't gobble. */
1607}
1608
1609/************************************************************************/
1612static gboolean handle_edit_key_press_with_shift(GdkEventKey *ev)
1613{
1614 enum editor_tool_type ett;
1615
1616 ett = editor_get_tool();
1617 switch (ev->keyval) {
1618 case GDK_KEY_D:
1620 break;
1621 case GDK_KEY_C:
1624 break;
1625 case GDK_KEY_V:
1628 break;
1629 case GDK_KEY_T:
1631 break;
1632 case GDK_KEY_R:
1634 break;
1635 case GDK_KEY_S:
1637 break;
1638 case GDK_KEY_P:
1640 break;
1641 case GDK_KEY_M:
1643 break;
1644 case GDK_KEY_U:
1646 break;
1647 default:
1648 return TRUE; /* Gobble? */
1649 break;
1650 }
1651
1653
1654 return TRUE;
1655}
1656
1657/************************************************************************/
1660gboolean handle_edit_key_press(GdkEventKey *ev)
1661{
1662 enum editor_tool_type ett, new_ett = NUM_EDITOR_TOOL_TYPES;
1663
1664 /* Check ctrl before shift - this is correct also for the case where
1665 * they are both active. */
1666 if (ev->state & GDK_CONTROL_MASK) {
1668 }
1669
1670 if (ev->state & GDK_SHIFT_MASK) {
1672 }
1673
1674 ett = editor_get_tool();
1675
1676 switch (ev->keyval) {
1677 case GDK_KEY_t:
1678 new_ett = ETT_TERRAIN;
1679 break;
1680 case GDK_KEY_r:
1681 new_ett = ETT_TERRAIN_RESOURCE;
1682 break;
1683 case GDK_KEY_s:
1684 new_ett = ETT_TERRAIN_SPECIAL;
1685 break;
1686 case GDK_KEY_p:
1687 new_ett = ETT_ROAD;
1688 break;
1689 case GDK_KEY_m:
1690 new_ett = ETT_MILITARY_BASE;
1691 break;
1692 case GDK_KEY_u:
1693 new_ett = ETT_UNIT;
1694 break;
1695 case GDK_KEY_c:
1696 new_ett = ETT_CITY;
1697 break;
1698 case GDK_KEY_v:
1699 new_ett = ETT_VISION;
1700 break;
1701 case GDK_KEY_b:
1702 new_ett = ETT_STARTPOS;
1703 break;
1704 case GDK_KEY_plus:
1705 case GDK_KEY_equal:
1706 case GDK_KEY_KP_Add:
1707 if (editor_tool_has_size(ett)) {
1708 int size = editor_tool_get_size(ett);
1709 editor_tool_set_size(ett, size + 1);
1710 } else if (editor_tool_has_count(ett)) {
1711 int count = editor_tool_get_count(ett);
1712 editor_tool_set_count(ett, count + 1);
1713 }
1714 break;
1715 case GDK_KEY_minus:
1716 case GDK_KEY_underscore:
1717 case GDK_KEY_KP_Subtract:
1718 if (editor_tool_has_size(ett)) {
1719 int size = editor_tool_get_size(ett);
1720 editor_tool_set_size(ett, size - 1);
1721 } else if (editor_tool_has_count(ett)) {
1722 int count = editor_tool_get_count(ett);
1723 editor_tool_set_count(ett, count - 1);
1724 }
1725 break;
1726 case GDK_KEY_1:
1727 case GDK_KEY_2:
1728 case GDK_KEY_3:
1729 case GDK_KEY_4:
1730 case GDK_KEY_5:
1731 case GDK_KEY_6:
1732 case GDK_KEY_7:
1733 case GDK_KEY_8:
1734 case GDK_KEY_9:
1735 if (editor_tool_has_size(ett)) {
1736 editor_tool_set_size(ett, ev->keyval - GDK_KEY_1 + 1);
1737 } else if (editor_tool_has_count(ett)) {
1738 editor_tool_set_count(ett, ev->keyval - GDK_KEY_1 + 1);
1739 }
1740 break;
1741 case GDK_KEY_space:
1743 break;
1744 case GDK_KEY_Tab:
1746 break;
1747 case GDK_KEY_F1:
1748 case GDK_KEY_F2:
1749 case GDK_KEY_F3:
1750 case GDK_KEY_F4:
1751 case GDK_KEY_F5:
1752 case GDK_KEY_F6:
1753 case GDK_KEY_F7:
1754 case GDK_KEY_F8:
1755 case GDK_KEY_F9:
1756 case GDK_KEY_F10:
1757 case GDK_KEY_F11:
1758 case GDK_KEY_F12:
1759 return FALSE; /* Allow default handler. */
1760 break;
1761 default:
1762 return TRUE; /* Gobbled... */
1763 break;
1764 }
1765
1766 if (new_ett != NUM_EDITOR_TOOL_TYPES) {
1767 try_to_set_editor_tool(new_ett);
1768 }
1769
1771
1772 return TRUE;
1773}
1774
1775/************************************************************************/
1778gboolean handle_edit_key_release(GdkEventKey *ev)
1779{
1780 return FALSE;
1781}
1782
1783/************************************************************************/
1787{
1788 return editor_toolbar;
1789}
1790
1791/************************************************************************/
1795{
1796 struct property_editor *pe;
1797 struct editbar *eb;
1798 struct editinfobox *ei;
1799
1800 eb = editgui_get_editbar();
1801 if (eb != NULL) {
1802 editbar_refresh(eb);
1803 }
1804
1806 if (ei != NULL) {
1808 }
1809
1811 if (!editor_is_active()) {
1814 }
1815}
1816
1817/************************************************************************/
1821{
1822 if (editor_toolbar == NULL) {
1824 }
1825 if (editor_infobox == NULL) {
1827 }
1828}
1829
1830/************************************************************************/
1834{
1835 struct editbar *eb = editgui_get_editbar();
1836 struct editinfobox *ei = editgui_get_editinfobox();
1837
1838 if (ei != NULL) {
1839 /* We have extra ref for ei->widget that has protected
1840 * it from getting destroyed when editinfobox_refresh()
1841 * moves widgets around. Free that extra ref here. */
1842 g_object_unref(ei->widget);
1843 }
1844
1846}
1847
1848/************************************************************************/
1852{
1853 return editor_infobox;
1854}
1855
1856/************************************************************************/
1865
1866/************************************************************************/
1873void editgui_popup_properties(const struct tile_list *tiles, int objtype)
1874{
1875 struct property_editor *pe;
1876
1881 property_editor_load_tiles(pe, tiles);
1883}
1884
1885/************************************************************************/
1889{
1890 struct property_editor *pe;
1891
1894 property_editor_clear(pe); /* And clear it. */
1895}
1896
1897/************************************************************************/
1905void editgui_notify_object_changed(int objtype, int object_id, bool removal)
1906{
1907 struct property_editor *pe;
1908
1909 if (!editor_is_active()) {
1910 return;
1911 }
1912
1914 property_editor_handle_object_changed(pe, objtype, object_id, removal);
1915}
1916
1917/************************************************************************/
1921{
1922 struct property_editor *pe;
1923
1924 if (!editor_is_active()) {
1925 return;
1926 }
1927
1930}
int send_chat_printf(const char *format,...)
int send_chat(const char *message)
bool client_is_global_observer(void)
int client_player_number(void)
#define client_player()
int objtype
Definition editgui_g.h:28
editgui_notify_object_changed
Definition editgui_g.h:27
int int id
Definition editgui_g.h:28
int edit_buffer_get_status_string(const struct edit_buffer *ebuf, char *buf, int buflen)
Definition editor.c:1870
bool editor_is_active(void)
Definition editor.c:346
int editor_tool_get_size(enum editor_tool_type ett)
Definition editor.c:1218
void editor_set_tool(enum editor_tool_type ett)
Definition editor.c:270
void editor_mouse_button_press(int canvas_x, int canvas_y, int button, int modifiers)
Definition editor.c:661
void editor_mouse_move(int canvas_x, int canvas_y, int modifiers)
Definition editor.c:866
enum editor_tool_mode editor_tool_get_mode(enum editor_tool_type ett)
Definition editor.c:335
void editor_tool_set_count(enum editor_tool_type ett, int count)
Definition editor.c:1263
void editor_tool_toggle_mode(enum editor_tool_type ett, enum editor_tool_mode etm)
Definition editor.c:1041
const char * editor_tool_get_value_name(enum editor_tool_type emt, int value)
Definition editor.c:1172
int editor_tool_get_value(enum editor_tool_type ett)
Definition editor.c:409
struct edit_buffer * editor_get_copy_buffer(void)
Definition editor.c:1767
void editor_tool_set_applied_player(enum editor_tool_type ett, int player_no)
Definition editor.c:1352
const char * editor_tool_get_mode_name(enum editor_tool_type ett, enum editor_tool_mode etm)
Definition editor.c:1778
void editor_tool_cycle_mode(enum editor_tool_type ett)
Definition editor.c:1058
int editor_tool_get_count(enum editor_tool_type ett)
Definition editor.c:1252
void editor_tool_set_size(enum editor_tool_type ett, int size)
Definition editor.c:1229
const char * editor_get_mode_tooltip(enum editor_tool_mode etm)
Definition editor.c:1815
bool editor_tool_is_usable(enum editor_tool_type ett)
Definition editor.c:359
enum editor_tool_type editor_get_tool(void)
Definition editor.c:286
bool editor_tool_has_applied_player(enum editor_tool_type ett)
Definition editor.c:1365
bool editor_tool_has_size(enum editor_tool_type ett)
Definition editor.c:1207
void editor_tool_set_value(enum editor_tool_type ett, int value)
Definition editor.c:397
bool editor_tool_has_value(enum editor_tool_type ett)
Definition editor.c:385
const char * editor_tool_get_tooltip(enum editor_tool_type ett)
Definition editor.c:1327
void editor_mouse_button_release(int canvas_x, int canvas_y, int button, int modifiers)
Definition editor.c:798
struct sprite * editor_tool_get_sprite(enum editor_tool_type ett)
Definition editor.c:1275
void editor_apply_tool_to_selection(void)
Definition editor.c:1128
bool editor_tool_has_mode(enum editor_tool_type ett, enum editor_tool_mode etm)
Definition editor.c:313
int editor_tool_get_applied_player(enum editor_tool_type ett)
Definition editor.c:1341
bool editor_tool_has_count(enum editor_tool_type ett)
Definition editor.c:1241
struct sprite * editor_get_mode_sprite(enum editor_tool_mode etm)
Definition editor.c:1837
const char * editor_tool_get_name(enum editor_tool_type ett)
Definition editor.c:1159
void editor_tool_set_mode(enum editor_tool_type ett, enum editor_tool_mode etm)
Definition editor.c:298
editor_tool_mode
Definition editor.h:52
@ ETM_PASTE
Definition editor.h:56
@ NUM_EDITOR_TOOL_MODES
Definition editor.h:58
@ ETM_ERASE
Definition editor.h:54
@ ETM_PAINT
Definition editor.h:53
@ ETM_COPY
Definition editor.h:55
@ MOUSE_BUTTON_OTHER
Definition editor.h:124
@ MOUSE_BUTTON_LEFT
Definition editor.h:126
@ MOUSE_BUTTON_RIGHT
Definition editor.h:128
@ MOUSE_BUTTON_MIDDLE
Definition editor.h:127
@ OBJTYPE_PLAYER
Definition editor.h:30
@ OBJTYPE_GAME
Definition editor.h:31
@ EKM_CTRL
Definition editor.h:120
@ EKM_SHIFT
Definition editor.h:118
@ EKM_ALT
Definition editor.h:119
@ EKM_NONE
Definition editor.h:117
editor_tool_type
Definition editor.h:36
@ ETT_ROAD
Definition editor.h:40
@ ETT_TERRAIN_SPECIAL
Definition editor.h:39
@ ETT_UNIT
Definition editor.h:42
@ ETT_VISION
Definition editor.h:44
@ ETT_STARTPOS
Definition editor.h:46
@ NUM_EDITOR_TOOL_TYPES
Definition editor.h:49
@ ETT_CITY
Definition editor.h:43
@ ETT_COPYPASTE
Definition editor.h:47
@ ETT_TERRAIN_RESOURCE
Definition editor.h:38
@ ETT_MILITARY_BASE
Definition editor.h:41
@ ETT_TERRAIN
Definition editor.h:37
struct extra_type * extra_by_number(int id)
Definition extras.c:175
const char * extra_name_translation(const struct extra_type *pextra)
Definition extras.c:186
#define extra_index(_e_)
Definition extras.h:177
#define extra_type_by_cause_iterate_end
Definition extras.h:315
#define extra_type_by_cause_iterate(_cause, _extra)
Definition extras.h:309
#define EC_SPECIAL
Definition fc_types.h:968
#define _(String)
Definition fcintl.h:67
#define FC_STATIC_CANVAS_INIT
Definition canvas.h:27
static void refresh_all_buttons(struct editbar *eb)
Definition editgui.c:113
static void editbar_player_properties_button_clicked(GtkButton *b, gpointer userdata)
Definition editgui.c:463
tool_applied_player_columns
Definition editgui.c:72
@ TAP_COL_FLAG
Definition editgui.c:73
@ TAP_COL_NAME
Definition editgui.c:74
@ TAP_COL_PLAYER_NO
Definition editgui.c:75
@ TAP_NUM_COLS
Definition editgui.c:77
static bool editgui_run_tool_selection(enum editor_tool_type ett)
Definition editgui.c:356
static void editinfobox_refresh(struct editinfobox *ei)
Definition editgui.c:1518
void editgui_popup_properties(const struct tile_list *tiles, int objtype)
Definition editgui.c:1874
void editgui_notify_object_created(int tag, int id)
Definition editgui.c:1921
gboolean handle_edit_key_release(GdkEventKey *ev)
Definition editgui.c:1779
static void clear_tool_stores(struct editbar *eb)
Definition editgui.c:752
static int convert_modifiers(int gdk_event_state)
Definition editgui.c:897
static void try_to_set_editor_tool(enum editor_tool_type ett)
Definition editgui.c:177
static void clear_tool_store(GtkListStore *store)
Definition editgui.c:730
gboolean handle_edit_mouse_button_release(GdkEventButton *ev)
Definition editgui.c:955
static gboolean handle_edit_key_press_with_ctrl(GdkEventKey *ev)
Definition editgui.c:1605
gboolean handle_edit_key_press(GdkEventKey *ev)
Definition editgui.c:1661
static int tool_value_selector_run(struct tool_value_selector *tvs)
Definition editgui.c:319
static void editbar_add_mode_button(struct editbar *eb, enum editor_tool_mode etm)
Definition editgui.c:478
static void editbar_tool_button_toggled(GtkToggleButton *tb, gpointer userdata)
Definition editgui.c:201
static void editbar_add_tool_button(struct editbar *eb, enum editor_tool_type ett)
Definition editgui.c:407
void editgui_create_widgets(void)
Definition editgui.c:1821
static struct editbar * editbar_create(void)
Definition editgui.c:519
void editgui_popdown_all(void)
Definition editgui.c:1889
static void editbar_reload_tileset(struct editbar *eb)
Definition editgui.c:765
static void tool_value_selector_treeview_row_activated(GtkTreeView *view, GtkTreePath *path, GtkTreeViewColumn *col, gpointer user_data)
Definition editgui.c:979
static void editinfobox_spin_button_value_changed(GtkSpinButton *spinbutton, gpointer userdata)
Definition editgui.c:1106
static void refresh_tool_applied_player_combo(struct editinfobox *ei)
Definition editgui.c:1338
static void editbar_mode_button_toggled(GtkToggleButton *tb, gpointer userdata)
Definition editgui.c:154
static void editinfobox_tool_applied_player_changed(GtkComboBox *combo, gpointer userdata)
Definition editgui.c:1142
static gboolean handle_edit_key_press_with_shift(GdkEventKey *ev)
Definition editgui.c:1613
static void refresh_all_tool_value_selectors(struct editbar *eb)
Definition editgui.c:651
spin_button_types
Definition editgui.c:80
@ SPIN_BUTTON_COUNT
Definition editgui.c:82
@ SPIN_BUTTON_SIZE
Definition editgui.c:81
gboolean handle_edit_mouse_button_press(GdkEventButton *ev)
Definition editgui.c:939
static struct tool_value_selector * create_tool_value_selector(struct editbar *eb_parent, enum editor_tool_type ett)
Definition editgui.c:993
static GdkPixbuf * create_terrain_pixbuf(struct terrain *pterrain)
Definition editgui.c:696
static gboolean editbar_tool_button_mouse_click(GtkWidget *w, GdkEventButton *ev, gpointer userdata)
Definition editgui.c:389
static void replace_widget(GtkWidget *old, GtkWidget *new)
Definition editgui.c:1500
static void editbar_player_pov_combobox_changed(GtkComboBox *combo, gpointer user_data)
Definition editgui.c:277
struct editinfobox * editgui_get_editinfobox(void)
Definition editgui.c:1852
static int convert_mouse_button(int gdk_mouse_button)
Definition editgui.c:917
gboolean handle_edit_mouse_move(GdkEventMotion *ev)
Definition editgui.c:970
tool_value_selector_columns
Definition editgui.c:56
@ TVS_COL_ID
Definition editgui.c:58
@ TVS_COL_NAME
Definition editgui.c:59
@ TVS_COL_IMAGE
Definition editgui.c:57
@ TVS_NUM_COLS
Definition editgui.c:61
player_pov_combo_columns
Definition editgui.c:64
@ PPV_COL_FLAG
Definition editgui.c:65
@ PPV_COL_PLAYER_NO
Definition editgui.c:67
@ PPV_COL_NAME
Definition editgui.c:66
@ PPV_NUM_COLS
Definition editgui.c:69
void editgui_refresh(void)
Definition editgui.c:1795
static void editbar_refresh(struct editbar *eb)
Definition editgui.c:669
static struct editinfobox * editor_infobox
Definition editgui.c:107
static struct editbar * editor_toolbar
Definition editgui.c:106
static gboolean editinfobox_handle_tool_image_button_press(GtkWidget *evbox, GdkEventButton *ev, gpointer data)
Definition editgui.c:1082
struct editbar * editgui_get_editbar(void)
Definition editgui.c:1787
static void refresh_tool_value_selector(struct editbar *eb, enum editor_tool_type ett)
Definition editgui.c:612
static struct editinfobox * editinfobox_create(void)
Definition editgui.c:1167
static void refresh_player_pov_indicator(struct editbar *eb)
Definition editgui.c:223
void editgui_tileset_changed(void)
Definition editgui.c:1861
void editgui_free(void)
Definition editgui.c:1834
static gboolean editinfobox_handle_mode_image_button_press(GtkWidget *evbox, GdkEventButton *ev, gpointer data)
Definition editgui.c:1093
static GdkPixbuf * get_tool_value_pixbuf(enum editor_tool_type ett, int value)
Definition editgui.c:1406
static GdkPixbuf * get_tool_mode_pixbuf(enum editor_tool_mode etm)
Definition editgui.c:1476
void property_editor_handle_object_changed(struct property_editor *pe, enum editor_object_type objtype, int object_id, bool remove)
Definition editprop.c:6284
void property_editor_handle_object_created(struct property_editor *pe, int tag, int object_id)
Definition editprop.c:6306
void property_editor_popup(struct property_editor *pe, enum editor_object_type objtype)
Definition editprop.c:6254
struct property_editor * editprop_get_property_editor(void)
Definition editprop.c:6207
void property_editor_reload(struct property_editor *pe, enum editor_object_type objtype)
Definition editprop.c:6344
void property_editor_popdown(struct property_editor *pe)
Definition editprop.c:6272
void property_editor_load_tiles(struct property_editor *pe, const struct tile_list *tiles)
Definition editprop.c:6218
void property_editor_clear(struct property_editor *pe)
Definition editprop.c:6324
GtkWidget * unit_info_box
Definition gui_main.c:151
GtkWidget * toplevel
Definition gui_main.c:124
void disable_gobject_callback(GObject *obj, GCallback cb)
Definition gui_stuff.c:1087
void enable_gobject_callback(GObject *obj, GCallback cb)
Definition gui_stuff.c:1103
GdkPixbuf * get_flag(const struct nation_type *nation)
Definition plrdlg.c:609
const char * tooltip
Definition repodlgs.c:1314
GdkPixbuf * surface_get_pixbuf(cairo_surface_t *surf, int width, int height)
Definition sprite.c:418
GdkPixbuf * sprite_get_pixbuf(struct sprite *sprite)
Definition sprite.c:402
GdkPixbuf * create_extra_pixbuf(const struct extra_type *pextra)
Definition sprite.c:494
#define fc_assert_ret(condition)
Definition log.h:191
void put_drawn_sprites(struct canvas *pcanvas, float zoom, int canvas_x, int canvas_y, int count, struct drawn_sprite *pdrawn, bool fog)
#define fc_calloc(n, esz)
Definition mem.h:38
#define NO_NATION_SELECTED
Definition nation.h:29
struct player * player_by_number(const int player_id)
Definition player.c:840
int player_count(void)
Definition player.c:808
int player_number(const struct player *pplayer)
Definition player.c:828
const char * player_name(const struct player *pplayer)
Definition player.c:886
#define players_iterate_end
Definition player.h:535
#define players_iterate(_pplayer)
Definition player.h:530
size_t size
Definition specvec.h:72
cairo_surface_t * surface
Definition canvas.h:22
GtkWidget * tool_buttons[NUM_EDITOR_TOOL_TYPES]
Definition editgui.h:31
GtkWidget * player_pov_combobox
Definition editgui.h:35
GtkListStore * player_pov_store
Definition editgui.h:34
GtkSizeGroup * size_group
Definition editgui.h:28
GtkWidget * widget
Definition editgui.h:26
GtkWidget * mode_buttons[NUM_EDITOR_TOOL_MODES]
Definition editgui.h:30
struct tool_value_selector * tool_selectors[NUM_EDITOR_TOOL_TYPES]
Definition editgui.h:32
GtkWidget * tool_label
Definition editgui.h:50
GtkListStore * tool_applied_player_store
Definition editgui.h:59
GtkWidget * size_spin_button
Definition editgui.h:55
GtkWidget * tool_applied_player_combobox
Definition editgui.h:60
GtkWidget * mode_image
Definition editgui.h:47
GtkWidget * count_hbox
Definition editgui.h:56
GtkWidget * tool_image
Definition editgui.h:52
GtkWidget * tool_value_label
Definition editgui.h:51
GtkWidget * count_spin_button
Definition editgui.h:57
GtkWidget * widget
Definition editgui.h:45
GtkWidget * mode_label
Definition editgui.h:48
GtkWidget * size_hbox
Definition editgui.h:54
struct sprite * city
Definition tilespec.h:348
struct sprite * vision
Definition tilespec.h:349
struct sprite * copypaste
Definition tilespec.h:342
struct sprite * startpos
Definition tilespec.h:343
struct sprite * properties
Definition tilespec.h:351
char name[MAX_LEN_NAME]
Definition player.h:251
GtkWidget * view
Definition editgui.c:91
struct editbar * editbar_parent
Definition editgui.c:86
GtkListStore * store
Definition editgui.c:90
GtkWidget * dialog
Definition editgui.c:88
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
const char * terrain_name_translation(const struct terrain *pterrain)
Definition terrain.c:226
struct terrain * terrain_by_number(const Terrain_type_id type)
Definition terrain.c:144
Terrain_type_id terrain_number(const struct terrain *pterrain)
Definition terrain.c:135
#define terrain_type_iterate(_p)
Definition terrain.h:358
#define terrain_type_iterate_end
Definition terrain.h:364
struct sprite * get_unittype_sprite(const struct tileset *t, const struct unit_type *punittype, enum direction8 facing)
Definition tilespec.c:6516
int fill_basic_terrain_layer_sprite_array(struct tileset *t, struct drawn_sprite *sprs, int layer, struct terrain *pterrain)
Definition tilespec.c:6864
int tileset_tile_height(const struct tileset *t)
Definition tilespec.c:728
int tileset_tile_width(const struct tileset *t)
Definition tilespec.c:716
const struct editor_sprites * get_editor_sprites(const struct tileset *t)
Definition tilespec.c:6645
struct unit_type * utype_by_number(const Unit_type_id id)
Definition unittype.c:112
Unit_type_id utype_number(const struct unit_type *punittype)
Definition unittype.c:100
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