]> Pileus Git - ~andy/gtk/blob - gtk/gtkarrow.c
GtkType and macro fixups. widget argument implementation:
[~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 enum {
25   ARG_0,
26   ARG_ARROW_TYPE,
27   ARG_SHADOW_TYPE
28 };
29
30
31 static void gtk_arrow_class_init (GtkArrowClass  *klass);
32 static void gtk_arrow_init       (GtkArrow       *arrow);
33 static gint gtk_arrow_expose     (GtkWidget      *widget,
34                                   GdkEventExpose *event);
35 static void gtk_arrow_set_arg    (GtkObject      *object,
36                                   GtkArg         *arg,
37                                   guint           arg_id);
38 static void gtk_arrow_get_arg    (GtkObject      *object,
39                                   GtkArg         *arg,
40                                   guint           arg_id);
41
42
43 GtkType
44 gtk_arrow_get_type (void)
45 {
46   static GtkType arrow_type = 0;
47
48   if (!arrow_type)
49     {
50       GtkTypeInfo arrow_info =
51       {
52         "GtkArrow",
53         sizeof (GtkArrow),
54         sizeof (GtkArrowClass),
55         (GtkClassInitFunc) gtk_arrow_class_init,
56         (GtkObjectInitFunc) gtk_arrow_init,
57         /* reversed_1 */ NULL,
58         /* reversed_2 */ NULL,
59         (GtkClassInitFunc) NULL,
60       };
61
62       arrow_type = gtk_type_unique (GTK_TYPE_MISC, &arrow_info);
63     }
64
65   return arrow_type;
66 }
67
68 static void
69 gtk_arrow_class_init (GtkArrowClass *class)
70 {
71   GtkObjectClass *object_class;
72   GtkWidgetClass *widget_class;
73
74   object_class = (GtkObjectClass*) class;
75   widget_class = (GtkWidgetClass*) class;
76
77   gtk_object_add_arg_type ("GtkArrow::arrow_type", GTK_TYPE_ARROW_TYPE, GTK_ARG_READWRITE, ARG_ARROW_TYPE);
78   gtk_object_add_arg_type ("GtkArrow::shadow_type", GTK_TYPE_SHADOW_TYPE, GTK_ARG_READWRITE, ARG_SHADOW_TYPE);
79
80   object_class->set_arg = gtk_arrow_set_arg;
81   object_class->get_arg = gtk_arrow_get_arg;
82
83   widget_class->expose_event = gtk_arrow_expose;
84 }
85
86 static void
87 gtk_arrow_set_arg (GtkObject      *object,
88                    GtkArg         *arg,
89                    guint           arg_id)
90 {
91   GtkArrow *arrow;
92
93   arrow = GTK_ARROW (object);
94
95   switch (arg_id)
96     {
97     case ARG_ARROW_TYPE:
98       gtk_arrow_set (arrow,
99                      GTK_VALUE_ENUM (*arg),
100                      arrow->shadow_type);
101       break;
102     case ARG_SHADOW_TYPE:
103       gtk_arrow_set (arrow,
104                      arrow->arrow_type,
105                      GTK_VALUE_ENUM (*arg));
106       break;
107     default:
108       break;
109     }
110 }
111
112 static void
113 gtk_arrow_get_arg (GtkObject      *object,
114                    GtkArg         *arg,
115                    guint           arg_id)
116 {
117   GtkArrow *arrow;
118
119   arrow = GTK_ARROW (object);
120   switch (arg_id)
121     {
122     case ARG_ARROW_TYPE:
123       GTK_VALUE_ENUM (*arg) = arrow->arrow_type;
124       break;
125     case ARG_SHADOW_TYPE:
126       GTK_VALUE_ENUM (*arg) = arrow->shadow_type;
127       break;
128     default:
129       arg->type = GTK_TYPE_INVALID;
130       break;
131     }
132 }
133
134 static void
135 gtk_arrow_init (GtkArrow *arrow)
136 {
137   GTK_WIDGET_SET_FLAGS (arrow, GTK_NO_WINDOW);
138
139   arrow->arrow_type = GTK_ARROW_RIGHT;
140   arrow->shadow_type = GTK_SHADOW_OUT;
141 }
142
143 GtkWidget*
144 gtk_arrow_new (GtkArrowType  arrow_type,
145                GtkShadowType shadow_type)
146 {
147   GtkArrow *arrow;
148
149   arrow = gtk_type_new (gtk_arrow_get_type ());
150
151   GTK_WIDGET (arrow)->requisition.width = MIN_ARROW_SIZE + GTK_MISC (arrow)->xpad * 2;
152   GTK_WIDGET (arrow)->requisition.height = MIN_ARROW_SIZE + GTK_MISC (arrow)->ypad * 2;
153
154   arrow->arrow_type = arrow_type;
155   arrow->shadow_type = shadow_type;
156
157   return GTK_WIDGET (arrow);
158 }
159
160 void
161 gtk_arrow_set (GtkArrow      *arrow,
162                GtkArrowType   arrow_type,
163                GtkShadowType  shadow_type)
164 {
165   g_return_if_fail (arrow != NULL);
166   g_return_if_fail (GTK_IS_ARROW (arrow));
167
168   if (((GtkArrowType) arrow->arrow_type != arrow_type) ||
169       ((GtkShadowType) arrow->shadow_type != shadow_type))
170     {
171       arrow->arrow_type = arrow_type;
172       arrow->shadow_type = shadow_type;
173
174       if (GTK_WIDGET_DRAWABLE (arrow))
175         {
176           gdk_window_clear_area (GTK_WIDGET (arrow)->window,
177                                  GTK_WIDGET (arrow)->allocation.x + 1,
178                                  GTK_WIDGET (arrow)->allocation.y + 1,
179                                  GTK_WIDGET (arrow)->allocation.width - 2,
180                                  GTK_WIDGET (arrow)->allocation.height - 2);
181           gtk_widget_queue_draw (GTK_WIDGET (arrow));
182         }
183     }
184 }
185
186
187 static gint
188 gtk_arrow_expose (GtkWidget      *widget,
189                   GdkEventExpose *event)
190 {
191   GtkArrow *arrow;
192   GtkMisc *misc;
193   GtkShadowType shadow_type;
194   gint width, height;
195   gint x, y;
196   gint extent;
197
198   g_return_val_if_fail (widget != NULL, FALSE);
199   g_return_val_if_fail (GTK_IS_ARROW (widget), FALSE);
200   g_return_val_if_fail (event != NULL, FALSE);
201
202   if (GTK_WIDGET_DRAWABLE (widget))
203     {
204       arrow = GTK_ARROW (widget);
205       misc = GTK_MISC (widget);
206
207       width = widget->allocation.width - misc->xpad * 2;
208       height = widget->allocation.height - misc->ypad * 2;
209       extent = MIN (width, height);
210
211       x = ((widget->allocation.x + misc->xpad) * (1.0 - misc->xalign) +
212            (widget->allocation.x + widget->allocation.width - extent - misc->ypad) * misc->xalign);
213       y = ((widget->allocation.y + misc->ypad) * (1.0 - misc->yalign) +
214            (widget->allocation.y + widget->allocation.height - extent - misc->ypad) * misc->yalign);
215
216       shadow_type = arrow->shadow_type;
217
218       if (widget->state == GTK_STATE_ACTIVE)
219         {
220           if (shadow_type == GTK_SHADOW_IN)
221             shadow_type = GTK_SHADOW_OUT;
222           else if (shadow_type == GTK_SHADOW_OUT)
223             shadow_type = GTK_SHADOW_IN;
224           else if (shadow_type == GTK_SHADOW_ETCHED_IN)
225             shadow_type = GTK_SHADOW_ETCHED_OUT;
226           else if (shadow_type == GTK_SHADOW_ETCHED_OUT)
227             shadow_type = GTK_SHADOW_ETCHED_IN;
228         }
229
230       gtk_draw_arrow (widget->style, widget->window,
231                       widget->state, shadow_type, arrow->arrow_type, TRUE,
232                       x, y, extent, extent);
233     }
234
235   return FALSE;
236 }