gtkbutton.c 30.1 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 <string.h>
Elliot Lee's avatar
Elliot Lee committed
28 29 30 31
#include "gtkbutton.h"
#include "gtklabel.h"
#include "gtkmain.h"
#include "gtksignal.h"
32 33 34 35
#include "gtkimage.h"
#include "gtkhbox.h"
#include "gtkstock.h"
#include "gtkiconfactory.h"
36
#include "gtkintl.h"
Elliot Lee's avatar
Elliot Lee committed
37 38 39 40

#define CHILD_SPACING     1
#define DEFAULT_SPACING   7

41 42 43 44
/* Time out before giving up on getting a key release when animatng
 * the close button.
 */
#define ACTIVATE_TIMEOUT 250
Elliot Lee's avatar
Elliot Lee committed
45 46 47 48 49 50 51

enum {
  PRESSED,
  RELEASED,
  CLICKED,
  ENTER,
  LEAVE,
52
  ACTIVATE,
Elliot Lee's avatar
Elliot Lee committed
53 54
  LAST_SIGNAL
};
55

56
enum {
57 58 59
  PROP_0,
  PROP_LABEL,
  PROP_RELIEF
60
};
61

Elliot Lee's avatar
Elliot Lee committed
62 63
static void gtk_button_class_init     (GtkButtonClass   *klass);
static void gtk_button_init           (GtkButton        *button);
64 65 66 67 68 69 70 71
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);
Elliot Lee's avatar
Elliot Lee committed
72
static void gtk_button_realize        (GtkWidget        *widget);
73
static void gtk_button_unrealize      (GtkWidget        *widget);
Elliot Lee's avatar
Elliot Lee committed
74 75 76 77 78 79 80 81 82 83 84 85
static void gtk_button_size_request   (GtkWidget        *widget,
				       GtkRequisition   *requisition);
static void gtk_button_size_allocate  (GtkWidget        *widget,
				       GtkAllocation    *allocation);
static void gtk_button_paint          (GtkWidget        *widget,
				       GdkRectangle     *area);
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);
86 87
static gint gtk_button_key_release    (GtkWidget        *widget,
				       GdkEventKey      *event);
Elliot Lee's avatar
Elliot Lee committed
88 89 90 91 92 93 94 95 96 97 98 99
static gint gtk_button_enter_notify   (GtkWidget        *widget,
				       GdkEventCrossing *event);
static gint gtk_button_leave_notify   (GtkWidget        *widget,
				       GdkEventCrossing *event);
static void gtk_button_add            (GtkContainer     *container,
				       GtkWidget        *widget);
static void gtk_button_remove         (GtkContainer     *container,
				       GtkWidget        *widget);
static void gtk_real_button_pressed   (GtkButton        *button);
static void gtk_real_button_released  (GtkButton        *button);
static void gtk_real_button_enter     (GtkButton        *button);
static void gtk_real_button_leave     (GtkButton        *button);
100
static void gtk_real_button_activate (GtkButton         *button);
101
static GtkType gtk_button_child_type  (GtkContainer     *container);
Elliot Lee's avatar
Elliot Lee committed
102

103 104
static void gtk_button_finish_activate (GtkButton *button,
					gboolean   do_it);
Elliot Lee's avatar
Elliot Lee committed
105

106
static GtkBinClass *parent_class = NULL;
107
static guint button_signals[LAST_SIGNAL] = { 0 };
Elliot Lee's avatar
Elliot Lee committed
108 109


110
GtkType
111
gtk_button_get_type (void)
Elliot Lee's avatar
Elliot Lee committed
112
{
113
  static GtkType button_type = 0;
Elliot Lee's avatar
Elliot Lee committed
114 115 116

  if (!button_type)
    {
117
      static const GTypeInfo button_info =
Elliot Lee's avatar
Elliot Lee committed
118 119
      {
	sizeof (GtkButtonClass),
120 121 122 123 124 125 126 127
	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
128 129
      };

130
      button_type = g_type_register_static (GTK_TYPE_BIN, "GtkButton", &button_info, 0);
Elliot Lee's avatar
Elliot Lee committed
131 132 133 134 135 136 137 138 139 140 141 142 143 144 145
    }

  return button_type;
}

