static guint signals [n_SIGNALS];
+/* Callback for when an item disappears from the source list.
+ By implication, this means that the item has been inserted into the
+ destination.
+ */
+static void
+on_row_deleted (PsppireSelector *selector)
+{
+ g_signal_emit (selector, signals [SELECTED], 0);
+}
+
+/* Callback for when a new item appears in the source list.
+ By implication, this means that an item has been deleted from the
+ destination.
+ */
+static void
+on_row_inserted (PsppireSelector *selector)
+{
+ g_signal_emit (selector, signals [DE_SELECTED], 0);
+}
+
GType
psppire_selector_get_type (void)
static void on_click (GtkButton *b);
+static void on_realize (GtkWidget *selector);
+
static void update_subjects (PsppireSelector *selector);
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
GtkButtonClass *button_class = GTK_BUTTON_CLASS (class);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
GParamSpec *orientation_spec =
g_param_spec_enum ("orientation",
"Orientation",
button_class->clicked = on_click;
+ widget_class->realize = on_realize;
+
object_class->set_property = psppire_selector_set_property;
object_class->get_property = psppire_selector_get_property;
static void
-on_realize (PsppireSelector *selector)
+on_realize (GtkWidget *w)
{
+ PsppireSelector *selector = PSPPIRE_SELECTOR (w);
PsppireSelectorClass *class = g_type_class_peek (PSPPIRE_SELECTOR_TYPE);
GtkTreeSelection* selection ;
GList *list = g_hash_table_lookup (class->source_hash, selector->source);
+ if (GTK_WIDGET_CLASS (parent_class)->realize)
+ GTK_WIDGET_CLASS (parent_class)->realize (w);
+
if ( NULL == list)
return;
selector->filter = NULL;
selector->arrow = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE);
- selector->filtered_source = NULL;
gtk_container_add (GTK_CONTAINER (selector), selector->arrow);
selector->row_activate_id = 0;
selector->source_select_id = 0;
-
- g_signal_connect (selector, "realize",
- G_CALLBACK (on_realize), NULL);
-
}
}
+/* Callback which causes the filter to be refiltered.
+ Called when the DEST GtkEntry is activated (Enter is pressed), or when it
+ looses focus.
+*/
+static gboolean
+refilter (PsppireSelector *selector)
+{
+ GtkTreeModel *model = gtk_tree_view_get_model (GTK_TREE_VIEW (selector->source));
+ gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (model));
+ return FALSE;
+}
+
/* Removes something from the DEST widget */
static void
de_select_selection (PsppireSelector *selector)
selector->selecting = FALSE;
- gtk_tree_model_filter_refilter (selector->filtered_source);
+ refilter (selector);
g_signal_emit (selector, signals [DE_SELECTED], 0);
}
GList *selected_rows =
gtk_tree_selection_get_selected_rows (selection, NULL);
- GtkTreeModel *childmodel = gtk_tree_model_filter_get_model
- (selector->filtered_source);
+ GtkTreeModel *childmodel =
+ gtk_tree_model_filter_get_model (GTK_TREE_MODEL_FILTER (gtk_tree_view_get_model (GTK_TREE_VIEW (selector->source))));
g_return_if_fail (selector->select_items);
GtkTreeIter iter;
GtkTreePath *path = item->data;
- gtk_tree_model_get_iter (GTK_TREE_MODEL (selector->filtered_source),
+ gtk_tree_model_get_iter (GTK_TREE_MODEL (gtk_tree_view_get_model (GTK_TREE_VIEW (selector->source))),
&iter, path);
gtk_tree_model_filter_convert_iter_to_child_iter
- (selector->filtered_source,
+ (GTK_TREE_MODEL_FILTER (gtk_tree_view_get_model (GTK_TREE_VIEW (selector->source))),
&child_iter,
&iter);
g_list_foreach (selected_rows, (GFunc) gtk_tree_path_free, NULL);
g_list_free (selected_rows);
- gtk_tree_model_filter_refilter (selector->filtered_source);
+ refilter (selector);
g_signal_emit (selector, signals [SELECTED], 0);
if (GTK_BUTTON_CLASS (parent_class)->clicked)
GTK_BUTTON_CLASS (parent_class)->clicked (b);
-
}
static gboolean
/* set the source widget to SOURCE */
static void
-set_tree_view_source (PsppireSelector *selector,
- GtkTreeView *source)
+set_tree_view_source (PsppireSelector *selector)
{
GList *list = NULL;
PsppireSelectorClass *class = g_type_class_peek (PSPPIRE_SELECTOR_TYPE);
- GtkTreeModel *model = gtk_tree_view_get_model (source);
-
- if ( ! (list = g_hash_table_lookup (class->source_hash, source)))
+ if ( ! (list = g_hash_table_lookup (class->source_hash, selector->source)))
{
- selector->filtered_source =
- GTK_TREE_MODEL_FILTER (gtk_tree_model_filter_new (model, NULL));
-
- gtk_tree_view_set_model (source,
- GTK_TREE_MODEL (selector->filtered_source));
-
list = g_list_append (list, selector);
- g_hash_table_insert (class->source_hash, source, list);
-
-
- gtk_tree_model_filter_set_visible_func (selector->filtered_source,
- is_source_item_visible,
- selector,
- NULL);
+ g_hash_table_insert (class->source_hash, selector->source, list);
}
else
{ /* Append this selector to the list and push the <source,list>
pair onto the hash table */
- selector->filtered_source = GTK_TREE_MODEL_FILTER (model);
-
if ( NULL == g_list_find (list, selector) )
{
if ( selector->primary_requested )
list = g_list_prepend (list, selector);
else
list = g_list_append (list, selector);
- g_hash_table_replace (class->source_hash, source, list);
+ g_hash_table_replace (class->source_hash, selector->source, list);
}
}
}
+
+/* This function is a callback which occurs when the
+ SOURCE's model has changed */
+static void
+update_model (
+ GtkTreeView *source,
+ GParamSpec *psp,
+ PsppireSelector *selector
+ )
+{
+ GtkTreeModel *model = gtk_tree_view_get_model (source);
+
+ g_assert (source == GTK_TREE_VIEW (selector->source));
+
+ if (model && (model == g_object_get_data (G_OBJECT (source), "model-copy")))
+ return;
+
+ if (model != NULL)
+ {
+ GtkTreeModel *new_model = gtk_tree_model_filter_new (model, NULL);
+
+ g_object_set_data (G_OBJECT (source), "model-copy", new_model);
+
+ gtk_tree_view_set_model (source, new_model);
+
+ gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER (new_model),
+ is_source_item_visible,
+ selector,
+ NULL);
+
+ g_signal_connect_swapped (new_model,
+ "row-deleted",
+ G_CALLBACK (on_row_deleted), selector);
+
+ g_signal_connect_swapped (new_model,
+ "row-inserted",
+ G_CALLBACK (on_row_inserted), selector);
+ }
+}
+
+
+
/*
Callback for when the destination treeview's data changes
*/
if ( selector->selecting) return;
- gtk_tree_model_filter_refilter (selector->filtered_source);
+ refilter (selector);
}
if ( selector->selecting ) return;
- gtk_tree_model_filter_refilter (selector->filtered_source);
+ refilter (selector);
}
}
-/* Callback which causes the filter to be refiltered.
- Called when the DEST GtkEntry is activated (Enter is pressed), or when it
- looses focus.
-*/
-static gboolean
-refilter (PsppireSelector *selector)
-{
- gtk_tree_model_filter_refilter (selector->filtered_source);
- return FALSE;
-}
/* Callback for when the DEST GtkEntry is selected (clicked) */
}
-/* Callback for when an item disappears from the source list.
- By implication, this means that the item has been inserted into the
- destination.
- */
-static void
-on_row_deleted (PsppireSelector *selector)
-{
- g_signal_emit (selector, signals [SELECTED], 0);
-}
-
-/* Callback for when a new item appears in the source list.
- By implication, this means that an item has been deleted from the
- destination.
- */
-static void
-on_row_inserted (PsppireSelector *selector)
-{
- g_signal_emit (selector, signals [DE_SELECTED], 0);
-}
-
-
/* Set DEST to be the destination GtkEntry widget */
static void
selector);
- g_signal_connect_swapped (selector->filtered_source, "row-deleted",
- G_CALLBACK (on_row_deleted), selector);
-
- g_signal_connect_swapped (selector->filtered_source, "row-inserted",
- G_CALLBACK (on_row_inserted), selector);
}
static void
}
}
+
static void
update_subjects (PsppireSelector *selector)
{
- GtkTreeModel *model = NULL;
-
if ( NULL == selector->dest )
return;
if ( NULL == selector->source )
return;
- g_signal_connect_swapped (selector->source, "notify::model",
- G_CALLBACK (update_subjects), selector);
-
- model = gtk_tree_view_get_model (GTK_TREE_VIEW (selector->source));
-
- if ( NULL == model)
- return;
+ if ( GTK_IS_TREE_VIEW (selector->source))
+ {
+ set_tree_view_source (selector);
+ g_signal_connect (selector->source, "notify::model",
+ G_CALLBACK (update_model), selector);
- if ( GTK_IS_TREE_VIEW (selector->source))
- set_tree_view_source (selector, GTK_TREE_VIEW (selector->source) );
+ update_model (GTK_TREE_VIEW (selector->source), 0, selector);
+ }
else
g_error ("Unsupported source widget: %s", G_OBJECT_TYPE_NAME (selector->source));