pspp-sheet-view: Rename GtkTreeSelectMode to PsppSheetSelectMode.
[pspp] / src / ui / gui / oneway-anova-dialog.c
1 /* PSPPIRE - a graphical user interface for PSPP.
2    Copyright (C) 2007, 2010, 2011, 2012  Free Software Foundation
3
4    This program is free software: you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation, either version 3 of the License, or
7    (at your option) any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program.  If not, see <http://www.gnu.org/licenses/>.
16 */
17
18 #include <config.h>
19 #include <gtk/gtk.h>
20 #include "oneway-anova-dialog.h"
21 #include "psppire-dict.h"
22 #include "psppire-var-view.h"
23 #include "builder-wrapper.h"
24 #include "psppire-data-window.h"
25 #include "psppire-dialog.h"
26 #include "dialog-common.h"
27 #include "psppire-acr.h"
28 #include "psppire-selector.h"
29 #include "dict-display.h"
30
31 #include "executor.h"
32 #include "helper.h"
33
34 #include "gettext.h"
35 #define _(msgid) gettext (msgid)
36 #define N_(msgid) msgid
37
38
39 struct contrasts_subdialog;
40 struct oneway_anova_dialog;
41
42
43 static void run_contrasts_dialog (struct oneway_anova_dialog *);
44 static void push_new_store (GArray *, struct contrasts_subdialog *);
45
46 static void next (GtkWidget *, gpointer);
47 static void prev (GtkWidget *, gpointer);
48
49
50 struct contrasts_subdialog
51 {
52   /* Contrasts Dialog widgets */
53   GtkWidget *contrasts_dialog;
54   GtkWidget *stack_label;
55   PsppireAcr *acr;
56
57
58   /* Gets copied into contrasts when "Continue"
59      is clicked */
60   GArray *temp_contrasts;
61
62   /* Index into the temp_contrasts */
63   guint c;
64
65   GtkWidget *prev;
66   GtkWidget *next;
67
68   GtkWidget *ctotal;
69 };
70
71
72 struct oneway_anova_dialog
73 {
74   PsppireDict *dict;
75
76   GtkWidget *factor_entry;
77   GtkWidget *vars_treeview;
78   GtkWindow *dialog;
79   GArray *contrasts_array;
80
81   GtkToggleButton *descriptives;
82   GtkToggleButton *homogeneity;
83
84   struct contrasts_subdialog contrasts;
85 };
86
87 static gboolean
88 dialog_state_valid (gpointer data)
89 {
90   struct oneway_anova_dialog *ow = data;
91
92   GtkTreeModel *vars =
93     gtk_tree_view_get_model (GTK_TREE_VIEW (ow->vars_treeview));
94
95   GtkTreeIter notused;
96
97   if ( !gtk_tree_model_get_iter_first (vars, &notused) )
98     return FALSE;
99
100   if ( 0 == strcmp ("", gtk_entry_get_text (GTK_ENTRY (ow->factor_entry))))
101     return FALSE;
102
103   return TRUE;
104 }
105
106 static gchar * generate_syntax (const struct oneway_anova_dialog *);
107
108
109 static void
110 refresh (struct oneway_anova_dialog *ow)
111 {
112   GtkTreeModel *model =
113     gtk_tree_view_get_model (GTK_TREE_VIEW (ow->vars_treeview));
114
115   gtk_entry_set_text (GTK_ENTRY (ow->factor_entry), "");
116
117   gtk_list_store_clear (GTK_LIST_STORE (model));
118 }
119
120
121
122 /* Pops up the dialog box */
123 void
124 oneway_anova_dialog (PsppireDataWindow *de)
125 {
126   gint response;
127
128   struct oneway_anova_dialog ow;
129
130   GtkBuilder *builder = builder_new ("oneway.ui");
131
132   GtkWidget *dict_view =
133     get_widget_assert (builder, "oneway-anova-treeview1");
134
135   GtkWidget *selector2 =
136     get_widget_assert (builder, "oneway-anova-selector2");
137
138   GtkWidget *contrasts_button =
139     get_widget_assert (builder, "contrasts-button");
140
141
142   g_signal_connect_swapped (contrasts_button, "clicked",
143                     G_CALLBACK (run_contrasts_dialog), &ow);
144
145
146   ow.factor_entry = get_widget_assert (builder, "oneway-anova-entry");
147   ow.vars_treeview =
148     get_widget_assert (builder, "oneway-anova-treeview2");
149
150   ow.descriptives =
151     GTK_TOGGLE_BUTTON (get_widget_assert (builder, "checkbutton1"));
152
153   ow.homogeneity =
154     GTK_TOGGLE_BUTTON (get_widget_assert (builder, "checkbutton2"));
155
156   g_object_get (de->data_editor, "dictionary", &ow.dict, NULL);
157
158   ow.dialog =
159     GTK_WINDOW (get_widget_assert (builder, "oneway-anova-dialog"));
160
161   gtk_window_set_transient_for (ow.dialog, GTK_WINDOW (de));
162
163   g_object_set (dict_view, "model", ow.dict, NULL);
164
165
166   psppire_selector_set_filter_func (PSPPIRE_SELECTOR (selector2),
167                                     is_currently_in_entry);
168
169
170   g_signal_connect_swapped (ow.dialog, "refresh", G_CALLBACK (refresh),  &ow);
171
172
173
174   psppire_dialog_set_valid_predicate (PSPPIRE_DIALOG (ow.dialog),
175                                       dialog_state_valid, &ow);
176
177
178   {
179     struct contrasts_subdialog *cd = &ow.contrasts;
180     GtkEntry *entry = GTK_ENTRY (get_widget_assert (builder, "entry1"));
181
182     cd->acr = PSPPIRE_ACR (get_widget_assert (builder, "psppire-acr1"));
183     cd->contrasts_dialog = get_widget_assert (builder, "contrasts-dialog");
184
185     cd->next = get_widget_assert (builder, "next-button");
186     cd->prev = get_widget_assert (builder, "prev-button");
187     cd->ctotal = get_widget_assert (builder, "entry2");
188
189     cd->stack_label = get_widget_assert (builder, "contrast-stack-label");
190
191     /* Contrasts */
192     ow.contrasts_array = g_array_new (FALSE, FALSE, sizeof (GtkListStore *));
193
194     g_signal_connect (cd->next, "clicked", G_CALLBACK (next), cd);
195     g_signal_connect (cd->prev, "clicked", G_CALLBACK (prev), cd);
196
197     psppire_acr_set_entry (cd->acr, entry);
198
199     gtk_window_set_transient_for (GTK_WINDOW (cd->contrasts_dialog),
200                                   GTK_WINDOW (de));
201   }
202
203   response = psppire_dialog_run (PSPPIRE_DIALOG (ow.dialog));
204
205   switch (response)
206     {
207     case GTK_RESPONSE_OK:
208       g_free (execute_syntax_string (de, generate_syntax (&ow)));
209       break;
210     case PSPPIRE_RESPONSE_PASTE:
211       g_free (paste_syntax_to_window (generate_syntax (&ow)));
212       break;
213     default:
214       break;
215     }
216
217   g_array_free (ow.contrasts_array, FALSE);
218
219   g_object_unref (builder);
220 }
221
222
223 static gchar * generate_syntax (const struct oneway_anova_dialog *ow)
224 {
225   gchar *text;
226   gint i;
227   gboolean descriptives = gtk_toggle_button_get_active (ow->descriptives);
228   gboolean homogeneity = gtk_toggle_button_get_active (ow->homogeneity);
229   struct string dss;
230
231   ds_init_cstr (&dss, "ONEWAY /VARIABLES=");
232
233   psppire_var_view_append_names_str (PSPPIRE_VAR_VIEW (ow->vars_treeview), 0, &dss);
234
235   ds_put_cstr (&dss, " BY ");
236
237   ds_put_cstr (&dss, gtk_entry_get_text (GTK_ENTRY (ow->factor_entry)));
238
239   if (descriptives || homogeneity )
240     {
241       ds_put_cstr (&dss, "\n\t/STATISTICS=");
242       if (descriptives)
243         ds_put_cstr (&dss, "DESCRIPTIVES ");
244       if (homogeneity)
245         ds_put_cstr (&dss, "HOMOGENEITY ");
246     }
247
248   for (i = 0 ; i < ow->contrasts_array->len ; ++i )
249     {
250       GtkListStore *ls = g_array_index (ow->contrasts_array, GtkListStore*, i);
251       GtkTreeIter iter;
252       gboolean ok;
253
254       ds_put_cstr (&dss, "\n\t/CONTRAST=");
255
256       for (ok = gtk_tree_model_get_iter_first (GTK_TREE_MODEL(ls),
257                                                &iter);
258            ok;
259            ok = gtk_tree_model_iter_next (GTK_TREE_MODEL (ls), &iter))
260         {
261           gdouble v;
262
263           gtk_tree_model_get (GTK_TREE_MODEL (ls), &iter, 0, &v, -1);
264
265           ds_put_c_format (&dss, " %g", v);
266         }
267     }
268
269   ds_put_cstr (&dss, ".\n");
270
271   text = ds_steal_cstr (&dss);
272   ds_destroy (&dss);
273
274   return text;
275 }
276
277
278 \f
279 /* Contrasts stuff */
280
281
282
283
284 /* Callback for when the list store currently associated with the
285    treeview has changed.  It sets the widgets of the subdialog
286    to reflect the store's new state.
287 */
288 static void
289 list_store_changed (struct contrasts_subdialog *csd)
290 {
291   gboolean ok;
292   gdouble total = 0.0;
293   GtkTreeIter iter;
294   GtkTreeModel *ls = NULL;
295   gchar *text =
296     g_strdup_printf (_("Contrast %d of %d"),
297                      csd->c, csd->temp_contrasts->len);
298
299   gtk_label_set_label (GTK_LABEL (csd->stack_label), text);
300
301   g_free (text);
302
303   gtk_widget_set_sensitive (csd->prev, csd->c > 1);
304
305   if ( csd->c > 0 )
306     ls = g_array_index (csd->temp_contrasts, GtkTreeModel*, csd->c - 1);
307
308   psppire_acr_set_model (csd->acr, GTK_LIST_STORE (ls));
309
310   /* Sensitive iff the liststore has two items or more */
311   gtk_widget_set_sensitive (csd->next,
312                             gtk_tree_model_iter_nth_child
313                             (ls, &iter,  NULL, 1));
314
315   for (ok = gtk_tree_model_get_iter_first (ls, &iter);
316        ok;
317        ok = gtk_tree_model_iter_next (ls, &iter)
318        )
319     {
320       gdouble v;
321       gtk_tree_model_get (ls, &iter, 0, &v, -1);
322       total += v;
323     }
324
325   text = g_strdup_printf ("%g", total);
326
327   gtk_entry_set_text (GTK_ENTRY (csd->ctotal), text);
328
329   g_free (text);
330 }
331
332
333
334 /* Copy the contrasts array into the local array */
335 static GArray *
336 clone_contrasts_array (GArray *src_array)
337 {
338   gint i;
339
340   GArray *dest_array =
341     g_array_sized_new (FALSE, FALSE, sizeof (GtkListStore *),
342                        src_array->len);
343
344   for (i = 0 ; i < src_array->len ; ++i )
345     {
346
347       GtkTreeIter src_iter;
348       GtkListStore *src = g_array_index (src_array, GtkListStore*, i);
349       GtkListStore *dest;
350
351       /* Refuse to copy empty stores */
352       if (! gtk_tree_model_get_iter_first (GTK_TREE_MODEL (src),
353                                            &src_iter))
354         continue;
355
356       dest = clone_list_store (src);
357
358       g_array_append_val (dest_array, dest);
359     }
360
361   return dest_array;
362 }
363
364
365 static void
366 run_contrasts_dialog (struct oneway_anova_dialog *ow)
367 {
368   struct contrasts_subdialog *csd = &ow->contrasts;
369   gint response;
370
371   csd->temp_contrasts = clone_contrasts_array (ow->contrasts_array);
372
373   csd->c = 1;
374
375   push_new_store (csd->temp_contrasts, csd);
376
377   response = psppire_dialog_run (PSPPIRE_DIALOG (csd->contrasts_dialog));
378
379   if ( response == PSPPIRE_RESPONSE_CONTINUE )
380     {
381       ow->contrasts_array = clone_contrasts_array (csd->temp_contrasts);
382     }
383
384   /* Destroy the temp contrasts here */
385
386 }
387
388
389 static void
390 push_new_store (GArray *contrast_stack, struct contrasts_subdialog *csd)
391 {
392   GtkListStore *ls = gtk_list_store_new (1, G_TYPE_DOUBLE);
393
394   g_array_append_val (contrast_stack, ls);
395
396   g_signal_connect_swapped (ls, "row-deleted",
397                             G_CALLBACK (list_store_changed), csd);
398
399   g_signal_connect_swapped (ls, "row-changed",
400                             G_CALLBACK (list_store_changed), csd);
401
402   list_store_changed (csd);
403 }
404
405 static void
406 next (GtkWidget *widget, gpointer data)
407 {
408   struct contrasts_subdialog *csd = data;
409
410   if (csd->c >= csd->temp_contrasts->len)
411     push_new_store (csd->temp_contrasts, csd);
412
413   csd->c++;
414
415   list_store_changed (csd);
416 }
417
418
419 static void
420 prev (GtkWidget *widget, gpointer data)
421 {
422   struct contrasts_subdialog *csd = data;
423
424   if ( csd->c > 0 )
425     --csd->c;
426
427   list_store_changed (csd);
428 }
429
430