2 * This file is a part of hildon
4 * Copyright (C) 2009 Nokia Corporation, all rights reserved.
6 * Contact: Rodrigo Novo <rodrigo.novo@nokia.com>
8 * This widget is based on GtkViewport from gtk+
9 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU Lesser Public License as published by
14 * the Free Software Foundation; version 2 of the license.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU Lesser Public License for more details.
24 #include "hildon-viewport.h"
25 #include "hildon-pannable-area.h"
26 #include "hildon-marshalers.h"
27 #include "hildon-enum-types.h"
38 static void hildon_viewport_finalize (GObject *object);
39 static void hildon_viewport_destroy (GtkObject *object);
40 static void hildon_viewport_set_property (GObject *object,
44 static void hildon_viewport_get_property (GObject *object,
48 static void hildon_viewport_set_scroll_adjustments (HildonViewport *viewport,
49 GtkAdjustment *hadjustment,
50 GtkAdjustment *vadjustment);
51 static void hildon_viewport_realize (GtkWidget *widget);
52 static void hildon_viewport_unrealize (GtkWidget *widget);
53 static void hildon_viewport_paint (GtkWidget *widget,
55 static gint hildon_viewport_expose (GtkWidget *widget,
56 GdkEventExpose *event);
57 static void hildon_viewport_add (GtkContainer *container,
59 static void hildon_viewport_size_request (GtkWidget *widget,
60 GtkRequisition *requisition);
61 static void hildon_viewport_size_allocate (GtkWidget *widget,
62 GtkAllocation *allocation);
63 static void hildon_viewport_adjustment_value_changed (GtkAdjustment *adjustment,
65 static void hildon_viewport_style_set (GtkWidget *widget,
66 GtkStyle *previous_style);
67 static void hildon_viewport_paint_offscreen_children (cairo_t *cr,
69 GdkEventExpose *event,
72 static void hildon_viewport_expose_window (GdkWindow *window);
75 G_DEFINE_TYPE (HildonViewport, hildon_viewport, GTK_TYPE_BIN)
77 #define HILDON_VIEWPORT_PRIVATE(o) \
78 (G_TYPE_INSTANCE_GET_PRIVATE ((o), HILDON_TYPE_VIEWPORT, \
79 HildonViewportPrivate))
81 struct _HildonViewportPrivate {
88 hildon_viewport_class_init (HildonViewportClass *class)
90 GtkObjectClass *object_class;
91 GObjectClass *gobject_class;
92 GtkWidgetClass *widget_class;
93 GtkContainerClass *container_class;
95 g_type_class_add_private (class, sizeof (HildonViewportPrivate));
97 object_class = (GtkObjectClass*) class;
98 gobject_class = G_OBJECT_CLASS (class);
99 widget_class = (GtkWidgetClass*) class;
100 container_class = (GtkContainerClass*) class;
102 gobject_class->finalize = hildon_viewport_finalize;
103 gobject_class->set_property = hildon_viewport_set_property;
104 gobject_class->get_property = hildon_viewport_get_property;
105 object_class->destroy = hildon_viewport_destroy;
107 widget_class->realize = hildon_viewport_realize;
108 widget_class->unrealize = hildon_viewport_unrealize;
109 widget_class->expose_event = hildon_viewport_expose;
110 widget_class->size_request = hildon_viewport_size_request;
111 widget_class->size_allocate = hildon_viewport_size_allocate;
112 widget_class->style_set = hildon_viewport_style_set;
114 container_class->add = hildon_viewport_add;
116 class->set_scroll_adjustments = hildon_viewport_set_scroll_adjustments;
118 g_object_class_install_property (gobject_class,
120 g_param_spec_object ("hadjustment",
121 "Horizontal adjustment",
122 "The GtkAdjustment that determines the values of the horizontal position for this viewport",
124 G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
126 g_object_class_install_property (gobject_class,
128 g_param_spec_object ("vadjustment",
129 "Vertical adjustment",
130 "The GtkAdjustment that determines the values of the vertical position for this viewport",
132 G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
134 g_object_class_install_property (gobject_class,
136 g_param_spec_enum ("shadow-type",
138 "Determines how the shadowed box around the viewport is drawn",
139 GTK_TYPE_SHADOW_TYPE,
143 widget_class->set_scroll_adjustments_signal =
144 g_signal_new ("set_scroll_adjustments",
145 G_OBJECT_CLASS_TYPE (gobject_class),
146 G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
147 G_STRUCT_OFFSET (HildonViewportClass, set_scroll_adjustments),
149 _hildon_marshal_VOID__OBJECT_OBJECT,
152 GTK_TYPE_ADJUSTMENT);
156 hildon_viewport_set_property (GObject *object,
161 HildonViewport *viewport;
163 viewport = HILDON_VIEWPORT (object);
167 case PROP_HADJUSTMENT:
168 hildon_viewport_set_hadjustment (viewport, g_value_get_object (value));
170 case PROP_VADJUSTMENT:
171 hildon_viewport_set_vadjustment (viewport, g_value_get_object (value));
173 case PROP_SHADOW_TYPE:
174 hildon_viewport_set_shadow_type (viewport, g_value_get_enum (value));
177 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
183 hildon_viewport_get_property (GObject *object,
188 HildonViewport *viewport;
190 viewport = HILDON_VIEWPORT (object);
194 case PROP_HADJUSTMENT:
195 g_value_set_object (value, viewport->hadjustment);
197 case PROP_VADJUSTMENT:
198 g_value_set_object (value, viewport->vadjustment);
200 case PROP_SHADOW_TYPE:
201 g_value_set_enum (value, viewport->shadow_type);
204 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
210 hildon_viewport_init (HildonViewport *viewport)
212 HildonViewportPrivate *priv = HILDON_VIEWPORT_PRIVATE (viewport);
214 GTK_WIDGET_UNSET_FLAGS (viewport, GTK_NO_WINDOW);
216 gtk_widget_set_redraw_on_allocate (GTK_WIDGET (viewport), FALSE);
217 gtk_container_set_resize_mode (GTK_CONTAINER (viewport), GTK_RESIZE_QUEUE);
219 viewport->priv = priv;
220 viewport->shadow_type = GTK_SHADOW_IN;
221 viewport->view_window = NULL;
222 viewport->bin_window = NULL;
223 viewport->hadjustment = NULL;
224 viewport->vadjustment = NULL;
231 * hildon_viewport_new:
232 * @hadjustment: horizontal adjustment.
233 * @vadjustment: vertical adjustment.
234 * @returns: a new #HildonViewport.
236 * Creates a new #HildonViewport with the given adjustments.
240 hildon_viewport_new (GtkAdjustment *hadjustment,
241 GtkAdjustment *vadjustment)
245 viewport = gtk_widget_new (HILDON_TYPE_VIEWPORT,
246 "hadjustment", hadjustment,
247 "vadjustment", vadjustment,
253 #define ADJUSTMENT_POINTER(viewport, orientation) \
254 (((orientation) == GTK_ORIENTATION_HORIZONTAL) ? \
255 &(viewport)->hadjustment : &(viewport)->vadjustment)
258 viewport_disconnect_adjustment (HildonViewport *viewport,
259 GtkOrientation orientation)
261 GtkAdjustment **adjustmentp = ADJUSTMENT_POINTER (viewport, orientation);
265 g_signal_handlers_disconnect_by_func (*adjustmentp,
266 hildon_viewport_adjustment_value_changed,
268 g_object_unref (*adjustmentp);
274 hildon_viewport_finalize (GObject *object)
276 HildonViewport *viewport = HILDON_VIEWPORT (object);
278 viewport_disconnect_adjustment (viewport, GTK_ORIENTATION_HORIZONTAL);
279 viewport_disconnect_adjustment (viewport, GTK_ORIENTATION_VERTICAL);
281 G_OBJECT_CLASS (hildon_viewport_parent_class)->finalize (object);
285 hildon_viewport_destroy (GtkObject *object)
287 HildonViewport *viewport = HILDON_VIEWPORT (object);
289 viewport_disconnect_adjustment (viewport, GTK_ORIENTATION_HORIZONTAL);
290 viewport_disconnect_adjustment (viewport, GTK_ORIENTATION_VERTICAL);
292 GTK_OBJECT_CLASS (hildon_viewport_parent_class)->destroy (object);
296 * hildon_viewport_get_hadjustment:
297 * @viewport: a #HildonViewport.
299 * Returns the horizontal adjustment of the viewport.
301 * Return value: the horizontal adjustment of @viewport.
304 hildon_viewport_get_hadjustment (HildonViewport *viewport)
306 g_return_val_if_fail (HILDON_IS_VIEWPORT (viewport), NULL);
308 if (!viewport->hadjustment)
309 hildon_viewport_set_hadjustment (viewport, NULL);
311 return viewport->hadjustment;
315 * hildon_viewport_get_vadjustment:
316 * @viewport: a #HildonViewport.
318 * Returns the vertical adjustment of the viewport.
320 * Return value: the vertical adjustment of @viewport.
323 hildon_viewport_get_vadjustment (HildonViewport *viewport)
325 g_return_val_if_fail (HILDON_IS_VIEWPORT (viewport), NULL);
327 if (!viewport->vadjustment)
328 hildon_viewport_set_vadjustment (viewport, NULL);
330 return viewport->vadjustment;
334 viewport_get_view_allocation (HildonViewport *viewport,
335 GtkAllocation *view_allocation)
337 GtkWidget *widget = GTK_WIDGET (viewport);
338 GtkAllocation *allocation = &widget->allocation;
339 gint border_width = GTK_CONTAINER (viewport)->border_width;
341 view_allocation->x = 0;
342 view_allocation->y = 0;
344 if (viewport->shadow_type != GTK_SHADOW_NONE)
346 view_allocation->x = widget->style->xthickness;
347 view_allocation->y = widget->style->ythickness;
350 view_allocation->width = MAX (1, allocation->width - view_allocation->x * 2 - border_width * 2);
351 view_allocation->height = MAX (1, allocation->height - view_allocation->y * 2 - border_width * 2);
355 viewport_reclamp_adjustment (GtkAdjustment *adjustment,
356 gboolean *value_changed)
358 gdouble value = adjustment->value;
360 value = CLAMP (value, 0, adjustment->upper - adjustment->page_size);
361 if (value != adjustment->value)
363 adjustment->value = value;
365 *value_changed = TRUE;
367 else if (value_changed)
368 *value_changed = FALSE;
372 viewport_set_hadjustment_values (HildonViewport *viewport,
373 gboolean *value_changed)
375 GtkBin *bin = GTK_BIN (viewport);
376 GtkAllocation view_allocation;
377 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
378 gdouble old_page_size;
382 viewport_get_view_allocation (viewport, &view_allocation);
384 old_page_size = hadjustment->page_size;
385 old_upper = hadjustment->upper;
386 old_value = hadjustment->value;
387 hadjustment->page_size = view_allocation.width;
388 hadjustment->step_increment = view_allocation.width * 0.1;
389 hadjustment->page_increment = view_allocation.width * 0.9;
391 hadjustment->lower = 0;
393 if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
395 GtkRequisition child_requisition;
397 gtk_widget_get_child_requisition (bin->child, &child_requisition);
398 hadjustment->upper = MAX (child_requisition.width, view_allocation.width);
401 hadjustment->upper = view_allocation.width;
403 if (gtk_widget_get_direction (GTK_WIDGET (viewport)) == GTK_TEXT_DIR_RTL)
405 gdouble dist = old_upper - (old_value + old_page_size);
406 hadjustment->value = hadjustment->upper - dist - hadjustment->page_size;
407 viewport_reclamp_adjustment (hadjustment, value_changed);
408 *value_changed = (old_value != hadjustment->value);
411 viewport_reclamp_adjustment (hadjustment, value_changed);
415 viewport_set_vadjustment_values (HildonViewport *viewport,
416 gboolean *value_changed)
418 GtkBin *bin = GTK_BIN (viewport);
419 GtkAllocation view_allocation;
420 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
422 viewport_get_view_allocation (viewport, &view_allocation);
424 vadjustment->page_size = view_allocation.height;
425 vadjustment->step_increment = view_allocation.height * 0.1;
426 vadjustment->page_increment = view_allocation.height * 0.9;
428 vadjustment->lower = 0;
430 if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
432 GtkRequisition child_requisition;
434 gtk_widget_get_child_requisition (bin->child, &child_requisition);
435 vadjustment->upper = MAX (child_requisition.height, view_allocation.height);
438 vadjustment->upper = view_allocation.height;
440 viewport_reclamp_adjustment (vadjustment, value_changed);
444 viewport_set_adjustment (HildonViewport *viewport,
445 GtkOrientation orientation,
446 GtkAdjustment *adjustment)
448 GtkAdjustment **adjustmentp = ADJUSTMENT_POINTER (viewport, orientation);
449 gboolean value_changed;
451 if (adjustment && adjustment == *adjustmentp)
455 adjustment = GTK_ADJUSTMENT (gtk_adjustment_new (0.0, 0.0, 0.0,
457 viewport_disconnect_adjustment (viewport, orientation);
458 *adjustmentp = adjustment;
459 g_object_ref_sink (adjustment);
461 if (orientation == GTK_ORIENTATION_HORIZONTAL)
462 viewport_set_hadjustment_values (viewport, &value_changed);
464 viewport_set_vadjustment_values (viewport, &value_changed);
466 g_signal_connect (adjustment, "value_changed",
467 G_CALLBACK (hildon_viewport_adjustment_value_changed),
470 gtk_adjustment_changed (adjustment);
473 gtk_adjustment_value_changed (adjustment);
475 hildon_viewport_adjustment_value_changed (adjustment, viewport);
479 * hildon_viewport_set_hadjustment:
480 * @viewport: a #HildonViewport.
481 * @adjustment: a #GtkAdjustment.
483 * Sets the horizontal adjustment of the viewport.
486 hildon_viewport_set_hadjustment (HildonViewport *viewport,
487 GtkAdjustment *adjustment)
489 g_return_if_fail (HILDON_IS_VIEWPORT (viewport));
491 g_return_if_fail (GTK_IS_ADJUSTMENT (adjustment));
493 viewport_set_adjustment (viewport, GTK_ORIENTATION_HORIZONTAL, adjustment);
495 g_object_notify (G_OBJECT (viewport), "hadjustment");
499 * hildon_viewport_set_vadjustment:
500 * @viewport: a #HildonViewport.
501 * @adjustment: a #GtkAdjustment.
503 * Sets the vertical adjustment of the viewport.
506 hildon_viewport_set_vadjustment (HildonViewport *viewport,
507 GtkAdjustment *adjustment)
509 g_return_if_fail (HILDON_IS_VIEWPORT (viewport));
511 g_return_if_fail (GTK_IS_ADJUSTMENT (adjustment));
513 viewport_set_adjustment (viewport, GTK_ORIENTATION_VERTICAL, adjustment);
515 g_object_notify (G_OBJECT (viewport), "vadjustment");
519 hildon_viewport_set_scroll_adjustments (HildonViewport *viewport,
520 GtkAdjustment *hadjustment,
521 GtkAdjustment *vadjustment)
523 hildon_viewport_set_hadjustment (viewport, hadjustment);
524 hildon_viewport_set_vadjustment (viewport, vadjustment);
528 * hildon_viewport_set_shadow_type:
529 * @viewport: a #HildonViewport.
530 * @type: the new shadow type.
532 * Sets the shadow type of the viewport.
535 hildon_viewport_set_shadow_type (HildonViewport *viewport,
538 g_return_if_fail (HILDON_IS_VIEWPORT (viewport));
540 if ((GtkShadowType) viewport->shadow_type != type)
542 viewport->shadow_type = type;
544 if (GTK_WIDGET_VISIBLE (viewport))
546 gtk_widget_size_allocate (GTK_WIDGET (viewport), &(GTK_WIDGET (viewport)->allocation));
547 gtk_widget_queue_draw (GTK_WIDGET (viewport));
550 g_object_notify (G_OBJECT (viewport), "shadow-type");
555 * hildon_viewport_get_shadow_type:
556 * @viewport: a #HildonViewport
558 * Gets the shadow type of the #HildonViewport. See
559 * hildon_viewport_set_shadow_type().
561 * Return value: the shadow type
564 hildon_viewport_get_shadow_type (HildonViewport *viewport)
566 g_return_val_if_fail (HILDON_IS_VIEWPORT (viewport), GTK_SHADOW_NONE);
568 return viewport->shadow_type;
572 hildon_viewport_expose_window (GdkWindow *window)
576 gboolean is_double_buffered;
579 gdk_window_get_user_data (window, &user_data);
581 event.expose.type = GDK_EXPOSE;
582 event.expose.window = g_object_ref (window);
583 event.expose.send_event = FALSE;
584 event.expose.count = 0;
585 event.expose.area.x = 0;
586 event.expose.area.y = 0;
587 gdk_drawable_get_size (GDK_DRAWABLE (window),
588 &event.expose.area.width,
589 &event.expose.area.height);
590 event.expose.region = gdk_region_rectangle (&event.expose.area);
593 is_double_buffered = GTK_WIDGET_DOUBLE_BUFFERED (GTK_WIDGET (user_data));
595 is_double_buffered = FALSE;
597 /* If this is not double buffered, force a double buffer so that
598 redirection works. */
599 if (!is_double_buffered)
600 gdk_window_begin_paint_region (window, event.expose.region);
602 gtk_main_do_event (&event);
604 if (!is_double_buffered)
605 gdk_window_end_paint (window);
607 children = gdk_window_peek_children (window);
608 for (l = children; l != NULL; l = l->next)
610 GdkWindow *child = l->data;
612 /* Don't expose input-only windows */
613 if (gdk_drawable_get_depth (GDK_DRAWABLE (child)) != 0)
614 hildon_viewport_expose_window (l->data);
619 hildon_viewport_freeze (HildonViewport *viewport)
621 HildonViewportPrivate *priv = viewport->priv;
630 g_return_if_fail (GTK_WIDGET_REALIZED (viewport));
631 g_return_if_fail (priv->pixmap == NULL);
633 widget = GTK_WIDGET (viewport);
634 bin = GTK_BIN (viewport);
635 child = gtk_bin_get_child (bin);
637 gdk_drawable_get_size (viewport->bin_window, &w, &h);
639 event.expose.type = GDK_EXPOSE;
640 event.expose.window = g_object_ref (viewport->bin_window);
641 event.expose.send_event = FALSE;
642 event.expose.count = 0;
643 event.expose.area.x = 0;
644 event.expose.area.y = 0;
645 event.expose.area.width = w;
646 event.expose.area.height = h;
647 event.expose.region = gdk_region_rectangle (&event.expose.area);
649 /* ensure the children are drawn in their offscreen windows */
650 hildon_viewport_expose_window (viewport->bin_window);
652 cr = gdk_cairo_create (viewport->bin_window);
654 /* blit the children to the bin window */
655 hildon_viewport_paint_offscreen_children (cr,
656 viewport->bin_window,
657 (GdkEventExpose *)&event,
660 g_object_unref (viewport->bin_window);
664 priv->pixmap = gdk_pixmap_new (viewport->bin_window,
667 /* copy the bin window to a pixmap, consider XCompositeNameWindowPixmap */
668 gdk_draw_drawable (priv->pixmap,
669 widget->style->bg_gc[GTK_STATE_NORMAL],
670 viewport->bin_window,
673 gdk_window_hide (viewport->bin_window);
674 gtk_widget_queue_draw (widget);
678 hildon_viewport_thaw (HildonViewport *viewport)
680 HildonViewportPrivate *priv = viewport->priv;
681 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
682 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
686 widget = GTK_WIDGET (viewport);
689 g_object_unref (priv->pixmap);
693 new_x = - hadjustment->value;
694 new_y = - vadjustment->value;
696 gdk_window_show (viewport->bin_window);
697 gdk_window_move (viewport->bin_window, new_x, new_y);
702 hildon_viewport_realize (GtkWidget *widget)
704 HildonViewport *viewport = HILDON_VIEWPORT (widget);
705 GtkBin *bin = GTK_BIN (widget);
706 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
707 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
708 gint border_width = GTK_CONTAINER (widget)->border_width;
710 GtkAllocation view_allocation;
711 GdkWindowAttr attributes;
712 gint attributes_mask;
715 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
717 attributes.x = widget->allocation.x + border_width;
718 attributes.y = widget->allocation.y + border_width;
719 attributes.width = widget->allocation.width - border_width * 2;
720 attributes.height = widget->allocation.height - border_width * 2;
721 attributes.window_type = GDK_WINDOW_CHILD;
722 attributes.wclass = GDK_INPUT_OUTPUT;
723 attributes.visual = gtk_widget_get_visual (widget);
724 attributes.colormap = gtk_widget_get_colormap (widget);
726 event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
727 /* We select on button_press_mask so that button 4-5 scrolls are trapped.
729 attributes.event_mask = event_mask | GDK_BUTTON_PRESS_MASK;
731 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
733 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget),
734 &attributes, attributes_mask);
735 gdk_window_set_user_data (widget->window, viewport);
737 viewport_get_view_allocation (viewport, &view_allocation);
739 attributes.x = view_allocation.x;
740 attributes.y = view_allocation.y;
741 attributes.width = view_allocation.width;
742 attributes.height = view_allocation.height;
743 attributes.event_mask = event_mask;
745 viewport->view_window = gdk_window_new (widget->window, &attributes, attributes_mask);
746 gdk_window_set_user_data (viewport->view_window, viewport);
748 attributes.x = - hadjustment->value;
749 attributes.y = - vadjustment->value;
750 attributes.width = hadjustment->upper;
751 attributes.height = vadjustment->upper;
753 viewport->bin_window = gdk_window_new (viewport->view_window, &attributes, attributes_mask);
754 gdk_window_set_user_data (viewport->bin_window, viewport);
757 gtk_widget_set_parent_window (bin->child, viewport->bin_window);
759 widget->style = gtk_style_attach (widget->style, widget->window);
760 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
761 gtk_style_set_background (widget->style, viewport->view_window, GTK_STATE_NORMAL);
762 gtk_style_set_background (widget->style, viewport->bin_window, GTK_STATE_NORMAL);
764 /* Call paint here to allow a theme to set the background without flashing
766 gtk_paint_flat_box(widget->style, viewport->bin_window, GTK_STATE_NORMAL,
768 NULL, widget, "viewportbin",
771 gdk_window_set_composited (viewport->bin_window, TRUE);
773 gdk_window_show (viewport->bin_window);
774 gdk_window_show (viewport->view_window);
778 hildon_viewport_unrealize (GtkWidget *widget)
780 HildonViewport *viewport = HILDON_VIEWPORT (widget);
782 gdk_window_set_user_data (viewport->view_window, NULL);
783 gdk_window_destroy (viewport->view_window);
784 viewport->view_window = NULL;
786 gdk_window_set_user_data (viewport->bin_window, NULL);
787 gdk_window_destroy (viewport->bin_window);
788 viewport->bin_window = NULL;
790 if (GTK_WIDGET_CLASS (hildon_viewport_parent_class)->unrealize)
791 (* GTK_WIDGET_CLASS (hildon_viewport_parent_class)->unrealize) (widget);
795 hildon_viewport_paint (GtkWidget *widget,
798 HildonViewport *viewport;
800 g_return_if_fail (HILDON_IS_VIEWPORT (widget));
801 g_return_if_fail (area != NULL);
803 if (GTK_WIDGET_DRAWABLE (widget))
805 viewport = HILDON_VIEWPORT (widget);
807 gtk_paint_shadow (widget->style, widget->window,
808 GTK_STATE_NORMAL, viewport->shadow_type,
809 area, widget, "viewport",
815 hildon_viewport_paint_offscreen_children (cairo_t *cr,
817 GdkEventExpose *event,
823 for (subwindow = gdk_window_peek_children (window);
825 subwindow = subwindow->next)
827 /* avoid input-only windows */
828 if (gdk_drawable_get_depth (GDK_DRAWABLE (subwindow->data)) != 0)
832 gdk_window_get_position (subwindow->data,
838 gdk_drawable_get_size (subwindow->data, &w, &h);
840 /* do not draw the widgets outside the event */
841 if ((x+w < event->area.x) ||
842 (y+h < event->area.y) ||
843 (x > event->area.x+event->area.width) ||
844 (y > event->area.y+event->area.height))
847 gdk_cairo_set_source_pixmap (cr, subwindow->data, x, y);
849 cairo_rectangle (cr, x, y, w, h);
854 hildon_viewport_paint_offscreen_children (cr,
863 hildon_viewport_expose (GtkWidget *widget,
864 GdkEventExpose *event)
866 HildonViewport *viewport = HILDON_VIEWPORT (widget);
867 HildonViewportPrivate *priv = viewport->priv;
869 if (GTK_WIDGET_DRAWABLE (widget))
871 if (event->window == widget->window)
872 hildon_viewport_paint (widget, &event->area);
873 else if ((event->window == viewport->bin_window))
877 gtk_paint_flat_box(widget->style,
878 viewport->bin_window,
879 GTK_STATE_NORMAL, GTK_SHADOW_NONE,
880 &event->area, widget, "viewportbin",
883 (* GTK_WIDGET_CLASS (hildon_viewport_parent_class)->expose_event) (widget, event);
885 cr = gdk_cairo_create (viewport->bin_window);
887 hildon_viewport_paint_offscreen_children (cr,
888 viewport->bin_window,
893 else if (event->window == viewport->view_window)
895 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
896 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
898 gdk_draw_drawable (viewport->view_window,
899 widget->style->bg_gc[GTK_STATE_NORMAL],
900 priv->pixmap ? priv->pixmap : viewport->bin_window,
901 hadjustment->value + event->area.x,
902 vadjustment->value + event->area.y,
903 event->area.x, event->area.y,
904 event->area.width, event->area.height);
912 hildon_viewport_add (GtkContainer *container,
917 g_return_if_fail (GTK_IS_WIDGET (child));
919 bin = GTK_BIN (container);
920 g_return_if_fail (bin->child == NULL);
922 gtk_widget_set_parent_window (child, HILDON_VIEWPORT (bin)->bin_window);
924 GTK_CONTAINER_CLASS (hildon_viewport_parent_class)->add (container, child);
928 hildon_viewport_size_request (GtkWidget *widget,
929 GtkRequisition *requisition)
932 GtkRequisition child_requisition;
934 bin = GTK_BIN (widget);
936 requisition->width = GTK_CONTAINER (widget)->border_width;
938 requisition->height = GTK_CONTAINER (widget)->border_width;
940 if (HILDON_VIEWPORT (widget)->shadow_type != GTK_SHADOW_NONE)
942 requisition->width += 2 * widget->style->xthickness;
943 requisition->height += 2 * widget->style->ythickness;
946 if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
948 gtk_widget_size_request (bin->child, &child_requisition);
949 requisition->width += child_requisition.width;
950 requisition->height += child_requisition.height;
955 hildon_viewport_size_allocate (GtkWidget *widget,
956 GtkAllocation *allocation)
958 HildonViewport *viewport = HILDON_VIEWPORT (widget);
959 GtkBin *bin = GTK_BIN (widget);
960 gint border_width = GTK_CONTAINER (widget)->border_width;
961 gboolean hadjustment_value_changed, vadjustment_value_changed;
962 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
963 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
964 GtkAllocation child_allocation;
966 /* If our size changed, and we have a shadow, queue a redraw on widget->window to
967 * redraw the shadow correctly.
969 if (GTK_WIDGET_MAPPED (widget) &&
970 viewport->shadow_type != GTK_SHADOW_NONE &&
971 (widget->allocation.width != allocation->width ||
972 widget->allocation.height != allocation->height))
973 gdk_window_invalidate_rect (widget->window, NULL, FALSE);
975 widget->allocation = *allocation;
977 viewport_set_hadjustment_values (viewport, &hadjustment_value_changed);
978 viewport_set_vadjustment_values (viewport, &vadjustment_value_changed);
980 child_allocation.x = 0;
981 child_allocation.y = 0;
982 child_allocation.width = hadjustment->upper;
983 child_allocation.height = vadjustment->upper;
984 if (GTK_WIDGET_REALIZED (widget))
986 GtkAllocation view_allocation;
988 gdk_window_move_resize (widget->window,
989 allocation->x + border_width,
990 allocation->y + border_width,
991 allocation->width - border_width * 2,
992 allocation->height - border_width * 2);
994 viewport_get_view_allocation (viewport, &view_allocation);
995 gdk_window_move_resize (viewport->view_window,
998 view_allocation.width,
999 view_allocation.height);
1000 gdk_window_move_resize (viewport->bin_window,
1001 - hadjustment->value,
1002 - vadjustment->value,
1003 child_allocation.width,
1004 child_allocation.height);
1006 if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
1007 gtk_widget_size_allocate (bin->child, &child_allocation);
1009 gtk_adjustment_changed (hadjustment);
1010 gtk_adjustment_changed (vadjustment);
1011 if (hadjustment_value_changed)
1012 gtk_adjustment_value_changed (hadjustment);
1013 if (vadjustment_value_changed)
1014 gtk_adjustment_value_changed (vadjustment);
1018 hildon_viewport_adjustment_value_changed (GtkAdjustment *adjustment,
1022 HildonViewport *viewport;
1024 HildonViewportPrivate *priv;
1026 g_return_if_fail (GTK_IS_ADJUSTMENT (adjustment));
1027 g_return_if_fail (HILDON_IS_VIEWPORT (data));
1029 viewport = HILDON_VIEWPORT (data);
1030 widget = GTK_WIDGET (viewport);
1031 bin = GTK_BIN (data);
1033 priv = viewport->priv;
1035 if (bin->child && GTK_WIDGET_VISIBLE (bin->child) &&
1036 GTK_WIDGET_REALIZED (viewport))
1038 GtkAdjustment *hadjustment = hildon_viewport_get_hadjustment (viewport);
1039 GtkAdjustment *vadjustment = hildon_viewport_get_vadjustment (viewport);
1042 new_x = - hadjustment->value;
1043 new_y = - vadjustment->value;
1045 if (new_x != priv->old_x || new_y != priv->old_y)
1048 gdk_window_invalidate_rect (viewport->view_window, NULL, FALSE);
1050 gdk_window_move (viewport->bin_window, new_x, new_y);
1053 gdk_window_process_updates (viewport->view_window, TRUE);
1055 priv->old_x = new_x;
1056 priv->old_y = new_y;
1062 hildon_viewport_style_set (GtkWidget *widget,
1063 GtkStyle *previous_style)
1065 HildonViewport *viewport;
1067 if (GTK_WIDGET_REALIZED (widget) &&
1068 !GTK_WIDGET_NO_WINDOW (widget))
1070 HildonViewportPrivate *priv;
1071 viewport = HILDON_VIEWPORT (widget);
1073 priv = viewport->priv;
1075 gtk_style_set_background (widget->style, viewport->bin_window, GTK_STATE_NORMAL);
1076 gtk_style_set_background (widget->style, viewport->view_window, GTK_STATE_NORMAL);
1077 gtk_style_set_background (widget->style, widget->window, widget->state);
1081 #define __HILDON_VIEWPORT_C__