New module psppire-var-view
[pspp-builds.git] / src / ui / gui / dialog-common.c
index 3f72522117a08dd8034605125d5a130fc3ca45c7..7edbbe504920a594e22bba139dba576f9e45b213 100644 (file)
    along with this program.  If not, see <http://www.gnu.org/licenses/>. */
 
 #include <config.h>
+
+#include <libpspp/i18n.h>
 #include "dialog-common.h"
 
+#include "psppire-var-ptr.h"
+
 #include "helper.h"
 
+/* A (*GtkTreeCellDataFunc) function.
+   This function expects TREEMODEL to hold G_TYPE_BOXED, which is a pointer to a variable.
+   It renders the name of the variable into CELL.
+*/
+void
+XXX_cell_var_name (GtkTreeViewColumn *tree_column,
+                  GtkCellRenderer *cell,
+                  GtkTreeModel *treemodel,
+                  GtkTreeIter *iter,
+                  gpointer data)
+{
+  const struct variable *var;
+  GValue value = {0};
+
+  GtkTreePath *path = gtk_tree_model_get_path (treemodel, iter);
+
+  gtk_tree_model_get_value (treemodel, iter, 0, &value);
+
+  gtk_tree_path_free (path);
+
+  var = g_value_get_boxed (&value);
 
-/* Append the names of selected variables to STRING.
-   TREEVIEW is the treeview containing the variables.
-   DICT is the dictionary for those variables.
+  g_value_unset (&value);
+
+  g_object_set (cell, "text", var_get_name (var), NULL);
+}
+
+
+
+/* Returns FALSE if the variables represented by the union of the rows
+   currently selected by SOURCE widget, and contents of the DEST
+   widget, are of different types.
+
+   In other words, this function when passed as the argument to
+   psppire_selector_set_allow, ensures that the selector selects only
+   string  variables, or only numeric variables, not a mixture.
 */
-gint
-append_variable_names (GString *string,
-                      PsppireDict *dict, GtkTreeView *treeview)
+gboolean
+homogeneous_types (GtkWidget *source, GtkWidget *dest)
 {
-  gint n_vars = 0;
+  gboolean ok;
   GtkTreeIter iter;
+  gboolean retval = TRUE;
 
-  GtkTreeModel *list_store =
-    gtk_tree_view_get_model (treeview);
+  GtkTreeModel *model = gtk_tree_view_get_model (GTK_TREE_VIEW (source));
 
-  if ( gtk_tree_model_get_iter_first (list_store, &iter) )
+  PsppireDict *dict;
+  GtkTreeSelection *selection;
+  enum val_type type = -1;
+  GList *list, *l;
+
+  while (GTK_IS_TREE_MODEL_FILTER (model))
     {
-      do
-       {
-         GValue value = {0};
-         struct variable *var;
-         GtkTreePath *path = gtk_tree_model_get_path (list_store, &iter);
+      model = gtk_tree_model_filter_get_model (GTK_TREE_MODEL_FILTER (model));
+    }
 
-         gtk_tree_model_get_value (list_store, &iter, 0, &value);
+  dict = PSPPIRE_DICT (model);
 
-         var = psppire_dict_get_variable (dict, g_value_get_int (&value));
-         g_value_unset (&value);
+  selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (source));
 
-         g_string_append (string, " ");
-         g_string_append (string, var_get_name (var));
+  list = gtk_tree_selection_get_selected_rows (selection, &model);
 
-         gtk_tree_path_free (path);
-         n_vars++;
-       }
-      while (gtk_tree_model_iter_next (list_store, &iter));
-    }
+  /* Iterate through the selection of the source treeview */
+  for (l = list; l ; l = l->next)
+    {
+      GtkTreePath *path = l->data;
 
-  return n_vars;
-}
+      GtkTreePath *fpath =
+       gtk_tree_model_filter_convert_path_to_child_path (GTK_TREE_MODEL_FILTER (model), path);
 
+      gint *idx = gtk_tree_path_get_indices (fpath);
 
+      const struct variable *v = psppire_dict_get_variable (dict, idx[0]);
 
-struct variable *
-get_selected_variable (GtkTreeModel *treemodel,
-                      GtkTreeIter *iter,
-                      PsppireDict *dict)
-{
-  struct variable *var;
-  GValue value = {0};
+      gtk_tree_path_free (fpath);
 
-  GtkTreePath *path = gtk_tree_model_get_path (treemodel, iter);
+      if ( type != -1 )
+       {
+         if ( var_get_type (v) != type )
+           {
+             retval = FALSE;
+             break;
+           }
+       }
 
-  gtk_tree_model_get_value (treemodel, iter, 0, &value);
+      type = var_get_type (v);
+    }
 
