Blame demos/gtk-demo/editable_cells.c

Packit Service fb6fa5
/* Tree View/Editable Cells
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * This demo demonstrates the use of editable cells in a GtkTreeView. If
Packit Service fb6fa5
 * you're new to the GtkTreeView widgets and associates, look into
Packit Service fb6fa5
 * the GtkListStore example first. It also shows how to use the
Packit Service fb6fa5
 * GtkCellRenderer::editing-started signal to do custom setup of the
Packit Service fb6fa5
 * editable widget.
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * The cell renderers used in this demo are GtkCellRendererText, 
Packit Service fb6fa5
 * GtkCellRendererCombo and GtkCellRendererProgress.
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
#include <gtk/gtk.h>
Packit Service fb6fa5
#include <string.h>
Packit Service fb6fa5
#include <stdlib.h>
Packit Service fb6fa5
Packit Service fb6fa5
static GtkWidget *window = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
typedef struct
Packit Service fb6fa5
{
Packit Service fb6fa5
  gint   number;
Packit Service fb6fa5
  gchar *product;
Packit Service fb6fa5
  gint   yummy;
Packit Service fb6fa5
}
Packit Service fb6fa5
Item;
Packit Service fb6fa5
Packit Service fb6fa5
enum
Packit Service fb6fa5
{
Packit Service fb6fa5
  COLUMN_ITEM_NUMBER,
Packit Service fb6fa5
  COLUMN_ITEM_PRODUCT,
Packit Service fb6fa5
  COLUMN_ITEM_YUMMY,
Packit Service fb6fa5
  NUM_ITEM_COLUMNS
Packit Service fb6fa5
};
Packit Service fb6fa5
Packit Service fb6fa5
enum
Packit Service fb6fa5
{
Packit Service fb6fa5
  COLUMN_NUMBER_TEXT,
Packit Service fb6fa5
  NUM_NUMBER_COLUMNS
Packit Service fb6fa5
};
Packit Service fb6fa5
Packit Service fb6fa5
static GArray *articles = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
add_items (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  Item foo;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_if_fail (articles != NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 3;
Packit Service fb6fa5
  foo.product = "bottles of coke";
Packit Service fb6fa5
  foo.yummy = 20;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 5;
Packit Service fb6fa5
  foo.product = "packages of noodles";
Packit Service fb6fa5
  foo.yummy = 50;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 2;
Packit Service fb6fa5
  foo.product = "packages of chocolate chip cookies";
Packit Service fb6fa5
  foo.yummy = 90;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 1;
Packit Service fb6fa5
  foo.product = "can vanilla ice cream";
Packit Service fb6fa5
  foo.yummy = 60;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 6;
Packit Service fb6fa5
  foo.product = "eggs";
Packit Service fb6fa5
  foo.yummy = 10;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static GtkTreeModel *
Packit Service fb6fa5
create_items_model (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  gint i = 0;
Packit Service fb6fa5
  GtkListStore *model;
Packit Service fb6fa5
  GtkTreeIter iter;
Packit Service fb6fa5
Packit Service fb6fa5
  /* create array */
Packit Service fb6fa5
  articles = g_array_sized_new (FALSE, FALSE, sizeof (Item), 1);
Packit Service fb6fa5
Packit Service fb6fa5
  add_items ();
Packit Service fb6fa5
Packit Service fb6fa5
  /* create list store */
Packit Service fb6fa5
  model = gtk_list_store_new (NUM_ITEM_COLUMNS, G_TYPE_INT, G_TYPE_STRING,
Packit Service fb6fa5
                              G_TYPE_INT, G_TYPE_BOOLEAN);
Packit Service fb6fa5
Packit Service fb6fa5
  /* add items */
Packit Service fb6fa5
  for (i = 0; i < articles->len; i++)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gtk_list_store_append (model, &iter);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_list_store_set (model, &iter,
Packit Service fb6fa5
                          COLUMN_ITEM_NUMBER,
Packit Service fb6fa5
                          g_array_index (articles, Item, i).number,
Packit Service fb6fa5
                          COLUMN_ITEM_PRODUCT,
Packit Service fb6fa5
                          g_array_index (articles, Item, i).product,
Packit Service fb6fa5
                          COLUMN_ITEM_YUMMY,
Packit Service fb6fa5
                          g_array_index (articles, Item, i).yummy,
Packit Service fb6fa5
                          -1);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  return GTK_TREE_MODEL (model);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static GtkTreeModel *