static void
gtk_button_class_init (GtkButtonClass *klass)
{
  GtkObjectClass *object_class;
  GtkWidgetClass *widget_class;
  GtkContainerClass *container_class;

  object_class = (GtkObjectClass*) klass;
  widget_class = (GtkWidgetClass*) klass;
  container_class = (GtkContainerClass*) klass;
146 147
  
  parent_class = g_type_class_peek_parent (klass);
148

149 150
  G_OBJECT_CLASS(object_class)->set_property = gtk_button_set_property;
  G_OBJECT_CLASS(object_class)->get_property = gtk_button_get_property;
151 152

  widget_class->realize = gtk_button_realize;
153
  widget_class->unrealize = gtk_button_unrealize;
154 155 156 157 158
  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;
159
  widget_class->key_release_event = gtk_button_key_release;
160 161 162 163 164 165 166 167 168 169 170 171
  widget_class->enter_notify_event = gtk_button_enter_notify;
  widget_class->leave_notify_event = gtk_button_leave_notify;

  container_class->add = gtk_button_add;
  container_class->remove = gtk_button_remove;
  container_class->child_type = gtk_button_child_type;

  klass->pressed = gtk_real_button_pressed;
  klass->released = gtk_real_button_released;
  klass->clicked = NULL;
  klass->enter = gtk_real_button_enter;
  klass->leave = gtk_real_button_leave;
172
  klass->activate = gtk_real_button_activate;
173

174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
  g_object_class_install_property (G_OBJECT_CLASS(object_class),
                                   PROP_LABEL,
                                   g_param_spec_string ("label",
                                                        _("Label"),
                                                        _("Text of the label widget inside the button, if the button contains a label widget."),
                                                        NULL,
                                                        G_PARAM_READABLE | G_PARAM_WRITABLE));
  
  g_object_class_install_property (G_OBJECT_CLASS(object_class),
                                   PROP_RELIEF,
                                   g_param_spec_enum ("relief",
                                                      _("Border relief"),
                                                      _("The border relief style."),
                                                      GTK_TYPE_RELIEF_STYLE,
                                                      GTK_RELIEF_NORMAL,
                                                      G_PARAM_READABLE | G_PARAM_WRITABLE));
Elliot Lee's avatar
Elliot Lee committed
190 191 192 193

  button_signals[PRESSED] =
    gtk_signal_new ("pressed",
                    GTK_RUN_FIRST,
194
                    GTK_CLASS_TYPE (object_class),
Elliot Lee's avatar
Elliot Lee committed
195
                    GTK_SIGNAL_OFFSET (GtkButtonClass, pressed),
196
                    gtk_marshal_VOID__VOID,
Elliot Lee's avatar
Elliot Lee committed
197 198 199 200
		    GTK_TYPE_NONE, 0);
  button_signals[RELEASED] =
    gtk_signal_new ("released",
                    GTK_RUN_FIRST,
201
                    GTK_CLASS_TYPE (object_class),
Elliot Lee's avatar
Elliot Lee committed
202
                    GTK_SIGNAL_OFFSET (GtkButtonClass, released),
203
                    gtk_marshal_VOID__VOID,
Elliot Lee's avatar
Elliot Lee committed
204 205 206
		    GTK_TYPE_NONE, 0);
  button_signals[CLICKED] =
    gtk_signal_new ("clicked",
207
                    GTK_RUN_FIRST | GTK_RUN_ACTION,
208
                    GTK_CLASS_TYPE (object_class),
Elliot Lee's avatar
Elliot Lee committed
209
                    GTK_SIGNAL_OFFSET (GtkButtonClass, clicked),
210
                    gtk_marshal_VOID__VOID,
Elliot Lee's avatar
Elliot Lee committed
211 212 213 214
		    GTK_TYPE_NONE, 0);
  button_signals[ENTER] =
    gtk_signal_new ("enter",
                    GTK_RUN_FIRST,
215
                    GTK_CLASS_TYPE (object_class),
216
                    GTK_SIGNAL_OFFSET (GtkButtonClass, enter),
217
                    gtk_marshal_VOID__VOID,
Elliot Lee's avatar
Elliot Lee committed
218 219 220 221
		    GTK_TYPE_NONE, 0);
  button_signals[LEAVE] =
    gtk_signal_new ("leave",
                    GTK_RUN_FIRST,
222
                    GTK_CLASS_TYPE (object_class),
223
                    GTK_SIGNAL_OFFSET (GtkButtonClass, leave),
224
                    gtk_marshal_VOID__VOID,
Elliot Lee's avatar
Elliot Lee committed
225
		    GTK_TYPE_NONE, 0);
226 227 228 229 230 231 232 233
  button_signals[ACTIVATE] =
    gtk_signal_new ("activate",
                    GTK_RUN_FIRST,
                    GTK_CLASS_TYPE (object_class),
                    GTK_SIGNAL_OFFSET (GtkButtonClass, activate),
                    gtk_marshal_VOID__VOID,
		    GTK_TYPE_NONE, 0);
  widget_class->activate_signal = button_signals[ACTIVATE];
234 235 236 237 238 239 240 241 242

  gtk_widget_class_install_style_property (widget_class,
					   g_param_spec_int ("default_spacing",
							     _("Default Spacing"),
							     _("Extra space to add for CAN_DEFAULT buttons"),
							     0,
							     G_MAXINT,
							     DEFAULT_SPACING,
							     G_PARAM_READABLE));
Elliot Lee's avatar
Elliot Lee committed
243 244 245 246 247
}

static void
gtk_button_init (GtkButton *button)
{
248
  GTK_WIDGET_SET_FLAGS (button, GTK_CAN_FOCUS | GTK_RECEIVES_DEFAULT);
249
  GTK_WIDGET_UNSET_FLAGS (button, GTK_NO_WINDOW);
Elliot Lee's avatar
Elliot Lee committed
250 251 252 253

  button->child = NULL;
  button->in_button = FALSE;
  button->button_down = FALSE;
254
  button->relief = GTK_RELIEF_NORMAL;
Elliot Lee's avatar
Elliot Lee committed
255 256
}