-  gtk_tree_path_free (path);
+  g_list_foreach (list, (GFunc) gtk_tree_path_free, NULL);
+  g_list_free (list);
 
-  var =  psppire_dict_get_variable (dict, g_value_get_int (&value));
+  if ( retval == FALSE )
+    return FALSE;
 
-  g_value_unset (&value);
+  /* now deal with the dest widget */
+  model = gtk_tree_view_get_model (GTK_TREE_VIEW (dest));
 
-  return var;
-}
+  for (ok = gtk_tree_model_get_iter_first (model, &iter);
+       ok;
+       ok = gtk_tree_model_iter_next (model, &iter))
+    {
+      const struct variable *v;
+      gtk_tree_model_get (model, &iter, 0, &v, -1);
 
+      if ( type != -1 )
+       {
+         if ( var_get_type (v) != type )
+           {
+             retval = FALSE;
+             break;
+           }
+       }
 
+      type = var_get_type (v);
+    }
 
+  return retval;
+}
 
-/* A (*GtkTreeCellDataFunc) function.
-   This function expects TREEMODEL to hold G_TYPE_INT.  The ints it holds
-   are the indices of the variables in the dictionary, which DATA points to.
-   It renders the name of the variable into CELL.
-*/
-void
-cell_var_name (GtkTreeViewColumn *tree_column,
-              GtkCellRenderer *cell,
-              GtkTreeModel *tree_model,
-              GtkTreeIter *iter,
-              gpointer data)
+
+
+/* Returns true iff the variable selected by SOURCE is numeric */
+gboolean
+numeric_only (GtkWidget *source, GtkWidget *dest)
 {
-  PsppireDict *dict = data;
-  struct variable *var;
-  gchar *name;
+  gboolean retval = TRUE;
 
-  var = get_selected_variable (tree_model, iter, dict);
+  GtkTreeModel *model = gtk_tree_view_get_model (GTK_TREE_VIEW (source));
 
-  name = pspp_locale_to_utf8 (var_get_name (var), -1, NULL);
-  g_object_set (cell, "text", name, NULL);
-  g_free (name);
-}
+  PsppireDict *dict;
+  GtkTreeSelection *selection;
+  GList *list, *l;
 
+  while (GTK_IS_TREE_MODEL_FILTER (model))
+    {
+      model = gtk_tree_model_filter_get_model (GTK_TREE_MODEL_FILTER (model));
+    }
 
+  dict = PSPPIRE_DICT (model);
 
-/* Set a model for DEST, which is an GtkListStore of g_int's
-   whose values are the indices into DICT */
-void
-set_dest_model (GtkTreeView *dest, PsppireDict *dict)
-{
-  GtkTreeViewColumn *col;
-  GtkListStore *dest_list = gtk_list_store_new (1, G_TYPE_INT);
-  GtkCellRenderer *renderer = gtk_cell_renderer_text_new ();
+  selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (source));
+
+  list = gtk_tree_selection_get_selected_rows (selection, &model);
+
+  /* Iterate through the selection of the source treeview */
+  for (l = list; l ; l = l->next)
+    {
+      GtkTreePath *path = l->data;
+      GtkTreePath *fpath = gtk_tree_model_filter_convert_path_to_child_path
+       (GTK_TREE_MODEL_FILTER (model), path);
 
-  gtk_tree_view_set_model (GTK_TREE_VIEW (dest), GTK_TREE_MODEL (dest_list));
+      gint *idx = gtk_tree_path_get_indices (fpath);
 
-  col = gtk_tree_view_column_new_with_attributes ("Var",
-                                                 renderer,
-                                                 "text",
-                                                 0,
-                                                 NULL);
+      const struct variable *v = psppire_dict_get_variable (dict, idx[0]);
 
-  gtk_tree_view_column_set_cell_data_func (col, renderer,
-                                          cell_var_name,
-                                          dict, 0);
+      gtk_tree_path_free (fpath);
 
-  /* FIXME: make this a value in terms of character widths */
-  g_object_set (col, "min-width",  100, NULL);
+      if ( var_is_alpha (v))
+       {
+         retval = FALSE;
+         break;
+       }
+    }
 
-  gtk_tree_view_column_set_sizing (col, GTK_TREE_VIEW_COLUMN_FIXED);
+  g_list_foreach (list, (GFunc) gtk_tree_path_free, NULL);
+  g_list_free (list);
 
-  gtk_tree_view_append_column (GTK_TREE_VIEW(dest), col);
+  return retval;
 }