Packit Service fb6fa5
create_numbers_model (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
#define N_NUMBERS 10
Packit Service fb6fa5
  gint i = 0;
Packit Service fb6fa5
  GtkListStore *model;
Packit Service fb6fa5
  GtkTreeIter iter;
Packit Service fb6fa5
Packit Service fb6fa5
  /* create list store */
Packit Service fb6fa5
  model = gtk_list_store_new (NUM_NUMBER_COLUMNS, G_TYPE_STRING, G_TYPE_INT);
Packit Service fb6fa5
Packit Service fb6fa5
  /* add numbers */
Packit Service fb6fa5
  for (i = 0; i < N_NUMBERS; i++)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      char str[2];
Packit Service fb6fa5
Packit Service fb6fa5
      str[0] = '0' + i;
Packit Service fb6fa5
      str[1] = '\0';
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_list_store_append (model, &iter);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_list_store_set (model, &iter,
Packit Service fb6fa5
                          COLUMN_NUMBER_TEXT, str,
Packit Service fb6fa5
                          -1);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  return GTK_TREE_MODEL (model);
Packit Service fb6fa5
Packit Service fb6fa5
#undef N_NUMBERS
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
add_item (GtkWidget *button, gpointer data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  Item foo;
Packit Service fb6fa5
  GtkTreeIter iter;
Packit Service fb6fa5
  GtkTreeModel *model = (GtkTreeModel *)data;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_if_fail (articles != NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  foo.number = 0;
Packit Service fb6fa5
  foo.product = g_strdup ("Description here");
Packit Service fb6fa5
  foo.yummy = 50;
Packit Service fb6fa5
  g_array_append_vals (articles, &foo, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_list_store_append (GTK_LIST_STORE (model), &iter);
Packit Service fb6fa5
  gtk_list_store_set (GTK_LIST_STORE (model), &iter,
Packit Service fb6fa5
                      COLUMN_ITEM_NUMBER, foo.number,
Packit Service fb6fa5
                      COLUMN_ITEM_PRODUCT, foo.product,
Packit Service fb6fa5
                      COLUMN_ITEM_YUMMY, foo.yummy,
Packit Service fb6fa5
                      -1);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
remove_item (GtkWidget *widget, gpointer data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GtkTreeIter iter;
Packit Service fb6fa5
  GtkTreeView *treeview = (GtkTreeView *)data;
Packit Service fb6fa5
  GtkTreeModel *model = gtk_tree_view_get_model (treeview);
Packit Service fb6fa5
  GtkTreeSelection *selection = gtk_tree_view_get_selection (treeview);
Packit Service fb6fa5
Packit Service fb6fa5
  if (gtk_tree_selection_get_selected (selection, NULL, &iter))
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gint i;
Packit Service fb6fa5
      GtkTreePath *path;
Packit Service fb6fa5
Packit Service fb6fa5
      path = gtk_tree_model_get_path (model, &iter);
Packit Service fb6fa5
      i = gtk_tree_path_get_indices (path)[0];
Packit Service fb6fa5
      gtk_list_store_remove (GTK_LIST_STORE (model), &iter);
Packit Service fb6fa5
Packit Service fb6fa5
      g_array_remove_index (articles, i);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_tree_path_free (path);
Packit Service fb6fa5
    }
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static gboolean
Packit Service fb6fa5
separator_row (GtkTreeModel *model,
Packit Service fb6fa5
               GtkTreeIter  *iter,
Packit Service fb6fa5
               gpointer      data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GtkTreePath *path;
Packit Service fb6fa5
  gint idx;
Packit Service fb6fa5
Packit Service fb6fa5
  path = gtk_tree_model_get_path (model, iter);
Packit Service fb6fa5
  idx = gtk_tree_path_get_indices (path)[0];
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_path_free (path);
Packit Service fb6fa5
Packit Service fb6fa5
  return idx == 5;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
editing_started (GtkCellRenderer *cell,
Packit Service fb6fa5
                 GtkCellEditable *editable,
Packit Service fb6fa5
                 const gchar     *path,
Packit Service fb6fa5
                 gpointer         data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  gtk_combo_box_set_row_separator_func (GTK_COMBO_BOX (editable), 
Packit Service fb6fa5
                                        separator_row, NULL, NULL);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
cell_edited (GtkCellRendererText *cell,
Packit Service fb6fa5
             const gchar         *path_string,
Packit Service fb6fa5
             const gchar         *new_text,
Packit Service fb6fa5
             gpointer             data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GtkTreeModel *model = (GtkTreeModel *)data;
Packit Service fb6fa5
  GtkTreePath *path = gtk_tree_path_new_from_string (path_string);
Packit Service fb6fa5
  GtkTreeIter iter;
Packit Service fb6fa5
Packit Service fb6fa5
  gint column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "column"));
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_model_get_iter (model, &iter, path);
Packit Service fb6fa5
Packit Service fb6fa5
  switch (column)
Packit Service fb6fa5
    {
Packit Service fb6fa5
    case COLUMN_ITEM_NUMBER:
Packit Service fb6fa5
      {
Packit Service fb6fa5
        gint i;
Packit Service fb6fa5
Packit Service fb6fa5
        i = gtk_tree_path_get_indices (path)[0];
Packit Service fb6fa5
        g_array_index (articles, Item, i).number = atoi (new_text);
Packit Service fb6fa5
Packit Service fb6fa5
        gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
Packit Service fb6fa5
                            g_array_index (articles, Item, i).number, -1);
Packit Service fb6fa5
      }
Packit Service fb6fa5
      break;
Packit Service fb6fa5
Packit Service fb6fa5
    case COLUMN_ITEM_PRODUCT:
Packit Service fb6fa5
      {
Packit Service fb6fa5
        gint i;
Packit Service fb6fa5
        gchar *old_text;
Packit Service fb6fa5
Packit Service fb6fa5
        gtk_tree_model_get (model, &iter, column, &old_text, -1);
Packit Service fb6fa5
        g_free (old_text);
Packit Service fb6fa5
Packit Service fb6fa5
        i = gtk_tree_path_get_indices (path)[0];
Packit Service fb6fa5
        g_free (g_array_index (articles, Item, i).product);
Packit Service fb6fa5
        g_array_index (articles, Item, i).product = g_strdup (new_text);
Packit Service fb6fa5
Packit Service fb6fa5
        gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
Packit Service fb6fa5
                            g_array_index (articles, Item, i).product, -1);
Packit Service fb6fa5
      }
Packit Service fb6fa5
      break;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_path_free (path);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
add_columns (GtkTreeView  *treeview, 
Packit Service fb6fa5
             GtkTreeModel *items_model,
Packit Service fb6fa5
             GtkTreeModel *numbers_model)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GtkCellRenderer *renderer;
Packit Service fb6fa5
Packit Service fb6fa5
  /* number column */
Packit Service fb6fa5
  renderer = gtk_cell_renderer_combo_new ();
Packit Service fb6fa5
  g_object_set (renderer,
Packit Service fb6fa5
                "model", numbers_model,
Packit Service fb6fa5
                "text-column", COLUMN_NUMBER_TEXT,
Packit Service fb6fa5
                "has-entry", FALSE,
Packit Service fb6fa5
                "editable", TRUE,
Packit Service fb6fa5
                NULL);
Packit Service fb6fa5
  g_signal_connect (renderer, "edited",
Packit Service fb6fa5
                    G_CALLBACK (cell_edited), items_model);
Packit Service fb6fa5
  g_signal_connect (renderer, "editing-started",
Packit Service fb6fa5
                    G_CALLBACK (editing_started), NULL);
Packit Service fb6fa5
  g_object_set_data (G_OBJECT (renderer), "column", GINT_TO_POINTER (COLUMN_ITEM_NUMBER));
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview),
Packit Service fb6fa5
                                               -1, "Number", renderer,
Packit Service fb6fa5
                                               "text", COLUMN_ITEM_NUMBER,
Packit Service fb6fa5
                                               NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  /* product column */
Packit Service fb6fa5
  renderer = gtk_cell_renderer_text_new ();
Packit Service fb6fa5
  g_object_set (renderer,
Packit Service fb6fa5
                "editable", TRUE,
Packit Service fb6fa5
                NULL);
Packit Service fb6fa5
  g_signal_connect (renderer, "edited",
Packit Service fb6fa5
                    G_CALLBACK (cell_edited), items_model);
Packit Service fb6fa5
  g_object_set_data (G_OBJECT (renderer), "column", GINT_TO_POINTER (COLUMN_ITEM_PRODUCT));
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview),
Packit Service fb6fa5
                                               -1, "Product", renderer,
Packit Service fb6fa5
                                               "text", COLUMN_ITEM_PRODUCT,
Packit Service fb6fa5
                                               NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  /* yummy column */
Packit Service fb6fa5
  renderer = gtk_cell_renderer_progress_new ();
Packit Service fb6fa5
  g_object_set_data (G_OBJECT (renderer), "column", GINT_TO_POINTER (COLUMN_ITEM_YUMMY));
Packit Service fb6fa5
Packit Service fb6fa5
  gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview),
Packit Service fb6fa5
                                               -1, "Yummy", renderer,
Packit Service fb6fa5
                                               "value", COLUMN_ITEM_YUMMY,
Packit Service fb6fa5
                                               NULL);