257 258 259
static GtkType
gtk_button_child_type  (GtkContainer     *container)
{
260
  if (!GTK_BIN (container)->child)
261 262 263 264 265
    return GTK_TYPE_WIDGET;
  else
    return GTK_TYPE_NONE;
}

Elliot Lee's avatar
Elliot Lee committed
266
static void
267 268 269 270
gtk_button_set_property (GObject         *object,
                         guint            prop_id,
                         const GValue    *value,
                         GParamSpec      *pspec)
Elliot Lee's avatar
Elliot Lee committed
271
{
272 273 274
  GtkButton *button;

  button = GTK_BUTTON (object);
Elliot Lee's avatar
Elliot Lee committed
275

276
  switch (prop_id)
277
    {
278
      GtkWidget *child;
279

280
    case PROP_LABEL:
281 282 283 284 285 286 287
      child = GTK_BIN (button)->child;
      if (!child)
	child = gtk_widget_new (GTK_TYPE_LABEL,
				"visible", TRUE,
				"parent", button,
				NULL);
      if (GTK_IS_LABEL (child))
288 289 290 291
        {
          gtk_label_set_text (GTK_LABEL (child),
                              g_value_get_string (value) ? g_value_get_string (value) : "");
        }
292
      break;
293 294
    case PROP_RELIEF:
      gtk_button_set_relief (button, g_value_get_enum (value));
295
      break;
296
    default:
297
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
298 299 300 301 302
      break;
    }
}

static void
303 304 305 306
gtk_button_get_property (GObject         *object,
                         guint            prop_id,
                         GValue          *value,
                         GParamSpec      *pspec)
307
{
308 309 310 311
  GtkButton *button;

  button = GTK_BUTTON (object);

312
  switch (prop_id)
313
    {
314
    case PROP_LABEL:
315
      if (GTK_BIN (button)->child && GTK_IS_LABEL (GTK_BIN (button)->child))
Alexander Larsson's avatar
Alexander Larsson committed
316
	 g_value_set_string (value, GTK_LABEL (GTK_BIN (button)->child)->label); 
317
      else
Alexander Larsson's avatar
Alexander Larsson committed
318
	 g_value_set_string (value, NULL);
319
      break;
320
    case PROP_RELIEF:
Alexander Larsson's avatar
Alexander Larsson committed
321
      g_value_set_enum (value, gtk_button_get_relief (button));
322
      break;
Tim Janik's avatar
Tim Janik committed
323
    default:
324
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Tim Janik's avatar
Tim Janik committed
325
      break;
Elliot Lee's avatar
Elliot Lee committed
326 327 328 329
    }
}

GtkWidget*
330
gtk_button_new (void)
Elliot Lee's avatar
Elliot Lee committed
331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350
{
  return GTK_WIDGET (gtk_type_new (gtk_button_get_type ()));
}

GtkWidget*
gtk_button_new_with_label (const gchar *label)
{
  GtkWidget *button;
  GtkWidget *label_widget;

  button = gtk_button_new ();
  label_widget = gtk_label_new (label);
  gtk_misc_set_alignment (GTK_MISC (label_widget), 0.5, 0.5);

  gtk_container_add (GTK_CONTAINER (button), label_widget);
  gtk_widget_show (label_widget);

  return button;
}

351 352 353 354 355 356 357 358 359
/**
 * gtk_button_new_from_stock:
 * @stock_id: the name of the stock item 
 * @returns: a new #GtkButton
 *
 * Creates a new #GtkButton containing the image and text from a stock item.
 * Some stock ids have preprocessor macros like #GTK_STOCK_BUTTON_OK and
 * #GTK_STOCK_BUTTON_APPLY.
 **/
360
GtkWidget*
361
gtk_button_new_from_stock (const gchar   *stock_id)
362 363 364 365 366 367 368 369 370 371 372 373
{
  GtkWidget *button;
  GtkStockItem item;

  if (gtk_stock_lookup (stock_id, &item))
    {
      GtkWidget *label;
      GtkWidget *image;
      GtkWidget *hbox;
      
      button = gtk_button_new ();

374 375 376
      label = gtk_label_new_with_mnemonic (item.label);

      gtk_label_set_mnemonic_widget (GTK_LABEL (label), button);
377 378

      image = gtk_image_new_from_stock (stock_id, GTK_ICON_SIZE_BUTTON);
Havoc Pennington's avatar
Havoc Pennington committed
379
      hbox = gtk_hbox_new (FALSE, 1);
380

Havoc Pennington's avatar
Havoc Pennington committed
381 382
      gtk_box_pack_start (GTK_BOX (hbox), image, FALSE, FALSE, 0);
      gtk_box_pack_end (GTK_BOX (hbox), label, TRUE, TRUE, 0);
383 384 385 386 387 388
      
      gtk_container_add (GTK_CONTAINER (button), hbox);
      gtk_widget_show_all (hbox);
    }
  else
    {
389
      button = gtk_button_new_with_mnemonic (stock_id);
390 391 392 393 394
    }
  
  return button;
}

