625 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			625 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GTK - The GIMP Toolkit
 | |
|  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Library General Public
 | |
|  * License as published by the Free Software Foundation; either
 | |
|  * version 2 of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This library is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|  * Library General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Library General Public
 | |
|  * License along with this library; if not, write to the Free
 | |
|  * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | |
|  */
 | |
| #include "gtksignal.h"
 | |
| #include "gtkviewport.h"
 | |
| 
 | |
| 
 | |
| static void gtk_viewport_class_init               (GtkViewportClass *klass);
 | |
| static void gtk_viewport_init                     (GtkViewport      *viewport);
 | |
| static void gtk_viewport_map                      (GtkWidget        *widget);
 | |
| static void gtk_viewport_unmap                    (GtkWidget        *widget);
 | |
| static void gtk_viewport_realize                  (GtkWidget        *widget);
 | |
| static void gtk_viewport_unrealize                (GtkWidget        *widget);
 | |
| static void gtk_viewport_paint                    (GtkWidget        *widget,
 | |
| 						   GdkRectangle     *area);
 | |
| static void gtk_viewport_draw                     (GtkWidget        *widget,
 | |
| 						   GdkRectangle     *area);
 | |
| static gint gtk_viewport_expose                   (GtkWidget        *widget,
 | |
| 						   GdkEventExpose   *event);
 | |
| static void gtk_viewport_size_request             (GtkWidget        *widget,
 | |
| 						   GtkRequisition   *requisition);
 | |
| static void gtk_viewport_size_allocate            (GtkWidget        *widget,
 | |
| 						   GtkAllocation    *allocation);
 | |
| static gint gtk_viewport_need_resize              (GtkContainer     *container);
 | |
| static void gtk_viewport_adjustment_changed       (GtkAdjustment    *adjustment,
 | |
| 						   gpointer          data);
 | |
| static void gtk_viewport_adjustment_value_changed (GtkAdjustment    *adjustment,
 | |
| 						   gpointer          data);
 | |
| 
 | |
| 
 | |
| guint
 | |
| gtk_viewport_get_type ()
 | |
| {
 | |
|   static guint viewport_type = 0;
 | |
| 
 | |
|   if (!viewport_type)
 | |
|     {
 | |
|       GtkTypeInfo viewport_info =
 | |
|       {
 | |
| 	"GtkViewport",
 | |
| 	sizeof (GtkViewport),
 | |
| 	sizeof (GtkViewportClass),
 | |
| 	(GtkClassInitFunc) gtk_viewport_class_init,
 | |
| 	(GtkObjectInitFunc) gtk_viewport_init,
 | |
| 	(GtkArgFunc) NULL,
 | |
|       };
 | |
| 
 | |
|       viewport_type = gtk_type_unique (gtk_bin_get_type (), &viewport_info);
 | |
|     }
 | |
| 
 | |
|   return viewport_type;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_class_init (GtkViewportClass *class)
 | |
| {
 | |
|   GtkWidgetClass *widget_class;
 | |
|   GtkContainerClass *container_class;
 | |
| 
 | |
|   widget_class = (GtkWidgetClass*) class;
 | |
|   container_class = (GtkContainerClass*) class;
 | |
| 
 | |
|   widget_class->map = gtk_viewport_map;
 | |
|   widget_class->unmap = gtk_viewport_unmap;
 | |
|   widget_class->realize = gtk_viewport_realize;
 | |
|   widget_class->unrealize = gtk_viewport_unrealize;
 | |
|   widget_class->draw = gtk_viewport_draw;
 | |
|   widget_class->expose_event = gtk_viewport_expose;
 | |
|   widget_class->size_request = gtk_viewport_size_request;
 | |
|   widget_class->size_allocate = gtk_viewport_size_allocate;
 | |
| 
 | |
|   container_class->need_resize = gtk_viewport_need_resize;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_init (GtkViewport *viewport)
 | |
| {
 | |
|   GTK_WIDGET_UNSET_FLAGS (viewport, GTK_NO_WINDOW);
 | |
|   GTK_WIDGET_SET_FLAGS (viewport, GTK_BASIC);
 | |
| 
 | |
|   viewport->shadow_type = GTK_SHADOW_IN;
 | |
|   viewport->main_window = NULL;
 | |
|   viewport->view_window = NULL;
 | |
|   viewport->hadjustment = NULL;
 | |
|   viewport->vadjustment = NULL;
 | |
| }
 | |
| 
 | |
| GtkWidget*
 | |
| gtk_viewport_new (GtkAdjustment *hadjustment,
 | |
| 		  GtkAdjustment *vadjustment)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
| 
 | |
|   viewport = gtk_type_new (gtk_viewport_get_type ());
 | |
| 
 | |
|   if (!hadjustment)
 | |
|     hadjustment = (GtkAdjustment*) gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0);
 | |
| 
 | |
|   if (!vadjustment)
 | |
|     vadjustment = (GtkAdjustment*) gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0);
 | |
