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