gtkbutton.c 50.8 KB
Newer Older
Elliot Lee's avatar
Elliot Lee committed
1 2 3 4
/* 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
5
 * modify it under the terms of the GNU Lesser General Public
Elliot Lee's avatar
Elliot Lee committed
6 7 8 9 10 11
 * 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
12
 * Lesser General Public License for more details.
Elliot Lee's avatar
Elliot Lee committed
13
 *
14
 * You should have received a copy of the GNU Lesser General Public
15 16 17
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
Elliot Lee's avatar
Elliot Lee committed
18
 */
19 20

/*
21
 * Modified by the GTK+ Team and others 1997-2001.  See the AUTHORS
22 23 24 25 26
 * file for a list of people on the GTK+ Team.  See the ChangeLog
 * files for a list of changes.  These files are distributed with
 * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
 */

27
#include <config.h>
28
#include <string.h>
29
#include "gtkalignment.h"
Elliot Lee's avatar
Elliot Lee committed
30 31 32
#include "gtkbutton.h"
#include "gtklabel.h"
#include "gtkmain.h"
33
#include "gtkmarshalers.h"
34 35 36 37
#include "gtkimage.h"
#include "gtkhbox.h"
#include "gtkstock.h"
#include "gtkiconfactory.h"
38
#include "gtkprivate.h"
39
#include "gtkintl.h"
40
#include "gtkalias.h"
Elliot Lee's avatar
Elliot Lee committed
41 42

#define CHILD_SPACING     1
43

44 45
static const GtkBorder default_default_border = { 1, 1, 1, 1 };
static const GtkBorder default_default_outside_border = { 0, 0, 0, 0 };
Elliot Lee's avatar
Elliot Lee committed
46

Matthias Clasen's avatar
Matthias Clasen committed
47
/* Time out before giving up on getting a key release when animating
48 49 50
 * the close button.
 */
#define ACTIVATE_TIMEOUT 250
Elliot Lee's avatar
Elliot Lee committed
51 52 53 54 55 56 57

enum {
  PRESSED,
  RELEASED,
  CLICKED,
  ENTER,
  LEAVE,
58
  ACTIVATE,
Elliot Lee's avatar
Elliot Lee committed
59 60
  LAST_SIGNAL
};
61

62
enum {
63 64
  PROP_0,
  PROP_LABEL,
Matthias Clasen's avatar
Matthias Clasen committed
65
  PROP_IMAGE,
66 67
  PROP_RELIEF,
  PROP_USE_UNDERLINE,
Soeren Sandmann's avatar
Soeren Sandmann committed
68
  PROP_USE_STOCK,
69 70
  PROP_FOCUS_ON_CLICK,
  PROP_XALIGN,
Matthias Clasen's avatar
Matthias Clasen committed
71
  PROP_YALIGN,
72 73 74 75 76 77 78
};

#define GTK_BUTTON_GET_PRIVATE(o)       (G_TYPE_INSTANCE_GET_PRIVATE ((o), GTK_TYPE_BUTTON, GtkButtonPrivate))
typedef struct _GtkButtonPrivate GtkButtonPrivate;

struct _GtkButtonPrivate
{
Matthias Clasen's avatar
Matthias Clasen committed
79 80 81
  gfloat       xalign;
  gfloat       yalign;
  GtkWidget   *image;
Matthias Clasen's avatar
Matthias Clasen committed
82
  guint        align_set : 1;
Matthias Clasen's avatar
Matthias Clasen committed
83
  guint        image_is_stock : 1;
84
};
85

Elliot Lee's avatar
Elliot Lee committed
86 87
static void gtk_button_class_init     (GtkButtonClass   *klass);
static void gtk_button_init           (GtkButton        *button);
88
static void gtk_button_destroy        (GtkObject        *object);
89 90 91 92 93 94 95 96
static void gtk_button_set_property   (GObject         *object,
                                       guint            prop_id,
                                       const GValue    *value,
                                       GParamSpec      *pspec);
static void gtk_button_get_property   (GObject         *object,
                                       guint            prop_id,
                                       GValue          *value,
                                       GParamSpec      *pspec);
Matthias Clasen's avatar
Matthias Clasen committed
97 98
static void gtk_button_screen_changed (GtkWidget        *widget,
				       GdkScreen        *previous_screen);
Elliot Lee's avatar
Elliot Lee committed
99
static void gtk_button_realize        (GtkWidget        *widget);
100
static void gtk_button_unrealize      (GtkWidget        *widget);
101 102
static void gtk_button_map            (GtkWidget        *widget);
static void gtk_button_unmap          (GtkWidget        *widget);
Elliot Lee's avatar
Elliot Lee committed
103 104 105 106 107 108 109 110 111 112
static void gtk_button_size_request   (GtkWidget        *widget,
				       GtkRequisition   *requisition);
static void gtk_button_size_allocate  (GtkWidget        *widget,
				       GtkAllocation    *allocation);
static gint gtk_button_expose         (GtkWidget        *widget,
				       GdkEventExpose   *event);
static gint gtk_button_button_press   (GtkWidget        *widget,
				       GdkEventButton   *event);
static gint gtk_button_button_release (GtkWidget        *widget,
				       GdkEventButton   *event);
113 114
static gint gtk_button_key_release    (GtkWidget        *widget,
				       GdkEventKey      *event);
Elliot Lee's avatar
Elliot Lee committed
115 116 117 118 119 120
static gint gtk_button_enter_notify   (GtkWidget        *widget,
				       GdkEventCrossing *event);
static gint gtk_button_leave_notify   (GtkWidget        *widget,
				       GdkEventCrossing *event);
