1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 2012 Red Hat, Inc.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser 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.
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 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser 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.
22 #include "gtkcolorswatch.h"
24 #include "gtkroundedboxprivate.h"
25 #include "gtkthemingbackgroundprivate.h"
27 #include "gtkicontheme.h"
30 #include "gtkmenuitem.h"
31 #include "gtkmenushell.h"
32 #include "gtkprivate.h"
36 struct _GtkColorSwatchPrivate
44 guint contains_pointer : 1;
62 static guint signals[LAST_SIGNAL];
64 G_DEFINE_TYPE (GtkColorSwatch, gtk_color_swatch, GTK_TYPE_DRAWING_AREA)
67 gtk_color_swatch_init (GtkColorSwatch *swatch)
69 swatch->priv = G_TYPE_INSTANCE_GET_PRIVATE (swatch,
70 GTK_TYPE_COLOR_SWATCH,
71 GtkColorSwatchPrivate);
73 gtk_widget_set_can_focus (GTK_WIDGET (swatch), TRUE);
74 gtk_widget_set_events (GTK_WIDGET (swatch), GDK_BUTTON_PRESS_MASK
75 | GDK_BUTTON_RELEASE_MASK
77 | GDK_ENTER_NOTIFY_MASK
78 | GDK_LEAVE_NOTIFY_MASK);
79 swatch->priv->use_alpha = TRUE;
82 #define INTENSITY(r, g, b) ((r) * 0.30 + (g) * 0.59 + (b) * 0.11)
84 static cairo_pattern_t *
85 get_checkered_pattern (void)
87 /* need to respect pixman's stride being a multiple of 4 */
88 static unsigned char data[8] = { 0xFF, 0x00, 0x00, 0x00,
89 0x00, 0xFF, 0x00, 0x00 };
90 static cairo_surface_t *checkered = NULL;
91 cairo_pattern_t *pattern;
93 if (checkered == NULL)
94 checkered = cairo_image_surface_create_for_data (data,
98 pattern = cairo_pattern_create_for_surface (checkered);
99 cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
100 cairo_pattern_set_filter (pattern, CAIRO_FILTER_NEAREST);
106 swatch_draw (GtkWidget *widget,
109 GtkColorSwatch *swatch = (GtkColorSwatch*)widget;
110 GtkThemingBackground background;
113 gdouble width, height;
114 GtkStyleContext *context;
117 GtkIconInfo *icon_info = NULL;
119 theme = gtk_icon_theme_get_default ();
120 context = gtk_widget_get_style_context (widget);
121 state = gtk_widget_get_state_flags (widget);
122 width = gtk_widget_get_allocated_width (widget);
123 height = gtk_widget_get_allocated_height (widget);
127 gtk_style_context_save (context);
128 gtk_style_context_set_state (context, state);
130 _gtk_theming_background_init_from_context (&background, context,
134 if (swatch->priv->has_color)
136 cairo_pattern_t *pattern;
137 cairo_matrix_t matrix;
139 if (swatch->priv->use_alpha)
143 _gtk_rounded_box_path (&background.clip_box, cr);
144 cairo_clip_preserve (cr);
146 cairo_set_source_rgb (cr, 0.33, 0.33, 0.33);
147 cairo_fill_preserve (cr);
149 pattern = get_checkered_pattern ();
150 cairo_matrix_init_scale (&matrix, 0.125, 0.125);
151 cairo_pattern_set_matrix (pattern, &matrix);
153 cairo_set_source_rgb (cr, 0.66, 0.66, 0.66);
154 cairo_mask (cr, pattern);
155 cairo_pattern_destroy (pattern);
159 background.bg_color = swatch->priv->color;
163 background.bg_color = swatch->priv->color;
164 background.bg_color.alpha = 1.0;
167 _gtk_theming_background_render (&background, cr);
170 gtk_render_frame (context, cr,
171 0, 0, width, height);
173 if (gtk_widget_has_visible_focus (widget))
175 cairo_set_line_width (cr, 2);
176 if (swatch->priv->has_color && INTENSITY (swatch->priv->color.red, swatch->priv->color.green, swatch->priv->color.blue) < 0.5)
177 cairo_set_source_rgba (cr, 1., 1., 1., 0.4);
179 cairo_set_source_rgba (cr, 0., 0., 0., 0.4);
180 _gtk_rounded_box_shrink (&background.clip_box, 3, 3, 3, 3);
181 _gtk_rounded_box_path (&background.clip_box, cr);
185 if (swatch->priv->icon)
187 icon_info = gtk_icon_theme_lookup_icon (theme, swatch->priv->icon, 16,
188 GTK_ICON_LOOKUP_GENERIC_FALLBACK
189 | GTK_ICON_LOOKUP_USE_BUILTIN);
191 else if (swatch->priv->selected)
194 GtkBorder border_width;
197 gtk_style_context_add_class (context, "color-active-badge");
198 gtk_style_context_get_background_color (context, state, &bg);
199 gtk_style_context_get_border_color (context, state, &border);
200 gtk_style_context_get_border (context, state, &border_width);
202 cairo_new_sub_path (cr);
203 cairo_arc (cr, width / 2, height / 2, 10, 0, 2 * G_PI);
204 cairo_close_path (cr);
205 gdk_cairo_set_source_rgba (cr, &bg);
206 cairo_fill_preserve (cr);
208 gdk_cairo_set_source_rgba (cr, &border);
209 cairo_set_line_width (cr, border_width.left);
212 gicon = g_themed_icon_new ("object-select-symbolic");
213 /* fallback for themes that don't have object-select-symbolic */
214 g_themed_icon_append_name (G_THEMED_ICON (gicon), "gtk-apply");
216 icon_info = gtk_icon_theme_lookup_by_gicon (theme, gicon, 16,
217 GTK_ICON_LOOKUP_GENERIC_FALLBACK
218 | GTK_ICON_LOOKUP_USE_BUILTIN);
219 g_object_unref (gicon);
222 if (icon_info != NULL)
226 pixbuf = gtk_icon_info_load_symbolic_for_context (icon_info, context,
231 gtk_render_icon (context, cr, pixbuf,
232 (width - gdk_pixbuf_get_width (pixbuf)) / 2,
233 (height - gdk_pixbuf_get_height (pixbuf)) / 2);
234 g_object_unref (pixbuf);
237 gtk_icon_info_free (icon_info);
241 gtk_style_context_restore (context);
247 drag_set_color_icon (GdkDragContext *context,
248 const GdkRGBA *color)
250 cairo_surface_t *surface;
253 surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, 48, 32);
254 cr = cairo_create (surface);
255 gdk_cairo_set_source_rgba (cr, color);
258 cairo_surface_set_device_offset (surface, -4, -4);
259 gtk_drag_set_icon_surface (context, surface);
262 cairo_surface_destroy (surface);
266 swatch_drag_begin (GtkWidget *widget,
267 GdkDragContext *context)
269 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
272 gtk_color_swatch_get_rgba (swatch, &color);
273 drag_set_color_icon (context, &color);
277 swatch_drag_data_get (GtkWidget *widget,
278 GdkDragContext *context,
279 GtkSelectionData *selection_data,
283 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
287 gtk_color_swatch_get_rgba (swatch, &color);
289 vals[0] = color.red * 0xffff;
290 vals[1] = color.green * 0xffff;
291 vals[2] = color.blue * 0xffff;
292 vals[3] = color.alpha * 0xffff;
294 gtk_selection_data_set (selection_data,
295 gdk_atom_intern_static_string ("application/x-color"),
296 16, (guchar *)vals, 8);
300 swatch_drag_data_received (GtkWidget *widget,
301 GdkDragContext *context,
304 GtkSelectionData *selection_data,
312 length = gtk_selection_data_get_length (selection_data);
317 /* We accept drops with the wrong format, since the KDE color
318 * chooser incorrectly drops application/x-color with format 8.
322 g_warning ("Received invalid color data\n");
326 vals = (guint16 *) gtk_selection_data_get_data (selection_data);
328 color.red = (gdouble)vals[0] / 0xffff;
329 color.green = (gdouble)vals[1] / 0xffff;
330 color.blue = (gdouble)vals[2] / 0xffff;
331 color.alpha = (gdouble)vals[3] / 0xffff;
333 gtk_color_swatch_set_rgba (GTK_COLOR_SWATCH (widget), &color);
337 swatch_get_preferred_width (GtkWidget *widget,
345 swatch_get_preferred_height (GtkWidget *widget,
353 swatch_key_press (GtkWidget *widget,
356 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
358 if (event->keyval == GDK_KEY_space ||
359 event->keyval == GDK_KEY_Return ||
360 event->keyval == GDK_KEY_ISO_Enter||
361 event->keyval == GDK_KEY_KP_Enter ||
362 event->keyval == GDK_KEY_KP_Space)
364 if (swatch->priv->has_color && !swatch->priv->selected)
365 gtk_color_swatch_set_selected (swatch, TRUE);
367 g_signal_emit (swatch, signals[ACTIVATE], 0);
371 if (GTK_WIDGET_CLASS (gtk_color_swatch_parent_class)->key_press_event (widget, event))
378 swatch_enter_notify (GtkWidget *widget,
379 GdkEventCrossing *event)
381 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
382 swatch->priv->contains_pointer = TRUE;
387 swatch_leave_notify (GtkWidget *widget,
388 GdkEventCrossing *event)
390 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
391 swatch->priv->contains_pointer = FALSE;
396 emit_customize (GtkColorSwatch *swatch)
398 g_signal_emit (swatch, signals[CUSTOMIZE], 0);
402 popup_position_func (GtkMenu *menu,
413 GdkRectangle monitor;
416 widget = GTK_WIDGET (user_data);
417 g_return_if_fail (gtk_widget_get_realized (widget));
418 window = gtk_widget_get_window (widget);
420 screen = gtk_widget_get_screen (widget);
421 monitor_num = gdk_screen_get_monitor_at_window (screen, window);
424 gtk_menu_set_monitor (menu, monitor_num);
426 gdk_window_get_origin (window, &root_x, &root_y);
427 gtk_widget_get_preferred_size (GTK_WIDGET (menu), &req, NULL);
429 /* Put corner of menu centered on swatch */
430 *x = root_x + gtk_widget_get_allocated_width (widget) / 2;
431 *y = root_y + gtk_widget_get_allocated_height (widget) / 2;
434 gdk_screen_get_monitor_workarea (screen, monitor_num, &monitor);
435 *x = CLAMP (*x, monitor.x, MAX (monitor.x, monitor.width - req.width));
436 *y = CLAMP (*y, monitor.y, MAX (monitor.y, monitor.height - req.height));
440 do_popup (GtkWidget *swatch,
441 GdkEventButton *event)
446 menu = gtk_menu_new ();
447 item = gtk_menu_item_new_with_mnemonic (_("_Customize"));
448 gtk_menu_attach_to_widget (GTK_MENU (menu), swatch, NULL);
450 g_signal_connect_swapped (item, "activate",
451 G_CALLBACK (emit_customize), swatch);
453 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
455 gtk_widget_show_all (item);
458 gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
459 NULL, NULL, event->button, event->time);
461 gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
462 popup_position_func, swatch,
463 0, gtk_get_current_event_time ());
467 swatch_button_press (GtkWidget *widget,
468 GdkEventButton *event)
470 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
472 gtk_widget_grab_focus (widget);
474 if (gdk_event_triggers_context_menu ((GdkEvent *) event) &&
475 swatch->priv->has_color)
477 do_popup (widget, event);
480 else if (event->type == GDK_2BUTTON_PRESS &&
481 event->button == GDK_BUTTON_PRIMARY)
483 g_signal_emit (swatch, signals[ACTIVATE], 0);
491 swatch_button_release (GtkWidget *widget,
492 GdkEventButton *event)
494 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (widget);
496 if (event->button == GDK_BUTTON_PRIMARY &&
497 swatch->priv->contains_pointer)
499 if (!swatch->priv->has_color)
501 g_signal_emit (swatch, signals[ACTIVATE], 0);
504 else if (!swatch->priv->selected)
506 gtk_color_swatch_set_selected (swatch, TRUE);
515 swatch_popup_menu (GtkWidget *swatch)
517 do_popup (swatch, NULL);
521 /* GObject implementation {{{1 */
524 swatch_get_property (GObject *object,
529 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (object);
535 gtk_color_swatch_get_rgba (swatch, &color);
536 g_value_set_boxed (value, &color);
539 g_value_set_boolean (value, swatch->priv->selected);
542 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
548 swatch_set_property (GObject *object,
553 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (object);
558 gtk_color_swatch_set_rgba (swatch, g_value_get_boxed (value));
561 gtk_color_swatch_set_selected (swatch, g_value_get_boolean (value));
564 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
570 swatch_finalize (GObject *object)
572 GtkColorSwatch *swatch = GTK_COLOR_SWATCH (object);
574 g_free (swatch->priv->icon);
576 G_OBJECT_CLASS (gtk_color_swatch_parent_class)->finalize (object);
580 gtk_color_swatch_class_init (GtkColorSwatchClass *class)
582 GtkWidgetClass *widget_class = (GtkWidgetClass *)class;
583 GObjectClass *object_class = (GObjectClass *)class;
585 object_class->get_property = swatch_get_property;
586 object_class->set_property = swatch_set_property;
587 object_class->finalize = swatch_finalize;
589 widget_class->get_preferred_width = swatch_get_preferred_width;
590 widget_class->get_preferred_height = swatch_get_preferred_height;
591 widget_class->draw = swatch_draw;
592 widget_class->drag_begin = swatch_drag_begin;
593 widget_class->drag_data_get = swatch_drag_data_get;
594 widget_class->drag_data_received = swatch_drag_data_received;
595 widget_class->key_press_event = swatch_key_press;
596 widget_class->popup_menu = swatch_popup_menu;
597 widget_class->button_press_event = swatch_button_press;
598 widget_class->button_release_event = swatch_button_release;
599 widget_class->enter_notify_event = swatch_enter_notify;
600 widget_class->leave_notify_event = swatch_leave_notify;
603 g_signal_new ("activate",
604 GTK_TYPE_COLOR_SWATCH,
606 G_STRUCT_OFFSET (GtkColorSwatchClass, activate),
607 NULL, NULL, NULL, G_TYPE_NONE, 0);
610 g_signal_new ("customize",
611 GTK_TYPE_COLOR_SWATCH,
613 G_STRUCT_OFFSET (GtkColorSwatchClass, customize),
614 NULL, NULL, NULL, G_TYPE_NONE, 0);
616 g_object_class_install_property (object_class, PROP_RGBA,
617 g_param_spec_boxed ("rgba", P_("RGBA Color"), P_("Color as RGBA"),
618 GDK_TYPE_RGBA, GTK_PARAM_READWRITE));
620 g_object_class_install_property (object_class, PROP_SELECTED,
621 g_param_spec_boolean ("selected", P_("Selected"), P_("Selected"),
622 FALSE, GTK_PARAM_READWRITE));
624 g_type_class_add_private (object_class, sizeof (GtkColorSwatchPrivate));
627 /* Public API {{{1 */
630 gtk_color_swatch_new (void)
632 return (GtkWidget *) g_object_new (GTK_TYPE_COLOR_SWATCH, NULL);
635 static const GtkTargetEntry dnd_targets[] = {
636 { "application/x-color", 0 }
640 gtk_color_swatch_set_rgba (GtkColorSwatch *swatch,
641 const GdkRGBA *color)
643 GtkStyleContext *context;
645 context = gtk_widget_get_style_context (GTK_WIDGET (swatch));
647 if (!swatch->priv->has_color)
649 gtk_drag_source_set (GTK_WIDGET (swatch),
650 GDK_BUTTON1_MASK | GDK_BUTTON3_MASK,
651 dnd_targets, G_N_ELEMENTS (dnd_targets),
652 GDK_ACTION_COPY | GDK_ACTION_MOVE);
656 gtk_style_context_remove_class (context, "color-light");
657 gtk_style_context_remove_class (context, "color-dark");
660 swatch->priv->has_color = TRUE;
661 swatch->priv->color = *color;
663 if (INTENSITY (swatch->priv->color.red, swatch->priv->color.green, swatch->priv->color.blue) > 0.5)
664 gtk_style_context_add_class (context, "color-light");
666 gtk_style_context_add_class (context, "color-dark");
668 gtk_widget_queue_draw (GTK_WIDGET (swatch));
669 g_object_notify (G_OBJECT (swatch), "rgba");
673 gtk_color_swatch_get_rgba (GtkColorSwatch *swatch,
676 if (swatch->priv->has_color)
678 color->red = swatch->priv->color.red;
679 color->green = swatch->priv->color.green;
680 color->blue = swatch->priv->color.blue;
681 color->alpha = swatch->priv->color.alpha;
695 gtk_color_swatch_set_corner_radii (GtkColorSwatch *swatch,
698 gdouble bottom_right,
701 swatch->priv->radius[0] = top_left;
702 swatch->priv->radius[1] = top_right;
703 swatch->priv->radius[2] = bottom_right;
704 swatch->priv->radius[3] = bottom_left;
706 gtk_widget_queue_draw (GTK_WIDGET (swatch));
710 gtk_color_swatch_set_selected (GtkColorSwatch *swatch,
713 if (swatch->priv->selected != selected)
715 swatch->priv->selected = selected;
716 gtk_widget_queue_draw (GTK_WIDGET (swatch));
717 g_object_notify (G_OBJECT (swatch), "selected");
722 gtk_color_swatch_set_icon (GtkColorSwatch *swatch,
725 swatch->priv->icon = g_strdup (icon);
726 gtk_widget_queue_draw (GTK_WIDGET (swatch));
730 gtk_color_swatch_set_can_drop (GtkColorSwatch *swatch,
733 if (!swatch->priv->can_drop)
734 gtk_drag_dest_set (GTK_WIDGET (swatch),
735 GTK_DEST_DEFAULT_HIGHLIGHT |
736 GTK_DEST_DEFAULT_MOTION |
737 GTK_DEST_DEFAULT_DROP,
738 dnd_targets, G_N_ELEMENTS (dnd_targets),
741 swatch->priv->can_drop = can_drop;
745 gtk_color_swatch_set_use_alpha (GtkColorSwatch *swatch,
748 swatch->priv->use_alpha = use_alpha;
749 gtk_widget_queue_draw (GTK_WIDGET (swatch));
752 /* vim:set foldmethod=marker: */