395 396 397 398 399 400 401 402 403 404 405
/**
 * 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.
 * If characters in @label are preceded by an underscore, they are underlined
 * indicating that they represent a keyboard accelerator called a mnemonic.
 * Pressing Alt and that key activates the button.
 **/
406
GtkWidget*
407
gtk_button_new_with_mnemonic (const gchar *label)
408 409
{
  GtkWidget *button;
410
  GtkWidget *label_widget;
411 412 413

  button = gtk_button_new ();
  
414
  label_widget = gtk_label_new_with_mnemonic (label);
415

416
  gtk_label_set_mnemonic_widget (GTK_LABEL (label_widget), button);
417
  
418 419
  gtk_container_add (GTK_CONTAINER (button), label_widget);
  gtk_widget_show (label_widget);
420 421 422 423

  return button;
}

Elliot Lee's avatar
Elliot Lee committed
424 425 426
void
gtk_button_pressed (GtkButton *button)
{
427 428 429
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

Elliot Lee's avatar
Elliot Lee committed
430 431 432 433 434 435
  gtk_signal_emit (GTK_OBJECT (button), button_signals[PRESSED]);
}

void
gtk_button_released (GtkButton *button)
{
436 437 438
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

Elliot Lee's avatar
Elliot Lee committed
439 440 441 442 443 444
  gtk_signal_emit (GTK_OBJECT (button), button_signals[RELEASED]);
}

void
gtk_button_clicked (GtkButton *button)
{
445 446 447
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

Elliot Lee's avatar
Elliot Lee committed
448 449 450 451 452 453
  gtk_signal_emit (GTK_OBJECT (button), button_signals[CLICKED]);
}

void
gtk_button_enter (GtkButton *button)
{
454 455 456
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

Elliot Lee's avatar
Elliot Lee committed
457 458 459 460 461 462
  gtk_signal_emit (GTK_OBJECT (button), button_signals[ENTER]);
}

void
gtk_button_leave (GtkButton *button)
{
463 464 465
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

Elliot Lee's avatar
Elliot Lee committed
466 467 468
  gtk_signal_emit (GTK_OBJECT (button), button_signals[LEAVE]);
}

469 470 471 472 473 474 475 476
void
gtk_button_set_relief (GtkButton *button,
		       GtkReliefStyle newrelief)
{
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

  button->relief = newrelief;
477
  g_object_notify(G_OBJECT(button), "relief");
478
  gtk_widget_queue_draw (GTK_WIDGET (button));
479 480 481
}

GtkReliefStyle
482
gtk_button_get_relief (GtkButton *button)
483 484 485 486 487 488 489
{
  g_return_val_if_fail (button != NULL, GTK_RELIEF_NORMAL);
  g_return_val_if_fail (GTK_IS_BUTTON (button), GTK_RELIEF_NORMAL);

  return button->relief;
}

Elliot Lee's avatar
Elliot Lee committed
490 491 492 493 494 495
static void
gtk_button_realize (GtkWidget *widget)
{
  GtkButton *button;
  GdkWindowAttr attributes;
  gint attributes_mask;
Owen Taylor's avatar
Owen Taylor committed
496
  gint border_width;
Elliot Lee's avatar
Elliot Lee committed
497 498 499 500 501 502 503

  g_return_if_fail (widget != NULL);
  g_return_if_fail (GTK_IS_BUTTON (widget));

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

Owen Taylor's avatar
Owen Taylor committed
504 505
  border_width = GTK_CONTAINER (widget)->border_width;

Elliot Lee's avatar
Elliot Lee committed
506
  attributes.window_type = GDK_WINDOW_CHILD;
Owen Taylor's avatar
Owen Taylor committed
507 508 509 510
  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;
Elliot Lee's avatar
Elliot Lee committed
511 512 513 514 515 516 517 518 519 520 521 522
  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);
  attributes.event_mask |= (GDK_EXPOSURE_MASK |
			    GDK_BUTTON_PRESS_MASK |
			    GDK_BUTTON_RELEASE_MASK |
			    GDK_ENTER_NOTIFY_MASK |
			    GDK_LEAVE_NOTIFY_MASK);

  attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;

523
  widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
Elliot Lee's avatar
Elliot Lee committed
524 525 526 527 528 529
  gdk_window_set_user_data (widget->window, button);

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

530 531 532 533 534 535 536 537 538 539 540
static void
gtk_button_unrealize (GtkWidget *widget)
{
  GtkButton *button = GTK_BUTTON (widget);

  if (button->activate_timeout)
    gtk_button_finish_activate (button, FALSE);
    
  GTK_WIDGET_CLASS (parent_class)->unrealize (widget);
}

Elliot Lee's avatar
Elliot Lee committed
541
static void
542 543 544
gtk_button_get_props (GtkButton *button,
		      gint      *default_spacing,
		      gboolean  *interior_focus)