static void gtk_real_button_pressed   (GtkButton        *button);
static void gtk_real_button_released  (GtkButton        *button);
Manish Singh's avatar
Manish Singh committed
121
static void gtk_real_button_activate  (GtkButton         *button);
122
static void gtk_button_update_state   (GtkButton        *button);
123 124
static void gtk_button_add            (GtkContainer   *container,
			               GtkWidget      *widget);
Manish Singh's avatar
Manish Singh committed
125
static GType gtk_button_child_type    (GtkContainer     *container);
126 127
static void gtk_button_finish_activate (GtkButton *button,
					gboolean   do_it);
Elliot Lee's avatar
Elliot Lee committed
128

129 130 131
static GObject*	gtk_button_constructor (GType                  type,
					guint                  n_construct_properties,
					GObjectConstructParam *construct_params);
Matthias Clasen's avatar
Matthias Clasen committed
132
static void gtk_button_construct_child (GtkButton             *button);
133 134 135 136 137
static void gtk_button_state_changed   (GtkWidget             *widget,
					GtkStateType           previous_state);
static void gtk_button_grab_notify     (GtkWidget             *widget,
					gboolean               was_grabbed);

138 139


140
static GtkBinClass *parent_class = NULL;
141
static guint button_signals[LAST_SIGNAL] = { 0 };
Elliot Lee's avatar
Elliot Lee committed
142 143


Manish Singh's avatar
Manish Singh committed
144
GType
145
gtk_button_get_type (void)
Elliot Lee's avatar
Elliot Lee committed
146
{
Manish Singh's avatar
Manish Singh committed
147
  static GType button_type = 0;
Elliot Lee's avatar
Elliot Lee committed
148 149 150

  if (!button_type)
    {
151
      static const GTypeInfo button_info =
Elliot Lee's avatar
Elliot Lee committed
152 153
      {
	sizeof (GtkButtonClass),
154 155 156 157 158 159 160 161
	NULL,		/* base_init */
	NULL,		/* base_finalize */
	(GClassInitFunc) gtk_button_class_init,
	NULL,		/* class_finalize */
	NULL,		/* class_data */
	sizeof (GtkButton),
	16,		/* n_preallocs */
	(GInstanceInitFunc) gtk_button_init,
Elliot Lee's avatar
Elliot Lee committed
162 163
      };

Manish Singh's avatar
Manish Singh committed
164 165
      button_type = g_type_register_static (GTK_TYPE_BIN, "GtkButton",
					    &button_info, 0);
Elliot Lee's avatar
Elliot Lee committed
166 167 168 169 170 171 172 173
    }

  return button_type;
}