Packit Service fb6fa5
  
Packit Service fb6fa5
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GtkWidget *
Packit Service fb6fa5
do_editable_cells (GtkWidget *do_widget)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (!window)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      GtkWidget *vbox;
Packit Service fb6fa5
      GtkWidget *hbox;
Packit Service fb6fa5
      GtkWidget *sw;
Packit Service fb6fa5
      GtkWidget *treeview;
Packit Service fb6fa5
      GtkWidget *button;
Packit Service fb6fa5
      GtkTreeModel *items_model;
Packit Service fb6fa5
      GtkTreeModel *numbers_model;
Packit Service fb6fa5
Packit Service fb6fa5
      /* create window, etc */
Packit Service fb6fa5
      window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
Packit Service fb6fa5
      gtk_window_set_screen (GTK_WINDOW (window),
Packit Service fb6fa5
                             gtk_widget_get_screen (do_widget));
Packit Service fb6fa5
      gtk_window_set_title (GTK_WINDOW (window), "Shopping list");
Packit Service fb6fa5
      gtk_container_set_border_width (GTK_CONTAINER (window), 5);
Packit Service fb6fa5
      g_signal_connect (window, "destroy",
Packit Service fb6fa5
                        G_CALLBACK (gtk_widget_destroyed), &window);
Packit Service fb6fa5
Packit Service fb6fa5
      vbox = gtk_vbox_new (FALSE, 5);
