]> Pileus Git - ~andy/gtk/blob - gtk/gtkcolorchooserwidget.c
Initial cut at implementing a new color chooser
[~andy/gtk] / gtk / gtkcolorchooserwidget.c
1 /* GTK - The GIMP Toolkit
2  *
3  * Copyright (C) 2012 Red Hat, Inc.
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the
17  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18  * Boston, MA 02111-1307, USA.
19  */
20
21 #include "config.h"
22
23 #include "gtkcolorchooserprivate.h"
24 #include "gtkcolorchooserwidget.h"
25 #include "gtkcoloreditor.h"
26 #include "gtkcolorswatch.h"
27 #include "gtkbox.h"
28 #include "gtkgrid.h"
29 #include "gtkhsv.h"
30 #include "gtklabel.h"
31 #include "gtkorientable.h"
32 #include "gtkintl.h"
33
34 struct _GtkColorChooserWidgetPrivate
35 {
36   GtkWidget *palette;
37   GtkWidget *editor;
38
39   GtkWidget *colors;
40   GtkWidget *grays;
41   GtkWidget *custom;
42
43   GtkColorSwatch *current;
44
45   GSettings *settings;
46 };
47
48 enum
49 {
50   PROP_ZERO,
51   PROP_COLOR
52 };
53
54 static void gtk_color_chooser_widget_iface_init (GtkColorChooserInterface *iface);
55
56 G_DEFINE_TYPE_WITH_CODE (GtkColorChooserWidget, gtk_color_chooser_widget, GTK_TYPE_BOX,
57                          G_IMPLEMENT_INTERFACE (GTK_TYPE_COLOR_CHOOSER,
58                                                 gtk_color_chooser_widget_iface_init))
59
60 static void
61 select_swatch (GtkColorChooserWidget *cc,
62                GtkColorSwatch        *swatch)
63 {
64   if (cc->priv->current == swatch)
65     return;
66   if (cc->priv->current != NULL)
67     gtk_color_swatch_set_selected (cc->priv->current, FALSE);
68   gtk_color_swatch_set_selected (swatch, TRUE);
69   cc->priv->current = swatch;
70   g_object_notify (G_OBJECT (cc), "color");
71 }
72
73 static void add_custom (GtkColorChooserWidget *cc, GtkColorSwatch *swatch);
74 static void save_custom (GtkColorChooserWidget *cc);
75
76 static void
77 button_activate (GtkColorSwatch        *swatch,
78                  GtkColorChooserWidget *cc)
79 {
80   add_custom (cc, swatch);
81 }
82
83 static void
84 swatch_activate (GtkColorSwatch        *swatch,
85                  GtkColorChooserWidget *cc)
86 {
87   GdkRGBA color;
88
89   gtk_color_swatch_get_color (swatch, &color);
90   _gtk_color_chooser_color_activated (GTK_COLOR_CHOOSER (cc), &color);
91 }
92
93 static void
94 swatch_customize (GtkColorSwatch        *swatch,
95                   GtkColorChooserWidget *cc)
96 {
97   GdkRGBA color;
98
99   gtk_color_swatch_get_color (swatch, &color);
100   gtk_color_chooser_set_color (GTK_COLOR_CHOOSER (cc->priv->editor), &color);
101
102   gtk_widget_hide (cc->priv->palette);
103   gtk_widget_show (cc->priv->editor);
104   g_print ("TODO: customize color\n");
105 }
106
107 static void
108 swatch_selected (GtkColorSwatch        *swatch,
109                  GParamSpec            *pspec,
110                  GtkColorChooserWidget *cc)
111 {
112   select_swatch (cc, swatch);
113 }
114
115 static void
116 connect_swatch_signals (GtkWidget *p, gpointer data)
117 {
118   g_signal_connect (p, "activate", G_CALLBACK (swatch_activate), data);
119   g_signal_connect (p, "customize", G_CALLBACK (swatch_customize), data);
120   g_signal_connect (p, "notify::selected", G_CALLBACK (swatch_selected), data);
121 }
122
123 static void
124 connect_button_signals (GtkWidget *p, gpointer data)
125 {
126   g_signal_connect (p, "activate", G_CALLBACK (button_activate), data);
127 }
128
129 static void
130 connect_custom_signals (GtkWidget *p, gpointer data)
131 {
132   connect_swatch_signals (p, data);
133   g_signal_connect_swapped (p, "notify::color", G_CALLBACK (save_custom), data);
134 }
135
136 static void
137 save_custom (GtkColorChooserWidget *cc)
138 {
139   GVariantBuilder builder;
140   GVariant *variant;
141   GdkRGBA color;
142   GList *children, *l;
143
144   g_variant_builder_init (&builder, G_VARIANT_TYPE ("a(dddd)"));
145
146   children = gtk_container_get_children (GTK_CONTAINER (cc->priv->custom));
147   for (l = children; l; l = l->next)
148     {
149       if (gtk_color_swatch_get_color (GTK_COLOR_SWATCH (l->data), &color))
150         {
151           g_variant_builder_add (&builder, "(dddd)",
152                                  color.red, color.green, color.blue, color.alpha);
153         }
154     }
155
156   g_list_free (children);
157
158   variant = g_variant_builder_end (&builder);
159   g_settings_set_value (cc->priv->settings, "custom-colors", variant);
160 }
161
162 static void
163 add_custom (GtkColorChooserWidget *cc, GtkColorSwatch *swatch)
164 {
165   GList *children;
166   gint n_children;
167   GdkRGBA color;
168   GtkWidget *p;
169
170   children = gtk_container_get_children (GTK_CONTAINER (cc->priv->custom));
171   n_children = g_list_length (children);
172   g_list_free (children);
173   if (n_children >= 9)
174     {
175       gtk_widget_error_bell (GTK_WIDGET (cc));
176       return;
177     }
178
179   gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (swatch), 10, 1, 1, 10);
180
181   color.red = g_random_double_range (0, 1);
182   color.green = g_random_double_range (0, 1);
183   color.blue = g_random_double_range (0, 1);
184   color.alpha = 1.0;
185
186   p = gtk_color_swatch_new ();
187   gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color);
188   gtk_color_swatch_set_can_drop (GTK_COLOR_SWATCH (p), TRUE);
189   connect_custom_signals (p, cc);
190
191   if (n_children == 1)
192     gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 10, 10, 1);
193   else
194     gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 1, 1, 1);
195
196   gtk_grid_insert_next_to (GTK_GRID (cc->priv->custom), GTK_WIDGET (swatch), GTK_POS_RIGHT);
197   gtk_grid_attach (GTK_GRID (cc->priv->custom), p, 1, 0, 1, 1);
198
199   gtk_widget_show (p);
200
201   save_custom (cc);
202 }
203
204 static void
205 gtk_color_chooser_widget_init (GtkColorChooserWidget *cc)
206 {
207   GtkWidget *grid;
208   GtkWidget *p;
209   GtkWidget *label;
210   gint i;
211   GdkRGBA color, color1, color2;
212   gdouble h, s, v;
213   GVariant *variant;
214   GVariantIter iter;
215   gboolean selected;
216   const gchar *default_palette[9] = {
217     "red",
218     "orange",
219     "yellow",
220     "green",
221     "blue",
222     "purple",
223     "brown",
224     "darkgray",
225     "gray"
226   };
227
228   cc->priv = G_TYPE_INSTANCE_GET_PRIVATE (cc, GTK_TYPE_COLOR_CHOOSER_WIDGET, GtkColorChooserWidgetPrivate);
229
230   gtk_orientable_set_orientation (GTK_ORIENTABLE (cc), GTK_ORIENTATION_VERTICAL);
231   cc->priv->palette = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
232   gtk_container_add (GTK_CONTAINER (cc), cc->priv->palette);
233
234   cc->priv->colors = grid = gtk_grid_new ();
235   g_object_set (grid, "margin", 12, NULL);
236   gtk_widget_set_margin_bottom (grid, 12);
237   gtk_grid_set_row_spacing (GTK_GRID (grid), 2);
238   gtk_grid_set_column_spacing (GTK_GRID (grid), 4);
239   gtk_container_add (GTK_CONTAINER (cc->priv->palette), grid);
240
241   for (i = 0; i < 9; i++)
242     {
243        gdk_rgba_parse (&color, default_palette[i]);
244        gtk_rgb_to_hsv (color.red, color.green, color.blue, &h, &s, &v);
245        gtk_hsv_to_rgb (h, s / 2, (v + 1) / 2, &color1.red, &color1.green, &color1.blue);
246        color1.alpha = color.alpha;
247
248        gtk_hsv_to_rgb (h, s, v * 3 / 4, &color2.red, &color2.green, &color2.blue);
249        color2.alpha = color.alpha;
250
251        p = gtk_color_swatch_new ();
252        connect_swatch_signals (p, cc);
253        gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 10, 10, 1, 1);
254        gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color1);
255
256        gtk_grid_attach (GTK_GRID (grid), p, i, 0, 1, 1);
257
258        p = gtk_color_swatch_new ();
259        connect_swatch_signals (p, cc);
260        gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 1, 1, 1);
261        gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color);
262        gtk_grid_attach (GTK_GRID (grid), p, i, 1, 1, 1);
263
264        p = gtk_color_swatch_new ();
265        connect_swatch_signals (p, cc);
266        gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 1, 10, 10);
267        gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color2);
268        gtk_grid_attach (GTK_GRID (grid), p, i, 2, 1, 1);
269     }
270
271   cc->priv->grays = grid = gtk_grid_new ();
272   g_object_set (grid, "margin", 12, "margin-top", 0, NULL);
273   gtk_grid_set_column_spacing (GTK_GRID (grid), 4);
274   gtk_container_add (GTK_CONTAINER (cc->priv->palette), grid);
275
276   for (i = 0; i < 9; i++)
277     {
278        color.red = color.green = color.blue = i / 8.0;
279        color.alpha = 1.0;
280
281        p = gtk_color_swatch_new ();
282        connect_swatch_signals (p, cc);
283        if (i == 0)
284          gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 10, 1, 1, 10);
285        else if (i == 8)
286          gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 10, 10, 1);
287        else
288          gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 1, 1, 1);
289
290        gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color);
291        gtk_grid_attach (GTK_GRID (grid), p, i, 0, 1, 1);
292     }
293
294   label = gtk_label_new (_("Custom color"));
295   gtk_widget_set_halign (label, GTK_ALIGN_START);
296   gtk_container_add (GTK_CONTAINER (cc->priv->palette), label);
297   g_object_set (grid, "margin", 12, "margin-top", 0, NULL);
298
299   cc->priv->custom = grid = gtk_grid_new ();
300   g_object_set (grid, "margin", 12, "margin-top", 0, NULL);
301   gtk_grid_set_column_spacing (GTK_GRID (grid), 4);
302   gtk_container_add (GTK_CONTAINER (cc->priv->palette), grid);
303
304   p = gtk_color_swatch_new ();
305   gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 10, 10, 10, 10);
306   connect_button_signals (p, cc);
307   gtk_color_swatch_set_icon (GTK_COLOR_SWATCH (p), "list-add-symbolic");
308   gtk_grid_attach (GTK_GRID (grid), p, 0, 0, 1, 1);
309
310   cc->priv->settings = g_settings_new_with_path ("org.gtk.Settings.ColorChooser",
311                                                  "/org/gtk/settings/color-chooser/");
312   variant = g_settings_get_value (cc->priv->settings, "custom-colors");
313   g_variant_iter_init (&iter, variant);
314   i = 0;
315   while (g_variant_iter_loop (&iter, "(dddd)", &color.red, &color.green, &color.blue, &color.alpha))
316     {
317       i++;
318       p = gtk_color_swatch_new ();
319       gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 1, 1, 1);
320       gtk_color_swatch_set_color (GTK_COLOR_SWATCH (p), &color);
321       gtk_color_swatch_set_can_drop (GTK_COLOR_SWATCH (p), TRUE);
322       connect_custom_signals (p, cc);
323       gtk_grid_attach (GTK_GRID (grid), p, i, 0, 1, 1);
324
325       if (i == 8)
326         break;
327     }
328   if (i > 0)
329     gtk_color_swatch_set_corner_radii (GTK_COLOR_SWATCH (p), 1, 10, 10, 1);
330   g_variant_unref (variant);
331
332   cc->priv->editor = gtk_color_editor_new ();
333   gtk_container_add (GTK_CONTAINER (cc), cc->priv->editor);
334
335   g_settings_get (cc->priv->settings, "selected-color", "(bdddd)",
336                   &selected,
337                   &color.red, &color.green, &color.blue, &color.alpha);
338   if (selected)
339     gtk_color_chooser_set_color (GTK_COLOR_CHOOSER (cc), &color);
340
341   gtk_widget_show_all (GTK_WIDGET (cc));
342   gtk_widget_hide (GTK_WIDGET (cc->priv->editor));
343   gtk_widget_hide (GTK_WIDGET (cc));
344
345   gtk_widget_set_no_show_all (cc->priv->palette, TRUE);
346   gtk_widget_set_no_show_all (cc->priv->editor, TRUE);
347 }
348
349 static void
350 gtk_color_chooser_widget_get_property (GObject    *object,
351                                        guint       prop_id,
352                                        GValue     *value,
353                                        GParamSpec *pspec)
354 {
355   GtkColorChooser *cc = GTK_COLOR_CHOOSER (object);
356
357   switch (prop_id)
358     {
359     case PROP_COLOR:
360       {
361         GdkRGBA color;
362
363         gtk_color_chooser_get_color (cc, &color);
364         g_value_set_boxed (value, &color);
365       }
366     break;
367     default:
368       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
369       break;
370     }
371 }
372
373 static void
374 gtk_color_chooser_widget_set_property (GObject      *object,
375                                        guint         prop_id,
376                                        const GValue *value,
377                                        GParamSpec   *pspec)
378 {
379   GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (object);
380
381   switch (prop_id)
382     {
383     case PROP_COLOR:
384       gtk_color_chooser_set_color (GTK_COLOR_CHOOSER (cc),
385                                    g_value_get_boxed (value));
386     break;
387     default:
388       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
389       break;
390     }
391 }
392
393 static void
394 gtk_color_chooser_widget_finalize (GObject *object)
395 {
396   GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (object);
397
398   g_object_unref (cc->priv->settings);
399
400   G_OBJECT_CLASS (gtk_color_chooser_widget_parent_class)->finalize (object);
401 }
402
403 static void
404 gtk_color_chooser_widget_class_init (GtkColorChooserWidgetClass *class)
405 {
406   GObjectClass *object_class = G_OBJECT_CLASS (class);
407
408   object_class->get_property = gtk_color_chooser_widget_get_property;
409   object_class->set_property = gtk_color_chooser_widget_set_property;
410   object_class->finalize = gtk_color_chooser_widget_finalize;
411
412   g_object_class_override_property (object_class, PROP_COLOR, "color");
413
414   g_type_class_add_private (object_class, sizeof (GtkColorChooserWidgetPrivate));
415 }
416
417 static void
418 gtk_color_chooser_widget_get_color (GtkColorChooser *chooser,
419                                     GdkRGBA         *color)
420 {
421   GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (chooser);
422
423   if (gtk_widget_get_visible (cc->priv->editor))
424     gtk_color_chooser_get_color (GTK_COLOR_CHOOSER (cc->priv->editor), color);
425   else if (cc->priv->current)
426     gtk_color_swatch_get_color (cc->priv->current, color);
427   else
428     {
429       color->red = 1.0;
430       color->green = 1.0;
431       color->blue = 1.0;
432       color->alpha = 1.0;
433     }
434 }
435
436 static void
437 gtk_color_chooser_widget_set_color (GtkColorChooser *chooser,
438                                     const GdkRGBA   *color)
439 {
440   GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (chooser);
441   GList *children, *l;
442   GtkColorSwatch *swatch;
443   GdkRGBA c;
444   GtkWidget *grids[3];
445   gint i;
446
447   grids[0] = cc->priv->colors;
448   grids[1] = cc->priv->grays;
449   grids[2] = cc->priv->custom;
450
451   for (i = 0; i < 3; i++)
452     {
453       children = gtk_container_get_children (GTK_CONTAINER (grids[i]));
454       for (l = children; l; l = l->next)
455         {
456           swatch = l->data;
457           gtk_color_swatch_get_color (swatch, &c);
458           if (gdk_rgba_equal (color, &c))
459             {
460               select_swatch (cc, swatch);
461               g_list_free (children);
462               return;
463             }
464         }
465       g_list_free (children);
466     }
467
468   /* FIXME: add new custom color */
469 }
470
471 static void
472 gtk_color_chooser_widget_iface_init (GtkColorChooserInterface *iface)
473 {
474   iface->get_color = gtk_color_chooser_widget_get_color;
475   iface->set_color = gtk_color_chooser_widget_set_color;
476 }
477
478 GtkWidget *
479 gtk_color_chooser_widget_new (void)
480 {
481   return g_object_new (GTK_TYPE_COLOR_CHOOSER_WIDGET, NULL);
482 }