Added custom psppire-selector widget.
[pspp-builds.git] / src / ui / gui / transpose-dialog.c
1 /*
2     PSPPIRE --- A Graphical User Interface for PSPP
3     Copyright (C) 2007  Free Software Foundation
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
18     02110-1301, USA. */
19
20 #include <config.h>
21
22 #include "transpose-dialog.h"
23 #include "psppire-selector.h"
24 #include "psppire-dialog.h"
25 #include "helper.h"
26 #include "data-editor.h"
27 #include "dict-display.h"
28 #include <language/syntax-string-source.h>
29 #include "syntax-editor.h"
30
31 #include <gtk/gtk.h>
32 #include <glade/glade.h>
33
34 #include <gettext.h>
35
36 #define _(msgid) gettext (msgid)
37 #define N_(msgid) msgid
38
39
40 /* FIXME: These shouldn't be here */
41 #include <gtksheet/gtksheet.h>
42 #include "psppire-var-store.h"
43
44
45
46 static struct variable *
47 get_selected_variable (GtkTreeModel *treemodel,
48                        GtkTreeIter *iter,
49                        PsppireDict *dict)
50 {
51   struct variable *var;
52   GValue value = {0};
53
54   GtkTreePath *path = gtk_tree_model_get_path (treemodel, iter);
55
56   gtk_tree_model_get_value (treemodel, iter, 0, &value);
57
58   gtk_tree_path_free (path);
59
60   var =  psppire_dict_get_variable (dict, g_value_get_int (&value));
61
62   g_value_unset (&value);
63
64   return var;
65 }
66
67
68 /* A (*GtkTreeCellDataFunc) function.
69    This function expects TREEMODEL to hold G_TYPE_INT.  The ints it holds
70    are the indices of the variables in the dictionary, which DATA points to.
71    It renders the name of the variable into CELL.
72 */
73 static void
74 cell_var_name (GtkTreeViewColumn *tree_column,
75                GtkCellRenderer *cell,
76                GtkTreeModel *tree_model,
77                GtkTreeIter *iter,
78                gpointer data)
79 {
80   PsppireDict *dict = data;
81   struct variable *var;
82   gchar *name;
83
84   var = get_selected_variable (tree_model, iter, dict);
85
86   name = pspp_locale_to_utf8 (var_get_name (var), -1, NULL);
87   g_object_set (cell, "text", name, NULL);
88   g_free (name);
89 }
90
91 static gchar * generate_syntax (PsppireDict *dict, GladeXML *xml);
92
93 void
94 transpose_dialog (GObject *o, gpointer data)
95 {
96   gint response ;
97   struct data_editor *de = data;
98
99   GladeXML *xml = glade_xml_new (PKGDATADIR "/psppire.glade",
100                                  "transpose-dialog", NULL);
101
102   GtkSheet *var_sheet =
103     GTK_SHEET (get_widget_assert (de->xml, "variable_sheet"));
104
105   PsppireVarStore *vs = PSPPIRE_VAR_STORE (gtk_sheet_get_model (var_sheet));
106
107   GtkWidget *dialog = get_widget_assert (xml, "transpose-dialog");
108   GtkWidget *source = get_widget_assert (xml, "source-treeview");
109   GtkWidget *dest = get_widget_assert (xml, "variables-treeview");
110   GtkWidget *selector1 = get_widget_assert (xml, "psppire-selector2");
111   GtkWidget *selector2 = get_widget_assert (xml, "psppire-selector3");
112   GtkWidget *new_name_entry = get_widget_assert (xml, "new-name-entry");
113
114   attach_dictionary_to_treeview (GTK_TREE_VIEW (source),
115                                  vs->dict,
116                                  GTK_SELECTION_MULTIPLE, NULL);
117   {
118     GtkTreeViewColumn *col;
119     GtkListStore *dest_list = gtk_list_store_new (1, G_TYPE_INT);
120     GtkCellRenderer *renderer = gtk_cell_renderer_text_new ();
121
122     gtk_tree_view_set_model (GTK_TREE_VIEW (dest), GTK_TREE_MODEL (dest_list));
123
124     col = gtk_tree_view_column_new_with_attributes (_("Var"),
125                                                     renderer,
126                                                     "text",
127                                                     0,
128                                                     NULL);
129
130     gtk_tree_view_column_set_cell_data_func (col, renderer,
131                                              cell_var_name,
132                                              vs->dict, 0);
133
134     /* FIXME: make this a value in terms of character widths */
135     g_object_set (col, "min-width",  100, NULL);
136
137     gtk_tree_view_column_set_sizing (col, GTK_TREE_VIEW_COLUMN_FIXED);
138
139     gtk_tree_view_append_column (GTK_TREE_VIEW(dest), col);
140   }
141
142
143   psppire_selector_set_subjects (PSPPIRE_SELECTOR (selector1),
144                                  source, dest,
145                                  insert_source_row_into_tree_view,
146                                  NULL);
147
148
149   psppire_selector_set_subjects (PSPPIRE_SELECTOR (selector2),
150                                  source, new_name_entry,
151                                  insert_source_row_into_entry,
152                                  is_currently_in_entry);
153
154
155   response = psppire_dialog_run (PSPPIRE_DIALOG (dialog));
156
157   switch (response)
158     {
159     case GTK_RESPONSE_OK:
160       {
161         gchar *syntax = generate_syntax (vs->dict, xml);
162         struct getl_interface *sss = create_syntax_string_source (syntax);
163         execute_syntax (sss);
164
165         g_free (syntax);
166       }
167       break;
168     case PSPPIRE_RESPONSE_PASTE:
169       {
170         gchar *syntax = generate_syntax (vs->dict, xml);
171
172         struct syntax_editor *se =
173           (struct syntax_editor *) window_create (WINDOW_SYNTAX, NULL);
174
175         gtk_text_buffer_insert_at_cursor (se->buffer, syntax, -1);
176
177         g_free (syntax);
178       }
179       break;
180     default:
181       break;
182     }
183
184   g_object_unref (xml);
185 }
186
187
188   /*
189      FLIP /VARIABLES=var_list /NEWNAMES=var_name.
190   */
191 static gchar *
192 generate_syntax (PsppireDict *dict, GladeXML *xml)
193 {
194   GtkTreeIter iter;
195   const gchar *text;
196   GString *string = g_string_new ("FLIP");
197   gchar *syntax ;
198
199   GtkWidget *dest = get_widget_assert (xml, "variables-treeview");
200   GtkWidget *entry = get_widget_assert (xml, "new-name-entry");
201
202   GtkTreeModel *list_store = gtk_tree_view_get_model (GTK_TREE_VIEW (dest));
203
204   if ( gtk_tree_model_get_iter_first (list_store, &iter) )
205     {
206       g_string_append (string, " /VARIABLES =");
207       do
208         {
209           GValue value = {0};
210           struct variable *var;
211           GtkTreePath *path = gtk_tree_model_get_path (list_store, &iter);
212
213           gtk_tree_model_get_value (list_store, &iter, 0, &value);
214
215           var = psppire_dict_get_variable (dict, g_value_get_int (&value));
216           g_value_unset (&value);
217
218           g_string_append (string, " ");
219           g_string_append (string, var_get_name (var));
220
221           gtk_tree_path_free (path);
222         }
223       while (gtk_tree_model_iter_next (list_store, &iter));
224     }
225
226   text = gtk_entry_get_text (GTK_ENTRY (entry));
227
228   if ( text)
229     g_string_append_printf (string, " /NEWNAME = %s", text);
230
231   g_string_append (string, ".");
232
233   syntax = string->str;
234
235   g_string_free (string, FALSE);
236
237   return syntax;
238 }