forked from AuroraMiddleware/gtk
890 lines
27 KiB
C
890 lines
27 KiB
C
/* GTK - The GIMP Toolkit
|
|
*
|
|
* Copyright (C) 2012 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkcolorchooserprivate.h"
|
|
#include "gtkcolorchooserwidget.h"
|
|
#include "gtkcoloreditorprivate.h"
|
|
#include "gtkcolorswatchprivate.h"
|
|
#include "gtkgrid.h"
|
|
#include "gtklabel.h"
|
|
#include "gtkorientable.h"
|
|
#include "gtkprivate.h"
|
|
#include "gtkintl.h"
|
|
#include "gtksizegroup.h"
|
|
#include "gtkstylecontext.h"
|
|
#include "gtkboxlayout.h"
|
|
|
|
#include "a11y/gtkcompositeaccessible.h"
|
|
|
|
#include <math.h>
|
|
|
|
/**
|
|
* SECTION:gtkcolorchooserwidget
|
|
* @Short_description: A widget for choosing colors
|
|
* @Title: GtkColorChooserWidget
|
|
* @See_also: #GtkColorChooserDialog
|
|
*
|
|
* The #GtkColorChooserWidget widget lets the user select a
|
|
* color. By default, the chooser presents a predefined palette
|
|
* of colors, plus a small number of settable custom colors.
|
|
* It is also possible to select a different color with the
|
|
* single-color editor. To enter the single-color editing mode,
|
|
* use the context menu of any color of the palette, or use the
|
|
* '+' button to add a new custom color.
|
|
*
|
|
* The chooser automatically remembers the last selection, as well
|
|
* as custom colors.
|
|
*
|
|
* To change the initially selected color, use gtk_color_chooser_set_rgba().
|
|
* To get the selected color use gtk_color_chooser_get_rgba().
|
|
*
|
|
* The #GtkColorChooserWidget is used in the #GtkColorChooserDialog
|
|
* to provide a dialog for selecting colors.
|
|
*
|
|
* # CSS names
|
|
*
|
|
* GtkColorChooserWidget has a single CSS node with name colorchooser.
|
|
*/
|
|
|
|
typedef struct _GtkColorChooserWidgetPrivate GtkColorChooserWidgetPrivate;
|
|
typedef struct _GtkColorChooserWidgetClass GtkColorChooserWidgetClass;
|
|
|
|
struct _GtkColorChooserWidget
|
|
{
|
|
GtkWidget parent_instance;
|
|
};
|
|
|
|
struct _GtkColorChooserWidgetClass
|
|
{
|
|
GtkWidgetClass parent_class;
|
|
};
|
|
|
|
struct _GtkColorChooserWidgetPrivate
|
|
{
|
|
GtkWidget *palette;
|
|
GtkWidget *editor;
|
|
GtkSizeGroup *size_group;
|
|
|
|
GtkWidget *custom_label;
|
|
GtkWidget *custom;
|
|
|
|
GtkWidget *button;
|
|
GtkColorSwatch *current;
|
|
|
|
gboolean use_alpha;
|
|
gboolean has_default_palette;
|
|
|
|
GSettings *settings;
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_ZERO,
|
|
PROP_RGBA,
|
|
PROP_USE_ALPHA,
|
|
PROP_SHOW_EDITOR
|
|
};
|
|
|
|
static void gtk_color_chooser_widget_iface_init (GtkColorChooserInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkColorChooserWidget, gtk_color_chooser_widget, GTK_TYPE_WIDGET,
|
|
G_ADD_PRIVATE (GtkColorChooserWidget)
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_COLOR_CHOOSER,
|
|
gtk_color_chooser_widget_iface_init))
|
|
|
|
static void
|
|
select_swatch (GtkColorChooserWidget *cc,
|
|
GtkColorSwatch *swatch)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GdkRGBA color;
|
|
|
|
if (priv->current == swatch)
|
|
return;
|
|
|
|
if (priv->current != NULL)
|
|
gtk_widget_unset_state_flags (GTK_WIDGET (priv->current), GTK_STATE_FLAG_SELECTED);
|
|
|
|
gtk_widget_set_state_flags (GTK_WIDGET (swatch), GTK_STATE_FLAG_SELECTED, FALSE);
|
|
priv->current = swatch;
|
|
|
|
gtk_color_swatch_get_rgba (swatch, &color);
|
|
|
|
g_settings_set (priv->settings, "selected-color", "(bdddd)",
|
|
TRUE, color.red, color.green, color.blue, color.alpha);
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (priv->editor)))
|
|
gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (priv->editor), &color);
|
|
else
|
|
g_object_notify (G_OBJECT (cc), "rgba");
|
|
}
|
|
|
|
static void
|
|
swatch_selected (GtkColorSwatch *swatch,
|
|
GtkStateFlags previous,
|
|
GtkColorChooserWidget *cc)
|
|
{
|
|
GtkStateFlags flags;
|
|
|
|
flags = gtk_widget_get_state_flags (GTK_WIDGET (swatch));
|
|
if ((flags & GTK_STATE_FLAG_SELECTED) != (previous & GTK_STATE_FLAG_SELECTED) &&
|
|
(flags & GTK_STATE_FLAG_SELECTED) != 0)
|
|
select_swatch (cc, swatch);
|
|
}
|
|
|
|
static void
|
|
connect_swatch_signals (GtkWidget *p,
|
|
gpointer data)
|
|
{
|
|
g_signal_connect (p, "state-flags-changed", G_CALLBACK (swatch_selected), data);
|
|
}
|
|
|
|
static void
|
|
connect_button_signals (GtkWidget *p,
|
|
gpointer data)
|
|
{
|
|
// g_signal_connect (p, "activate", G_CALLBACK (button_activate), data);
|
|
}
|
|
|
|
static void
|
|
save_custom_colors (GtkColorChooserWidget *cc)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GVariantBuilder builder;
|
|
GVariant *variant;
|
|
GdkRGBA color;
|
|
GList *children, *l;
|
|
GtkWidget *child;
|
|
|
|
g_variant_builder_init (&builder, G_VARIANT_TYPE ("a(dddd)"));
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (priv->custom));
|
|
for (l = g_list_nth (children, 1); l != NULL; l = l->next)
|
|
{
|
|
child = l->data;
|
|
if (gtk_color_swatch_get_rgba (GTK_COLOR_SWATCH (child), &color))
|
|
g_variant_builder_add (&builder, "(dddd)",
|
|
color.red, color.green, color.blue, color.alpha);
|
|
}
|
|
|
|
variant = g_variant_builder_end (&builder);
|
|
g_settings_set_value (priv->settings, "custom-colors", variant);
|
|
|
|
g_list_free (children);
|
|
}
|
|
|
|
static void
|
|
connect_custom_signals (GtkWidget *p,
|
|
gpointer data)
|
|
{
|
|
connect_swatch_signals (p, data);
|
|
g_signal_connect_swapped (p, "notify::rgba",
|
|
G_CALLBACK (save_custom_colors), data);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_set_use_alpha (GtkColorChooserWidget *cc,
|
|
gboolean use_alpha)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GList *children, *l;
|
|
GList *palettes, *p;
|
|
GtkWidget *swatch;
|
|
GtkWidget *grid;
|
|
|
|
if (priv->use_alpha == use_alpha)
|
|
return;
|
|
|
|
priv->use_alpha = use_alpha;
|
|
gtk_color_chooser_set_use_alpha (GTK_COLOR_CHOOSER (priv->editor), use_alpha);
|
|
|
|
palettes = gtk_container_get_children (GTK_CONTAINER (priv->palette));
|
|
for (p = palettes; p; p = p->next)
|
|
{
|
|
grid = p->data;
|
|
|
|
if (!GTK_IS_CONTAINER (grid))
|
|
continue;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (grid));
|
|
for (l = children; l; l = l->next)
|
|
{
|
|
swatch = l->data;
|
|
gtk_color_swatch_set_use_alpha (GTK_COLOR_SWATCH (swatch), use_alpha);
|
|
}
|
|
g_list_free (children);
|
|
}
|
|
g_list_free (palettes);
|
|
|
|
gtk_widget_queue_draw (GTK_WIDGET (cc));
|
|
g_object_notify (G_OBJECT (cc), "use-alpha");
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_set_show_editor (GtkColorChooserWidget *cc,
|
|
gboolean show_editor)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
|
|
if (show_editor)
|
|
{
|
|
GdkRGBA color = { 0.75, 0.25, 0.25, 1.0 };
|
|
|
|
if (priv->current)
|
|
gtk_color_swatch_get_rgba (priv->current, &color);
|
|
gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (priv->editor), &color);
|
|
}
|
|
|
|
gtk_widget_set_visible (priv->editor, show_editor);
|
|
gtk_widget_set_visible (priv->palette, !show_editor);
|
|
}
|
|
|
|
static void
|
|
update_from_editor (GtkColorEditor *editor,
|
|
GParamSpec *pspec,
|
|
GtkColorChooserWidget *widget)
|
|
{
|
|
if (gtk_widget_get_visible (GTK_WIDGET (editor)))
|
|
g_object_notify (G_OBJECT (widget), "rgba");
|
|
}
|
|
|
|
/* UI construction {{{1 */
|
|
|
|
static guint
|
|
scale_round (gdouble value, gdouble scale)
|
|
{
|
|
value = floor (value * scale + 0.5);
|
|
value = MAX (value, 0);
|
|
value = MIN (value, scale);
|
|
return (guint)value;
|
|
}
|
|
|
|
static gchar *
|
|
accessible_color_name (GdkRGBA *color)
|
|
{
|
|
if (color->alpha < 1.0)
|
|
return g_strdup_printf (_("Red %d%%, Green %d%%, Blue %d%%, Alpha %d%%"),
|
|
scale_round (color->red, 100),
|
|
scale_round (color->green, 100),
|
|
scale_round (color->blue, 100),
|
|
scale_round (color->alpha, 100));
|
|
else
|
|
return g_strdup_printf (_("Red %d%%, Green %d%%, Blue %d%%"),
|
|
scale_round (color->red, 100),
|
|
scale_round (color->green, 100),
|
|
scale_round (color->blue, 100));
|
|
}
|
|
|
|
static void
|
|
remove_palette (GtkColorChooserWidget *cc)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GList *children, *l;
|
|
GtkWidget *widget;
|
|
|
|
if (priv->current != NULL &&
|
|
gtk_widget_get_parent (GTK_WIDGET (priv->current)) != priv->custom)
|
|
priv->current = NULL;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (priv->palette));
|
|
for (l = children; l; l = l->next)
|
|
{
|
|
widget = l->data;
|
|
if (widget == priv->custom_label || widget == priv->custom)
|
|
continue;
|
|
gtk_container_remove (GTK_CONTAINER (priv->palette), widget);
|
|
}
|
|
g_list_free (children);
|
|
}
|
|
|
|
static void
|
|
add_palette (GtkColorChooserWidget *cc,
|
|
GtkOrientation orientation,
|
|
gint colors_per_line,
|
|
gint n_colors,
|
|
GdkRGBA *colors,
|
|
const gchar **names)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GtkWidget *grid;
|
|
GtkWidget *p;
|
|
AtkObject *atk_obj;
|
|
gint line, pos;
|
|
gint i;
|
|
gint left, right;
|
|
|
|
if (colors == NULL)
|
|
{
|
|
remove_palette (cc);
|
|
return;
|
|
}
|
|
|
|
grid = gtk_grid_new ();
|
|
gtk_widget_set_margin_bottom (grid, 12);
|
|
gtk_grid_set_row_spacing (GTK_GRID (grid), 2);
|
|
gtk_grid_set_column_spacing (GTK_GRID (grid), 4);
|
|
gtk_container_add (GTK_CONTAINER (priv->palette), grid);
|
|
|
|
left = 0;
|
|
right = colors_per_line - 1;
|
|
if (gtk_widget_get_direction (GTK_WIDGET (cc)) == GTK_TEXT_DIR_RTL)
|
|
{
|
|
i = left;
|
|
left = right;
|
|
right = i;
|
|
}
|
|
|
|
for (i = 0; i < n_colors; i++)
|
|
{
|
|
p = gtk_color_swatch_new ();
|
|
atk_obj = gtk_widget_get_accessible (p);
|
|
if (names)
|
|
{
|
|
atk_object_set_name (atk_obj,
|
|
g_dpgettext2 (GETTEXT_PACKAGE, "Color name", names[i]));
|
|
}
|
|
else
|
|
{
|
|
gchar *text, *name;
|
|
|
|
name = accessible_color_name (&colors[i]);
|
|
text = g_strdup_printf (_("Color: %s"), name);
|
|
atk_object_set_name (atk_obj, text);
|
|
g_free (text);
|
|
g_free (name);
|
|
}
|
|
gtk_color_swatch_set_rgba (GTK_COLOR_SWATCH (p), &colors[i]);
|
|
connect_swatch_signals (p, cc);
|
|
|
|
line = i / colors_per_line;
|
|
pos = i % colors_per_line;
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
if (pos == left)
|
|
gtk_widget_add_style_class (p, GTK_STYLE_CLASS_LEFT);
|
|
else if (pos == right)
|
|
gtk_widget_add_style_class (p, GTK_STYLE_CLASS_RIGHT);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), p, pos, line, 1, 1);
|
|
}
|
|
else
|
|
{
|
|
if (pos == 0)
|
|
gtk_widget_add_style_class (p, GTK_STYLE_CLASS_TOP);
|
|
else if (pos == colors_per_line - 1)
|
|
gtk_widget_add_style_class (p, GTK_STYLE_CLASS_BOTTOM);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), p, line, pos, 1, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
remove_default_palette (GtkColorChooserWidget *cc)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
|
|
if (!priv->has_default_palette)
|
|
return;
|
|
|
|
remove_palette (cc);
|
|
priv->has_default_palette = FALSE;
|
|
}
|
|
|
|
static void
|
|
add_default_palette (GtkColorChooserWidget *cc)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
const gchar *default_colors[9][3] = {
|
|
{ "#ef2929", "#cc0000", "#a40000" }, /* Scarlet Red */
|
|
{ "#fcaf3e", "#f57900", "#ce5c00" }, /* Orange */
|
|
{ "#fce94f", "#edd400", "#c4a000" }, /* Butter */
|
|
{ "#8ae234", "#73d216", "#4e9a06" }, /* Chameleon */
|
|
{ "#729fcf", "#3465a4", "#204a87" }, /* Sky Blue */
|
|
{ "#ad7fa8", "#75507b", "#5c3566" }, /* Plum */
|
|
{ "#e9b96e", "#c17d11", "#8f5902" }, /* Chocolate */
|
|
{ "#888a85", "#555753", "#2e3436" }, /* Aluminum 1 */
|
|
{ "#eeeeec", "#d3d7cf", "#babdb6" } /* Aluminum 2 */
|
|
};
|
|
const gchar *color_names[] = {
|
|
NC_("Color name", "Light Scarlet Red"),
|
|
NC_("Color name", "Scarlet Red"),
|
|
NC_("Color name", "Dark Scarlet Red"),
|
|
NC_("Color name", "Light Orange"),
|
|
NC_("Color name", "Orange"),
|
|
NC_("Color name", "Dark Orange"),
|
|
NC_("Color name", "Light Butter"),
|
|
NC_("Color name", "Butter"),
|
|
NC_("Color name", "Dark Butter"),
|
|
NC_("Color name", "Light Chameleon"),
|
|
NC_("Color name", "Chameleon"),
|
|
NC_("Color name", "Dark Chameleon"),
|
|
NC_("Color name", "Light Sky Blue"),
|
|
NC_("Color name", "Sky Blue"),
|
|
NC_("Color name", "Dark Sky Blue"),
|
|
NC_("Color name", "Light Plum"),
|
|
NC_("Color name", "Plum"),
|
|
NC_("Color name", "Dark Plum"),
|
|
NC_("Color name", "Light Chocolate"),
|
|
NC_("Color name", "Chocolate"),
|
|
NC_("Color name", "Dark Chocolate"),
|
|
NC_("Color name", "Light Aluminum 1"),
|
|
NC_("Color name", "Aluminum 1"),
|
|
NC_("Color name", "Dark Aluminum 1"),
|
|
NC_("Color name", "Light Aluminum 2"),
|
|
NC_("Color name", "Aluminum 2"),
|
|
NC_("Color name", "Dark Aluminum 2")
|
|
};
|
|
const gchar *default_grays[9] = {
|
|
"#000000", /* black */
|
|
"#2e3436", /* very dark gray */
|
|
"#555753", /* darker gray */
|
|
"#888a85", /* dark gray */
|
|
"#babdb6", /* medium gray */
|
|
"#d3d7cf", /* light gray */
|
|
"#eeeeec", /* lighter gray */
|
|
"#f3f3f3", /* very light gray */
|
|
"#ffffff" /* white */
|
|
};
|
|
const gchar *gray_names[] = {
|
|
NC_("Color name", "Black"),
|
|
NC_("Color name", "Very Dark Gray"),
|
|
NC_("Color name", "Darker Gray"),
|
|
NC_("Color name", "Dark Gray"),
|
|
NC_("Color name", "Medium Gray"),
|
|
NC_("Color name", "Light Gray"),
|
|
NC_("Color name", "Lighter Gray"),
|
|
NC_("Color name", "Very Light Gray"),
|
|
NC_("Color name", "White")
|
|
};
|
|
GdkRGBA colors[9*3];
|
|
gint i, j;
|
|
|
|
for (i = 0; i < 9; i++)
|
|
for (j = 0; j < 3; j++)
|
|
gdk_rgba_parse (&colors[i*3 + j], default_colors[i][j]);
|
|
|
|
add_palette (cc, GTK_ORIENTATION_VERTICAL, 3, 9*3, colors, color_names);
|
|
|
|
for (i = 0; i < 9; i++)
|
|
gdk_rgba_parse (&colors[i], default_grays[i]);
|
|
|
|
add_palette (cc, GTK_ORIENTATION_HORIZONTAL, 9, 9, colors, gray_names);
|
|
|
|
priv->has_default_palette = TRUE;
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_activate_color_customize (GtkWidget *widget,
|
|
const char *name,
|
|
GVariant *parameter)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (widget);
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GdkRGBA color;
|
|
|
|
g_variant_get (parameter, "(dddd)", &color.red, &color.green, &color.blue, &color.alpha);
|
|
|
|
gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (priv->editor), &color);
|
|
|
|
gtk_widget_hide (priv->palette);
|
|
gtk_widget_show (priv->editor);
|
|
g_object_notify (G_OBJECT (cc), "show-editor");
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_activate_color_select (GtkWidget *widget,
|
|
const char *name,
|
|
GVariant *parameter)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (widget);
|
|
GdkRGBA color;
|
|
|
|
g_variant_get (parameter, "(dddd)", &color.red, &color.green, &color.blue, &color.alpha);
|
|
|
|
_gtk_color_chooser_color_activated (GTK_COLOR_CHOOSER (cc), &color);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_init (GtkColorChooserWidget *cc)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GtkWidget *box;
|
|
GtkWidget *p;
|
|
GtkWidget *button;
|
|
GtkWidget *label;
|
|
gint i;
|
|
double color[4];
|
|
GdkRGBA rgba;
|
|
GVariant *variant;
|
|
GVariantIter iter;
|
|
gboolean selected;
|
|
AtkObject *atk_obj;
|
|
gchar *text, *name;
|
|
|
|
priv->use_alpha = TRUE;
|
|
|
|
priv->palette = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
gtk_widget_set_parent (priv->palette, GTK_WIDGET (cc));
|
|
|
|
add_default_palette (cc);
|
|
|
|
/* translators: label for the custom section in the color chooser */
|
|
priv->custom_label = label = gtk_label_new (_("Custom"));
|
|
gtk_widget_set_halign (label, GTK_ALIGN_START);
|
|
gtk_container_add (GTK_CONTAINER (priv->palette), label);
|
|
|
|
priv->custom = box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
|
|
g_object_set (box, "margin-top", 12, NULL);
|
|
gtk_container_add (GTK_CONTAINER (priv->palette), box);
|
|
|
|
priv->button = button = gtk_color_swatch_new ();
|
|
gtk_widget_set_name (button, "add-color-button");
|
|
atk_obj = gtk_widget_get_accessible (button);
|
|
atk_object_set_name (atk_obj, _("Custom color"));
|
|
atk_object_set_description (atk_obj, _("Create a custom color"));
|
|
connect_button_signals (button, cc);
|
|
gtk_color_swatch_set_icon (GTK_COLOR_SWATCH (button), "list-add-symbolic");
|
|
gtk_color_swatch_set_selectable (GTK_COLOR_SWATCH (button), FALSE);
|
|
gtk_container_add (GTK_CONTAINER (box), button);
|
|
|
|
priv->settings = g_settings_new ("org.gtk.gtk4.Settings.ColorChooser");
|
|
variant = g_settings_get_value (priv->settings, I_("custom-colors"));
|
|
g_variant_iter_init (&iter, variant);
|
|
i = 0;
|
|
p = NULL;
|
|
while (g_variant_iter_loop (&iter, "(dddd)", &color[0], &color[1], &color[2], &color[3]))
|
|
{
|
|
i++;
|
|
p = gtk_color_swatch_new ();
|
|
|
|
rgba.red = color[0];
|
|
rgba.green = color[1];
|
|
rgba.blue = color[2];
|
|
rgba.alpha = color[3];
|
|
|
|
gtk_color_swatch_set_rgba (GTK_COLOR_SWATCH (p), &rgba);
|
|
gtk_color_swatch_set_can_drop (GTK_COLOR_SWATCH (p), TRUE);
|
|
atk_obj = gtk_widget_get_accessible (p);
|
|
name = accessible_color_name (&rgba);
|
|
text = g_strdup_printf (_("Custom color %d: %s"), i, name);
|
|
atk_object_set_name (atk_obj, text);
|
|
g_free (text);
|
|
g_free (name);
|
|
connect_custom_signals (p, cc);
|
|
gtk_container_add (GTK_CONTAINER (box), p);
|
|
|
|
if (i == 8)
|
|
break;
|
|
}
|
|
g_variant_unref (variant);
|
|
|
|
priv->editor = gtk_color_editor_new ();
|
|
gtk_widget_set_halign (priv->editor, GTK_ALIGN_CENTER);
|
|
gtk_widget_set_hexpand (priv->editor, TRUE);
|
|
g_signal_connect (priv->editor, "notify::rgba",
|
|
G_CALLBACK (update_from_editor), cc);
|
|
|
|
gtk_widget_set_parent (priv->editor, GTK_WIDGET (cc));
|
|
|
|
g_settings_get (priv->settings, I_("selected-color"), "(bdddd)",
|
|
&selected,
|
|
&color[0], &color[1], &color[2], &color[3]);
|
|
if (selected)
|
|
{
|
|
rgba.red = color[0];
|
|
rgba.green = color[1];
|
|
rgba.blue = color[2];
|
|
rgba.alpha = color[3];
|
|
gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (cc), &rgba);
|
|
}
|
|
|
|
gtk_widget_hide (GTK_WIDGET (priv->editor));
|
|
}
|
|
|
|
/* GObject implementation {{{1 */
|
|
|
|
static void
|
|
gtk_color_chooser_widget_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkColorChooserWidget *cw = GTK_COLOR_CHOOSER_WIDGET (object);
|
|
GtkColorChooser *cc = GTK_COLOR_CHOOSER (object);
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cw);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_RGBA:
|
|
{
|
|
GdkRGBA color;
|
|
|
|
gtk_color_chooser_get_rgba (cc, &color);
|
|
g_value_set_boxed (value, &color);
|
|
}
|
|
break;
|
|
case PROP_USE_ALPHA:
|
|
g_value_set_boolean (value, priv->use_alpha);
|
|
break;
|
|
case PROP_SHOW_EDITOR:
|
|
g_value_set_boolean (value, gtk_widget_get_visible (priv->editor));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_RGBA:
|
|
gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (cc),
|
|
g_value_get_boxed (value));
|
|
break;
|
|
case PROP_USE_ALPHA:
|
|
gtk_color_chooser_widget_set_use_alpha (cc,
|
|
g_value_get_boolean (value));
|
|
break;
|
|
case PROP_SHOW_EDITOR:
|
|
gtk_color_chooser_widget_set_show_editor (cc,
|
|
g_value_get_boolean (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_finalize (GObject *object)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (object);
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
|
|
g_object_unref (priv->settings);
|
|
|
|
gtk_widget_unparent (priv->editor);
|
|
gtk_widget_unparent (priv->palette);
|
|
|
|
G_OBJECT_CLASS (gtk_color_chooser_widget_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_class_init (GtkColorChooserWidgetClass *class)
|
|
{
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
object_class->get_property = gtk_color_chooser_widget_get_property;
|
|
object_class->set_property = gtk_color_chooser_widget_set_property;
|
|
object_class->finalize = gtk_color_chooser_widget_finalize;
|
|
|
|
g_object_class_override_property (object_class, PROP_RGBA, "rgba");
|
|
g_object_class_override_property (object_class, PROP_USE_ALPHA, "use-alpha");
|
|
|
|
/**
|
|
* GtkColorChooserWidget:show-editor:
|
|
*
|
|
* The ::show-editor property is %TRUE when the color chooser
|
|
* is showing the single-color editor. It can be set to switch
|
|
* the color chooser into single-color editing mode.
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_SHOW_EDITOR,
|
|
g_param_spec_boolean ("show-editor", P_("Show editor"), P_("Show editor"),
|
|
FALSE, GTK_PARAM_READWRITE));
|
|
|
|
gtk_widget_class_set_css_name (widget_class, I_("colorchooser"));
|
|
gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_BOX_LAYOUT);
|
|
|
|
/**
|
|
* GtkColorChooserWidget|color.select:
|
|
* @red: the red value, between 0 and 1
|
|
* @green: the green value, between 0 and 1
|
|
* @blue: the blue value, between 0 and 1
|
|
* @alpha: the alpha value, between 0 and 1
|
|
*
|
|
* Emits the #GtkColorChooser::color-activated signal for
|
|
* the given color.
|
|
*/
|
|
gtk_widget_class_install_action (widget_class, "color.select", "(dddd)",
|
|
gtk_color_chooser_widget_activate_color_select);
|
|
|
|
/**
|
|
* GtkColorChooserWidget|color.customize:
|
|
* @red: the red value, between 0 and 1
|
|
* @green: the green value, between 0 and 1
|
|
* @blue: the blue value, between 0 and 1
|
|
* @alpha: the alpha value, between 0 and 1
|
|
*
|
|
* Activates the color editor for the given color.
|
|
*/
|
|
gtk_widget_class_install_action (widget_class, "color.customize", "(dddd)",
|
|
gtk_color_chooser_widget_activate_color_customize);
|
|
|
|
gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_COMPOSITE_ACCESSIBLE);
|
|
}
|
|
|
|
/* GtkColorChooser implementation {{{1 */
|
|
|
|
static void
|
|
gtk_color_chooser_widget_get_rgba (GtkColorChooser *chooser,
|
|
GdkRGBA *color)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (chooser);
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
|
|
if (gtk_widget_get_visible (priv->editor))
|
|
gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (priv->editor), color);
|
|
else if (priv->current)
|
|
gtk_color_swatch_get_rgba (priv->current, color);
|
|
else
|
|
{
|
|
color->red = 1.0;
|
|
color->green = 1.0;
|
|
color->blue = 1.0;
|
|
color->alpha = 1.0;
|
|
}
|
|
|
|
if (!priv->use_alpha)
|
|
color->alpha = 1.0;
|
|
}
|
|
|
|
static void
|
|
add_custom_color (GtkColorChooserWidget *cc,
|
|
const GdkRGBA *color)
|
|
{
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GtkWidget *last;
|
|
GtkWidget *p;
|
|
GList *children;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (priv->custom));
|
|
if (g_list_length (children) >= 9)
|
|
{
|
|
last = g_list_last (children)->data;
|
|
if (last == GTK_WIDGET (priv->current))
|
|
priv->current = NULL;
|
|
|
|
gtk_widget_destroy (last);
|
|
}
|
|
|
|
g_list_free (children);
|
|
|
|
p = gtk_color_swatch_new ();
|
|
gtk_color_swatch_set_rgba (GTK_COLOR_SWATCH (p), color);
|
|
gtk_color_swatch_set_can_drop (GTK_COLOR_SWATCH (p), TRUE);
|
|
connect_custom_signals (p, cc);
|
|
|
|
gtk_box_insert_child_after (GTK_BOX (priv->custom), p, gtk_widget_get_first_child (priv->custom));
|
|
gtk_widget_show (p);
|
|
|
|
select_swatch (cc, GTK_COLOR_SWATCH (p));
|
|
save_custom_colors (cc);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_set_rgba (GtkColorChooser *chooser,
|
|
const GdkRGBA *color)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (chooser);
|
|
GtkColorChooserWidgetPrivate *priv = gtk_color_chooser_widget_get_instance_private (cc);
|
|
GList *children, *l;
|
|
GList *palettes, *p;
|
|
GtkColorSwatch *swatch;
|
|
GtkWidget *w;
|
|
GdkRGBA c;
|
|
|
|
palettes = gtk_container_get_children (GTK_CONTAINER (priv->palette));
|
|
for (p = palettes; p; p = p->next)
|
|
{
|
|
w = p->data;
|
|
if (!GTK_IS_GRID (w) && !GTK_IS_BOX (w))
|
|
continue;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (w));
|
|
for (l = children; l; l = l->next)
|
|
{
|
|
swatch = l->data;
|
|
gtk_color_swatch_get_rgba (swatch, &c);
|
|
if (!priv->use_alpha)
|
|
c.alpha = color->alpha;
|
|
if (gdk_rgba_equal (color, &c))
|
|
{
|
|
select_swatch (cc, swatch);
|
|
g_list_free (children);
|
|
g_list_free (palettes);
|
|
return;
|
|
}
|
|
}
|
|
g_list_free (children);
|
|
}
|
|
g_list_free (palettes);
|
|
|
|
add_custom_color (cc, color);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_add_palette (GtkColorChooser *chooser,
|
|
GtkOrientation orientation,
|
|
gint colors_per_line,
|
|
gint n_colors,
|
|
GdkRGBA *colors)
|
|
{
|
|
GtkColorChooserWidget *cc = GTK_COLOR_CHOOSER_WIDGET (chooser);
|
|
|
|
remove_default_palette (cc);
|
|
add_palette (cc, orientation, colors_per_line, n_colors, colors, NULL);
|
|
}
|
|
|
|
static void
|
|
gtk_color_chooser_widget_iface_init (GtkColorChooserInterface *iface)
|
|
{
|
|
iface->get_rgba = gtk_color_chooser_widget_get_rgba;
|
|
iface->set_rgba = gtk_color_chooser_widget_set_rgba;
|
|
iface->add_palette = gtk_color_chooser_widget_add_palette;
|
|
}
|
|
|
|
/* Public API {{{1 */
|
|
|
|
/**
|
|
* gtk_color_chooser_widget_new:
|
|
*
|
|
* Creates a new #GtkColorChooserWidget.
|
|
*
|
|
* Returns: a new #GtkColorChooserWidget
|
|
*/
|
|
GtkWidget *
|
|
gtk_color_chooser_widget_new (void)
|
|
{
|
|
return g_object_new (GTK_TYPE_COLOR_CHOOSER_WIDGET, NULL);
|
|
}
|
|
|
|
/* vim:set foldmethod=marker: */
|