Elliot Lee's avatar
Elliot Lee committed
545
{
546
  GtkWidget *widget =  GTK_WIDGET (button);
Elliot Lee's avatar
Elliot Lee committed
547

548 549
  if (default_spacing)
    gtk_widget_style_get (widget, "default_spacing", default_spacing, NULL);
Elliot Lee's avatar
Elliot Lee committed
550

551 552 553 554 555 556 557 558 559 560 561
  if (interior_focus)
    gtk_widget_style_get (widget, "interior_focus", interior_focus, NULL);
}
	
static void
gtk_button_size_request (GtkWidget      *widget,
			 GtkRequisition *requisition)
{
  GtkButton *button = GTK_BUTTON (widget);
  gint default_spacing;
  gboolean interior_focus;
Elliot Lee's avatar
Elliot Lee committed
562

563 564
  gtk_button_get_props (button, &default_spacing, &interior_focus);
  
Elliot Lee's avatar
Elliot Lee committed
565
  requisition->width = (GTK_CONTAINER (widget)->border_width + CHILD_SPACING +
566
			GTK_WIDGET (widget)->style->xthickness) * 2;
Elliot Lee's avatar
Elliot Lee committed
567
  requisition->height = (GTK_CONTAINER (widget)->border_width + CHILD_SPACING +
568
			 GTK_WIDGET (widget)->style->ythickness) * 2;
Elliot Lee's avatar
Elliot Lee committed
569 570 571

  if (GTK_WIDGET_CAN_DEFAULT (widget))
    {
572
      requisition->width += (GTK_WIDGET (widget)->style->xthickness * 2 +
573
			     default_spacing);
574
      requisition->height += (GTK_WIDGET (widget)->style->ythickness * 2 +
575
			      default_spacing);
Elliot Lee's avatar
Elliot Lee committed
576 577
    }

578
  if (GTK_BIN (button)->child && GTK_WIDGET_VISIBLE (GTK_BIN (button)->child))
Elliot Lee's avatar
Elliot Lee committed
579
    {
580
      GtkRequisition child_requisition;
Elliot Lee's avatar
Elliot Lee committed
581

582 583 584 585
      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
586
    }
587 588 589 590 591 592

  if (interior_focus)
    {
      requisition->width += 2;
      requisition->height += 2;
    }
Elliot Lee's avatar
Elliot Lee committed
593 594 595 596 597 598
}

static void
gtk_button_size_allocate (GtkWidget     *widget,
			  GtkAllocation *allocation)
{
599
  GtkButton *button = GTK_BUTTON (widget);
Elliot Lee's avatar
Elliot Lee committed
600 601
  GtkAllocation child_allocation;

602 603 604 605
  gint border_width = GTK_CONTAINER (widget)->border_width;
  gint xthickness = GTK_WIDGET (widget)->style->xthickness;
  gint ythickness = GTK_WIDGET (widget)->style->ythickness;
  gint default_spacing;
Elliot Lee's avatar
Elliot Lee committed
606

607 608
  gtk_button_get_props (button, &default_spacing, NULL);
  
Elliot Lee's avatar
Elliot Lee committed
609 610 611 612 613 614 615 616 617
  widget->allocation = *allocation;

  if (GTK_WIDGET_REALIZED (widget))
    gdk_window_move_resize (widget->window,
			    widget->allocation.x + border_width,
			    widget->allocation.y + border_width,
			    widget->allocation.width - border_width * 2,
			    widget->allocation.height - border_width * 2);

618
  if (GTK_BIN (button)->child && GTK_WIDGET_VISIBLE (GTK_BIN (button)->child))
Elliot Lee's avatar
Elliot Lee committed
619
    {
620 621
      child_allocation.x = (CHILD_SPACING + xthickness);
      child_allocation.y = (CHILD_SPACING + ythickness);
Elliot Lee's avatar
Elliot Lee committed
622

623
      child_allocation.width = MAX (1, (gint)widget->allocation.width - child_allocation.x * 2 -
624
	                         border_width * 2);
625
      child_allocation.height = MAX (1, (gint)widget->allocation.height - child_allocation.y * 2 -
626
	                          border_width * 2);
Elliot Lee's avatar
Elliot Lee committed
627 628 629

      if (GTK_WIDGET_CAN_DEFAULT (button))
	{
630
	  child_allocation.x += (GTK_WIDGET (widget)->style->xthickness +
631
				 (1 + default_spacing) / 2);
632
	  child_allocation.y += (GTK_WIDGET (widget)->style->ythickness +
633
				 (1 + default_spacing) / 2);
634
	  child_allocation.width =  MAX (1, (gint)child_allocation.width -
635
					 (gint)(GTK_WIDGET (widget)->style->xthickness * 2 + default_spacing));
636
	  child_allocation.height = MAX (1, (gint)child_allocation.height -
637
					 (gint)(GTK_WIDGET (widget)->style->xthickness * 2 + default_spacing));
Elliot Lee's avatar
Elliot Lee committed
638 639
	}

640
      gtk_widget_size_allocate (GTK_BIN (button)->child, &child_allocation);
Elliot Lee's avatar
Elliot Lee committed
641 642 643
    }
}

