merge branch work back into trunk
[xonotic/netradiant.git] / plugins / surface_ufoai / surfaceflagsdialog_ufoai.cpp
1 /*
2 Copyright (C) 1999-2007 id Software, Inc. and contributors.
3 For a list of contributors, see the accompanying CONTRIBUTORS file.
4
5 This file is part of GtkRadiant.
6
7 GtkRadiant is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
11
12 GtkRadiant is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with GtkRadiant; if not, write to the Free Software
19 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
20 */
21
22 #include <gtk/gtk.h>
23 #include <glib/gi18n.h>
24 #include <gdk/gdkkeysyms.h>
25 #include <glib/gi18n.h>
26
27 #include "surfdlg_plugin.h"
28
29 #include "surfaceflagsdialog_ufoai.h"
30
31 GtkWidget *notebook1;
32
33 // 32 bit is the max
34 #define MAX_BUTTONS 32
35
36 GtkWidget *surface_buttons[MAX_BUTTONS];
37 GtkWidget *content_buttons[MAX_BUTTONS];
38
39 GtkWidget *value_entry;
40 gboolean setup_buttons = TRUE;
41
42 int working_surface_flags;
43 int surface_mask;
44 int working_content_flags;
45 int content_mask;
46 int working_value;
47
48 inline void set_inconsistent(GtkWidget *toggle_button)
49 {
50         gtk_toggle_button_set_inconsistent(GTK_TOGGLE_BUTTON (toggle_button), TRUE);
51 }
52
53 inline void clear_inconsistent(GtkWidget *toggle_button)
54 {
55         if ( gtk_toggle_button_get_inconsistent(GTK_TOGGLE_BUTTON (toggle_button)) )
56         {
57                 gtk_toggle_button_set_inconsistent(GTK_TOGGLE_BUTTON (toggle_button), FALSE);
58         }
59 }
60
61 void clear_all_inconsistent(void)
62 {
63         int i;
64
65         for (i = 0; i < MAX_BUTTONS; i++) {
66                 clear_inconsistent( surface_buttons[i] );
67                 clear_inconsistent( content_buttons[i] );
68         }
69 }
70
71 void clear_all_buttons_and_values()
72 {
73         int i;
74
75         for (i = 0; i < MAX_BUTTONS; i++) {
76                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON ( surface_buttons[i] ), FALSE);
77                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON ( content_buttons[i] ), FALSE);
78         }
79
80         gtk_entry_set_text( (GtkEntry *)value_entry, "");
81 }
82
83 void SetFlagButtons_UFOAI(texdef_to_face_t *texdef_face_list, bool b_isListEmpty)
84 {
85         int contents = 0;
86         int flags = 0;
87         int value = 0;
88         int diff_contents = 0;
89         int diff_flags = 0;
90         gboolean diff_value = FALSE;
91         char tex_buff[11];
92         texdef_t* tmp_texdef;
93         texdef_to_face_t* temp_texdef_face_list;
94         int i;
95
96         setup_buttons = TRUE;
97         working_surface_flags = 0;
98         surface_mask = 0;
99         working_content_flags = 0;
100         content_mask = 0;
101         working_value = 0;
102
103         if (!b_isListEmpty) {
104                 tmp_texdef = &texdef_face_list->texdef;
105                 contents = tmp_texdef->contents;
106                 flags = tmp_texdef->flags;
107                 value = tmp_texdef->value;
108
109                 Sys_Printf("Surface: %d\tContents: %d\tValue: %d\ttmp_texdef\n",tmp_texdef->flags,tmp_texdef->contents,tmp_texdef->value);
110                 Sys_Printf("Surface: %d\tContents: %d\tValue: %d\n",flags,contents,value);
111
112                 for (temp_texdef_face_list = texdef_face_list->next; temp_texdef_face_list; temp_texdef_face_list = temp_texdef_face_list->next)
113                 {
114                         tmp_texdef = &temp_texdef_face_list->texdef;
115                         diff_contents |= contents ^ tmp_texdef->contents;  // Figure out which buttons are inconsistent
116                         diff_flags |= flags ^ tmp_texdef->flags;
117                         if (tmp_texdef->value != value)
118                                 diff_value = TRUE;
119
120                         Sys_Printf("Surface: %d\tContents: %d\tValue: %d\ttmp_texdef\n",tmp_texdef->flags,tmp_texdef->contents,tmp_texdef->value);
121                         Sys_Printf("Surface: %d\tContents: %d\tValue: %d\n",flags,contents,value);
122                 }
123         }
124
125         clear_all_inconsistent();
126
127         // If no faces/brushes are selected, clear everything and bail
128         if (b_isListEmpty) {
129                 clear_all_buttons_and_values();
130                 setup_buttons = FALSE;
131                 return;
132         }
133
134         for (i = 0; i < MAX_BUTTONS; i++) {
135                 // Set surface buttons to reflect brush/face flags, contents, and values
136                 if(diff_flags & (1 << i))
137                         set_inconsistent(surface_buttons[i]);
138                 else if(flags & (1 << i))
139                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON (surface_buttons[i]), TRUE);
140                 else
141                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON (surface_buttons[i]), FALSE);
142
143                 if(diff_contents & (1 << i))
144                         set_inconsistent(content_buttons[i]);
145                 else if(contents & (1 << i))
146                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON (content_buttons[i]), TRUE);
147                 else
148                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON (content_buttons[i]), FALSE);
149         }
150
151         // Set Value
152         if (diff_value)
153                 gtk_entry_set_text( (GtkEntry *)value_entry, "");
154         else {
155                 working_value = value;
156                 sprintf( tex_buff, "%d", value);
157                 gtk_entry_set_text( (GtkEntry *)value_entry, tex_buff);
158         }
159
160         setup_buttons = FALSE;
161 }
162
163 void SetChangeInFlags_Face_UFOAI (texdef_to_face_t *texdef_face_list)
164 {
165         texdef_to_face_t *temp_texdef_face_list;
166         texdef_t *tmp_texdef;
167
168         for (temp_texdef_face_list = texdef_face_list; temp_texdef_face_list; temp_texdef_face_list = temp_texdef_face_list->next)
169         {
170                 tmp_texdef = &temp_texdef_face_list->texdef;
171                 tmp_texdef->flags = (tmp_texdef->flags & ~surface_mask) | working_surface_flags;
172                 tmp_texdef->contents = (tmp_texdef->contents & ~content_mask) | working_content_flags;
173                 tmp_texdef->value = working_value;
174                 Sys_Printf("content_flag: %d     content_mask: %d\n",working_content_flags,content_mask);
175                 Sys_Printf("content: %d\n",tmp_texdef->contents);
176         }
177 }
178
179 inline void change_surfaceflag (GtkWidget *togglebutton, int sur_flag, gboolean change_flag_to)
180 {
181         if (!setup_buttons) // If we're setting up the buttons, we really don't need to
182         {                   // set flags that are already set
183                 if (gtk_toggle_button_get_inconsistent(GTK_TOGGLE_BUTTON (togglebutton))) // Clear out inconsistent, if set
184                         clear_inconsistent(GTK_WIDGET (togglebutton));
185
186                 surface_mask |= sur_flag;
187
188                 if (change_flag_to)
189                         working_surface_flags |= sur_flag;
190                 else
191                         working_surface_flags &= ~sur_flag;
192         }
193 }
194
195 inline void change_contentflag (GtkWidget *togglebutton, int content_flag, gboolean change_flag_to)
196 {
197         if ( (!setup_buttons) )  // If we're setting up the buttons, we really don't need to
198         {                        // set flags that are already set
199                 if (gtk_toggle_button_get_inconsistent(GTK_TOGGLE_BUTTON (togglebutton)))
200                         clear_inconsistent(togglebutton);
201                 //if (g_ptrSelectedFaces.GetSize() == 0)  // Only changing content flags on whole brushes, not faces.
202                 //{
203                 content_mask |= content_flag;
204
205                 if (change_flag_to)
206                         working_content_flags |= content_flag;
207                 else
208                         working_content_flags &= ~content_flag;
209                 //}
210                 Sys_Printf("content_flag: %d     content_mask: %d\n",content_flag,content_mask);
211         }
212 }
213
214 // Surface Flags Callbacks
215 void on_surface_button_toggled (GtkToggleButton *togglebutton, gpointer user_data)
216 {
217         int flag = GPOINTER_TO_INT(user_data);
218         change_surfaceflag(GTK_WIDGET (togglebutton), flag, (GTK_TOGGLE_BUTTON (togglebutton)->active));
219 }
220
221 // Content Flags Callbacks
222 void on_content_button_toggled (GtkToggleButton *togglebutton, gpointer user_data)
223 {
224         int flag = GPOINTER_TO_INT(user_data);
225         change_contentflag(GTK_WIDGET (togglebutton), flag, (GTK_TOGGLE_BUTTON (togglebutton)->active));
226 }
227
228 // Value Entry Callback
229 void on_value_entry_changed (GtkEditable *editable, gpointer user_data)
230 {
231         if ( (!setup_buttons) )  // If we're setting up the buttons, don't change value
232         working_value = atoi( gtk_entry_get_text( (GtkEntry*)editable) );
233 }
234
235 void on_value_entry_insert_text (GtkEditable *editable, gchar *new_text, gint new_text_length, gint *position, gpointer user_data)
236 {
237         int i, count=0;
238         gchar *result;
239
240         // Limit input to digits, throwing out anything else
241         // Modified from Gtk FAQ for text filtering of GtkEntry
242         result = g_new (gchar, new_text_length);
243
244         for (i=0; i < new_text_length; i++) {
245                 if (!isdigit(new_text[i]))
246                         continue;
247                 result[count++] = new_text[i];
248         }
249
250         if (count > 0) {
251                 gtk_signal_handler_block_by_func (GTK_OBJECT (editable),
252                                                 GTK_SIGNAL_FUNC (on_value_entry_insert_text),
253                                                 user_data);
254                 gtk_editable_insert_text (editable, result, count, position);
255                 gtk_signal_handler_unblock_by_func (GTK_OBJECT (editable),
256                                                 GTK_SIGNAL_FUNC (on_value_entry_insert_text),
257                                                 user_data);
258         }
259         gtk_signal_emit_stop_by_name (GTK_OBJECT (editable), "insert_text");
260
261         g_free (result);
262 }
263
264 void on_surfacebutton_clicked (GtkButton *button, gpointer user_data)
265 {
266         gtk_notebook_set_page (GTK_NOTEBOOK(notebook1), 0);
267 }
268
269 void on_contentbutton_clicked (GtkButton *button, gpointer user_data)
270 {
271         gtk_notebook_set_page (GTK_NOTEBOOK(notebook1), 1);
272 }
273
274 #define UFOAI_FLAG_BUTTON_BORDER 3
275
276 GtkWidget* Create_UFOAIFlagsDialog (GtkWidget* surfacedialog_widget)
277 {
278         GtkWidget *frame1;
279         GtkWidget *vbox1;
280         GtkWidget *vbox2;
281         GtkWidget *vbox3;
282         GtkWidget *vbox4;
283         GtkWidget *table4;
284         GtkWidget *hbox2;
285         GtkWidget *hbox3;
286         GtkWidget *hseparator1;
287         GtkWidget *value_label;
288         GtkWidget *label5;
289         GtkWidget *table3;
290         GtkWidget *label6;
291         int i, x, y;
292         const char *buttonLabel;
293         char buffer[8];
294
295         frame1 = gtk_frame_new (_("Flags"));
296         gtk_widget_show (frame1);
297         gtk_container_add (GTK_CONTAINER (surfacedialog_widget), frame1);
298
299         vbox1 = gtk_vbox_new (FALSE, 0);
300         gtk_widget_show (vbox1);
301         gtk_container_add (GTK_CONTAINER (frame1), vbox1);
302
303         notebook1 = gtk_notebook_new ();
304         gtk_widget_show (notebook1);
305         gtk_box_pack_start (GTK_BOX (vbox1), notebook1, TRUE, TRUE, 0);
306         gtk_notebook_set_show_tabs (GTK_NOTEBOOK (notebook1), TRUE);
307         gtk_container_set_border_width (GTK_CONTAINER (notebook1), 5);
308
309         vbox2 = gtk_vbox_new (FALSE, 0);
310         gtk_widget_show (vbox2);
311         gtk_container_add (GTK_CONTAINER (notebook1), vbox2);
312
313         table4 = gtk_table_new (8, 4, FALSE);
314         gtk_widget_show (table4);
315         gtk_box_pack_start (GTK_BOX (vbox2), table4, TRUE, TRUE, 0);
316
317         y = -1;
318         for (i = 0; i < MAX_BUTTONS; i++) {
319                 if (!(i % 4))
320                         y++;
321                 x = i % 4;
322                 snprintf(buffer, sizeof(buffer) - 1, "surf%i", i + 1);
323                 buttonLabel = g_FuncTable.m_pfnReadProjectKey(buffer);
324                 Sys_Printf("%s: %s\n", buffer, buttonLabel);
325                 surface_buttons[i] = gtk_toggle_button_new_with_label (buttonLabel);
326                 gtk_signal_connect(GTK_OBJECT (surface_buttons[i]), "toggled", GTK_SIGNAL_FUNC(on_surface_button_toggled), GINT_TO_POINTER(1 << i));
327                 gtk_widget_show(surface_buttons[i]);
328                 gtk_table_attach(GTK_TABLE (table4), surface_buttons[i], 0 + x, 1 + x, (0 + y), (1 + y),
329                                                         (GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
330                                                         (GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
331                 gtk_container_set_border_width (GTK_CONTAINER (surface_buttons[i]), UFOAI_FLAG_BUTTON_BORDER);
332         }
333
334         hseparator1 = gtk_hseparator_new ();
335         gtk_widget_show (hseparator1);
336         gtk_box_pack_start (GTK_BOX (vbox2), hseparator1, FALSE, FALSE, 0);
337         gtk_widget_set_usize (hseparator1, -2, 5);
338
339         hbox2 = gtk_hbox_new (FALSE, 0);
340         gtk_widget_show (hbox2);
341         gtk_box_pack_start (GTK_BOX (vbox2), hbox2, FALSE, FALSE, 0);
342
343         hbox3 = gtk_hbox_new (FALSE, 0);
344         gtk_widget_show (hbox3);
345         gtk_box_pack_start (GTK_BOX (hbox2), hbox3, TRUE, TRUE, 0);
346
347         vbox4 = gtk_vbox_new (FALSE, 0);
348         gtk_widget_show (vbox4);
349         gtk_box_pack_start (GTK_BOX (hbox3), vbox4, TRUE, TRUE, 0);
350
351         value_label = gtk_label_new (" Value: ");
352         gtk_widget_show (value_label);
353         gtk_box_pack_start (GTK_BOX (hbox3), value_label, FALSE, FALSE, 0);
354
355         value_entry = gtk_entry_new ();
356         gtk_signal_connect (GTK_OBJECT (value_entry), "changed",
357                                                 GTK_SIGNAL_FUNC (on_value_entry_changed),
358                                                 NULL);
359         gtk_signal_connect (GTK_OBJECT (value_entry), "insert_text",
360                                                 GTK_SIGNAL_FUNC (on_value_entry_insert_text),
361                                                 NULL);
362         gtk_entry_set_max_length( (GtkEntry *)value_entry, 11);
363         gtk_widget_show (value_entry);
364         gtk_box_pack_start (GTK_BOX (hbox3), value_entry, TRUE, TRUE, 0);
365
366         vbox3 = gtk_vbox_new (FALSE, 0);
367         gtk_widget_show (vbox3);
368         gtk_box_pack_start (GTK_BOX (hbox3), vbox3, TRUE, TRUE, 0);
369
370         label5 = gtk_label_new ("Surface Flags");
371         gtk_widget_show (label5);
372         gtk_notebook_set_tab_label (GTK_NOTEBOOK (notebook1), gtk_notebook_get_nth_page (GTK_NOTEBOOK (notebook1), 0), label5);
373
374         table3 = gtk_table_new (8, 4, FALSE);
375         gtk_widget_show (table3);
376         gtk_container_add (GTK_CONTAINER (notebook1), table3);
377
378         y = -1;
379         for (i = 0; i < MAX_BUTTONS; i++) {
380                 if (!(i % 4))
381                         y++;
382                 x = i % 4;
383                 snprintf(buffer, sizeof(buffer) - 1, "cont%i", i + 1);
384                 buttonLabel = g_FuncTable.m_pfnReadProjectKey(buffer);
385                 content_buttons[i] = gtk_toggle_button_new_with_label(buttonLabel);
386                 gtk_signal_connect(GTK_OBJECT (content_buttons[i]), "toggled", GTK_SIGNAL_FUNC (on_content_button_toggled), GINT_TO_POINTER(1 << i));
387                 gtk_widget_show(content_buttons[i]);
388                 gtk_table_attach(GTK_TABLE (table3), content_buttons[i], 0 + x, 1 + x, (0 + y), (1 + y),
389                                                 (GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
390                                                 (GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
391                 gtk_container_set_border_width (GTK_CONTAINER (content_buttons[i]), UFOAI_FLAG_BUTTON_BORDER);
392         }
393
394         label6 = gtk_label_new ("Content Flags");
395         gtk_widget_show (label6);
396         gtk_notebook_set_tab_label (GTK_NOTEBOOK (notebook1), gtk_notebook_get_nth_page (GTK_NOTEBOOK (notebook1), 1), label6);
397
398         return frame1;
399 }
400