comparison gtk/gtkcellview.c @ 14191:009db0b357b5

This is a hand-crafted commit to migrate across subversion revisions 16854:16861, due to some vagaries of the way the original renames were done. Witness that monotone can do in one revision what svn had to spread across several.
author Ethan Blanton <elb@pidgin.im>
date Sat, 16 Dec 2006 04:59:55 +0000
parents
children
comparison
equal deleted inserted replaced
14190:366be2ce35a7 14191:009db0b357b5
1 /* gtkellview.c
2 * Copyright (C) 2002, 2003 Kristian Rietveld <kris@gtk.org>
3 *
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
8 *
9 * This library 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 GNU
12 * Library General Public License for more details.
13 *
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
18 */
19
20 /*
21 #include <config.h>
22 */
23 #include "gtkcellview.h"
24 #include <gtk/gtkversion.h>
25 #if !GTK_CHECK_VERSION(2,6,0)
26 #if GTK_CHECK_VERSION(2,4,0)
27 #include <gtk/gtkcelllayout.h>
28 #else
29 #include "gtkcelllayout.h"
30 #endif
31 #include <gtk/gtksignal.h>
32 #include <gtk/gtkcellrenderertext.h>
33 #include <gtk/gtkcellrendererpixbuf.h>
34 #include <gobject/gmarshal.h>
35
36 #define P_(x) (x)
37
38 typedef struct _GtkCellViewCellInfo GtkCellViewCellInfo;
39 struct _GtkCellViewCellInfo
40 {
41 GtkCellRenderer *cell;
42
43 gint requested_width;
44 gint real_width;
45 guint expand : 1;
46 guint pack : 1;
47
48 GSList *attributes;
49
50 GtkCellLayoutDataFunc func;
51 gpointer func_data;
52 GDestroyNotify destroy;
53 };
54
55 struct _GtkCellViewPrivate
56 {
57 GtkTreeModel *model;
58 GtkTreeRowReference *displayed_row;
59 GList *cell_list;
60 gint spacing;
61
62 GdkColor background;
63 gboolean background_set;
64 };
65
66
67 static void gtk_cell_view_class_init (GtkCellViewClass *klass);
68 static void gtk_cell_view_cell_layout_init (GtkCellLayoutIface *iface);
69 static void gtk_cell_view_get_property (GObject *object,
70 guint param_id,
71 GValue *value,
72 GParamSpec *pspec);
73 static void gtk_cell_view_set_property (GObject *object,
74 guint param_id,
75 const GValue *value,
76 GParamSpec *pspec);
77 static void gtk_cell_view_init (GtkCellView *cellview);
78 static void gtk_cell_view_finalize (GObject *object);
79 static void gtk_cell_view_style_set (GtkWidget *widget,
80 GtkStyle *previous_style);
81 static void gtk_cell_view_size_request (GtkWidget *widget,
82 GtkRequisition *requisition);
83 static void gtk_cell_view_size_allocate (GtkWidget *widget,
84 GtkAllocation *allocation);
85 static gboolean gtk_cell_view_expose (GtkWidget *widget,
86 GdkEventExpose *event);
87 static void gtk_cell_view_set_valuesv (GtkCellView *cellview,
88 GtkCellRenderer *renderer,
89 va_list args);
90 static GtkCellViewCellInfo *gtk_cell_view_get_cell_info (GtkCellView *cellview,
91 GtkCellRenderer *renderer);
92 static void gtk_cell_view_set_cell_data (GtkCellView *cellview);
93
94
95 static void gtk_cell_view_cell_layout_pack_start (GtkCellLayout *layout,
96 GtkCellRenderer *renderer,
97 gboolean expand);
98 static void gtk_cell_view_cell_layout_pack_end (GtkCellLayout *layout,
99 GtkCellRenderer *renderer,
100 gboolean expand);
101 static void gtk_cell_view_cell_layout_add_attribute (GtkCellLayout *layout,
102 GtkCellRenderer *renderer,
103 const gchar *attribute,
104 gint column);
105 static void gtk_cell_view_cell_layout_clear (GtkCellLayout *layout);
106 static void gtk_cell_view_cell_layout_clear_attributes (GtkCellLayout *layout,
107 GtkCellRenderer *renderer);
108 static void gtk_cell_view_cell_layout_set_cell_data_func (GtkCellLayout *layout,
109 GtkCellRenderer *cell,
110 GtkCellLayoutDataFunc func,
111 gpointer func_data,
112 GDestroyNotify destroy);
113 static void gtk_cell_view_cell_layout_reorder (GtkCellLayout *layout,
114 GtkCellRenderer *cell,
115 gint position);
116
117
118 enum
119 {
120 PROP_0,
121 PROP_BACKGROUND,
122 PROP_BACKGROUND_GDK,
123 PROP_BACKGROUND_SET
124 };
125
126 static GtkObjectClass *parent_class = NULL;
127
128
129 GType
130 gtk_cell_view_get_type (void)
131 {
132 static GType cell_view_type = 0;
133
134 if (!cell_view_type)
135 {
136 static const GTypeInfo cell_view_info =
137 {
138 sizeof (GtkCellViewClass),
139 NULL, /* base_init */
140 NULL, /* base_finalize */
141 (GClassInitFunc) gtk_cell_view_class_init,
142 NULL, /* class_finalize */
143 NULL, /* class_data */
144 sizeof (GtkCellView),
145 0,
146 (GInstanceInitFunc) gtk_cell_view_init
147 };
148
149 static const GInterfaceInfo cell_layout_info =
150 {
151 (GInterfaceInitFunc) gtk_cell_view_cell_layout_init,
152 NULL,
153 NULL
154 };
155
156 cell_view_type = g_type_register_static (GTK_TYPE_WIDGET, "GaimGtkCellView",
157 &cell_view_info, 0);
158
159 g_type_add_interface_static (cell_view_type, GTK_TYPE_CELL_LAYOUT,
160 &cell_layout_info);
161 }
162
163 return cell_view_type;
164 }
165
166 static void
167 gtk_cell_view_class_init (GtkCellViewClass *klass)
168 {
169 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
170 GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
171
172 parent_class = g_type_class_peek_parent (klass);
173
174 gobject_class->get_property = gtk_cell_view_get_property;
175 gobject_class->set_property = gtk_cell_view_set_property;
176 gobject_class->finalize = gtk_cell_view_finalize;
177
178 widget_class->expose_event = gtk_cell_view_expose;
179 widget_class->size_allocate = gtk_cell_view_size_allocate;
180 widget_class->size_request = gtk_cell_view_size_request;
181 widget_class->style_set = gtk_cell_view_style_set;
182
183 /* properties */
184 g_object_class_install_property (gobject_class,
185 PROP_BACKGROUND,
186 g_param_spec_string ("background",
187 P_("Background color name"),
188 P_("Background color as a string"),
189 NULL,
190 G_PARAM_WRITABLE));
191 g_object_class_install_property (gobject_class,
192 PROP_BACKGROUND_GDK,
193 g_param_spec_boxed ("background_gdk",
194 P_("Background color"),
195 P_("Background color as a GdkColor"),
196 GDK_TYPE_COLOR,
197 G_PARAM_READABLE | G_PARAM_WRITABLE));
198
199 #define ADD_SET_PROP(propname, propval, nick, blurb) g_object_class_install_property (gobject_class, propval, g_param_spec_boolean (propname, nick, blurb, FALSE, G_PARAM_READABLE | G_PARAM_WRITABLE))
200
201 ADD_SET_PROP ("background_set", PROP_BACKGROUND_SET,
202 P_("Background set"),
203 P_("Whether this tag affects the background color"));
204 }
205
206 static void
207 gtk_cell_view_cell_layout_init (GtkCellLayoutIface *iface)
208 {
209 iface->pack_start = gtk_cell_view_cell_layout_pack_start;
210 iface->pack_end = gtk_cell_view_cell_layout_pack_end;
211 iface->clear = gtk_cell_view_cell_layout_clear;
212 iface->add_attribute = gtk_cell_view_cell_layout_add_attribute;
213 iface->set_cell_data_func = gtk_cell_view_cell_layout_set_cell_data_func;
214 iface->clear_attributes = gtk_cell_view_cell_layout_clear_attributes;
215 iface->reorder = gtk_cell_view_cell_layout_reorder;
216 }
217
218 static void
219 gtk_cell_view_get_property (GObject *object,
220 guint param_id,
221 GValue *value,
222 GParamSpec *pspec)
223 {
224 GtkCellView *view = GTK_CELL_VIEW (object);
225
226 switch (param_id)
227 {
228 case PROP_BACKGROUND_GDK:
229 {
230 GdkColor color;
231
232 color = view->priv->background;
233
234 g_value_set_boxed (value, &color);
235 }
236 break;
237 case PROP_BACKGROUND_SET:
238 g_value_set_boolean (value, view->priv->background_set);
239 break;
240 default:
241 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
242 break;
243 }
244 }
245
246 static void
247 gtk_cell_view_set_property (GObject *object,
248 guint param_id,
249 const GValue *value,
250 GParamSpec *pspec)
251 {
252 GtkCellView *view = GTK_CELL_VIEW (object);
253
254 switch (param_id)
255 {
256 case PROP_BACKGROUND:
257 {
258 GdkColor color;
259
260 if (!g_value_get_string (value))
261 gtk_cell_view_set_background_color (view, NULL);
262 else if (gdk_color_parse (g_value_get_string (value), &color))
263 gtk_cell_view_set_background_color (view, &color);
264 else
265 g_warning ("Don't know color `%s'", g_value_get_string (value));
266
267 g_object_notify (object, "background_gdk");
268 }
269 break;
270 case PROP_BACKGROUND_GDK:
271 gtk_cell_view_set_background_color (view, g_value_get_boxed (value));
272 break;
273 case PROP_BACKGROUND_SET:
274 view->priv->background_set = g_value_get_boolean (value);
275 break;
276 default:
277 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
278 break;
279 }
280 }
281
282 static void
283 gtk_cell_view_init (GtkCellView *cellview)
284 {
285 GTK_WIDGET_SET_FLAGS (cellview, GTK_NO_WINDOW);
286
287 cellview->priv = g_new0(GtkCellViewPrivate,1);
288 }
289
290 static void
291 gtk_cell_view_style_set (GtkWidget *widget,
292 GtkStyle *previous_style)
293 {
294 if (previous_style && GTK_WIDGET_REALIZED (widget))
295 gdk_window_set_background (widget->window,
296 &widget->style->base[GTK_WIDGET_STATE (widget)]);
297 }
298
299 static void
300 gtk_cell_view_finalize (GObject *object)
301 {
302 GtkCellView *cellview = GTK_CELL_VIEW (object);
303
304 gtk_cell_view_cell_layout_clear (GTK_CELL_LAYOUT (cellview));
305
306 if (cellview->priv->model)
307 g_object_unref (cellview->priv->model);
308
309 if (cellview->priv->displayed_row)
310 gtk_tree_row_reference_free (cellview->priv->displayed_row);
311
312 if (G_OBJECT_CLASS (parent_class)->finalize)
313 (* G_OBJECT_CLASS (parent_class)->finalize) (object);
314
315 g_free (cellview->priv);
316 }
317
318 static void
319 gtk_cell_view_size_request (GtkWidget *widget,
320 GtkRequisition *requisition)
321 {
322 GList *i;
323 gboolean first_cell = TRUE;
324 GtkCellView *cellview;
325
326 cellview = GTK_CELL_VIEW (widget);
327
328 requisition->width = 0;
329 requisition->height = 0;
330
331 if (cellview->priv->displayed_row)
332 gtk_cell_view_set_cell_data (cellview);
333
334 for (i = cellview->priv->cell_list; i; i = i->next)
335 {
336 gint width, height;
337 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
338
339 if (!info->cell->visible)
340 continue;
341
342 if (!first_cell)
343 requisition->width += cellview->priv->spacing;
344
345 gtk_cell_renderer_get_size (info->cell, widget, NULL, NULL, NULL,
346 &width, &height);
347
348 info->requested_width = width;
349 requisition->width += width;
350 requisition->height = MAX (requisition->height, height);
351
352 first_cell = FALSE;
353 }
354 }
355
356 static void
357 gtk_cell_view_size_allocate (GtkWidget *widget,
358 GtkAllocation *allocation)
359 {
360 GList *i;
361 gint expand_cell_count = 0;
362 gint full_requested_width = 0;
363 gint extra_space;
364 GtkCellView *cellview;
365
366 widget->allocation = *allocation;
367
368 cellview = GTK_CELL_VIEW (widget);
369
370 /* checking how much extra space we have */
371 for (i = cellview->priv->cell_list; i; i = i->next)
372 {
373 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
374
375 if (!info->cell->visible)
376 continue;
377
378 if (info->expand)
379 expand_cell_count++;
380
381 full_requested_width += info->requested_width;
382 }
383
384 extra_space = widget->allocation.width - full_requested_width;
385 if (extra_space < 0)
386 extra_space = 0;
387 else if (extra_space > 0 && expand_cell_count > 0)
388 extra_space /= expand_cell_count;
389
390 /* iterate list for PACK_START cells */
391 for (i = cellview->priv->cell_list; i; i = i->next)
392 {
393 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
394
395 if (info->pack == GTK_PACK_END)
396 continue;
397
398 if (!info->cell->visible)
399 continue;
400
401 info->real_width = info->requested_width + (info->expand ? extra_space : 0);
402 }
403
404 /* iterate list for PACK_END cells */
405 for (i = cellview->priv->cell_list; i; i = i->next)
406 {
407 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
408
409 if (info->pack == GTK_PACK_START)
410 continue;
411
412 if (!info->cell->visible)
413 continue;
414
415 info->real_width = info->requested_width + (info->expand ? extra_space : 0);
416 }
417 }
418
419 static gboolean
420 gtk_cell_view_expose (GtkWidget *widget,
421 GdkEventExpose *event)
422 {
423 GList *i;
424 GtkCellView *cellview;
425 GdkRectangle area;
426 GtkCellRendererState state;
427 gboolean rtl = (gtk_widget_get_direction(widget) == GTK_TEXT_DIR_RTL);
428
429 cellview = GTK_CELL_VIEW (widget);
430
431 if (! GTK_WIDGET_DRAWABLE (widget))
432 return FALSE;
433
434 /* "blank" background */
435 if (cellview->priv->background_set)
436 {
437 GdkGC *gc;
438
439 gc = gdk_gc_new (GTK_WIDGET (cellview)->window);
440 gdk_gc_set_rgb_fg_color (gc, &cellview->priv->background);
441
442 gdk_draw_rectangle (GTK_WIDGET (cellview)->window,
443 gc,
444 TRUE,
445
446 /*0, 0,*/
447 widget->allocation.x,
448 widget->allocation.y,
449
450 widget->allocation.width,
451 widget->allocation.height);
452
453 g_object_unref (G_OBJECT (gc));
454 }
455
456 /* set cell data (if available) */
457 if (cellview->priv->displayed_row)
458 gtk_cell_view_set_cell_data (cellview);
459 else if (cellview->priv->model)
460 return FALSE;
461
462 /* render cells */
463 area = widget->allocation;
464
465 /* we draw on our very own window, initialize x and y to zero */
466 area.x = widget->allocation.x + (rtl ? widget->allocation.width : 0);
467 area.y = widget->allocation.y;
468
469 if (GTK_WIDGET_STATE (widget) == GTK_STATE_PRELIGHT)
470 state = GTK_CELL_RENDERER_PRELIT;
471 else
472 state = 0;
473
474 /* PACK_START */
475 for (i = cellview->priv->cell_list; i; i = i->next)
476 {
477 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
478
479 if (info->pack == GTK_PACK_END)
480 continue;
481
482 if (!info->cell->visible)
483 continue;
484
485 area.width = info->real_width;
486 if (rtl)
487 area.x -= area.width;
488
489 gtk_cell_renderer_render (info->cell,
490 event->window,
491 widget,
492 /* FIXME! */
493 &area, &area, &event->area, state);
494
495 if (!rtl)
496 area.x += info->real_width;
497 }
498
499 area.x = rtl ? widget->allocation.x : (widget->allocation.x + widget->allocation.width);
500
501 /* PACK_END */
502 for (i = cellview->priv->cell_list; i; i = i->next)
503 {
504 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
505
506 if (info->pack == GTK_PACK_START)
507 continue;
508
509 if (!info->cell->visible)
510 continue;
511
512 area.width = info->real_width;
513 if (!rtl)
514 area.x -= area.width;
515
516 gtk_cell_renderer_render (info->cell,
517 widget->window,
518 widget,
519 /* FIXME ! */
520 &area, &area, &event->area, state);
521 if (rtl)
522 area.x += info->real_width;
523 }
524
525 return FALSE;
526 }
527
528 static GtkCellViewCellInfo *
529 gtk_cell_view_get_cell_info (GtkCellView *cellview,
530 GtkCellRenderer *renderer)
531 {
532 GList *i;
533
534 for (i = cellview->priv->cell_list; i; i = i->next)
535 {
536 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)i->data;
537
538 if (info->cell == renderer)
539 return info;
540 }
541
542 return NULL;
543 }
544
545 static void
546 gtk_cell_view_set_cell_data (GtkCellView *cellview)
547 {
548 GList *i;
549 GtkTreeIter iter;
550 GtkTreePath *path;
551
552 g_return_if_fail (cellview->priv->displayed_row != NULL);
553
554 path = gtk_tree_row_reference_get_path (cellview->priv->displayed_row);
555 gtk_tree_model_get_iter (cellview->priv->model, &iter, path);
556 gtk_tree_path_free (path);
557
558 for (i = cellview->priv->cell_list; i; i = i->next)
559 {
560 GSList *j;
561 GtkCellViewCellInfo *info = i->data;
562
563 g_object_freeze_notify (G_OBJECT (info->cell));
564
565 for (j = info->attributes; j && j->next; j = j->next->next)
566 {
567 gchar *property = j->data;
568 gint column = GPOINTER_TO_INT (j->next->data);
569 GValue value = {0, };
570
571 gtk_tree_model_get_value (cellview->priv->model, &iter,
572 column, &value);
573 g_object_set_property (G_OBJECT (info->cell),
574 property, &value);
575 g_value_unset (&value);
576 }
577
578 if (info->func)
579 (* info->func) (GTK_CELL_LAYOUT (cellview),
580 info->cell,
581 cellview->priv->model,
582 &iter,
583 info->func_data);
584
585 g_object_thaw_notify (G_OBJECT (info->cell));
586 }
587 }
588
589 /* GtkCellLayout implementation */
590 static void
591 gtk_cell_view_cell_layout_pack_start (GtkCellLayout *layout,
592 GtkCellRenderer *renderer,
593 gboolean expand)
594 {
595 GtkCellViewCellInfo *info;
596 GtkCellView *cellview = GTK_CELL_VIEW (layout);
597
598 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
599 g_return_if_fail (GTK_IS_CELL_RENDERER (renderer));
600 g_return_if_fail (!gtk_cell_view_get_cell_info (cellview, renderer));
601
602 g_object_ref (G_OBJECT (renderer));
603 gtk_object_sink (GTK_OBJECT (renderer));
604
605 info = g_new0 (GtkCellViewCellInfo, 1);
606 info->cell = renderer;
607 info->expand = expand ? TRUE : FALSE;
608 info->pack = GTK_PACK_START;
609
610 cellview->priv->cell_list = g_list_append (cellview->priv->cell_list, info);
611 }
612
613 static void
614 gtk_cell_view_cell_layout_pack_end (GtkCellLayout *layout,
615 GtkCellRenderer *renderer,
616 gboolean expand)
617 {
618 GtkCellViewCellInfo *info;
619 GtkCellView *cellview = GTK_CELL_VIEW (layout);
620
621 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
622 g_return_if_fail (GTK_IS_CELL_RENDERER (renderer));
623 g_return_if_fail (!gtk_cell_view_get_cell_info (cellview, renderer));
624
625 g_object_ref (G_OBJECT (renderer));
626 gtk_object_sink (GTK_OBJECT (renderer));
627
628 info = g_new0 (GtkCellViewCellInfo, 1);
629 info->cell = renderer;
630 info->expand = expand ? TRUE : FALSE;
631 info->pack = GTK_PACK_END;
632
633 cellview->priv->cell_list = g_list_append (cellview->priv->cell_list, info);
634 }
635
636 static void
637 gtk_cell_view_cell_layout_add_attribute (GtkCellLayout *layout,
638 GtkCellRenderer *renderer,
639 const gchar *attribute,
640 gint column)
641 {
642 GtkCellViewCellInfo *info;
643 GtkCellView *cellview = GTK_CELL_VIEW (layout);
644
645 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
646 info = gtk_cell_view_get_cell_info (cellview, renderer);
647 g_return_if_fail (info != NULL);
648
649 info->attributes = g_slist_prepend (info->attributes,
650 GINT_TO_POINTER (column));
651 info->attributes = g_slist_prepend (info->attributes,
652 g_strdup (attribute));
653 }
654
655 static void
656 gtk_cell_view_cell_layout_clear (GtkCellLayout *layout)
657 {
658 GtkCellView *cellview = GTK_CELL_VIEW (layout);
659
660 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
661
662 while (cellview->priv->cell_list)
663 {
664 GtkCellViewCellInfo *info = (GtkCellViewCellInfo *)cellview->priv->cell_list->data;
665
666 gtk_cell_view_cell_layout_clear_attributes (layout, info->cell);
667 g_object_unref (G_OBJECT (info->cell));
668 g_free (info);
669 cellview->priv->cell_list = g_list_delete_link (cellview->priv->cell_list,
670 cellview->priv->cell_list);
671 }
672 }
673
674 static void
675 gtk_cell_view_cell_layout_set_cell_data_func (GtkCellLayout *layout,
676 GtkCellRenderer *cell,
677 GtkCellLayoutDataFunc func,
678 gpointer func_data,
679 GDestroyNotify destroy)
680 {
681 GtkCellView *cellview = GTK_CELL_VIEW (layout);
682 GtkCellViewCellInfo *info;
683
684 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
685
686 info = gtk_cell_view_get_cell_info (cellview, cell);
687 g_return_if_fail (info != NULL);
688
689 if (info->destroy)
690 {
691 GDestroyNotify d = info->destroy;
692
693 info->destroy = NULL;
694 d (info->func_data);
695 }
696
697 info->func = func;
698 info->func_data = func_data;
699 info->destroy = destroy;
700 }
701
702 static void
703 gtk_cell_view_cell_layout_clear_attributes (GtkCellLayout *layout,
704 GtkCellRenderer *renderer)
705 {
706 GtkCellViewCellInfo *info;
707 GtkCellView *cellview = GTK_CELL_VIEW (layout);
708 GSList *list;
709
710 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
711 g_return_if_fail (GTK_IS_CELL_RENDERER (renderer));
712
713 info = gtk_cell_view_get_cell_info (cellview, renderer);
714 if (info != NULL)
715 {
716 list = info->attributes;
717 while (list && list->next)
718 {
719 g_free (list->data);
720 list = list->next->next;
721 }
722
723 g_slist_free (info->attributes);
724 info->attributes = NULL;
725 }
726 }
727
728 static void
729 gtk_cell_view_cell_layout_reorder (GtkCellLayout *layout,
730 GtkCellRenderer *cell,
731 gint position)
732 {
733 GList *link;
734 GtkCellViewCellInfo *info;
735 GtkCellView *cellview = GTK_CELL_VIEW (layout);
736
737 g_return_if_fail (GTK_IS_CELL_VIEW (cellview));
738 g_return_if_fail (GTK_IS_CELL_RENDERER (cell));
739
740 info = gtk_cell_view_get_cell_info (cellview, cell);
741
742 g_return_if_fail (info != NULL);
743 g_return_if_fail (position >= 0);
744
745 link = g_list_find (cellview->priv->cell_list, info);
746
747 g_return_if_fail (link != NULL);
748
749 cellview->priv->cell_list = g_list_remove_link (cellview->priv->cell_list,
750 link);
751 cellview->priv->cell_list = g_list_insert (cellview->priv->cell_list,
752 info, position);
753
754 gtk_widget_queue_draw (GTK_WIDGET (cellview));
755 }
756
757 /* public API */
758 GtkWidget *
759 gtk_cell_view_new (void)
760 {
761 GtkCellView *cellview;
762
763 cellview = GTK_CELL_VIEW (g_object_new (gtk_cell_view_get_type (), NULL));
764
765 return GTK_WIDGET (cellview);
766 }
767
768 GtkWidget *
769 gtk_cell_view_new_with_text (const gchar *text)
770 {
771 GtkCellView *cellview;
772 GtkCellRenderer *renderer;
773 GValue value = {0, };
774
775 cellview = GTK_CELL_VIEW (gtk_cell_view_new ());
776
777 renderer = gtk_cell_renderer_text_new ();
778 gtk_cell_view_cell_layout_pack_start (GTK_CELL_LAYOUT (cellview),
779 renderer, TRUE);
780
781 g_value_init (&value, G_TYPE_STRING);
782 g_value_set_string (&value, text);
783 gtk_cell_view_set_values (cellview, renderer, "text", &value, NULL);
784 g_value_unset (&value);
785
786 return GTK_WIDGET (cellview);
787 }
788
789 GtkWidget *
790 gtk_cell_view_new_with_markup (const gchar *markup)
791 {
792 GtkCellView *cellview;
793 GtkCellRenderer *renderer;
794 GValue value = {0, };
795
796 cellview = GTK_CELL_VIEW (gtk_cell_view_new ());
797
798 renderer = gtk_cell_renderer_text_new ();
799 gtk_cell_view_cell_layout_pack_start (GTK_CELL_LAYOUT (cellview),
800 renderer, TRUE);
801
802 g_value_init (&value, G_TYPE_STRING);
803 g_value_set_string (&value, markup);
804 gtk_cell_view_set_values (cellview, renderer, "markup", &value, NULL);
805 g_value_unset (&value);
806
807 return GTK_WIDGET (cellview);
808 }
809
810 GtkWidget *
811 gtk_cell_view_new_with_pixbuf (GdkPixbuf *pixbuf)
812 {
813 GtkCellView *cellview;
814 GtkCellRenderer *renderer;
815 GValue value = {0, };
816
817 cellview = GTK_CELL_VIEW (gtk_cell_view_new ());
818
819 renderer = gtk_cell_renderer_pixbuf_new ();
820 gtk_cell_view_cell_layout_pack_start (GTK_CELL_LAYOUT (cellview),
821 renderer, TRUE);
822
823 g_value_init (&value, GDK_TYPE_PIXBUF);
824 g_value_set_object (&value, pixbuf);
825 gtk_cell_view_set_values (cellview, renderer, "pixbuf", &value, NULL);
826 g_value_unset (&value);
827
828 return GTK_WIDGET (cellview);
829 }
830
831 void
832 gtk_cell_view_set_value (GtkCellView *cell_view,
833 GtkCellRenderer *renderer,
834 gchar *property,
835 GValue *value)
836 {
837 g_return_if_fail (GTK_IS_CELL_VIEW (cell_view));
838 g_return_if_fail (GTK_IS_CELL_RENDERER (renderer));
839
840 g_object_set_property (G_OBJECT (renderer), property, value);
841
842 /* force resize and redraw */
843 gtk_widget_queue_resize (GTK_WIDGET (cell_view));
844 gtk_widget_queue_draw (GTK_WIDGET (cell_view));
845 }
846
847 static void
848 gtk_cell_view_set_valuesv (GtkCellView *cell_view,
849 GtkCellRenderer *renderer,
850 va_list args)
851 {
852 gchar *attribute;
853 GValue *value;
854
855 attribute = va_arg (args, gchar *);
856
857 while (attribute)
858 {
859 value = va_arg (args, GValue *);
860 gtk_cell_view_set_value (cell_view, renderer, attribute, value);
861 attribute = va_arg (args, gchar *);
862 }
863 }
864
865 void
866 gtk_cell_view_set_values (GtkCellView *cell_view,
867 GtkCellRenderer *renderer,
868 ...)
869 {
870 va_list args;
871
872 g_return_if_fail (GTK_IS_CELL_VIEW (cell_view));
873 g_return_if_fail (GTK_IS_CELL_RENDERER (renderer));
874 g_return_if_fail (gtk_cell_view_get_cell_info (cell_view, renderer));
875
876 va_start (args, renderer);
877 gtk_cell_view_set_valuesv (cell_view, renderer, args);
878 va_end (args);
879 }
880
881 void
882 gtk_cell_view_set_model (GtkCellView *cell_view,
883 GtkTreeModel *model)
884 {
885 g_return_if_fail (GTK_IS_CELL_VIEW (cell_view));
886 g_return_if_fail (GTK_IS_TREE_MODEL (model));
887
888 if (cell_view->priv->model)
889 {
890 if (cell_view->priv->displayed_row)
891 gtk_tree_row_reference_free (cell_view->priv->displayed_row);
892 cell_view->priv->displayed_row = NULL;
893
894 g_object_unref (G_OBJECT (cell_view->priv->model));
895 cell_view->priv->model = NULL;
896 }
897
898 cell_view->priv->model = model;
899
900 if (cell_view->priv->model)
901 g_object_ref (G_OBJECT (cell_view->priv->model));
902 }
903
904 /**
905 * gtk_cell_view_set_displayed_row:
906 * @cell_view: a #GtkCellView
907 * @path: a #GtkTreePath or %NULL to unset.
908 *
909 * Sets the row of the model that is currently displayed
910 * by the #GtkCellView. If the path is unset, then the
911 * contents of the cellview "stick" at their last value;
912 * this is not normally a desired result, but may be
913 * a needed intermediate state if say, the model for
914 * the #GtkCellView becomes temporarily empty.
915 **/
916 void
917 gtk_cell_view_set_displayed_row (GtkCellView *cell_view,
918 GtkTreePath *path)
919 {
920 g_return_if_fail (GTK_IS_CELL_VIEW (cell_view));
921 g_return_if_fail (GTK_IS_TREE_MODEL (cell_view->priv->model));
922
923 if (cell_view->priv->displayed_row)
924 gtk_tree_row_reference_free (cell_view->priv->displayed_row);
925
926 if (path)
927 {
928 cell_view->priv->displayed_row =
929 gtk_tree_row_reference_new (cell_view->priv->model, path);
930 }
931 else
932 cell_view->priv->displayed_row = NULL;
933
934 /* force resize and redraw */
935 gtk_widget_queue_resize (GTK_WIDGET (cell_view));
936 gtk_widget_queue_draw (GTK_WIDGET (cell_view));
937 }
938
939 GtkTreePath *
940 gtk_cell_view_get_displayed_row (GtkCellView *cell_view)
941 {
942 g_return_val_if_fail (GTK_IS_CELL_VIEW (cell_view), NULL);
943
944 if (!cell_view->priv->displayed_row)
945 return NULL;
946
947 return gtk_tree_row_reference_get_path (cell_view->priv->displayed_row);
948 }
949
950 gboolean
951 gtk_cell_view_get_size_of_row (GtkCellView *cell_view,
952 GtkTreePath *path,
953 GtkRequisition *requisition)
954 {
955 GtkTreeRowReference *tmp;
956 GtkRequisition req;
957
958 g_return_val_if_fail (GTK_IS_CELL_VIEW (cell_view), FALSE);
959 g_return_val_if_fail (path != NULL, FALSE);
960 g_return_val_if_fail (requisition != NULL, FALSE);
961
962 tmp = cell_view->priv->displayed_row;
963 cell_view->priv->displayed_row =
964 gtk_tree_row_reference_new (cell_view->priv->model, path);
965
966 gtk_cell_view_size_request (GTK_WIDGET (cell_view), requisition);
967
968 gtk_tree_row_reference_free (cell_view->priv->displayed_row);
969 cell_view->priv->displayed_row = tmp;
970
971 /* restore actual size info */
972 gtk_cell_view_size_request (GTK_WIDGET (cell_view), &req);
973
974 return TRUE;
975 }
976
977 void
978 gtk_cell_view_set_background_color (GtkCellView *view,
979 const GdkColor *color)
980 {
981 g_return_if_fail (GTK_IS_CELL_VIEW (view));
982
983 if (color)
984 {
985 if (!view->priv->background_set)
986 {
987 view->priv->background_set = TRUE;
988 g_object_notify (G_OBJECT (view), "background_set");
989 }
990
991 view->priv->background = *color;
992 }
993 else
994 {
995 if (view->priv->background_set)
996 {
997 view->priv->background_set = FALSE;
998 g_object_notify (G_OBJECT (view), "background_set");
999 }
1000 }
1001 }
1002 #endif /* Gtk 2.6 */