644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669
/*
 * +------------------------------------------------+
 * |                   BORDER                       |
 * |  +------------------------------------------+  |
 * |  |\\\\\\\\\\\\\\\\DEFAULT\\\\\\\\\\\\\\\\\  |  |
 * |  |\\+------------------------------------+  |  |
 * |  |\\| |           SPACING       3      | |  |  |
 * |  |\\| +--------------------------------+ |  |  |
 * |  |\\| |########## FOCUS ###############| |  |  |
 * |  |\\| |#+----------------------------+#| |  |  |
 * |  |\\| |#|         RELIEF            \|#| |  |  |
 * |  |\\| |#|  +-----------------------+\|#| |  |  |
 * |  |\\|1|#|  +     THE TEXT          +\|#|2|  |  |
 * |  |\\| |#|  +-----------------------+\|#| |  |  |
 * |  |\\| |#| \\\\\ ythickness \\\\\\\\\\|#| |  |  |
 * |  |\\| |#+----------------------------+#| |  |  |
 * |  |\\| |########### 1 ##################| |  |  |
 * |  |\\| +--------------------------------+ |  |  |
 * |  |\\| |        default spacing   4     | |  |  |
 * |  |\\+------------------------------------+  |  |
 * |  |\            ythickness                   |  |
 * |  +------------------------------------------+  |
 * |                border_width                    |
 * +------------------------------------------------+
 */

Elliot Lee's avatar
Elliot Lee committed
670 671 672 673 674 675 676 677
static void
gtk_button_paint (GtkWidget    *widget,
		  GdkRectangle *area)
{
  GtkButton *button;
  GtkShadowType shadow_type;
  gint width, height;
  gint x, y;
678 679
  gint default_spacing;
  gboolean interior_focus;
680
   
Elliot Lee's avatar
Elliot Lee committed
681 682 683
  if (GTK_WIDGET_DRAWABLE (widget))
    {
      button = GTK_BUTTON (widget);
684 685

      gtk_button_get_props (button, &default_spacing, &interior_focus);
686
	
Elliot Lee's avatar
Elliot Lee committed
687 688 689 690 691
      x = 0;
      y = 0;
      width = widget->allocation.width - GTK_CONTAINER (widget)->border_width * 2;
      height = widget->allocation.height - GTK_CONTAINER (widget)->border_width * 2;

692 693 694 695 696 697 698 699 700 701 702 703
      gdk_window_set_back_pixmap (widget->window, NULL, TRUE);
      gdk_window_clear_area (widget->window, area->x, area->y, area->width, area->height);

      if (GTK_WIDGET_HAS_DEFAULT (widget) &&
	  GTK_BUTTON (widget)->relief == GTK_RELIEF_NORMAL)
	{
	  gtk_paint_box (widget->style, widget->window,
			 GTK_STATE_NORMAL, GTK_SHADOW_IN,
			 area, widget, "buttondefault",
			 x, y, width, height);
	}

Elliot Lee's avatar
Elliot Lee committed
704 705
      if (GTK_WIDGET_CAN_DEFAULT (widget))
	{
706 707
	  x += widget->style->xthickness;
	  y += widget->style->ythickness;
708 709 710 711
	  width -= 2 * x + default_spacing;
	  height -= 2 * y + default_spacing;
 	  x += (1 + default_spacing) / 2;
 	  y += (1 + default_spacing) / 2;
Elliot Lee's avatar
Elliot Lee committed
712
	}
713
       
714
      if (!interior_focus && GTK_WIDGET_HAS_FOCUS (widget))
Elliot Lee's avatar
Elliot Lee committed
715 716 717 718 719 720
	{
	  x += 1;
	  y += 1;
	  width -= 2;
	  height -= 2;
	}
721
	
Elliot Lee's avatar
Elliot Lee committed
722 723 724 725 726
      if (GTK_WIDGET_STATE (widget) == GTK_STATE_ACTIVE)
	shadow_type = GTK_SHADOW_IN;
      else
	shadow_type = GTK_SHADOW_OUT;

727 728 729 730 731 732 733 734
      if ((button->relief != GTK_RELIEF_NONE) ||
	  ((GTK_WIDGET_STATE(widget) != GTK_STATE_NORMAL) &&
	   (GTK_WIDGET_STATE(widget) != GTK_STATE_INSENSITIVE)))
	gtk_paint_box (widget->style, widget->window,
		       GTK_WIDGET_STATE (widget),
		       shadow_type, area, widget, "button",
		       x, y, width, height);
       
Elliot Lee's avatar
Elliot Lee committed
735 736
      if (GTK_WIDGET_HAS_FOCUS (widget))
	{
737 738 739 740 741 742 743 744 745 746 747 748 749 750
	  if (interior_focus)
	    {
	      x += widget->style->xthickness + 1;
	      y += widget->style->ythickness + 1;
	      width -= 2 * (widget->style->xthickness + 1);
	      height -=  2 * (widget->style->xthickness + 1);
	    }
	  else
	    {
	      x -= 1;
	      y -= 1;
	      width += 2;
	      height += 2;
	    }
751

752 753 754
	  gtk_paint_focus (widget->style, widget->window,
			   area, widget, "button",
			   x, y, width - 1, height - 1);
Elliot Lee's avatar
Elliot Lee committed
755 756 757 758
	}
    }
}

