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