Merge 'psppsheet' into 'master'.
[pspp] / src / ui / gui / aggregate-dialog.c
1 /* PSPPIRE - a graphical user interface for PSPP.
2    Copyright (C) 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 #include <config.h>
18
19 #include "dialog-common.h"
20
21 #include <gl/c-xvasprintf.h>
22 #include <language/stats/aggregate.h>
23
24 #include <ui/syntax-gen.h>
25 #include <libpspp/str.h>
26
27 #include "aggregate-dialog.h"
28 #include "psppire-selector.h"
29 #include "psppire-dictview.h"
30 #include "psppire-dialog.h"
31
32 #include "psppire-data-window.h"
33 #include "psppire-var-view.h"
34 #include "psppire-acr.h"
35
36 #include "dict-display.h"
37
38 #include "executor.h"
39 #include "builder-wrapper.h"
40 #include "helper.h"
41
42 #include <gtk/gtk.h>
43
44 #include "gettext.h"
45 #define _(msgid) gettext (msgid)
46 #define N_(msgid) msgid
47
48
49 enum
50   {
51     COMBO_MODEL_COL_DESC = 0,
52     COMBO_MODEL_COL_SYNTAX,
53     COMBO_MODEL_COL_SRC_VARS,
54     COMBO_MODEL_COL_ARITY
55   };
56
57
58 struct aggregate
59 {
60   GtkBuilder *xml;
61   PsppireDataWindow *de ;
62   PsppireDict *dict;
63   GtkWidget *break_variables;
64
65   GtkWidget *replace_radiobutton;
66   GtkWidget *add_radiobutton;
67   GtkWidget *filename_radiobutton;
68   GtkWidget *filename_button;
69   GtkWidget *filename_box;
70   GtkWidget *filename_label;
71
72   GtkWidget *function_combo;
73
74   GtkWidget *summary_acr;
75   GtkWidget *summary_var_name_entry;
76   GtkWidget *summary_var_label_entry;
77
78   GtkWidget *summary_sv;
79   GtkWidget *summary_sv_entry;
80
81   GtkWidget *summary_arg1;
82   GtkWidget *summary_arg2;
83
84   GtkWidget *summary_arg1_entry;
85   GtkWidget *summary_arg2_entry;
86
87   GtkWidget *sorted_button;
88   GtkWidget *needs_sort_button;
89
90   GtkWidget *pane;
91 };
92
93
94 static char * generate_syntax (const struct aggregate *rd);
95
96
97 static void update_arguments (struct aggregate *agg);
98
99
100 static void
101 refresh (struct aggregate *agg)
102 {
103   GtkTreeModel *liststore =
104     gtk_tree_view_get_model (GTK_TREE_VIEW (agg->break_variables));
105   gtk_list_store_clear (GTK_LIST_STORE (liststore));
106
107   gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (agg->add_radiobutton), TRUE);
108   gtk_label_set_text (GTK_LABEL (agg->filename_label), "");
109
110
111   gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (agg->needs_sort_button), TRUE);
112
113   gtk_entry_set_text (GTK_ENTRY (agg->summary_sv_entry), "");
114   gtk_entry_set_text (GTK_ENTRY (agg->summary_arg1_entry), "");
115   gtk_entry_set_text (GTK_ENTRY (agg->summary_arg2_entry), "");
116   gtk_entry_set_text (GTK_ENTRY (agg->summary_var_label_entry), "");
117   gtk_entry_set_text (GTK_ENTRY (agg->summary_var_name_entry), "N_BREAK");
118   gtk_editable_select_region (GTK_EDITABLE (agg->summary_var_name_entry), 0, -1);
119
120   gtk_combo_box_set_active (GTK_COMBO_BOX (agg->function_combo), N);
121
122   gtk_list_store_clear (PSPPIRE_ACR (agg->summary_acr)->list_store);
123
124   update_arguments (agg);
125 }
126
127
128 static gboolean
129 dialog_state_valid (gpointer data)
130 {
131   GtkTreeIter iter;
132   const struct aggregate *agg = data;
133
134   GtkTreeModel *liststore =
135     gtk_tree_view_get_model (GTK_TREE_VIEW (agg->break_variables));
136
137   if ( ! gtk_tree_model_get_iter_first  (liststore, &iter))
138     return FALSE;
139
140   liststore = GTK_TREE_MODEL (PSPPIRE_ACR (agg->summary_acr)->list_store);
141   
142   if ( ! gtk_tree_model_get_iter_first (liststore, &iter))
143     return FALSE;
144
145   return TRUE;
146 }
147
148
149 static void
150 choose_filename (struct aggregate *fd)
151 {
152   GtkFileFilter *filter;
153
154   GtkWidget *dialog = gtk_file_chooser_dialog_new (_("Aggregate destination file"),
155                                                            GTK_WINDOW (fd->de),
156                                                            GTK_FILE_CHOOSER_ACTION_SAVE,
157                                                            GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
158                                                            GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
159                                                            NULL);
160   g_object_set (dialog, "local-only", FALSE, NULL);
161
162   gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog), TRUE);
163
164
165   filter = gtk_file_filter_new ();
166   gtk_file_filter_set_name (filter, _("System Files (*.sav)"));
167   gtk_file_filter_add_mime_type (filter, "application/x-spss-sav");
168   gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
169
170   filter = gtk_file_filter_new ();
171   gtk_file_filter_set_name (filter, _("Portable Files (*.por) "));
172   gtk_file_filter_add_mime_type (filter, "application/x-spss-por");
173   gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
174
175
176   if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT)
177     {
178       char *filename;
179
180       filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
181
182       gtk_label_set_text (GTK_LABEL (fd->filename_label), filename);
183
184       g_free (filename);
185     }
186
187
188   gtk_widget_destroy (dialog);
189 }
190
191
192 static void
193 populate_combo_model (GtkComboBox *cb)
194 {
195   GtkListStore *list =  gtk_list_store_new (4, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT, G_TYPE_INT);
196   GtkTreeIter iter;
197   const struct agr_func *af = agr_func_tab;
198   GtkCellRenderer *renderer ;
199
200   for (af = agr_func_tab; af->name; ++af)
201     {
202       const gchar *s = af->description;
203       if (s == NULL)
204         continue;
205
206       gtk_list_store_append (list, &iter);
207       gtk_list_store_set (list, &iter,
208                           COMBO_MODEL_COL_DESC, gettext (s),
209                           COMBO_MODEL_COL_SYNTAX, af->name,
210                           COMBO_MODEL_COL_SRC_VARS, af->src_vars,
211                           COMBO_MODEL_COL_ARITY, af->n_args,
212                           -1);
213     }
214
215   renderer = gtk_cell_renderer_text_new ();
216   gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (cb), renderer, FALSE);
217
218   gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (cb), renderer, "text", 0);
219
220   gtk_combo_box_set_model (GTK_COMBO_BOX (cb), GTK_TREE_MODEL (list));
221   g_object_unref (list);
222 }
223
224
225
226 /* Returns TRUE iff all the necessary controls have been set to
227    completely specify a summary function */
228 static gboolean
229 summary_complete (const struct aggregate *agg)
230 {
231   GtkTreeIter iter;
232   int n_args;
233   enum agr_src_vars src_vars;
234   gboolean ok;
235   GtkTreeModel *model = gtk_combo_box_get_model (GTK_COMBO_BOX (agg->function_combo));
236
237   if ( 0 == strcmp ("", gtk_entry_get_text (GTK_ENTRY (agg->summary_var_name_entry))))
238     return FALSE;
239
240   ok = gtk_combo_box_get_active_iter (GTK_COMBO_BOX (agg->function_combo), &iter);
241
242   if (! ok)
243     return FALSE;
244
245
246
247   gtk_tree_model_get  (model,
248                        &iter,
249                        COMBO_MODEL_COL_ARITY,   &n_args,
250                        COMBO_MODEL_COL_SRC_VARS, &src_vars,
251                        -1);
252
253   if ( src_vars == AGR_SV_YES )
254     {
255       if (0 == strcmp ("", gtk_entry_get_text (GTK_ENTRY (agg->summary_sv_entry))))
256         return FALSE;
257     }
258
259   if ( n_args >= 2)
260     {
261       if (0 == strcmp ("", gtk_entry_get_text (GTK_ENTRY (agg->summary_arg2_entry))))
262         return FALSE;
263     }
264
265   if ( n_args >= 1)
266     {
267       if (0 == strcmp ("", gtk_entry_get_text (GTK_ENTRY (agg->summary_arg1_entry))))
268         return FALSE;
269     }
270
271
272   return TRUE;
273 }
274
275
276 static void
277 append_summary_spec (const struct aggregate *agg, GtkTreeIter *iter, GString *string);
278
279
280 static void
281 render_summary   (GtkTreeViewColumn *tree_column,
282                   GtkCellRenderer *cell,
283                   GtkTreeModel *tree_model,
284                   GtkTreeIter *iter,
285                   gpointer data)
286 {
287   struct aggregate *agg = data;
288   
289   GString *string = g_string_new ("");
290
291   append_summary_spec (agg, iter, string);
292
293   
294   g_object_set (cell, "text", string->str, NULL);
295
296   g_string_free (string, TRUE);
297 }
298
299
300 /* Enable/Disable the summary variable ACR */
301 static void
302 update_acr (struct aggregate *agg)
303 {
304   gboolean ready = summary_complete (agg);
305
306   psppire_acr_set_enabled (PSPPIRE_ACR (agg->summary_acr), ready);
307 }
308
309
310 /* Update the sensitivity of the summary variable argument fields */
311 static void
312 update_arguments (struct aggregate *agg)
313 {
314   GtkTreeIter iter;
315
316   gboolean ok = gtk_combo_box_get_active_iter (GTK_COMBO_BOX (agg->function_combo), &iter);
317
318   if ( ok)
319     {
320       GtkTreeModel *model = gtk_combo_box_get_model (GTK_COMBO_BOX (agg->function_combo));
321       int n_args;
322       enum agr_src_vars src_vars;
323       gtk_tree_model_get  (model,
324                            &iter,
325                            COMBO_MODEL_COL_ARITY,   &n_args,
326                            COMBO_MODEL_COL_SRC_VARS, &src_vars,
327                            -1);
328
329       gtk_widget_set_sensitive (agg->summary_sv, src_vars != AGR_SV_NO);
330       gtk_widget_set_sensitive (agg->summary_arg2, n_args >= 2);
331       gtk_widget_set_sensitive (agg->summary_arg1, n_args >= 1);
332     }
333   else
334     {
335       gtk_widget_set_sensitive (agg->summary_sv,   FALSE);
336       gtk_widget_set_sensitive (agg->summary_arg2, FALSE);
337       gtk_widget_set_sensitive (agg->summary_arg1, FALSE);
338     }
339 }
340
341 enum 
342   {
343     SUMMARY_COL_VARNAME = 0,
344     SUMMARY_COL_VARLABEL,
345     SUMMARY_COL_FUNCIDX,
346     SUMMARY_COL_SRCVAR,
347     SUMMARY_COL_ARG1,
348     SUMMARY_COL_ARG2
349   };
350
351 /* Set VAL to the value appropriate for COL according to the
352    current state of the dialog */
353 static gboolean
354 get_summary_spec (gint col, GValue *val, gpointer data)
355 {
356   const struct aggregate *agg = data;
357   switch (col)
358     {
359     case SUMMARY_COL_VARNAME:
360       g_value_init (val, G_TYPE_STRING);
361       g_value_set_string (val, gtk_entry_get_text (GTK_ENTRY (agg->summary_var_name_entry)));
362       break;
363     case SUMMARY_COL_VARLABEL:
364       g_value_init (val, G_TYPE_STRING);
365       g_value_set_string (val, gtk_entry_get_text (GTK_ENTRY (agg->summary_var_label_entry)));
366       break;
367     case SUMMARY_COL_SRCVAR:
368       g_value_init (val, G_TYPE_STRING);
369       g_value_set_string (val, gtk_entry_get_text (GTK_ENTRY (agg->summary_sv_entry)));
370       break;
371     case SUMMARY_COL_FUNCIDX:
372       g_value_init (val, G_TYPE_INT);
373       g_value_set_int (val, gtk_combo_box_get_active (GTK_COMBO_BOX (agg->function_combo)));
374       break;
375     case SUMMARY_COL_ARG1:
376       {
377         const gchar *text = gtk_entry_get_text (GTK_ENTRY (agg->summary_arg1_entry));
378         g_value_init (val, G_TYPE_DOUBLE);
379         g_value_set_double (val, g_strtod (text, 0));
380       }
381       break;
382     case SUMMARY_COL_ARG2:
383       {
384         const gchar *text = gtk_entry_get_text (GTK_ENTRY (agg->summary_arg2_entry));
385         g_value_init (val, G_TYPE_DOUBLE);
386         g_value_set_double (val, g_strtod (text, 0));
387       }
388       break;
389     default:
390       g_assert_not_reached ();
391       break;
392     }
393
394   return TRUE;
395 }
396
397
398 /* Update the status of the dialog box according to what row of the ACR's treeview
399    is selected */
400 static  void
401 on_acr_change (const struct aggregate *agg, GtkTreeView *tv)
402 {
403   const gchar *varname;
404   const gchar *label;
405   const gchar *srcvar;
406   gint f_idx;
407   double arg1, arg2;
408   gchar *text = NULL;
409     
410   GtkTreeIter iter;
411   GtkTreePath *path = NULL;
412   GtkTreeModel *model = gtk_tree_view_get_model (tv);
413   gtk_tree_view_get_cursor (tv, &path, NULL);
414   
415   gtk_tree_model_get_iter (model, &iter, path);
416
417   gtk_tree_model_get (model, &iter,
418                       SUMMARY_COL_VARNAME, &varname,
419                       SUMMARY_COL_VARLABEL, &label,
420                       SUMMARY_COL_FUNCIDX, &f_idx,
421                       SUMMARY_COL_SRCVAR, &srcvar,
422                       SUMMARY_COL_ARG1, &arg1,
423                       SUMMARY_COL_ARG2, &arg2,
424                       -1);
425
426   gtk_entry_set_text (GTK_ENTRY (agg->summary_var_name_entry), varname);
427   gtk_entry_set_text (GTK_ENTRY (agg->summary_var_label_entry), label);
428   gtk_entry_set_text (GTK_ENTRY (agg->summary_sv_entry), srcvar);
429   
430   text = c_xasprintf ("%g", arg1);
431   gtk_entry_set_text (GTK_ENTRY (agg->summary_arg1_entry), text);
432   g_free (text);
433
434   text = c_xasprintf ("%g", arg2);
435   gtk_entry_set_text (GTK_ENTRY (agg->summary_arg2_entry), text);
436   g_free (text);
437
438   gtk_combo_box_set_active (GTK_COMBO_BOX (agg->function_combo), f_idx);
439 }
440
441
442 /* Set the pane to 50% of its maximum size */
443 static void
444 set_initial_pos (GtkPaned *pane)
445 {
446   int max_pos;
447   g_object_get (pane, 
448                 "max-position", &max_pos,
449                 NULL);
450
451   gtk_paned_set_position (pane, max_pos);
452 }
453
454
455
456 /* Pops up the Aggregate dialog box */
457 void
458 aggregate_dialog (PsppireDataWindow *dw)
459 {
460   struct aggregate fd;
461   gint response;
462
463   GtkWidget *dialog ;
464   GtkWidget *source ;
465
466   GtkWidget *break_selector ;
467
468   fd.xml = builder_new ("aggregate.ui");
469
470   fd.de = dw;
471
472   dialog = get_widget_assert (fd.xml, "psppire-dialog1");
473   source = get_widget_assert   (fd.xml, "dict-view");
474   break_selector = get_widget_assert   (fd.xml, "break-selector");
475
476   fd.pane = get_widget_assert (fd.xml, "hbox1");
477   
478   fd.break_variables = get_widget_assert (fd.xml, "psppire-var-view1");
479   fd.filename_radiobutton = get_widget_assert (fd.xml, "filename-radiobutton");
480   fd.filename_button = get_widget_assert (fd.xml, "filename-button");
481   fd.filename_box = get_widget_assert (fd.xml, "filename-box");
482   fd.filename_label = get_widget_assert (fd.xml, "filename-label");
483   fd.replace_radiobutton = get_widget_assert (fd.xml, "replace-radiobutton");
484   fd.add_radiobutton = get_widget_assert (fd.xml, "add-radiobutton");
485   fd.function_combo = get_widget_assert (fd.xml, "function-combo");
486
487   fd.summary_acr = get_widget_assert (fd.xml, "psppire-acr1");
488   fd.summary_var_name_entry = get_widget_assert (fd.xml, "summary-var-name-entry");
489
490   fd.summary_arg1 = get_widget_assert (fd.xml, "summary-arg1");
491   fd.summary_arg2 = get_widget_assert (fd.xml, "summary-arg2");
492
493   fd.summary_arg1_entry = get_widget_assert (fd.xml, "summary-arg-entry1");
494   fd.summary_arg2_entry = get_widget_assert (fd.xml, "summary-arg-entry2");
495
496   fd.summary_var_label_entry = get_widget_assert (fd.xml, "summary-var-label-entry");
497
498   fd.summary_sv = get_widget_assert (fd.xml, "source-var");
499   fd.summary_sv_entry = get_widget_assert (fd.xml, "source-var-entry");
500
501   fd.sorted_button = get_widget_assert (fd.xml, "sorted-radiobutton");
502   fd.needs_sort_button = get_widget_assert (fd.xml, "needs-sort-radiobutton");
503
504   {
505     GtkTreeViewColumn *column ;
506
507     GList *l ;
508
509     GtkCellRenderer *cell_renderer ;
510
511     GtkListStore *list = gtk_list_store_new (6,
512                                              G_TYPE_STRING,
513                                              G_TYPE_STRING,
514                                              G_TYPE_INT, 
515                                              G_TYPE_STRING,
516                                              G_TYPE_DOUBLE,
517                                              G_TYPE_DOUBLE);
518
519     psppire_acr_set_model (PSPPIRE_ACR (fd.summary_acr), list);
520     g_object_unref (list);
521
522     psppire_acr_set_get_value_func (PSPPIRE_ACR (fd.summary_acr),
523                                                  get_summary_spec, &fd);
524
525     column = gtk_tree_view_get_column (PSPPIRE_ACR (fd.summary_acr)->tv, 0);
526
527     l = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
528
529     cell_renderer = l->data;
530
531     gtk_tree_view_column_set_cell_data_func (column,
532                                              cell_renderer,
533                                              render_summary, 
534                                              &fd,
535                                              NULL);
536
537     g_signal_connect_swapped (PSPPIRE_ACR (fd.summary_acr)->tv,
538                               "cursor-changed", G_CALLBACK (on_acr_change), &fd);
539   }
540   
541   g_signal_connect_swapped (fd.summary_var_name_entry, "changed", G_CALLBACK (update_acr),  &fd);
542   g_signal_connect_swapped (fd.function_combo, "changed", G_CALLBACK (update_acr),  &fd);
543   g_signal_connect_swapped (fd.summary_sv_entry, "changed", G_CALLBACK (update_acr),  &fd);  
544   g_signal_connect_swapped (fd.summary_arg1_entry, "changed", G_CALLBACK (update_acr),  &fd);  
545   g_signal_connect_swapped (fd.summary_arg2_entry, "changed", G_CALLBACK (update_acr),  &fd);  
546
547
548   g_signal_connect (fd.pane, "realize", G_CALLBACK (set_initial_pos),  &fd);  
549
550
551   g_signal_connect_swapped (fd.function_combo, "changed",
552                             G_CALLBACK (update_arguments),  &fd);
553
554   populate_combo_model (GTK_COMBO_BOX (fd.function_combo));
555
556   g_signal_connect_swapped (dialog, "refresh", G_CALLBACK (refresh),  &fd);
557
558   gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (fd.de));
559
560   g_object_get (fd.de->data_editor, "dictionary", &fd.dict, NULL);
561   g_object_set (source, "model", fd.dict, NULL);
562
563
564   psppire_selector_set_filter_func (PSPPIRE_SELECTOR (break_selector), NULL);
565
566
567   psppire_dialog_set_valid_predicate (PSPPIRE_DIALOG (dialog),
568                                       dialog_state_valid, &fd);
569
570   g_signal_connect (fd.filename_radiobutton, "toggled",
571                     G_CALLBACK (set_sensitivity_from_toggle), fd.filename_box );
572
573   g_signal_connect_swapped (fd.filename_button, "clicked",
574                     G_CALLBACK (choose_filename), &fd);
575
576
577   response = psppire_dialog_run (PSPPIRE_DIALOG (dialog));
578
579   switch (response)
580     {
581     case GTK_RESPONSE_OK:
582       g_free (execute_syntax_string (dw, generate_syntax (&fd)));
583       break;
584     case PSPPIRE_RESPONSE_PASTE:
585       g_free (paste_syntax_to_window (generate_syntax (&fd)));
586       break;
587     default:
588       break;
589     }
590
591   g_object_unref (fd.xml);
592 }
593
594
595 \f
596
597 static void
598 append_destination_filename (const struct aggregate *agg, GString *gs)
599 {
600   if ( gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (agg->filename_radiobutton)))
601     {
602       struct string ss;
603       const gchar *s = gtk_label_get_text (GTK_LABEL (agg->filename_label));
604       ds_init_empty (&ss);
605       syntax_gen_string (&ss, ss_cstr (s));
606       g_string_append (gs, ds_cstr (&ss));
607       ds_destroy (&ss);
608     }
609   else
610     {
611       g_string_append (gs, "* ");
612
613       if ( gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (agg->replace_radiobutton)))
614         g_string_append (gs, "MODE=REPLACE");
615       else
616         g_string_append (gs, "MODE=ADDVARIABLES");
617     }
618 }
619
620 /* Append the syntax of the summary function pointed to by ITER to STRING */
621 static void
622 append_summary_spec (const struct aggregate *agg, GtkTreeIter *iter, GString *string)
623 {
624   GtkTreeIter combo_iter;
625   char *varname = NULL;
626   char *funcname = NULL;
627
628   GtkTreeModel *acr_model = GTK_TREE_MODEL (PSPPIRE_ACR (agg->summary_acr)->list_store);
629   GtkTreeModel *combo_model = gtk_combo_box_get_model (GTK_COMBO_BOX (agg->function_combo));
630
631
632   /* This is an index into the combo_model.  Its used to get the function name */
633   int f_idx;
634   double arg1, arg2;
635   int arity;
636   enum agr_src_vars has_src_vars;
637   gchar *label = NULL;
638   gchar *srcvar = NULL;
639
640   gtk_tree_model_get (acr_model, iter,
641                       SUMMARY_COL_VARNAME, &varname,
642                       SUMMARY_COL_VARLABEL, &label,
643                       SUMMARY_COL_FUNCIDX, &f_idx,
644                       SUMMARY_COL_SRCVAR, &srcvar,
645                       SUMMARY_COL_ARG1, &arg1,
646                       SUMMARY_COL_ARG2, &arg2,
647                       -1);
648
649   gtk_tree_model_iter_nth_child (combo_model, &combo_iter, NULL, f_idx);
650
651   gtk_tree_model_get (combo_model, &combo_iter,
652                       COMBO_MODEL_COL_SYNTAX, &funcname,
653                       COMBO_MODEL_COL_ARITY, &arity,
654                       COMBO_MODEL_COL_SRC_VARS, &has_src_vars,
655                       -1);
656
657   g_string_append (string, varname);
658
659   if (0 != strcmp ("", label))
660     {
661       struct string ss;
662       ds_init_empty (&ss);
663       syntax_gen_string (&ss, ss_cstr (label));
664       g_string_append (string, " ");
665       g_string_append (string, ds_cstr (&ss));
666       ds_destroy (&ss);
667     }
668     
669   g_string_append_printf (string, " = %s", funcname);
670
671   if ( has_src_vars != AGR_SV_NO)
672     {
673       struct string dss;
674       ds_init_cstr (&dss, " (");
675       
676       ds_put_cstr (&dss, srcvar);
677
678       if ( arity > 0)
679         ds_put_c_format (&dss, ", %g", arg1);
680
681       if ( arity > 1)
682         ds_put_c_format (&dss, ", %g", arg2);
683
684       ds_put_cstr (&dss, ")");
685
686       g_string_append (string, ds_cstr (&dss));
687
688       ds_destroy (&dss);
689     }
690
691    free (label);
692    free (srcvar);
693    free (varname);
694    free (funcname);
695 }
696
697
698
699 static void
700 append_summary_variable_syntax (const struct aggregate *agg,  GString *string)
701 {
702   GtkTreeIter iter;
703   GtkTreeModel *acr_model = GTK_TREE_MODEL (PSPPIRE_ACR (agg->summary_acr)->list_store);
704
705
706   gboolean ok;
707
708   for (ok = gtk_tree_model_get_iter_first (GTK_TREE_MODEL (acr_model), &iter);
709        ok ;
710        ok = gtk_tree_model_iter_next (GTK_TREE_MODEL (acr_model), &iter)
711        )
712     {
713       g_string_append (string, "\n\t/");
714
715       append_summary_spec (agg, &iter, string);
716     }
717 }
718
719
720 static char *
721 generate_syntax (const struct aggregate *agg)
722 {
723   gchar *text;
724
725   GString *string = g_string_new ("AGGREGATE OUTFILE=");
726
727   append_destination_filename (agg, string);
728
729   if ( gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (agg->sorted_button)))
730     g_string_append (string, "\n\t/PRESORTED");
731
732   g_string_append (string, "\n\t/BREAK=");
733
734   psppire_var_view_append_names (PSPPIRE_VAR_VIEW (agg->break_variables), 0, string);
735
736   append_summary_variable_syntax (agg, string);
737
738   g_string_append (string, ".\n");
739
740   text = string->str;
741
742   g_string_free (string, FALSE);
743
744   return text;
745 }