759
static gboolean
Elliot Lee's avatar
Elliot Lee committed
760 761 762
gtk_button_expose (GtkWidget      *widget,
		   GdkEventExpose *event)
{
763
  GtkBin *bin;
Elliot Lee's avatar
Elliot Lee committed
764 765 766 767 768 769 770

  g_return_val_if_fail (widget != NULL, FALSE);
  g_return_val_if_fail (GTK_IS_BUTTON (widget), FALSE);
  g_return_val_if_fail (event != NULL, FALSE);

  if (GTK_WIDGET_DRAWABLE (widget))
    {
Owen Taylor's avatar
Owen Taylor committed
771
      bin = GTK_BIN (widget);
772
      
Elliot Lee's avatar
Elliot Lee committed
773
      gtk_button_paint (widget, &event->area);
774 775
      
      (* GTK_WIDGET_CLASS (parent_class)->expose_event) (widget, event);
Elliot Lee's avatar
Elliot Lee committed
776
    }
777
  
Elliot Lee's avatar
Elliot Lee committed
778 779 780
  return FALSE;
}

781
static gboolean
Elliot Lee's avatar
Elliot Lee committed
782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798
gtk_button_button_press (GtkWidget      *widget,
			 GdkEventButton *event)
{
  GtkButton *button;

  g_return_val_if_fail (widget != NULL, FALSE);
  g_return_val_if_fail (GTK_IS_BUTTON (widget), FALSE);
  g_return_val_if_fail (event != NULL, FALSE);

  if (event->type == GDK_BUTTON_PRESS)
    {
      button = GTK_BUTTON (widget);

      if (!GTK_WIDGET_HAS_FOCUS (widget))
	gtk_widget_grab_focus (widget);

      if (event->button == 1)
799
	gtk_button_pressed (button);
Elliot Lee's avatar
Elliot Lee committed
800 801 802 803 804
    }

  return TRUE;
}

805
static gboolean
Elliot Lee's avatar
Elliot Lee committed
806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823
gtk_button_button_release (GtkWidget      *widget,
			   GdkEventButton *event)
{
  GtkButton *button;

  g_return_val_if_fail (widget != NULL, FALSE);
  g_return_val_if_fail (GTK_IS_BUTTON (widget), FALSE);
  g_return_val_if_fail (event != NULL, FALSE);

  if (event->button == 1)
    {
      button = GTK_BUTTON (widget);
      gtk_button_released (button);
    }

  return TRUE;
}

824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840
static gboolean
gtk_button_key_release (GtkWidget   *widget,
			GdkEventKey *event)
{
  GtkButton *button = GTK_BUTTON (widget);

  if (button->activate_timeout)
    {
      gtk_button_finish_activate (button, TRUE);
      return TRUE;
    }
  else if (GTK_WIDGET_CLASS (parent_class)->key_release_event)
    return GTK_WIDGET_CLASS (parent_class)->key_release_event (widget, event);
  else
    return FALSE;
}

841
static gboolean
Elliot Lee's avatar
Elliot Lee committed
842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864
gtk_button_enter_notify (GtkWidget        *widget,
			 GdkEventCrossing *event)
{
  GtkButton *button;
  GtkWidget *event_widget;

  g_return_val_if_fail (widget != NULL, FALSE);
  g_return_val_if_fail (GTK_IS_BUTTON (widget), FALSE);
  g_return_val_if_fail (event != NULL, FALSE);

  button = GTK_BUTTON (widget);
  event_widget = gtk_get_event_widget ((GdkEvent*) event);

  if ((event_widget == widget) &&
      (event->detail != GDK_NOTIFY_INFERIOR))
    {
      button->in_button = TRUE;
      gtk_button_enter (button);
    }

  return FALSE;
}

865
static gboolean
Elliot Lee's avatar
Elliot Lee committed
866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895
gtk_button_leave_notify (GtkWidget        *widget,
			 GdkEventCrossing *event)
{
  GtkButton *button;
  GtkWidget *event_widget;

  g_return_val_if_fail (widget != NULL, FALSE);
  g_return_val_if_fail (GTK_IS_BUTTON (widget), FALSE);
  g_return_val_if_fail (event != NULL, FALSE);

  button = GTK_BUTTON (widget);
  event_widget = gtk_get_event_widget ((GdkEvent*) event);

  if ((event_widget == widget) &&
      (event->detail != GDK_NOTIFY_INFERIOR))
    {
      button->in_button = FALSE;
      gtk_button_leave (button);
    }

  return FALSE;
}

static void
gtk_button_add (GtkContainer *container,
		GtkWidget    *widget)
{
  g_return_if_fail (container != NULL);
  g_return_if_fail (widget != NULL);

896 897
  if (GTK_CONTAINER_CLASS (parent_class)->add)
    GTK_CONTAINER_CLASS (parent_class)->add (container, widget);
Elliot Lee's avatar
Elliot Lee committed
898

899
  GTK_BUTTON (container)->child = GTK_BIN (container)->child;
Elliot Lee's avatar
Elliot Lee committed
900 901 902 903 904 905 906
}

