Mercurial > pidgin
annotate src/gtkcellview.c @ 13918:61ba85cf05a6
[gaim-migrate @ 16421]
Paco-Paco thinks it would be good to make libgnt configure look for ncursesw.
Thanks to rekkanoryo and grim, we can do that.
committer: Tailor Script <tailor@pidgin.im>
| author | Sadrul Habib Chowdhury <imadil@gmail.com> |
|---|---|
| date | Tue, 04 Jul 2006 18:49:13 +0000 |
| parents | 7ab959eb857e |
| children |
| rev | line source |
|---|---|
| 10708 | 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 | |
|
13382
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
36 #define P_(x) (x) |
| 10708 | 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", | |
|
13382
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
187 P_("Background color name"), |
|
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
188 P_("Background color as a string"), |
| 10708 | 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", | |
|
13382
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
194 P_("Background color"), |
|
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
195 P_("Background color as a GdkColor"), |
| 10708 | 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, | |
|
13382
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
202 P_("Background set"), |
|
7ab959eb857e
[gaim-migrate @ 15755]
Richard Laager <rlaager@wiktel.com>
parents:
10708
diff
changeset
|
203 P_("Whether this tag affects the background color")); |
| 10708 | 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 */ |