static void
gtk_button_class_init (GtkButtonClass *klass)
{
Manish Singh's avatar
Manish Singh committed
174
  GObjectClass *gobject_class;
Elliot Lee's avatar
Elliot Lee committed
175 176 177 178
  GtkObjectClass *object_class;
  GtkWidgetClass *widget_class;
  GtkContainerClass *container_class;

Manish Singh's avatar
Manish Singh committed
179
  gobject_class = G_OBJECT_CLASS (klass);
Elliot Lee's avatar
Elliot Lee committed
180 181 182
  object_class = (GtkObjectClass*) klass;
  widget_class = (GtkWidgetClass*) klass;
  container_class = (GtkContainerClass*) klass;
183 184
  
  parent_class = g_type_class_peek_parent (klass);
185

Manish Singh's avatar
Manish Singh committed
186 187 188
  gobject_class->constructor = gtk_button_constructor;
  gobject_class->set_property = gtk_button_set_property;
  gobject_class->get_property = gtk_button_get_property;
189

190 191
  object_class->destroy = gtk_button_destroy;

Matthias Clasen's avatar
Matthias Clasen committed
192
  widget_class->screen_changed = gtk_button_screen_changed;
193
  widget_class->realize = gtk_button_realize;
194
  widget_class->unrealize = gtk_button_unrealize;
195 196
  widget_class->map = gtk_button_map;
  widget_class->unmap = gtk_button_unmap;
197 198 199 200 201
  widget_class->size_request = gtk_button_size_request;
  widget_class->size_allocate = gtk_button_size_allocate;
  widget_class->expose_event = gtk_button_expose;
  widget_class->button_press_event = gtk_button_button_press;
  widget_class->button_release_event = gtk_button_button_release;
202
  widget_class->key_release_event = gtk_button_key_release;
203 204
  widget_class->enter_notify_event = gtk_button_enter_notify;
  widget_class->leave_notify_event = gtk_button_leave_notify;
205 206
  widget_class->state_changed = gtk_button_state_changed;
  widget_class->grab_notify = gtk_button_grab_notify;
207 208

  container_class->child_type = gtk_button_child_type;
209
  container_class->add = gtk_button_add;
210 211 212 213

  klass->pressed = gtk_real_button_pressed;
  klass->released = gtk_real_button_released;
  klass->clicked = NULL;
214 215
  klass->enter = gtk_button_update_state;
  klass->leave = gtk_button_update_state;
216
  klass->activate = gtk_real_button_activate;
217

Manish Singh's avatar
Manish Singh committed
218
  g_object_class_install_property (gobject_class,
219 220
                                   PROP_LABEL,
                                   g_param_spec_string ("label",
221 222
                                                        P_("Label"),
                                                        P_("Text of the label widget inside the button, if the button contains a label widget"),
223
                                                        NULL,
224
                                                        GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT));
225
  
Manish Singh's avatar
Manish Singh committed
226
  g_object_class_install_property (gobject_class,
227
                                   PROP_USE_UNDERLINE,
Matthias Clasen's avatar
x  
Matthias Clasen committed
228
                                   g_param_spec_boolean ("use-underline",
229 230
							 P_("Use underline"),
							 P_("If set, an underline in the text indicates the next character should be used for the mnemonic accelerator key"),
231
                                                        FALSE,
232
                                                        GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT));
233
  
Manish Singh's avatar
Manish Singh committed
234
  g_object_class_install_property (gobject_class,
235
                                   PROP_USE_STOCK,
Matthias Clasen's avatar
x  
Matthias Clasen committed
236
                                   g_param_spec_boolean ("use-stock",
237 238
							 P_("Use stock"),
							 P_("If set, the label is used to pick a stock item instead of being displayed"),
239
                                                        FALSE,
240
                                                        GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT));
241
  
Soeren Sandmann's avatar
Soeren Sandmann committed
242 243
  g_object_class_install_property (gobject_class,
                                   PROP_FOCUS_ON_CLICK,
Matthias Clasen's avatar
x  
Matthias Clasen committed
244
                                   g_param_spec_boolean ("focus-on-click",
245 246
							 P_("Focus on click"),
							 P_("Whether the button grabs focus when it is clicked with the mouse"),
Soeren Sandmann's avatar
Soeren Sandmann committed
247
							 TRUE,
248
							 GTK_PARAM_READWRITE));
Soeren Sandmann's avatar
Soeren Sandmann committed
249
  
Manish Singh's avatar
Manish Singh committed
250
  g_object_class_install_property (gobject_class,
251 252
                                   PROP_RELIEF,
                                   g_param_spec_enum ("relief",
253 254
                                                      P_("Border relief"),
                                                      P_("The border relief style"),
255 256
                                                      GTK_TYPE_RELIEF_STYLE,
                                                      GTK_RELIEF_NORMAL,
257
                                                      GTK_PARAM_READWRITE));
258 259 260 261 262 263 264 265 266 267 268 269 270
  
  /**
   * GtkButton:xalign:
   *
   * If the child of the button is a #GtkMisc or #GtkAlignment, this property 
   * can be used to control it's horizontal alignment. 0.0 is left aligned, 
   * 1.0 is right aligned.
   * 
   * Since: 2.4
   */
  g_object_class_install_property (gobject_class,
                                   PROP_XALIGN,
                                   g_param_spec_float("xalign",
271 272
                                                      P_("Horizontal alignment for child"),
                                                      P_("Horizontal position of child in available space. 0.0 is left aligned, 1.0 is right aligned"),
273 274 275
                                                      0.0,
                                                      1.0,
                                                      0.5,
276
                                                      GTK_PARAM_READWRITE));
277 278 279 280 281 282 283 284 285 286 287 288 289

  /**
   * GtkButton:yalign:
   *
   * If the child of the button is a #GtkMisc or #GtkAlignment, this property 
   * can be used to control it's vertical alignment. 0.0 is top aligned, 
   * 1.0 is bottom aligned.
   * 
   * Since: 2.4
   */
  g_object_class_install_property (gobject_class,
                                   PROP_YALIGN,
                                   g_param_spec_float("yalign",
290 291
                                                      P_("Vertical alignment for child"),
                                                      P_("Vertical position of child in available space. 0.0 is top aligned, 1.0 is bottom aligned"),
292 293 294
                                                      0.0,
                                                      1.0,
                                                      0.5,
295
                                                      GTK_PARAM_READWRITE));
Elliot Lee's avatar
Elliot Lee committed
296

Matthias Clasen's avatar
Matthias Clasen committed
297 298 299 300 301 302 303 304 305 306 307 308 309
  /**
   * GtkButton::image:
   * 
   * The child widget to appear next to the button text.
   * 
   * Since: 2.6
   */
  g_object_class_install_property (gobject_class,
                                   PROP_IMAGE,
                                   g_param_spec_object ("image",
                                                        P_("Image widget"),
                                                        P_("Child widget to appear next to the button text"),
                                                        GTK_TYPE_WIDGET,
310
                                                        GTK_PARAM_READWRITE));
Matthias Clasen's avatar
Matthias Clasen committed
311

312 313 314 315 316 317 318 319
  /**
   * GtkButton::pressed:
   * @button: the object that received the signal
   *
   * Emitted when the button is pressed.
   * 
   * @Deprecated: Use the GtkWidget::button-press-event signal.
   */ 
Elliot Lee's avatar
Elliot Lee committed
320
  button_signals[PRESSED] =
Manish Singh's avatar
Manish Singh committed
321 322 323 324 325 326 327
    g_signal_new ("pressed",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GtkButtonClass, pressed),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
328 329 330 331 332 333 334 335 336

  /**
   * GtkButton::released:
   * @button: the object that received the signal
   *
   * Emitted when the button is released.
   * 
   * @Deprecated: Use the GtkWidget::button-release-event signal.
   */ 
Elliot Lee's avatar
Elliot Lee committed
337
  button_signals[RELEASED] =
Manish Singh's avatar
Manish Singh committed
338 339 340 341 342 343 344
    g_signal_new ("released",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GtkButtonClass, released),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
345 346 347 348 349 350 351

  /**
   * GtkButton::clicked:
   * @button: the object that received the signal
   *
   * Emitted when the button has been activated (pressed and released).
   */ 
Elliot Lee's avatar
Elliot Lee committed
352
  button_signals[CLICKED] =
Manish Singh's avatar
Manish Singh committed
353 354 355 356 357 358 359
    g_signal_new ("clicked",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
		  G_STRUCT_OFFSET (GtkButtonClass, clicked),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
360 361 362 363 364 365 366 367 368

  /**
   * GtkButton::enter:
   * @button: the object that received the signal
   *
   * Emitted when the pointer enters the button.
   * 
   * @Deprecated: Use the GtkWidget::enter-notify-event signal.
   */ 
Elliot Lee's avatar
Elliot Lee committed
369
  button_signals[ENTER] =
Manish Singh's avatar
Manish Singh committed
370 371 372 373 374 375 376
    g_signal_new ("enter",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GtkButtonClass, enter),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
377 378 379 380 381 382 383 384 385

  /**
   * GtkButton::leave:
   * @button: the object that received the signal
   *
   * Emitted when the pointer leaves the button.
   * 
   * @Deprecated: Use the GtkWidget::leave-notify-event signal.
   */ 
Elliot Lee's avatar
Elliot Lee committed
386
  button_signals[LEAVE] =
Manish Singh's avatar
Manish Singh committed
387 388 389 390 391 392 393
    g_signal_new ("leave",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GtkButtonClass, leave),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
394 395 396

  /**
   * GtkButton::activate:
Matthias Clasen's avatar
Matthias Clasen committed
397
   * @widget: the object which received the signal.
398 399 400 401 402 403
   *
   * The "activate" signal on GtkButton is an action signal and
   * emitting it causes the button to animate press then release. 
   * Applications should never connect to this signal, but use the
   * "clicked" signal.
   */
404
  button_signals[ACTIVATE] =
Manish Singh's avatar
Manish Singh committed
405 406 407 408 409 410 411
    g_signal_new ("activate",
		  G_OBJECT_CLASS_TYPE (object_class),
		  G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
		  G_STRUCT_OFFSET (GtkButtonClass, activate),
		  NULL, NULL,
		  _gtk_marshal_VOID__VOID,
		  G_TYPE_NONE, 0);
412
  widget_class->activate_signal = button_signals[ACTIVATE];
413 414

  gtk_widget_class_install_style_property (widget_class,
Matthias Clasen's avatar
x  
Matthias Clasen committed
415
					   g_param_spec_boxed ("default-border",
416 417
							       P_("Default Spacing"),
							       P_("Extra space to add for CAN_DEFAULT buttons"),
418
							       GTK_TYPE_BORDER,
419
							       GTK_PARAM_READABLE));
420 421

  gtk_widget_class_install_style_property (widget_class,
Matthias Clasen's avatar
x  
Matthias Clasen committed
422
					   g_param_spec_boxed ("default-outside-border",
423 424
							       P_("Default Outside Spacing"),
							       P_("Extra space to add for CAN_DEFAULT buttons that is always drawn outside the border"),
425
							       GTK_TYPE_BORDER,
426
							       GTK_PARAM_READABLE));
427
  gtk_widget_class_install_style_property (widget_class,
Matthias Clasen's avatar
x  
Matthias Clasen committed
428
					   g_param_spec_int ("child-displacement-x",
429 430
							     P_("Child X Displacement"),
							     P_("How far in the x direction to move the child when the button is depressed"),
431 432 433
							     G_MININT,
							     G_MAXINT,
							     0,
434
							     GTK_PARAM_READABLE));
435
  gtk_widget_class_install_style_property (widget_class,
Matthias Clasen's avatar
x  
Matthias Clasen committed
436
					   g_param_spec_int ("child-displacement-y",
437 438
							     P_("Child Y Displacement"),
							     P_("How far in the y direction to move the child when the button is depressed"),
439 440 441
							     G_MININT,
							     G_MAXINT,
							     0,
442
							     GTK_PARAM_READABLE));
443

444 445 446 447 448 449 450 451 452 453 454 455 456
  /**
   * GtkButton:displace-focus:
   *
   * Whether the child_displacement_x/child_displacement_y properties should also 
   * affect the focus rectangle.
   *
   * Since: 2.6
   */
  gtk_widget_class_install_style_property (widget_class,
					   g_param_spec_boolean ("displace-focus",
								 P_("Displace focus"),
								 P_("Whether the child_displacement_x/_y properties should also affect the focus rectangle"),
						       FALSE,
457
						       GTK_PARAM_READABLE));
458

Matthias Clasen's avatar
Matthias Clasen committed
459 460 461 462
  gtk_settings_install_property (g_param_spec_boolean ("gtk-button-images",
						       P_("Show button images"),
						       P_("Whether stock icons should be shown in buttons"),
						       TRUE,
463
						       GTK_PARAM_READWRITE));
Matthias Clasen's avatar
Matthias Clasen committed
464
  
465
  g_type_class_add_private (gobject_class, sizeof (GtkButtonPrivate));  
Elliot Lee's avatar
Elliot Lee committed
466 467 468 469 470
}

static void
gtk_button_init (GtkButton *button)
{
471 472
  GtkButtonPrivate *priv = GTK_BUTTON_GET_PRIVATE (button);

473
  GTK_WIDGET_SET_FLAGS (button, GTK_CAN_FOCUS | GTK_RECEIVES_DEFAULT);
474
  GTK_WIDGET_SET_FLAGS (button, GTK_NO_WINDOW);
Elliot Lee's avatar
Elliot Lee committed
475

476 477 478
  button->label_text = NULL;
  
  button->constructed = FALSE;
Elliot Lee's avatar
Elliot Lee committed
479 480
  button->in_button = FALSE;
  button->button_down = FALSE;
481
  button->relief = GTK_RELIEF_NORMAL;
482 483
  button->use_stock = FALSE;
  button->use_underline = FALSE;
484
  button->depressed = FALSE;
485
  button->depress_on_activate = TRUE;
Soeren Sandmann's avatar
Soeren Sandmann committed
486
  button->focus_on_click = TRUE;
487 488 489

  priv->xalign = 0.5;
  priv->yalign = 0.5;
Matthias Clasen's avatar
Matthias Clasen committed
490
  priv->align_set = 0;
491
  priv->image_is_stock = TRUE;
Elliot Lee's avatar
Elliot Lee committed
492 493
}

494 495 496 497 498 499 500 501 502 503 504 505 506 507
static void
gtk_button_destroy (GtkObject *object)
{
  GtkButton *button = GTK_BUTTON (object);
  
  if (button->label_text)
    {
      g_free (button->label_text);
      button->label_text = NULL;
    }
  
  (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
}

508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529
static GObject*
gtk_button_constructor (GType                  type,
			guint                  n_construct_properties,
			GObjectConstructParam *construct_params)
{
  GObject *object;
  GtkButton *button;

  object = (* G_OBJECT_CLASS (parent_class)->constructor) (type,
							   n_construct_properties,
							   construct_params);

  button = GTK_BUTTON (object);
  button->constructed = TRUE;

  if (button->label_text != NULL)
    gtk_button_construct_child (button);
  
  return object;
}


Manish Singh's avatar
Manish Singh committed
530
static GType
531 532
gtk_button_child_type  (GtkContainer     *container)
{
533
  if (!GTK_BIN (container)->child)
534 535
    return GTK_TYPE_WIDGET;
  else
Manish Singh's avatar
Manish Singh committed
536
    return G_TYPE_NONE;
537 538
}

539
static void
Matthias Clasen's avatar
Matthias Clasen committed
540 541
maybe_set_alignment (GtkButton *button,
		     GtkWidget *widget)
542
{
Matthias Clasen's avatar
Matthias Clasen committed
543 544
  GtkButtonPrivate *priv = GTK_BUTTON_GET_PRIVATE (button);

545 546 547
  if (GTK_IS_MISC (widget))
    {
      GtkMisc *misc = GTK_MISC (widget);
Matthias Clasen's avatar
Matthias Clasen committed
548 549 550
      
      if (priv->align_set)
	gtk_misc_set_alignment (misc, priv->xalign, priv->yalign);
551 552 553 554
    }
  else if (GTK_IS_ALIGNMENT (widget))
    {
      GtkAlignment *alignment = GTK_ALIGNMENT (widget);
Matthias Clasen's avatar
Matthias Clasen committed
555 556 557 558

      if (priv->align_set)
	gtk_alignment_set (alignment, priv->xalign, priv->yalign, 
			   alignment->xscale, alignment->yscale);
559 560 561 562 563 564 565
    }
}

static void
gtk_button_add (GtkContainer *container,
		GtkWidget    *widget)
{
Matthias Clasen's avatar
Matthias Clasen committed
566
  maybe_set_alignment (GTK_BUTTON (container), widget);
567 568 569 570

  GTK_CONTAINER_CLASS (parent_class)->add (container, widget);
}

Elliot Lee's avatar
Elliot Lee committed
571
static void
572 573 574 575
gtk_button_set_property (GObject         *object,
                         guint            prop_id,
                         const GValue    *value,
                         GParamSpec      *pspec)
Elliot Lee's avatar
Elliot Lee committed
576
{
577 578
  GtkButton *button = GTK_BUTTON (object);
  GtkButtonPrivate *priv = GTK_BUTTON_GET_PRIVATE (button);
Elliot Lee's avatar
Elliot Lee committed
579

580
  switch (prop_id)
581
    {
582
    case PROP_LABEL:
583
      gtk_button_set_label (button, g_value_get_string (value));
584
      break;
Matthias Clasen's avatar
Matthias Clasen committed
585 586 587
    case PROP_IMAGE:
      gtk_button_set_image (button, (GtkWidget *) g_value_get_object (value));
      break;
588 589
    case PROP_RELIEF:
      gtk_button_set_relief (button, g_value_get_enum (value));
590
      break;
591 592 593 594 595 596
    case PROP_USE_UNDERLINE:
      gtk_button_set_use_underline (button, g_value_get_boolean (value));
      break;
    case PROP_USE_STOCK:
      gtk_button_set_use_stock (button, g_value_get_boolean (value));
      break;
Soeren Sandmann's avatar
Soeren Sandmann committed
597 598 599
    case PROP_FOCUS_ON_CLICK:
      gtk_button_set_focus_on_click (button, g_value_get_boolean (value));
      break;
600 601 602 603 604 605
    case PROP_XALIGN:
      gtk_button_set_alignment (button, g_value_get_float (value), priv->yalign);
      break;
    case PROP_YALIGN:
      gtk_button_set_alignment (button, priv->xalign, g_value_get_float (value));
      break;
606
    default:
607
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
608 609 610 611 612
      break;
    }
}

static void
613 614 615 616
gtk_button_get_property (GObject         *object,
                         guint            prop_id,
                         GValue          *value,
                         GParamSpec      *pspec)
617
{
618 619
  GtkButton *button = GTK_BUTTON (object);
  GtkButtonPrivate *priv = GTK_BUTTON_GET_PRIVATE (button);
620

621
  switch (prop_id)
622
    {
623
    case PROP_LABEL:
624
      g_value_set_string (value, button->label_text);
625
      break;
Matthias Clasen's avatar
Matthias Clasen committed
626 627 628
    case PROP_IMAGE:
      g_value_set_object (value, (GObject *)priv->image);
      break;
629
    case PROP_RELIEF:
Alexander Larsson's avatar
Alexander Larsson committed
630
      g_value_set_enum (value, gtk_button_get_relief (button));
631
      break;
632 633 634 635 636 637
    case PROP_USE_UNDERLINE:
      g_value_set_boolean (value, button->use_underline);
      break;
    case PROP_USE_STOCK:
      g_value_set_boolean (value, button->use_stock);
      break;
Soeren Sandmann's avatar
Soeren Sandmann committed
638 639 640
    case PROP_FOCUS_ON_CLICK:
      g_value_set_boolean (value, button->focus_on_click);
      break;
641 642 643 644 645 646
    case PROP_XALIGN:
      g_value_set_float (value, priv->xalign);
      break;
    case PROP_YALIGN:
      g_value_set_float (value, priv->yalign);
      break;
Tim Janik's avatar
Tim Janik committed
647
    default:
648
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Tim Janik's avatar
Tim Janik committed
649
      break;
Elliot Lee's avatar
Elliot Lee committed
650 651 652 653
    }
}

GtkWidget*
654
gtk_button_new (void)
Elliot Lee's avatar
Elliot Lee committed
655
{
Manish Singh's avatar
Manish Singh committed
656
  return g_object_new (GTK_TYPE_BUTTON, NULL);
Elliot Lee's avatar
Elliot Lee committed
657 658
}

Matthias Clasen's avatar
Matthias Clasen committed
659 660 661 662 663 664 665 666 667 668 669
static gboolean
show_image (GtkButton *button)
{
  GtkSettings *settings = gtk_widget_get_settings (GTK_WIDGET (button));  
  gboolean show;

  g_object_get (settings, "gtk-button-images", &show, NULL);

  return show;
}

670 671
static void
gtk_button_construct_child (GtkButton *button)
Elliot Lee's avatar
Elliot Lee committed
672
{
Matthias Clasen's avatar
Matthias Clasen committed
673
  GtkButtonPrivate *priv = GTK_BUTTON_GET_PRIVATE (button);
674 675 676
  GtkStockItem item;
  GtkWidget *label;
  GtkWidget *hbox;
677
  GtkWidget *align;
Matthias Clasen's avatar
Matthias Clasen committed
678 679
  GtkWidget *image = NULL;
  gchar *label_text = NULL;
680
  
681 682 683 684 685
  if (!button->constructed)
    return;
  
  if (button->label_text == NULL)
    return;
Elliot Lee's avatar
Elliot Lee committed
686

687
  if (GTK_BIN (button)->child)
688
    {
Matthias Clasen's avatar
Matthias Clasen committed
689 690 691
      if (priv->image && !priv->image_is_stock)
	image = g_object_ref (priv->image);

692 693
      gtk_container_remove (GTK_CONTAINER (button),
  			    GTK_BIN (button)->child);
Matthias Clasen's avatar
Matthias Clasen committed
694
  
695 696
      priv->image = NULL;
    }
697 698 699
  
  if (button->use_stock &&
      gtk_stock_lookup (button->label_text, &item))
700
    {
Matthias Clasen's avatar
Matthias Clasen committed
701 702 703 704 705 706 707
      if (!image)
	image = g_object_ref (gtk_image_new_from_stock (button->label_text, GTK_ICON_SIZE_BUTTON));

      label_text = item.label;
    }
  else
    label_text = button->label_text;
708

Matthias Clasen's avatar
Matthias Clasen committed
709 710 711
  if (image)
    {
      label = gtk_label_new_with_mnemonic (label_text);
712 713
      gtk_label_set_mnemonic_widget (GTK_LABEL (label), GTK_WIDGET (button));
      
Matthias Clasen's avatar
Matthias Clasen committed
714 715
      priv->image = image;

Matthias Clasen's avatar
Matthias Clasen committed
716 717
      g_object_set (priv->image, 
		    "visible", show_image (button),
718
		    "no-show-all", TRUE,
Matthias Clasen's avatar
Matthias Clasen committed
719
		    NULL);
720
      hbox = gtk_hbox_new (FALSE, 2);
721

Matthias Clasen's avatar
Matthias Clasen committed
722 723 724 725 726
      if (priv->align_set)
	align = gtk_alignment_new (priv->xalign, priv->yalign, 0.0, 0.0);
      else
	align = gtk_alignment_new (0.5, 0.5, 0.0, 0.0);
	
Matthias Clasen's avatar
Matthias Clasen committed
727
      gtk_box_pack_start (GTK_BOX (hbox), priv->image, FALSE, FALSE, 0);
728
      gtk_box_pack_end (GTK_BOX (hbox), label, FALSE, FALSE, 0);
729
      
730 731 732
      gtk_container_add (GTK_CONTAINER (button), align);
      gtk_container_add (GTK_CONTAINER (align), hbox);
      gtk_widget_show_all (align);
733

Matthias Clasen's avatar
Matthias Clasen committed
734 735
      g_object_unref (image);

736
      return;
737
    }
Matthias Clasen's avatar
Matthias Clasen committed
738 739
  
 if (button->use_underline)
740
    {
741 742
      label = gtk_label_new_with_mnemonic (button->label_text);
      gtk_label_set_mnemonic_widget (GTK_LABEL (label), GTK_WIDGET (button));
743
    }
744 745
  else
    label = gtk_label_new (button->label_text);
746
  
Matthias Clasen's avatar
Matthias Clasen committed
747 748
  if (priv->align_set)
    gtk_misc_set_alignment (GTK_MISC (label), priv->xalign, priv->yalign);
749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767

  gtk_widget_show (label);
  gtk_container_add (GTK_CONTAINER (button), label);
}


GtkWidget*
gtk_button_new_with_label (const gchar *label)
{
  return g_object_new (GTK_TYPE_BUTTON, "label", label, NULL);
}

/**
 * gtk_button_new_from_stock:
 * @stock_id: the name of the stock item 
 *
 * Creates a new #GtkButton containing the image and text from a stock item.
 * Some stock ids have preprocessor macros like #GTK_STOCK_OK and
 * #GTK_STOCK_APPLY.
768 769 770 771 772
 *
 * If @stock_id is unknown, then it will be treated as a mnemonic
 * label (as for gtk_button_new_with_mnemonic()).
 *
 * Returns: a new #GtkButton
773 774
 **/
GtkWidget*
775
gtk_button_new_from_stock (const gchar *stock_id)
776
{
777 778
  return g_object_new (GTK_TYPE_BUTTON,
                       "label", stock_id,
779 780
                       "use-stock", TRUE,
                       "use-underline", TRUE,
781
                       NULL);
782 783
}

784 785 786 787 788 789 790
/**
 * gtk_button_new_with_mnemonic:
 * @label: The text of the button, with an underscore in front of the
 *         mnemonic character
 * @returns: a new #GtkButton
 *
 * Creates a new #GtkButton containing a label.
791 792 793 794
 * If characters in @label are preceded by an underscore, they are underlined.
 * If you need a literal underscore character in a label, use '__' (two 
 * underscores). The first underlined character represents a keyboard 
 * accelerator called a mnemonic.
795 796
 * Pressing Alt and that key activates the button.
 **/
797
GtkWidget*
798
gtk_button_new_with_mnemonic (const gchar *label)
799
{
800
  return g_object_new (GTK_TYPE_BUTTON, "label", label, "use-underline", TRUE,  NULL);
801 802
}

Elliot Lee's avatar
Elliot Lee committed
803 804 805
void
gtk_button_pressed (GtkButton *button)
{
806 807
  g_return_if_fail (GTK_IS_BUTTON (button));

Matthias Clasen's avatar
Matthias Clasen committed
808
  
Manish Singh's avatar
Manish Singh committed
809
  g_signal_emit (button, button_signals[PRESSED], 0);
Elliot Lee's avatar
Elliot Lee committed
810 811 812 813 814
}

void
gtk_button_released (GtkButton *button)
{
815 816
  g_return_if_fail (GTK_IS_BUTTON (button));

Manish Singh's avatar
Manish Singh committed
817
  g_signal_emit (button, button_signals[RELEASED], 0);
Elliot Lee's avatar
Elliot Lee committed
818 819 820 821 822
}

void
gtk_button_clicked (GtkButton *button)
{
823 824
  g_return_if_fail (GTK_IS_BUTTON (button));

Manish Singh's avatar
Manish Singh committed
825
  g_signal_emit (button, button_signals[CLICKED], 0);
Elliot Lee's avatar
Elliot Lee committed
826 827 828 829 830
}

void
gtk_button_enter (GtkButton *button)
{
831 832
  g_return_if_fail (GTK_IS_BUTTON (button));

Manish Singh's avatar
Manish Singh committed
833
  g_signal_emit (button, button_signals[ENTER], 0);
Elliot Lee's avatar
Elliot Lee committed
834 835 836 837 838
}

void
gtk_button_leave (GtkButton *button)
{
839 840
  g_return_if_fail (GTK_IS_BUTTON (button));

Manish Singh's avatar
Manish Singh committed
841
  g_signal_emit (button, button_signals[LEAVE], 0);
Elliot Lee's avatar
Elliot Lee committed
842 843
}

844 845 846 847 848 849
void
gtk_button_set_relief (GtkButton *button,
		       GtkReliefStyle newrelief)
{
  g_return_if_fail (GTK_IS_BUTTON (button));

850 851 852 853 854 855
  if (newrelief != button->relief) 
    {
       button->relief = newrelief;
       g_object_notify (G_OBJECT (button), "relief");
       gtk_widget_queue_draw (GTK_WIDGET (button));
    }
856 857 858
}

GtkReliefStyle
859
gtk_button_get_relief (GtkButton *button)
860 861 862 863 864 865
{
  g_return_val_if_fail (GTK_IS_BUTTON (button), GTK_RELIEF_NORMAL);

  return button->relief;
}

Elliot Lee's avatar
Elliot Lee committed
866 867 868 869 870 871
static void
gtk_button_realize (GtkWidget *widget)
{
  GtkButton *button;
  GdkWindowAttr attributes;
  gint attributes_mask;
Owen Taylor's avatar
Owen Taylor committed
872
  gint border_width;
Elliot Lee's avatar
Elliot Lee committed
873 874 875 876

  button = GTK_BUTTON (widget);
  GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);

Owen Taylor's avatar
Owen Taylor committed
877 878
  border_width = GTK_CONTAINER (widget)->border_width;

Elliot Lee's avatar
Elliot Lee committed
879
  attributes.window_type = GDK_WINDOW_CHILD;
Owen Taylor's avatar
Owen Taylor committed
880 881 882 883
  attributes.x = widget->allocation.x + border_width;
  attributes.y = widget->allocation.y + border_width;
  attributes.width = widget->allocation.width - border_width * 2;
  attributes.height = widget->allocation.height - border_width * 2;
884
  attributes.wclass = GDK_INPUT_ONLY;
Elliot Lee's avatar
Elliot Lee committed
885
  attributes.event_mask = gtk_widget_get_events (widget);
886
  attributes.event_mask |= (GDK_BUTTON_PRESS_MASK |
Elliot Lee's avatar
Elliot Lee committed
887 888 889 890
			    GDK_BUTTON_RELEASE_MASK |
			    GDK_ENTER_NOTIFY_MASK |
			    GDK_LEAVE_NOTIFY_MASK);

891
  attributes_mask = GDK_WA_X | GDK_WA_Y;
Elliot Lee's avatar
Elliot Lee committed
892

893
  widget->window = gtk_widget_get_parent_window (widget);
Manish Singh's avatar
Manish Singh committed
894
  g_object_ref (widget->window);
895 896 897 898
  
  button->event_window = gdk_window_new (gtk_widget_get_parent_window (widget),
					 &attributes, attributes_mask);
  gdk_window_set_user_data (button->event_window, button);
Elliot Lee's avatar
Elliot Lee committed
899 900 901 902

  widget->style = gtk_style_attach (widget->style, widget->window);
}

903 904 905 906 907 908 909
static void
gtk_button_unrealize (GtkWidget *widget)
{
  GtkButton *button = GTK_BUTTON (widget);

  if (button->activate_timeout)
    gtk_button_finish_activate (button, FALSE);
910 911 912 913 914 915 916 917

  if (button->event_window)
    {
      gdk_window_set_user_data (button->event_window, NULL);
      gdk_window_destroy (button->event_window);
      button->event_window = NULL;
    }
  
918 919 920
  GTK_WIDGET_CLASS (parent_class)->unrealize (widget);
}

921 922 923 924 925
static void
gtk_button_map (GtkWidget *widget)
{
  GtkButton *button = GTK_BUTTON (widget);
  
926 927
  GTK_WIDGET_CLASS (parent_class)->map (widget);

928 929 930 931 932 933 934 935 936 937 938 939 940 941 942
  if (button->event_window)
    gdk_window_show (button->event_window);
}

static void
gtk_button_unmap (GtkWidget *widget)
{
  GtkButton *button = GTK_BUTTON (widget);
    
  if (button->event_window)
    gdk_window_hide (button->event_window);

  GTK_WIDGET_CLASS (parent_class)->unmap (widget);
}

Elliot Lee's avatar
Elliot Lee committed
943
static void
944
gtk_button_get_props (GtkButton *button,
945 946
		      GtkBorder *default_border,
		      GtkBorder *default_outside_border,
947
		      gboolean  *interior_focus)
Elliot Lee's avatar
Elliot Lee committed
948
{
949
  GtkWidget *widget =  GTK_WIDGET (button);
950 951 952 953
  GtkBorder *tmp_border;

  if (default_border)
    {
954
      gtk_widget_style_get (widget, "default-border", &tmp_border, NULL);
955 956 957 958 959 960 961 962 963

      if (tmp_border)
	{
	  *default_border = *tmp_border;
	  g_free (tmp_border);
	}
      else
	*default_border = default_default_border;
    }
Elliot Lee's avatar
Elliot Lee committed
964

965 966
  if (default_outside_border)
    {
967
      gtk_widget_style_get (widget, "default-outside-border", &tmp_border, NULL);
968 969 970 971 972 973 974 975 976

      if (tmp_border)
	{
	  *default_outside_border = *tmp_border;
	  g_free (tmp_border);
	}
      else
	*default_outside_border = default_default_outside_border;
    }
Elliot Lee's avatar
Elliot Lee committed
977

978
  if (interior_focus)
979
    gtk_widget_style_get (widget, "interior-focus", interior_focus, NULL);
980 981 982 983 984 985 986
}
	
static void
gtk_button_size_request (GtkWidget      *widget,
			 GtkRequisition *requisition)
{
  GtkButton *button = GTK_BUTTON (widget);
987
  GtkBorder default_border;
988 989
  gint focus_width;
  gint focus_pad;
Elliot Lee's avatar
Elliot Lee committed
990

991
  gtk_button_get_props (button, &default_border, NULL, NULL);
992 993 994 995 996
  gtk_widget_style_get (GTK_WIDGET (widget),
			"focus-line-width", &focus_width,
			"focus-padding", &focus_pad,
			NULL);
 
Elliot Lee's avatar
Elliot Lee committed
997
  requisition->width = (GTK_CONTAINER (widget)->border_width + CHILD_SPACING +
998
			GTK_WIDGET (widget)->style->xthickness) * 2;
Elliot Lee's avatar
Elliot Lee committed
999
  requisition->height = (GTK_CONTAINER (widget)->border_width + CHILD_SPACING +
1000
			 GTK_WIDGET (widget)->style->ythickness) * 2;
Elliot Lee's avatar
Elliot Lee committed
1001 1002 1003

  if (GTK_WIDGET_CAN_DEFAULT (widget))
    {
1004 1005
      requisition->width += default_border.left + default_border.right;
      requisition->height += default_border.top + default_border.bottom;
Elliot Lee's avatar
Elliot Lee committed
1006 1007
    }

1008
  if (GTK_BIN (button)->child && GTK_WIDGET_VISIBLE (GTK_BIN (button)->child))
Elliot Lee's avatar
Elliot Lee committed
1009
    {
1010
      GtkRequisition child_requisition;
Elliot Lee's avatar
Elliot Lee committed
1011

1012 1013 1014 1015
      gtk_widget_size_request (GTK_BIN (button)->child, &child_requisition);

      requisition->width += child_requisition.width;
      requisition->height += child_requisition.height;
Elliot Lee's avatar
Elliot Lee committed
1016
    }