static void
gtk_button_remove (GtkContainer *container,
		   GtkWidget    *widget)
{
  g_return_if_fail (container != NULL);
907
  g_return_if_fail (widget != NULL);
Elliot Lee's avatar
Elliot Lee committed
908

909 910
  if (GTK_CONTAINER_CLASS (parent_class)->remove)
    GTK_CONTAINER_CLASS (parent_class)->remove (container, widget);
Elliot Lee's avatar
Elliot Lee committed
911

912
  GTK_BUTTON (container)->child = GTK_BIN (container)->child;
Elliot Lee's avatar
Elliot Lee committed
913 914 915 916 917 918 919 920 921 922
}

static void
gtk_real_button_pressed (GtkButton *button)
{
  GtkStateType new_state;

  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

923 924 925
  if (button->activate_timeout)
    return;
  
Elliot Lee's avatar
Elliot Lee committed
926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948
  button->button_down = TRUE;

  new_state = (button->in_button ? GTK_STATE_ACTIVE : GTK_STATE_NORMAL);

  if (GTK_WIDGET_STATE (button) != new_state)
    {
      gtk_widget_set_state (GTK_WIDGET (button), new_state);
      gtk_widget_queue_draw (GTK_WIDGET (button));
    }
}

static void
gtk_real_button_released (GtkButton *button)
{
  GtkStateType new_state;

  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

  if (button->button_down)
    {
      button->button_down = FALSE;

949 950 951
      if (button->activate_timeout)
	return;
  
Elliot Lee's avatar
Elliot Lee committed
952 953 954 955 956 957 958 959
      if (button->in_button)
	gtk_button_clicked (button);

      new_state = (button->in_button ? GTK_STATE_PRELIGHT : GTK_STATE_NORMAL);

      if (GTK_WIDGET_STATE (button) != new_state)
	{
	  gtk_widget_set_state (GTK_WIDGET (button), new_state);
960 961 962 963
	  /* We _draw () instead of queue_draw so that if the operation
	   * blocks, the label doesn't vanish.
	   */
	  gtk_widget_draw (GTK_WIDGET (button), NULL);
Elliot Lee's avatar
Elliot Lee committed
964 965 966 967 968 969 970 971 972 973 974 975 976 977
	}
    }
}

static void
gtk_real_button_enter (GtkButton *button)
{
  GtkStateType new_state;

  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

  new_state = (button->button_down ? GTK_STATE_ACTIVE : GTK_STATE_PRELIGHT);

978 979 980
  if (button->activate_timeout)
    return;
  
Elliot Lee's avatar
Elliot Lee committed
981 982 983 984 985 986 987 988 989 990 991 992
  if (GTK_WIDGET_STATE (button) != new_state)
    {
      gtk_widget_set_state (GTK_WIDGET (button), new_state);
      gtk_widget_queue_draw (GTK_WIDGET (button));
    }
}

static void
gtk_real_button_leave (GtkButton *button)
{
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));
993 994 995 996
  
  if (button->activate_timeout)
    return;
  
Elliot Lee's avatar
Elliot Lee committed
997 998 999 1000 1001 1002
  if (GTK_WIDGET_STATE (button) != GTK_STATE_NORMAL)
    {
      gtk_widget_set_state (GTK_WIDGET (button), GTK_STATE_NORMAL);
      gtk_widget_queue_draw (GTK_WIDGET (button));
    }
}
1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055

static gboolean
button_activate_timeout (gpointer data)
{
  gtk_button_finish_activate (data, TRUE);

  return FALSE;
}

static void
gtk_real_button_activate (GtkButton *button)
{
  GtkWidget *widget = GTK_WIDGET (button);
  
  g_return_if_fail (button != NULL);
  g_return_if_fail (GTK_IS_BUTTON (button));

  if (GTK_WIDGET_REALIZED (button) && !button->activate_timeout)
    {
      if (gdk_keyboard_grab (widget->window, TRUE,
			     gtk_get_current_event_time ()) == 0)
	{
	  gtk_grab_add (widget);
	  
	  button->activate_timeout = g_timeout_add (ACTIVATE_TIMEOUT,
						    button_activate_timeout,
						    button);
	  button->button_down = TRUE;
	  gtk_widget_set_state (widget, GTK_STATE_ACTIVE);
	}
    }
}

static void
gtk_button_finish_activate (GtkButton *button,
			    gboolean   do_it)
{
  GtkWidget *widget = GTK_WIDGET (button);
  
  g_source_remove (button->activate_timeout);
  button->activate_timeout = 0;

  gdk_keyboard_ungrab (gtk_get_current_event_time ());
  gtk_grab_remove (widget);

  button->button_down = FALSE;
  gtk_widget_set_state (GTK_WIDGET (button),
			button->in_button ? GTK_STATE_PRELIGHT : GTK_STATE_NORMAL);

  if (do_it)
    gtk_button_clicked (button);
}