Packit Service fb6fa5
      gtk_container_add (GTK_CONTAINER (window), vbox);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_box_pack_start (GTK_BOX (vbox),
Packit Service fb6fa5
                          gtk_label_new ("Shopping list (you can edit the cells!)"),
Packit Service fb6fa5
                          FALSE, FALSE, 0);
Packit Service fb6fa5
Packit Service fb6fa5
      sw = gtk_scrolled_window_new (NULL, NULL);
Packit Service fb6fa5
      gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw),
Packit Service fb6fa5
                                           GTK_SHADOW_ETCHED_IN);
Packit Service fb6fa5
      gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
Packit Service fb6fa5
                                      GTK_POLICY_AUTOMATIC,
Packit Service fb6fa5
                                      GTK_POLICY_AUTOMATIC);
Packit Service fb6fa5
      gtk_box_pack_start (GTK_BOX (vbox), sw, TRUE, TRUE, 0);
Packit Service fb6fa5
Packit Service fb6fa5
      /* create models */
Packit Service fb6fa5
      items_model = create_items_model ();
Packit Service fb6fa5
      numbers_model = create_numbers_model ();
Packit Service fb6fa5
Packit Service fb6fa5
      /* create tree view */
Packit Service fb6fa5
      treeview = gtk_tree_view_new_with_model (items_model);
Packit Service fb6fa5
      gtk_tree_view_set_rules_hint (GTK_TREE_VIEW (treeview), TRUE);
Packit Service fb6fa5
      gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (treeview)),
Packit Service fb6fa5
                                   GTK_SELECTION_SINGLE);
Packit Service fb6fa5
Packit Service fb6fa5
      add_columns (GTK_TREE_VIEW (treeview), items_model, numbers_model);
Packit Service fb6fa5
Packit Service fb6fa5
      g_object_unref (numbers_model);
Packit Service fb6fa5
      g_object_unref (items_model);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_container_add (GTK_CONTAINER (sw), treeview);
Packit Service fb6fa5
Packit Service fb6fa5
      /* some buttons */
Packit Service fb6fa5
      hbox = gtk_hbox_new (TRUE, 4);
Packit Service fb6fa5
      gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
Packit Service fb6fa5
Packit Service fb6fa5
      button = gtk_button_new_with_label ("Add item");
Packit Service fb6fa5
      g_signal_connect (button, "clicked",
Packit Service fb6fa5
                        G_CALLBACK (add_item), items_model);
Packit Service fb6fa5
      gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 0);
Packit Service fb6fa5
Packit Service fb6fa5
      button = gtk_button_new_with_label ("Remove item");
Packit Service fb6fa5
      g_signal_connect (button, "clicked",
Packit Service fb6fa5
                        G_CALLBACK (remove_item), treeview);
Packit Service fb6fa5
      gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 0);
Packit Service fb6fa5
Packit Service fb6fa5
      gtk_window_set_default_size (GTK_WINDOW (window), 320, 200);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  if (!gtk_widget_get_visible (window))
Packit Service fb6fa5
    gtk_widget_show_all (window);
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gtk_widget_destroy (window);
Packit Service fb6fa5
      window = NULL;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  return window;
Packit Service fb6fa5
}