]> Pileus Git - ~andy/gtk/blob - gtk/gtkarrow.c
configure.in acheader.h gdk/gdkwindow.c Check for Shape extension both on
[~andy/gtk] / gtk / gtkarrow.c
1 /* GTK - The GIMP Toolkit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
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 #include "gtkarrow.h"
20
21
22 #define MIN_ARROW_SIZE  11
23
24
25 static void gtk_arrow_class_init (GtkArrowClass  *klass);
26 static void gtk_arrow_init       (GtkArrow       *arrow);
27 static gint gtk_arrow_expose     (GtkWidget      *widget,
28                                   GdkEventExpose *event);
29
30
31 guint
32 gtk_arrow_get_type (void)
33 {
34   static guint arrow_type = 0;
35
36   if (!arrow_type)
37     {
38       GtkTypeInfo arrow_info =
39       {
40         "GtkArrow",
41         sizeof (GtkArrow),
42         sizeof (GtkArrowClass),
43         (GtkClassInitFunc) gtk_arrow_class_init,
44         (GtkObjectInitFunc) gtk_arrow_init,
45         (GtkArgSetFunc) NULL,
46         (GtkArgGetFunc) NULL,
47       };
48
49       arrow_type = gtk_type_unique (gtk_misc_get_type (), &arrow_info);
50     }
51
52   return arrow_type;
53 }
54
55 static void
56 gtk_arrow_class_init (GtkArrowClass *class)
57 {
58   GtkWidgetClass *widget_class;
59
60   widget_class = (GtkWidgetClass*) class;
61
62   widget_class->expose_event = gtk_arrow_expose;
63 }
64
65 static void
66 gtk_arrow_init (GtkArrow *arrow)
67 {
68   GTK_WIDGET_SET_FLAGS (arrow, GTK_NO_WINDOW);
69
70   arrow->arrow_type = GTK_ARROW_RIGHT;
71   arrow->shadow_type = GTK_SHADOW_OUT;
72 }
73
74 GtkWidget*
75 gtk_arrow_new (GtkArrowType  arrow_type,
76                GtkShadowType shadow_type)
77 {
78   GtkArrow *arrow;
79
80   arrow = gtk_type_new (gtk_arrow_get_type ());
81
82   GTK_WIDGET (arrow)->requisition.width = MIN_ARROW_SIZE + GTK_MISC (arrow)->xpad * 2;
83   GTK_WIDGET (arrow)->requisition.height = MIN_ARROW_SIZE + GTK_MISC (arrow)->ypad * 2;
84
85   arrow->arrow_type = arrow_type;
86   arrow->shadow_type = shadow_type;
87
88   return GTK_WIDGET (arrow);
89 }
90
91 void
92 gtk_arrow_set (GtkArrow      *arrow,
93                GtkArrowType   arrow_type,
94                GtkShadowType  shadow_type)
95 {
96   g_return_if_fail (arrow != NULL);
97   g_return_if_fail (GTK_IS_ARROW (arrow));
98
99   if (((GtkArrowType) arrow->arrow_type != arrow_type) ||
100       ((GtkShadowType) arrow->shadow_type != shadow_type))
101     {
102       arrow->arrow_type = arrow_type;
103       arrow->shadow_type = shadow_type;
104
105       if (GTK_WIDGET_DRAWABLE (arrow))
106         {
107           gdk_window_clear_area (GTK_WIDGET (arrow)->window,
108                                  GTK_WIDGET (arrow)->allocation.x + 1,
109                                  GTK_WIDGET (arrow)->allocation.y + 1,
110                                  GTK_WIDGET (arrow)->allocation.width - 2,
111                                  GTK_WIDGET (arrow)->allocation.height - 2);
112           gtk_widget_queue_draw (GTK_WIDGET (arrow));
113         }
114     }
115 }
116
117
118 static gint
119 gtk_arrow_expose (GtkWidget      *widget,
120                   GdkEventExpose *event)
121 {
122   GtkArrow *arrow;
123   GtkMisc *misc;
124   GtkShadowType shadow_type;
125   gint width, height;
126   gint x, y;
127   gint extent;
128
129   g_return_val_if_fail (widget != NULL, FALSE);
130   g_return_val_if_fail (GTK_IS_ARROW (widget), FALSE);
131   g_return_val_if_fail (event != NULL, FALSE);
132
133   if (GTK_WIDGET_DRAWABLE (widget))
134     {
135       arrow = GTK_ARROW (widget);
136       misc = GTK_MISC (widget);
137
138       width = widget->allocation.width - misc->xpad * 2;
139       height = widget->allocation.height - misc->ypad * 2;
140       extent = MIN (width, height);
141
142       x = ((widget->allocation.x + misc->xpad) * (1.0 - misc->xalign) +
143            (widget->allocation.x + widget->allocation.width - extent - misc->ypad) * misc->xalign);
144       y = ((widget->allocation.y + misc->ypad) * (1.0 - misc->yalign) +
145            (widget->allocation.y + widget->allocation.height - extent - misc->ypad) * misc->yalign);
146
147       shadow_type = arrow->shadow_type;
148
149       if (widget->state == GTK_STATE_ACTIVE)
150         {
151           if (shadow_type == GTK_SHADOW_IN)
152             shadow_type = GTK_SHADOW_OUT;
153           else if (shadow_type == GTK_SHADOW_OUT)
154             shadow_type = GTK_SHADOW_IN;
155           else if (shadow_type == GTK_SHADOW_ETCHED_IN)
156             shadow_type = GTK_SHADOW_ETCHED_OUT;
157           else if (shadow_type == GTK_SHADOW_ETCHED_OUT)
158             shadow_type = GTK_SHADOW_ETCHED_IN;
159         }
160
161       gtk_draw_arrow (widget->style, widget->window,
162                       widget->state, shadow_type, arrow->arrow_type, TRUE,
163                       x, y, extent, extent);
164     }
165
166   return FALSE;
167 }