322 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			322 lines
		
	
	
		
			8.4 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 "gtklabel.h"
 | |
| #include "gtkradiobutton.h"
 | |
| #include "gtksignal.h"
 | |
| 
 | |
| 
 | |
| #define CHECK_BUTTON_CLASS(w)  GTK_CHECK_BUTTON_CLASS (GTK_OBJECT (w)->klass)
 | |
| 
 | |
| 
 | |
| static void gtk_radio_button_class_init     (GtkRadioButtonClass  *klass);
 | |
| static void gtk_radio_button_init           (GtkRadioButton       *radio_button);
 | |
| static void gtk_radio_button_destroy        (GtkObject            *object);
 | |
| static void gtk_radio_button_clicked        (GtkButton            *button);
 | |
| static void gtk_radio_button_draw_indicator (GtkCheckButton       *check_button,
 | |
| 					     GdkRectangle         *area);
 | |
| 
 | |
| 
 | |
| static GtkCheckButtonClass *parent_class = NULL;
 | |
| 
 | |
| 
 | |
| guint
 | |
| gtk_radio_button_get_type ()
 | |
| {
 | |
|   static guint radio_button_type = 0;
 | |
| 
 | |
|   if (!radio_button_type)
 | |
|     {
 | |
|       GtkTypeInfo radio_button_info =
 | |
|       {
 | |
| 	"GtkRadioButton",
 | |
| 	sizeof (GtkRadioButton),
 | |
| 	sizeof (GtkRadioButtonClass),
 | |
| 	(GtkClassInitFunc) gtk_radio_button_class_init,
 | |
| 	(GtkObjectInitFunc) gtk_radio_button_init,
 | |
| 	(GtkArgFunc) NULL,
 | |
|       };
 | |
| 
 | |
|       radio_button_type = gtk_type_unique (gtk_check_button_get_type (), &radio_button_info);
 | |
|     }
 | |
| 
 | |
|   return radio_button_type;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_button_class_init (GtkRadioButtonClass *class)
 | |
| {
 | |
|   GtkObjectClass *object_class;
 | |
|   GtkButtonClass *button_class;
 | |
|   GtkCheckButtonClass *check_button_class;
 | |
| 
 | |
|   object_class = (GtkObjectClass*) class;
 | |
|   button_class = (GtkButtonClass*) class;
 | |
|   check_button_class = (GtkCheckButtonClass*) class;
 | |
| 
 | |
|   parent_class = gtk_type_class (gtk_check_button_get_type ());
 | |
| 
 | |
|   object_class->destroy = gtk_radio_button_destroy;
 | |
| 
 | |
|   button_class->clicked = gtk_radio_button_clicked;
 | |
| 
 | |
|   check_button_class->draw_indicator = gtk_radio_button_draw_indicator;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_button_init (GtkRadioButton *radio_button)
 | |
| {
 | |
|   radio_button->group = NULL;
 | |
| }
 | |
| 
 | |
| GtkWidget*
 | |
| gtk_radio_button_new (GSList *group)
 | |
| {
 | |
|   GtkRadioButton *radio_button;
 | |
|   GtkRadioButton *tmp_button;
 | |
|   GSList *tmp_list;
 | |
| 
 | |
|   radio_button = gtk_type_new (gtk_radio_button_get_type ());
 | |
| 
 | |
|   tmp_list = group;
 | |
|   radio_button->group = g_slist_prepend (group, radio_button);
 | |
| 
 | |
|   if (tmp_list)
 | |
|     {
 | |
|       while (tmp_list)
 | |
| 	{
 | |
| 	  tmp_button = tmp_list->data;
 | |
| 	  tmp_list = tmp_list->next;
 | |
| 
 | |
| 	  tmp_button->group = radio_button->group;
 | |
| 	}
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       GTK_TOGGLE_BUTTON (radio_button)->active = TRUE;
 | |
|       gtk_widget_set_state (GTK_WIDGET (radio_button), GTK_STATE_ACTIVE);
 | |
|     }
 | |
| 
 | |
|   return GTK_WIDGET (radio_button);
 | |
| }
 | |
| 
 | |
| GtkWidget*
 | |
| gtk_radio_button_new_with_label (GSList      *group,
 | |
| 				 const gchar *label)
 | |
| {
 | |
|   GtkWidget *radio_button;
 | |
|   GtkWidget *label_widget;
 | |
| 
 | |
|   radio_button = gtk_radio_button_new (group);
 | |
|   label_widget = gtk_label_new (label);
 | |
|   gtk_misc_set_alignment (GTK_MISC (label_widget), 0.0, 0.5);
 | |
| 
 | |
|   gtk_container_add (GTK_CONTAINER (radio_button), label_widget);
 | |
|   gtk_widget_show (label_widget);
 | |
| 
 | |
|   return radio_button;
 | |
| }
 | |
| 
 | |
| GtkWidget*
 | |
| gtk_radio_button_new_from_widget (GtkRadioButton *group)
 | |
| {
 | |
|   GSList *l = NULL;
 | |
|   if (group)
 | |
|     l = gtk_radio_button_group (group);
 | |
|   return gtk_radio_button_new (l);
 | |
| }
 | |
| 
 | |
| 
 | |
| GtkWidget*
 | |
| gtk_radio_button_new_with_label_from_widget (GtkRadioButton *group,
 | |
| 					     const gchar    *label)
 | |
| {
 | |
|   GSList *l = NULL;
 | |
|   if (group)
 | |
|     l = gtk_radio_button_group (group);
 | |
|   return gtk_radio_button_new_with_label (l, label);
 | |
| }
 | |
| 
 | |
| GSList*
 | |
| gtk_radio_button_group (GtkRadioButton *radio_button)
 | |
| {
 | |
|   g_return_val_if_fail (radio_button != NULL, NULL);
 | |
|   g_return_val_if_fail (GTK_IS_RADIO_BUTTON (radio_button), NULL);
 | |
| 
 | |
|   return radio_button->group;
 | |
| }
 | |
| 
 | |
| 
 | |
| static void
 | |
| gtk_radio_button_destroy (GtkObject *object)
 | |
| {
 | |
|   GtkRadioButton *radio_button;
 | |
|   GtkRadioButton *tmp_button;
 | |
|   GSList *tmp_list;
 | |
| 
 | |
|   g_return_if_fail (object != NULL);
 | |
|   g_return_if_fail (GTK_IS_RADIO_BUTTON (object));
 | |
| 
 | |
|   radio_button = GTK_RADIO_BUTTON (object);
 | |
| 
 | |
|   radio_button->group = g_slist_remove (radio_button->group, radio_button);
 | |
|   tmp_list = radio_button->group;
 | |
| 
 | |
|   while (tmp_list)
 | |
|     {
 | |
|       tmp_button = tmp_list->data;
 | |
|       tmp_list = tmp_list->next;
 | |
| 
 | |
|       tmp_button->group = radio_button->group;
 | |
|     }
 | |
| 
 | |
|   if (GTK_OBJECT_CLASS (parent_class)->destroy)
 | |
|     (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_button_clicked (GtkButton *button)
 | |
| {
 | |
|   GtkToggleButton *toggle_button;
 | |
|   GtkRadioButton *radio_button;
 | |
|   GtkToggleButton *tmp_button;
 | |
|   GtkStateType new_state;
 | |
|   GSList *tmp_list;
 | |
|   gint toggled;
 | |
| 
 | |
|   g_return_if_fail (button != NULL);
 | |
|   g_return_if_fail (GTK_IS_RADIO_BUTTON (button));
 | |
| 
 | |
|   radio_button = GTK_RADIO_BUTTON (button);
 | |
|   toggle_button = GTK_TOGGLE_BUTTON (button);
 | |
|   toggled = FALSE;
 | |
| 
 | |
|   if (toggle_button->active)
 | |
|     {
 | |
|       tmp_button = NULL;
 | |
|       tmp_list = radio_button->group;
 | |
| 
 | |
|       while (tmp_list)
 | |
| 	{
 | |
| 	  tmp_button = tmp_list->data;
 | |
| 	  tmp_list = tmp_list->next;
 | |
| 
 | |
| 	  if (tmp_button->active && (tmp_button != toggle_button))
 | |
| 	    break;
 | |
| 
 | |
| 	  tmp_button = NULL;
 | |
| 	}
 | |
| 
 | |
|       if (!tmp_button)
 | |
| 	{
 | |
| 	  new_state = (button->in_button ? GTK_STATE_PRELIGHT : GTK_STATE_ACTIVE);
 | |
| 	}
 | |
|       else
 | |
| 	{
 | |
| 	  toggled = TRUE;
 | |
| 	  toggle_button->active = !toggle_button->active;
 | |
| 	  new_state = (button->in_button ? GTK_STATE_PRELIGHT : GTK_STATE_NORMAL);
 | |
| 	}
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       toggled = TRUE;
 | |
|       toggle_button->active = !toggle_button->active;
 | |
| 
 | |
|       tmp_list = radio_button->group;
 | |
|       while (tmp_list)
 | |
| 	{
 | |
| 	  tmp_button = tmp_list->data;
 | |
| 	  tmp_list = tmp_list->next;
 | |
| 
 | |
| 	  if (tmp_button->active && (tmp_button != toggle_button))
 | |
| 	    {
 | |
| 	      gtk_button_clicked (GTK_BUTTON (tmp_button));
 | |
| 	      break;
 | |
| 	    }
 | |
| 	}
 | |
| 
 | |
|       new_state = (button->in_button ? GTK_STATE_PRELIGHT : GTK_STATE_ACTIVE);
 | |
|     }
 | |
| 
 | |
|   if (GTK_WIDGET_STATE (button) != new_state)
 | |
|     gtk_widget_set_state (GTK_WIDGET (button), new_state);
 | |
|   if (toggled)
 | |
|     gtk_toggle_button_toggled (toggle_button);
 | |
|   gtk_widget_queue_draw (GTK_WIDGET (button));
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_button_draw_indicator (GtkCheckButton *check_button,
 | |
| 				 GdkRectangle   *area)
 | |
| {
 | |
|   GtkWidget *widget;
 | |
|   GtkButton *button;
 | |
|   GtkToggleButton *toggle_button;
 | |
|   GtkStateType state_type;
 | |
|   GtkShadowType shadow_type;
 | |
|   GdkPoint pts[4];
 | |
|   gint width, height;
 | |
|   gint x, y;
 | |
| 
 | |
|   g_return_if_fail (check_button != NULL);
 | |
|   g_return_if_fail (GTK_IS_RADIO_BUTTON (check_button));
 | |
| 
 | |
|   if (GTK_WIDGET_VISIBLE (check_button) && GTK_WIDGET_MAPPED (check_button))
 | |
|     {
 | |
|       widget = GTK_WIDGET (check_button);
 | |
|       button = GTK_BUTTON (check_button);
 | |
|       toggle_button = GTK_TOGGLE_BUTTON (check_button);
 | |
| 
 | |
|       state_type = GTK_WIDGET_STATE (widget);
 | |
|       if ((state_type != GTK_STATE_NORMAL) &&
 | |
| 	  (state_type != GTK_STATE_PRELIGHT))
 | |
| 	state_type = GTK_STATE_NORMAL;
 | |
| 
 | |
|       gtk_style_set_background (widget->style, widget->window, state_type);
 | |
|       gdk_window_clear_area (widget->window, area->x, area->y, area->width, area->height);
 | |
| 
 | |
|       x = CHECK_BUTTON_CLASS (widget)->indicator_spacing + GTK_CONTAINER (widget)->border_width;
 | |
|       y = (widget->allocation.height - CHECK_BUTTON_CLASS (widget)->indicator_size) / 2;
 | |
|       width = CHECK_BUTTON_CLASS (widget)->indicator_size;
 | |
|       height = CHECK_BUTTON_CLASS (widget)->indicator_size;
 | |
| 
 | |
|       if (GTK_WIDGET_STATE (widget) == GTK_STATE_ACTIVE)
 | |
| 	shadow_type = GTK_SHADOW_IN;
 | |
|       else if ((GTK_WIDGET_STATE (widget) == GTK_STATE_PRELIGHT) && toggle_button->active)
 | |
| 	shadow_type = GTK_SHADOW_IN;
 | |
|       else
 | |
| 	shadow_type = GTK_SHADOW_OUT;
 | |
| 
 | |
|       pts[0].x = x + width / 2;
 | |
|       pts[0].y = y;
 | |
|       pts[1].x = x + width;
 | |
|       pts[1].y = y + height / 2;
 | |
|       pts[2].x = pts[0].x;
 | |
|       pts[2].y = y + height;
 | |
|       pts[3].x = x;
 | |
|       pts[3].y = pts[1].y;
 | |
| 
 | |
|       gdk_draw_polygon (widget->window,
 | |
| 			widget->style->bg_gc[GTK_WIDGET_STATE (widget)],
 | |
| 			TRUE, pts, 4);
 | |
|       gtk_draw_diamond (widget->style, widget->window,
 | |
| 			GTK_WIDGET_STATE (widget), shadow_type,
 | |
| 			x, y, width, height);
 | |
|     }
 | |
| }
 | 