| 
 | |
|   gtk_viewport_set_hadjustment (viewport, hadjustment);
 | |
|   gtk_viewport_set_vadjustment (viewport, vadjustment);
 | |
| 
 | |
|   return GTK_WIDGET (viewport);
 | |
| }
 | |
| 
 | |
| GtkAdjustment*
 | |
| gtk_viewport_get_hadjustment (GtkViewport *viewport)
 | |
| {
 | |
|   g_return_val_if_fail (viewport != NULL, NULL);
 | |
|   g_return_val_if_fail (GTK_IS_VIEWPORT (viewport), NULL);
 | |
| 
 | |
|   return viewport->hadjustment;
 | |
| }
 | |
| 
 | |
| GtkAdjustment*
 | |
| gtk_viewport_get_vadjustment (GtkViewport *viewport)
 | |
| {
 | |
|   g_return_val_if_fail (viewport != NULL, NULL);
 | |
|   g_return_val_if_fail (GTK_IS_VIEWPORT (viewport), NULL);
 | |
| 
 | |
|   return viewport->vadjustment;
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_viewport_set_hadjustment (GtkViewport   *viewport,
 | |
| 			      GtkAdjustment *adjustment)
 | |
| {
 | |
|   g_return_if_fail (viewport != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (viewport));
 | |
|   g_return_if_fail (adjustment != NULL);
 | |
| 
 | |
|   if (viewport->hadjustment != adjustment)
 | |
|     {
 | |
|       if (viewport->hadjustment)
 | |
| 	{
 | |
| 	  gtk_signal_disconnect_by_data (GTK_OBJECT (viewport->hadjustment),
 | |
| 					 (gpointer) viewport);
 | |
| 	  gtk_object_unref (GTK_OBJECT (viewport->hadjustment));
 | |
| 	}
 | |
| 
 | |
|       viewport->hadjustment = adjustment;
 | |
|       gtk_object_ref (GTK_OBJECT (viewport->hadjustment));
 | |
| 
 | |
|       gtk_signal_connect (GTK_OBJECT (adjustment), "changed",
 | |
| 			  (GtkSignalFunc) gtk_viewport_adjustment_changed,
 | |
| 			  (gpointer) viewport);
 | |
|       gtk_signal_connect (GTK_OBJECT (adjustment), "value_changed",
 | |
| 			  (GtkSignalFunc)gtk_viewport_adjustment_value_changed,
 | |
| 			  (gpointer) viewport);
 | |
| 
 | |
|       gtk_viewport_adjustment_changed (adjustment, (gpointer) viewport);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_viewport_set_vadjustment (GtkViewport   *viewport,
 | |
| 			      GtkAdjustment *adjustment)
 | |
| {
 | |
|   g_return_if_fail (viewport != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (viewport));
 | |
|   g_return_if_fail (adjustment != NULL);
 | |
| 
 | |
|   if (viewport->vadjustment != adjustment)
 | |
|     {
 | |
|       if (viewport->vadjustment)
 | |
| 	{
 | |
| 	  gtk_signal_disconnect_by_data (GTK_OBJECT (viewport->vadjustment),
 | |
| 					 (gpointer) viewport);
 | |
| 	  gtk_object_unref (GTK_OBJECT (viewport->vadjustment));
 | |
| 	}
 | |
| 
 | |
|       viewport->vadjustment = adjustment;
 | |
|       gtk_object_ref (GTK_OBJECT (viewport->vadjustment));
 | |
|       
 | |
|       gtk_signal_connect (GTK_OBJECT (adjustment), "changed",
 | |
| 			  (GtkSignalFunc) gtk_viewport_adjustment_changed,
 | |
| 			  (gpointer) viewport);
 | |
|       gtk_signal_connect (GTK_OBJECT (adjustment), "value_changed",
 | |
| 			  (GtkSignalFunc)gtk_viewport_adjustment_value_changed,
 | |
| 			  (gpointer) viewport);
 | |
| 
 | |
|       gtk_viewport_adjustment_changed (adjustment, (gpointer) viewport);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| gtk_viewport_set_shadow_type (GtkViewport   *viewport,
 | |
| 			      GtkShadowType  type)
 | |
| {
 | |
|   g_return_if_fail (viewport != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (viewport));
 | |
| 
 | |
|   if ((GtkShadowType) viewport->shadow_type != type)
 | |
|     {
 | |
|       viewport->shadow_type = type;
 | |
| 
 | |
|       if (GTK_WIDGET_VISIBLE (viewport))
 | |
| 	{
 | |
| 	  gtk_widget_size_allocate (GTK_WIDGET (viewport), &(GTK_WIDGET (viewport)->allocation));
 | |
| 	  gtk_widget_queue_draw (GTK_WIDGET (viewport));
 | |
| 	}
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static void
 | |
| gtk_viewport_map (GtkWidget *widget)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
| 
 | |
|   GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED);
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   bin = GTK_BIN (widget);
 | |
| 
 | |
|   gdk_window_show (viewport->main_window);
 | |
| 
 | |
|   if (bin->child &&
 | |
|       GTK_WIDGET_VISIBLE (bin->child) &&
 | |
|       !GTK_WIDGET_MAPPED (bin->child))
 | |
|     gtk_widget_map (bin->child);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_unmap (GtkWidget *widget)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
| 
 | |
|   GTK_WIDGET_UNSET_FLAGS (widget, GTK_MAPPED);
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   
 | |
|   gdk_window_hide (viewport->main_window);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_realize (GtkWidget *widget)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GdkWindowAttr attributes;
 | |
|   GtkRequisition *child_requisition;
 | |
|   gint attributes_mask;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
| 
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
 | |
| 
 | |
|   attributes.x = widget->allocation.x + GTK_CONTAINER (widget)->border_width;
 | |
|   attributes.y = widget->allocation.y + GTK_CONTAINER (widget)->border_width;
 | |
|   attributes.width = widget->allocation.width - GTK_CONTAINER (widget)->border_width * 2;
 | |
|   attributes.height = widget->allocation.height - GTK_CONTAINER (widget)->border_width * 2;
 | |
|   attributes.window_type = GDK_WINDOW_CHILD;
 | |
|   attributes.wclass = GDK_INPUT_OUTPUT;
 | |
|   attributes.visual = gtk_widget_get_visual (widget);
 | |
|   attributes.colormap = gtk_widget_get_colormap (widget);
 | |
|   attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
 | |
| 
 | |
|   attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
 | |
| 
 | |
|   viewport->main_window = gdk_window_new (widget->parent->window, &attributes, attributes_mask);
 | |
|   gdk_window_set_user_data (viewport->main_window, viewport);
 | |
| 
 | |
|   attributes.x += widget->style->klass->xthickness;
 | |
|   attributes.y += widget->style->klass->ythickness;
 | |
|   attributes.width -= widget->style->klass->xthickness * 2;
 | |
|   attributes.height -= widget->style->klass->ythickness * 2;
 | |
| 
 | |
|   viewport->view_window = gdk_window_new (viewport->main_window, &attributes, attributes_mask);
 | |
|   gdk_window_set_user_data (viewport->view_window, viewport);
 | |
| 
 | |
|   attributes.x = 0;
 | |
|   attributes.y = 0;
 | |
| 
 | |
|   if (GTK_BIN (viewport)->child)
 | |
|     {
 | |
|       child_requisition = >K_WIDGET (GTK_BIN (viewport)->child)->requisition;
 | |
|       attributes.width = child_requisition->width;
 | |
|       attributes.height = child_requisition->height;
 | |
|     }
 | |
| 
 | |
|   widget->window = gdk_window_new (viewport->view_window, &attributes, attributes_mask);
 | |
|   gdk_window_set_user_data (widget->window, viewport);
 | |
| 
 | |
|   widget->style = gtk_style_attach (widget->style, viewport->main_window);
 | |
|   gtk_style_set_background (widget->style, viewport->main_window, GTK_STATE_NORMAL);
 | |
|   gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
 | |
|   
 | |
|   gdk_window_show (widget->window);
 | |
|   gdk_window_show (viewport->view_window);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_unrealize (GtkWidget *widget)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
| 
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   GTK_WIDGET_UNSET_FLAGS (widget, GTK_REALIZED | GTK_MAPPED);
 | |
| 
 | |
|   gtk_style_detach (widget->style);
 | |
| 
 | |
|   gdk_window_destroy (widget->window);
 | |
|   gdk_window_destroy (viewport->view_window);
 | |
|   gdk_window_destroy (viewport->main_window);
 | |
| 
 | |
|   widget->window = NULL;
 | |
|   viewport->view_window = NULL;
 | |
|   viewport->main_window = NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_paint (GtkWidget    *widget,
 | |
| 		    GdkRectangle *area)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkStateType state;
 | |
|   gint x, y;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
|   g_return_if_fail (area != NULL);
 | |
| 
 | |
|   if (GTK_WIDGET_DRAWABLE (widget))
 | |
|     {
 | |
|       viewport = GTK_VIEWPORT (widget);
 | |
| 
 | |
|       state = widget->state;
 | |
|       if (!GTK_WIDGET_IS_SENSITIVE (widget))
 | |
|         state = GTK_STATE_INSENSITIVE;
 | |
| 
 | |
|       x = GTK_CONTAINER (viewport)->border_width;
 | |
|       y = GTK_CONTAINER (viewport)->border_width;
 | |
| 
 | |
|       gtk_draw_shadow (widget->style, viewport->main_window,
 | |
| 		       GTK_STATE_NORMAL, viewport->shadow_type,
 | |
| 		       0, 0, -1, -1);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_draw (GtkWidget    *widget,
 | |
| 		   GdkRectangle *area)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
|   GdkRectangle tmp_area;
 | |
|   GdkRectangle child_area;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
|   g_return_if_fail (area != NULL);
 | |
| 
 | |
|   if (GTK_WIDGET_DRAWABLE (widget))
 | |
|     {
 | |
|       viewport = GTK_VIEWPORT (widget);
 | |
|       bin = GTK_BIN (widget);
 | |
| 
 | |
|       gtk_viewport_paint (widget, area);
 | |
| 
 | |
|       if (bin->child)
 | |
| 	{
 | |
| 	  tmp_area = *area;
 | |
| 	  tmp_area.x += viewport->hadjustment->value;
 | |
| 	  tmp_area.y += viewport->vadjustment->value;
 | |
| 
 | |
| 	  if (gtk_widget_intersect (bin->child, &tmp_area, &child_area))
 | |
| 	    gtk_widget_draw (bin->child, &child_area);
 | |
| 	}
 | |
|     }
 | |
| }
 | |
| 
 | |
| static gint
 | |
| gtk_viewport_expose (GtkWidget      *widget,
 | |
| 		     GdkEventExpose *event)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
|   GdkEventExpose child_event;
 | |
| 
 | |
|   g_return_val_if_fail (widget != NULL, FALSE);
 | |
|   g_return_val_if_fail (GTK_IS_VIEWPORT (widget), FALSE);
 | |
|   g_return_val_if_fail (event != NULL, FALSE);
 | |
| 
 | |
|   if (GTK_WIDGET_DRAWABLE (widget))
 | |
|     {
 | |
|       viewport = GTK_VIEWPORT (widget);
 | |
|       bin = GTK_BIN (widget);
 | |
| 
 | |
|       if (event->window == viewport->main_window)
 | |
| 	gtk_viewport_paint (widget, &event->area);
 | |
| 
 | |
|       child_event = *event;
 | |
|       if (bin->child &&
 | |
| 	  GTK_WIDGET_NO_WINDOW (bin->child) &&
 | |
| 	  gtk_widget_intersect (bin->child, &event->area, &child_event.area))
 | |
| 	gtk_widget_event (bin->child, (GdkEvent*) &child_event);
 | |
|     }
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_size_request (GtkWidget      *widget,
 | |
| 			   GtkRequisition *requisition)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
|   g_return_if_fail (requisition != NULL);
 | |
| 
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   bin = GTK_BIN (widget);
 | |
| 
 | |
|   requisition->width = (GTK_CONTAINER (widget)->border_width +
 | |
| 			GTK_WIDGET (widget)->style->klass->xthickness) * 2 + 5;
 | |
| 
 | |
|   requisition->height = (GTK_CONTAINER (widget)->border_width * 2 +
 | |
| 			 GTK_WIDGET (widget)->style->klass->ythickness) * 2 + 5;
 | |
| 
 | |
|   if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
 | |
|     gtk_widget_size_request (bin->child, &bin->child->requisition);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_size_allocate (GtkWidget     *widget,
 | |
| 			    GtkAllocation *allocation)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
|   GtkAllocation child_allocation;
 | |
|   gint hval, vval;
 | |
| 
 | |
|   g_return_if_fail (widget != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (widget));
 | |
|   g_return_if_fail (allocation != NULL);
 | |
| 
 | |
|   widget->allocation = *allocation;
 | |
|   viewport = GTK_VIEWPORT (widget);
 | |
|   bin = GTK_BIN (widget);
 | |
| 
 | |
|   child_allocation.x = GTK_WIDGET (viewport)->style->klass->xthickness;
 | |
|   child_allocation.width = allocation->width - child_allocation.x * 2;
 | |
| 
 | |
|   child_allocation.y = GTK_WIDGET (viewport)->style->klass->ythickness;
 | |
|   child_allocation.height = allocation->height - child_allocation.y * 2;
 | |
| 
 | |
|   if (GTK_WIDGET_REALIZED (widget))
 | |
|     {
 | |
|       gdk_window_move_resize (viewport->main_window,
 | |
| 			      allocation->x + GTK_CONTAINER (viewport)->border_width,
 | |
| 			      allocation->y + GTK_CONTAINER (viewport)->border_width,
 | |
| 			      allocation->width - GTK_CONTAINER (viewport)->border_width * 2,
 | |
| 			      allocation->height - GTK_CONTAINER (viewport)->border_width * 2);
 | |
| 
 | |
|       gdk_window_move_resize (viewport->view_window,
 | |
| 			      child_allocation.x,
 | |
| 			      child_allocation.y,
 | |
| 			      child_allocation.width,
 | |
| 			      child_allocation.height);
 | |
|     }
 | |
| 
 | |
|   viewport->hadjustment->page_size = child_allocation.width;
 | |
|   viewport->hadjustment->page_increment = viewport->hadjustment->page_size / 2;
 | |
|   viewport->hadjustment->step_increment = 10;
 | |
| 
 | |
|   viewport->vadjustment->page_size = child_allocation.height;
 | |
|   viewport->vadjustment->page_increment = viewport->vadjustment->page_size / 2;
 | |
|   viewport->vadjustment->step_increment = 10;
 | |
| 
 | |
|   hval = viewport->hadjustment->value;
 | |
|   vval = viewport->vadjustment->value;
 | |
| 
 | |
|   if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
 | |
|     {
 | |
|       viewport->hadjustment->lower = 0;
 | |
|       viewport->hadjustment->upper = MAX (bin->child->requisition.width,
 | |
| 					  child_allocation.width);
 | |
| 
 | |
|       hval = CLAMP (hval, 0,
 | |
| 		    viewport->hadjustment->upper -
 | |
| 		    viewport->hadjustment->page_size);
 | |
| 
 | |
|       viewport->vadjustment->lower = 0;
 | |
|       viewport->vadjustment->upper = MAX (bin->child->requisition.height,
 | |
| 					  child_allocation.height);
 | |
| 
 | |
|       vval = CLAMP (vval, 0,
 | |
| 		    viewport->vadjustment->upper -
 | |
| 		    viewport->vadjustment->page_size);
 | |
|     }
 | |
| 
 | |
|   if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
 | |
|     {
 | |
|       child_allocation.x = 0;
 | |
|       child_allocation.y = 0;
 | |
| 
 | |
|       child_allocation.width = viewport->hadjustment->upper;
 | |
|       child_allocation.height = viewport->vadjustment->upper;
 | |
| 
 | |
|       if (!GTK_WIDGET_REALIZED (widget))
 | |
|         gtk_widget_realize (widget);
 | |
| 
 | |
|       gdk_window_resize (widget->window,
 | |
| 			 child_allocation.width,
 | |
| 			 child_allocation.height);
 | |
| 
 | |
|       child_allocation.x = 0;
 | |
|       child_allocation.y = 0;
 | |
|       gtk_widget_size_allocate (bin->child, &child_allocation);
 | |
|     }
 | |
| 
 | |
|   gtk_signal_emit_by_name (GTK_OBJECT (viewport->hadjustment), "changed");
 | |
|   gtk_signal_emit_by_name (GTK_OBJECT (viewport->vadjustment), "changed");
 | |
|   if (viewport->hadjustment->value != hval)
 | |
|     {
 | |
|       viewport->hadjustment->value = hval;
 | |
|       gtk_signal_emit_by_name (GTK_OBJECT (viewport->hadjustment), "value_changed");
 | |
|     }
 | |
|   if (viewport->vadjustment->value != vval)
 | |
|     {
 | |
|       viewport->vadjustment->value = vval;
 | |
|       gtk_signal_emit_by_name (GTK_OBJECT (viewport->vadjustment), "value_changed");
 | |
|     }
 | |
| }
 | |
| 
 | |
| static gint
 | |
| gtk_viewport_need_resize (GtkContainer *container)
 | |
| {
 | |
|   GtkBin *bin;
 | |
| 
 | |
|   g_return_val_if_fail (container != NULL, FALSE);
 | |
|   g_return_val_if_fail (GTK_IS_VIEWPORT (container), FALSE);
 | |
| 
 | |
|   if (GTK_WIDGET_REALIZED (container))
 | |
|     {
 | |
|       bin = GTK_BIN (container);
 | |
| 
 | |
|       gtk_widget_size_request (bin->child, &bin->child->requisition);
 | |
| 
 | |
|       gtk_widget_size_allocate (GTK_WIDGET (container),
 | |
| 				&(GTK_WIDGET (container)->allocation));
 | |
|     }
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_adjustment_changed (GtkAdjustment *adjustment,
 | |
| 				 gpointer       data)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
| 
 | |
|   g_return_if_fail (adjustment != NULL);
 | |
|   g_return_if_fail (data != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (data));
 | |
| 
 | |
|   viewport = GTK_VIEWPORT (data);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_viewport_adjustment_value_changed (GtkAdjustment *adjustment,
 | |
| 				       gpointer       data)
 | |
| {
 | |
|   GtkViewport *viewport;
 | |
|   GtkBin *bin;
 | |
|   GtkAllocation child_allocation;
 | |
|   gint width, height;
 | |
| 
 | |
|   g_return_if_fail (adjustment != NULL);
 | |
|   g_return_if_fail (data != NULL);
 | |
|   g_return_if_fail (GTK_IS_VIEWPORT (data));
 | |
| 
 | |
|   viewport = GTK_VIEWPORT (data);
 | |
|   bin = GTK_BIN (data);
 | |
| 
 | |
|   if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
 | |
|     {
 | |
|       gdk_window_get_size (viewport->view_window, &width, &height);
 | |
| 
 | |
|       child_allocation.x = 0;
 | |
|       child_allocation.y = 0;
 | |
| 
 | |
|       if (viewport->hadjustment->lower != (viewport->hadjustment->upper -
 | |
| 					   viewport->hadjustment->page_size))
 | |
| 	child_allocation.x =  viewport->hadjustment->lower - viewport->hadjustment->value;
 | |
| 
 | |
|       if (viewport->vadjustment->lower != (viewport->vadjustment->upper -
 | |
| 					   viewport->vadjustment->page_size))
 | |
| 	child_allocation.y = viewport->vadjustment->lower - viewport->vadjustment->value;
 | |
| 
 | |
|       if (GTK_WIDGET_REALIZED (viewport))
 | |
| 	gdk_window_move (GTK_WIDGET (viewport)->window,
 | |
| 			 child_allocation.x,
 | |
| 			 child_allocation.y);
 | |
|     }
 | |
| }
 | 
