gimpcurvestool.c 32.2 KB
Newer Older
1
/* GIMP - The GNU Image Manipulation Program
Elliot Lee's avatar
Elliot Lee committed
2 3
 * Copyright (C) 1995 Spencer Kimball and Peter Mattis
 *
4
 * This program is free software: you can redistribute it and/or modify
Elliot Lee's avatar
Elliot Lee committed
5
 * it under the terms of the GNU General Public License as published by
6
 * the Free Software Foundation; either version 3 of the License, or
Elliot Lee's avatar
Elliot Lee committed
7 8 9 10 11 12 13 14
 * (at your option) any later version.
 *
 * This program 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
15
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Elliot Lee's avatar
Elliot Lee committed
16
 */
Sven Neumann's avatar
Sven Neumann committed
17

18 19
#include "config.h"

20 21 22
#include <errno.h>

#include <glib/gstdio.h>
23
#include <gegl.h>
Sven Neumann's avatar
Sven Neumann committed
24 25
#include <gtk/gtk.h>

26
#include "libgimpbase/gimpbase.h"
27
#include "libgimpcolor/gimpcolor.h"
28
#include "libgimpconfig/gimpconfig.h"
29
#include "libgimpwidgets/gimpwidgets.h"
30

31
#include "tools-types.h"
Michael Natterer's avatar
Michael Natterer committed
32

33 34
#include "operations/gimpcurvesconfig.h"
#include "operations/gimpoperationcurves.h"
35

36
#include "core/gimp.h"
37
#include "core/gimpcurve.h"
Sven Neumann's avatar
Sven Neumann committed
38
#include "core/gimpcurve-map.h"
Michael Natterer's avatar
Michael Natterer committed
39
#include "core/gimpdrawable.h"
40
#include "core/gimpdrawable-histogram.h"
41
#include "core/gimperror.h"
42
#include "core/gimphistogram.h"
Michael Natterer's avatar
Michael Natterer committed
43 44
#include "core/gimpimage.h"

45
#include "widgets/gimpcolorbar.h"
46
#include "widgets/gimphelp-ids.h"
47
#include "widgets/gimpcurveview.h"
48
#include "widgets/gimpwidgets-utils.h"
Michael Natterer's avatar
Michael Natterer committed
49

50 51
#include "display/gimpdisplay.h"

Michael Natterer's avatar
Michael Natterer committed
52
#include "gimpcurvestool.h"
53
#include "gimphistogramoptions.h"
Sven Neumann's avatar
Sven Neumann committed
54

55
#include "gimp-intl.h"
56

57

58 59 60
#define GRAPH_SIZE 256
#define BAR_SIZE    12
#define RADIUS       4
Elliot Lee's avatar
Elliot Lee committed
61 62


Michael Natterer's avatar
Michael Natterer committed
63 64
/*  local function prototypes  */

65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
static gboolean   gimp_curves_tool_initialize      (GimpTool             *tool,
                                                    GimpDisplay          *display,
                                                    GError              **error);
static void       gimp_curves_tool_button_release  (GimpTool             *tool,
                                                    const GimpCoords     *coords,
                                                    guint32               time,
                                                    GdkModifierType       state,
                                                    GimpButtonReleaseType release_type,
                                                    GimpDisplay          *display);
static gboolean   gimp_curves_tool_key_press       (GimpTool             *tool,
                                                    GdkEventKey          *kevent,
                                                    GimpDisplay          *display);
static void       gimp_curves_tool_oper_update     (GimpTool             *tool,
                                                    const GimpCoords     *coords,
                                                    GdkModifierType       state,
                                                    gboolean              proximity,
                                                    GimpDisplay          *display);

static gchar    * gimp_curves_tool_get_operation   (GimpFilterTool       *filter_tool,
                                                    gchar               **title,
                                                    gchar               **description,
                                                    gchar               **undo_desc,
                                                    gchar               **icon_name,
88
                                                    gchar               **help_id,
89
                                                    gboolean             *has_settings);
90 91
static void       gimp_curves_tool_dialog          (GimpFilterTool       *filter_tool);
static void       gimp_curves_tool_reset           (GimpFilterTool       *filter_tool);
92 93 94
static void       gimp_curves_tool_config_notify   (GimpFilterTool       *filter_tool,
                                                    GimpConfig           *config,
                                                    const GParamSpec     *pspec);
95 96 97 98 99 100
static gboolean   gimp_curves_tool_settings_import (GimpFilterTool       *filter_tool,
                                                    GInputStream         *input,
                                                    GError              **error);
static gboolean   gimp_curves_tool_settings_export (GimpFilterTool       *filter_tool,
                                                    GOutputStream        *output,
                                                    GError              **error);
Ell's avatar
Ell committed
101 102 103 104 105 106
static void       gimp_curves_tool_color_picked    (GimpFilterTool       *filter_tool,
                                                    gpointer              identifier,
                                                    gdouble               x,
                                                    gdouble               y,
                                                    const Babl           *sample_format,
                                                    const GimpRGB        *color);
107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124

static void       gimp_curves_tool_export_setup    (GimpSettingsBox      *settings_box,
                                                    GtkFileChooserDialog *dialog,
                                                    gboolean              export,
                                                    GimpCurvesTool       *tool);
static void       gimp_curves_tool_update_channel  (GimpCurvesTool       *tool);

static void       curves_channel_callback          (GtkWidget            *widget,
                                                    GimpCurvesTool       *tool);
static void       curves_channel_reset_callback    (GtkWidget            *widget,
                                                    GimpCurvesTool       *tool);

static gboolean   curves_menu_sensitivity          (gint                  value,
                                                    gpointer              data);

static void       curves_curve_type_callback       (GtkWidget            *widget,
                                                    GimpCurvesTool       *tool);

125 126 127
static gboolean   curves_get_channel_color         (GtkWidget            *widget,
                                                    GimpHistogramChannel  channel,
                                                    GimpRGB              *color);
128 129 130


G_DEFINE_TYPE (GimpCurvesTool, gimp_curves_tool, GIMP_TYPE_FILTER_TOOL)
131 132

#define parent_class gimp_curves_tool_parent_class
133

Michael Natterer's avatar
Michael Natterer committed
134

135
/*  public functions  */
Michael Natterer's avatar
Michael Natterer committed
136 137

void
Nate Summers's avatar
Nate Summers committed
138
gimp_curves_tool_register (GimpToolRegisterCallback  callback,
139
                           gpointer                  data)
Michael Natterer's avatar
Michael Natterer committed
140
{
Nate Summers's avatar
Nate Summers committed
141
  (* callback) (GIMP_TYPE_CURVES_TOOL,
142
                GIMP_TYPE_HISTOGRAM_OPTIONS,
143
                gimp_color_options_gui,
144
                0,
145
                "gimp-curves-tool",
146
                _("Curves"),
147
                _("Curves Tool: Adjust color curves"),
148
                N_("_Curves..."), NULL,
149
                NULL, GIMP_HELP_TOOL_CURVES,
150
                GIMP_ICON_TOOL_CURVES,
151
                data);
Michael Natterer's avatar
Michael Natterer committed
152 153
}

154 155 156

/*  private functions  */

Michael Natterer's avatar
Michael Natterer committed
157 158 159
static void
gimp_curves_tool_class_init (GimpCurvesToolClass *klass)
{
160 161
  GimpToolClass       *tool_class        = GIMP_TOOL_CLASS (klass);
  GimpFilterToolClass *filter_tool_class = GIMP_FILTER_TOOL_CLASS (klass);
162

163 164 165 166
  tool_class->initialize             = gimp_curves_tool_initialize;
  tool_class->button_release         = gimp_curves_tool_button_release;
  tool_class->key_press              = gimp_curves_tool_key_press;
  tool_class->oper_update            = gimp_curves_tool_oper_update;
167

168 169 170
  filter_tool_class->get_operation   = gimp_curves_tool_get_operation;
  filter_tool_class->dialog          = gimp_curves_tool_dialog;
  filter_tool_class->reset           = gimp_curves_tool_reset;
171
  filter_tool_class->config_notify   = gimp_curves_tool_config_notify;
172 173
  filter_tool_class->settings_import = gimp_curves_tool_settings_import;
  filter_tool_class->settings_export = gimp_curves_tool_settings_export;
Ell's avatar
Ell committed
174
  filter_tool_class->color_picked    = gimp_curves_tool_color_picked;
Michael Natterer's avatar
Michael Natterer committed
175 176 177
}

static void
178
gimp_curves_tool_init (GimpCurvesTool *tool)
Michael Natterer's avatar
Michael Natterer committed
179
{
180
  gint i;
181

182 183
  for (i = 0; i < G_N_ELEMENTS (tool->picked_color); i++)
    tool->picked_color[i] = -1.0;
Michael Natterer's avatar
Michael Natterer committed
184 185
}

186
static gboolean
187 188 189
gimp_curves_tool_initialize (GimpTool     *tool,
                             GimpDisplay  *display,
                             GError      **error)
Michael Natterer's avatar
Michael Natterer committed
190
{
Ell's avatar
Ell committed
191 192 193 194
  GimpCurvesTool *c_tool      = GIMP_CURVES_TOOL (tool);
  GimpFilterTool *filter_tool = GIMP_FILTER_TOOL (tool);
  GimpImage      *image       = gimp_display_get_image (display);
  GimpDrawable   *drawable    = gimp_image_get_active_drawable (image);
195
  GimpHistogram  *histogram;
196

197 198 199 200
  if (! GIMP_TOOL_CLASS (parent_class)->initialize (tool, display, error))
    {
      return FALSE;
    }
201

202
  histogram = gimp_histogram_new (FALSE);
203
  gimp_drawable_calculate_histogram (drawable, histogram);
204
  gimp_histogram_view_set_background (GIMP_HISTOGRAM_VIEW (c_tool->graph),
205
                                      histogram);
206
  g_object_unref (histogram);
207

208
  if (gimp_drawable_get_component_type (drawable) == GIMP_COMPONENT_TYPE_U8)
209 210 211 212 213 214
    {
      gimp_curve_view_set_range_x (GIMP_CURVE_VIEW (c_tool->graph), 0, 255);
      gimp_curve_view_set_range_y (GIMP_CURVE_VIEW (c_tool->graph), 0, 255);
    }
  else
    {
215 216
      gimp_curve_view_set_range_x (GIMP_CURVE_VIEW (c_tool->graph), 0, 100);
      gimp_curve_view_set_range_y (GIMP_CURVE_VIEW (c_tool->graph), 0, 100);
217 218
    }

Ell's avatar
Ell committed
219 220 221
  /*  always pick colors  */
  gimp_filter_tool_enable_color_picking (filter_tool, NULL, FALSE);

222
  return TRUE;
Michael Natterer's avatar
Michael Natterer committed
223 224 225
}

static void
226
gimp_curves_tool_button_release (GimpTool              *tool,
227
                                 const GimpCoords      *coords,
228 229 230 231
                                 guint32                time,
                                 GdkModifierType        state,
                                 GimpButtonReleaseType  release_type,
                                 GimpDisplay           *display)
Michael Natterer's avatar
Michael Natterer committed
232
{
233 234 235
  GimpCurvesTool   *c_tool      = GIMP_CURVES_TOOL (tool);
  GimpFilterTool   *filter_tool = GIMP_FILTER_TOOL (tool);
  GimpCurvesConfig *config      = GIMP_CURVES_CONFIG (filter_tool->config);
Michael Natterer's avatar
Michael Natterer committed
236

237
  if (state & gimp_get_extend_selection_mask ())
Michael Natterer's avatar
Michael Natterer committed
238
    {
239
      GimpCurve *curve = config->curve[config->channel];
240
      gdouble    value = c_tool->picked_color[config->channel];
241
      gint       closest;
242

243
      closest = gimp_curve_get_closest_point (curve, value);
244

245
      gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
246
                                    closest);
247

248
      gimp_curve_set_point (curve, closest,
249
                            value, gimp_curve_map_value (curve, value));
Michael Natterer's avatar
Michael Natterer committed
250
    }
251
  else if (state & gimp_get_toggle_behavior_mask ())
Michael Natterer's avatar
Michael Natterer committed
252
    {
253
      GimpHistogramChannel channel;
254

255 256 257
      for (channel = GIMP_HISTOGRAM_VALUE;
           channel <= GIMP_HISTOGRAM_ALPHA;
           channel++)
258
        {
259 260
          GimpCurve *curve = config->curve[channel];
          gdouble    value = c_tool->picked_color[channel];
261
          gint       closest;
262

263 264 265
          if (value != -1)
            {
              closest = gimp_curve_get_closest_point (curve, value);
266

267 268
              gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
                                            closest);
269

270 271 272
              gimp_curve_set_point (curve, closest,
                                    value, gimp_curve_map_value (curve, value));
            }
273
        }
Michael Natterer's avatar
Michael Natterer committed
274 275
    }

276
  /*  chain up to halt the tool */
277 278
  GIMP_TOOL_CLASS (parent_class)->button_release (tool, coords, time, state,
                                                  release_type, display);
Michael Natterer's avatar
Michael Natterer committed
279 280
}

281
static gboolean
282 283
gimp_curves_tool_key_press (GimpTool    *tool,
                            GdkEventKey *kevent,
284
                            GimpDisplay *display)
285
{
286 287
  GimpCurvesTool *c_tool = GIMP_CURVES_TOOL (tool);

288 289 290 291 292
  if (tool->display && c_tool->graph)
    {
      if (gtk_widget_event (c_tool->graph, (GdkEvent *) kevent))
        return TRUE;
    }
293 294

  return GIMP_TOOL_CLASS (parent_class)->key_press (tool, kevent, display);
295 296 297
}

static void
298 299 300 301 302
gimp_curves_tool_oper_update (GimpTool         *tool,
                              const GimpCoords *coords,
                              GdkModifierType   state,
                              gboolean          proximity,
                              GimpDisplay      *display)
303
{
304 305
  if (gimp_filter_tool_on_guide (GIMP_FILTER_TOOL (tool),
                                 coords, display))
306
    {
307 308
      GIMP_TOOL_CLASS (parent_class)->oper_update (tool, coords, state, proximity,
                                                   display);
309
    }
310 311
  else
    {
312 313 314 315
      GimpColorPickMode  mode;
      gchar             *status      = NULL;
      GdkModifierType    extend_mask = gimp_get_extend_selection_mask ();
      GdkModifierType    toggle_mask = gimp_get_toggle_behavior_mask ();
316

317
      gimp_tool_pop_status (tool, display);
318

319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
      if (state & extend_mask)
        {
          mode   = GIMP_COLOR_PICK_MODE_PALETTE;
          status = g_strdup (_("Click to add a control point"));
        }
      else if (state & toggle_mask)
        {
          mode   = GIMP_COLOR_PICK_MODE_PALETTE;
          status = g_strdup (_("Click to add control points to all channels"));
        }
      else
        {
          mode   = GIMP_COLOR_PICK_MODE_NONE;
          status = gimp_suggest_modifiers (_("Click to locate on curve"),
                                           (extend_mask | toggle_mask) & ~state,
                                           _("%s: add control point"),
                                           _("%s: add control points to all channels"),
                                           NULL);
        }
338

339 340 341 342 343 344 345
      GIMP_COLOR_TOOL (tool)->pick_mode = mode;

      if (proximity)
        gimp_tool_push_status (tool, display, "%s", status);

      g_free (status);
    }
346 347
}

348
static gchar *
349 350 351 352 353
gimp_curves_tool_get_operation (GimpFilterTool  *filter_tool,
                                gchar          **title,
                                gchar          **description,
                                gchar          **undo_desc,
                                gchar          **icon_name,
354
                                gchar          **help_id,
355
                                gboolean        *has_settings)
356
{
357 358
  *description         = g_strdup (_("Adjust Color Curves"));
  *has_settings        = TRUE;
359

360
  return g_strdup ("gimp:curves");
361
}
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
362

363

364 365 366
/*******************/
/*  Curves dialog  */
/*******************/
Elliot Lee's avatar
Elliot Lee committed
367

368
static void
369
gimp_curves_tool_dialog (GimpFilterTool *filter_tool)
Elliot Lee's avatar
Elliot Lee committed
370
{
371 372 373
  GimpCurvesTool   *tool         = GIMP_CURVES_TOOL (filter_tool);
  GimpToolOptions  *tool_options = GIMP_TOOL_GET_OPTIONS (filter_tool);
  GimpCurvesConfig *config       = GIMP_CURVES_CONFIG (filter_tool->config);
374
  GtkListStore     *store;
375
  GtkWidget        *main_vbox;
376
  GtkWidget        *frame_vbox;
377 378 379 380
  GtkWidget        *vbox;
  GtkWidget        *hbox;
  GtkWidget        *hbox2;
  GtkWidget        *label;
381
  GtkWidget        *main_frame;
382 383 384 385
  GtkWidget        *frame;
  GtkWidget        *table;
  GtkWidget        *button;
  GtkWidget        *bar;
386
  GtkWidget        *combo;
387

388
  g_signal_connect (filter_tool->settings_box, "file-dialog-setup",
389
                    G_CALLBACK (gimp_curves_tool_export_setup),
390
                    filter_tool);
391

392
  main_vbox = gimp_filter_tool_dialog_get_vbox (filter_tool);
Elliot Lee's avatar
Elliot Lee committed
393

394
  /*  The combo box for selecting channels  */
395 396 397 398
  main_frame = gimp_frame_new (NULL);
  gtk_box_pack_start (GTK_BOX (main_vbox), main_frame, TRUE, TRUE, 0);
  gtk_widget_show (main_frame);

Michael Natterer's avatar
Michael Natterer committed
399
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
400
  gtk_frame_set_label_widget (GTK_FRAME (main_frame), hbox);
401 402
  gtk_widget_show (hbox);

403
  label = gtk_label_new_with_mnemonic (_("Cha_nnel:"));
404 405 406
  gimp_label_set_attributes (GTK_LABEL (label),
                             PANGO_ATTR_WEIGHT, PANGO_WEIGHT_BOLD,
                             -1);
407 408 409
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);

410 411 412
  store = gimp_enum_store_new_with_range (GIMP_TYPE_HISTOGRAM_CHANNEL,
                                          GIMP_HISTOGRAM_VALUE,
                                          GIMP_HISTOGRAM_ALPHA);
413 414
  tool->channel_menu =
    gimp_enum_combo_box_new_with_model (GIMP_ENUM_STORE (store));
415 416
  g_object_unref (store);

417 418
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                 config->channel);
419 420
  gimp_enum_combo_box_set_icon_prefix (GIMP_ENUM_COMBO_BOX (tool->channel_menu),
                                       "gimp-channel");
421 422
  gimp_int_combo_box_set_sensitivity (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                      curves_menu_sensitivity, filter_tool, NULL);
423 424
  gtk_box_pack_start (GTK_BOX (hbox), tool->channel_menu, FALSE, FALSE, 0);
  gtk_widget_show (tool->channel_menu);
425

426 427 428
  g_signal_connect (tool->channel_menu, "changed",
                    G_CALLBACK (curves_channel_callback),
                    tool);
429

430
  gtk_label_set_mnemonic_widget (GTK_LABEL (label), tool->channel_menu);
431

432
  button = gtk_button_new_with_mnemonic (_("R_eset Channel"));
433
  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
434
  gtk_widget_show (button);
435

436
  g_signal_connect (button, "clicked",
437
                    G_CALLBACK (curves_channel_reset_callback),
438
                    tool);
Elliot Lee's avatar
Elliot Lee committed
439

440
  /*  The histogram scale radio buttons  */
441 442 443
  hbox2 = gimp_prop_enum_icon_box_new (G_OBJECT (tool_options),
                                       "histogram-scale", "gimp-histogram",
                                       0, 0);
444 445
  gtk_box_pack_end (GTK_BOX (hbox), hbox2, FALSE, FALSE, 0);
  gtk_widget_show (hbox2);
446

447 448 449 450
  frame_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 4);
  gtk_container_add (GTK_CONTAINER (main_frame), frame_vbox);
  gtk_widget_show (frame_vbox);

451
  /*  The table for the color bars and the graph  */
Elliot Lee's avatar
Elliot Lee committed
452
  table = gtk_table_new (2, 2, FALSE);
453 454
  gtk_table_set_col_spacings (GTK_TABLE (table), 2);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
455
  gtk_box_pack_start (GTK_BOX (frame_vbox), table, TRUE, TRUE, 0);
456 457

  /*  The left color bar  */
Michael Natterer's avatar
Michael Natterer committed
458
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
459
  gtk_table_attach (GTK_TABLE (table), vbox, 0, 1, 0, 1,
460
                    GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
461
  gtk_widget_show (vbox);
Elliot Lee's avatar
Elliot Lee committed
462 463

  frame = gtk_frame_new (NULL);
464
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
465
  gtk_box_pack_start (GTK_BOX (vbox), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
466 467
  gtk_widget_show (frame);

468 469 470 471
  tool->yrange = gimp_color_bar_new (GTK_ORIENTATION_VERTICAL);
  gtk_widget_set_size_request (tool->yrange, BAR_SIZE, -1);
  gtk_container_add (GTK_CONTAINER (frame), tool->yrange);
  gtk_widget_show (tool->yrange);
472

Elliot Lee's avatar
Elliot Lee committed
473 474
  /*  The curves graph  */
  frame = gtk_frame_new (NULL);
475
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
476
  gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 0, 1,
477
                    GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
478
  gtk_widget_show (frame);
Elliot Lee's avatar
Elliot Lee committed
479

480
  tool->graph = gimp_curve_view_new ();
481 482
  gimp_curve_view_set_range_x (GIMP_CURVE_VIEW (tool->graph), 0, 255);
  gimp_curve_view_set_range_y (GIMP_CURVE_VIEW (tool->graph), 0, 255);
483 484 485 486
  gtk_widget_set_size_request (tool->graph,
                               GRAPH_SIZE + RADIUS * 2,
                               GRAPH_SIZE + RADIUS * 2);
  g_object_set (tool->graph,
487 488 489
                "border-width", RADIUS,
                "subdivisions", 1,
                NULL);
490 491
  gtk_container_add (GTK_CONTAINER (frame), tool->graph);
  gtk_widget_show (tool->graph);
492

493 494 495 496
  g_object_bind_property (G_OBJECT (tool_options), "histogram-scale",
                          G_OBJECT (tool->graph),  "histogram-scale",
                          G_BINDING_SYNC_CREATE |
                          G_BINDING_BIDIRECTIONAL);
497 498

  /*  The bottom color bar  */
Michael Natterer's avatar
Michael Natterer committed
499
  hbox2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
500
  gtk_table_attach (GTK_TABLE (table), hbox2, 1, 2, 1, 2,
501
                    GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
502
  gtk_widget_show (hbox2);
503

Elliot Lee's avatar
Elliot Lee committed
504
  frame = gtk_frame_new (NULL);
505
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
506
  gtk_box_pack_start (GTK_BOX (hbox2), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
507 508
  gtk_widget_show (frame);

Michael Natterer's avatar
Michael Natterer committed
509 510
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
  gtk_box_set_homogeneous (GTK_BOX (vbox), TRUE);
511 512
  gtk_container_add (GTK_CONTAINER (frame), vbox);
  gtk_widget_show (vbox);
513

514 515
  tool->xrange = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
  gtk_widget_set_size_request (tool->xrange, -1, BAR_SIZE / 2);
516
  gtk_box_pack_start (GTK_BOX (vbox), tool->xrange, TRUE, TRUE, 0);
517
  gtk_widget_show (tool->xrange);
518

519
  bar = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
520
  gtk_box_pack_start (GTK_BOX (vbox), bar, TRUE, TRUE, 0);
521 522
  gtk_widget_show (bar);

523
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
524

Michael Natterer's avatar
Michael Natterer committed
525
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
526
  gtk_box_pack_end (GTK_BOX (frame_vbox), hbox, FALSE, FALSE, 0);
527
  gtk_widget_show (hbox);
528

529 530 531
  label = gtk_label_new_with_mnemonic (_("Curve _type:"));
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);
532

533
  tool->curve_type = combo = gimp_enum_combo_box_new (GIMP_TYPE_CURVE_TYPE);
534 535
  gimp_enum_combo_box_set_icon_prefix (GIMP_ENUM_COMBO_BOX (combo),
                                       "gimp-curve");
536 537 538 539 540
  gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo), 0,
                              G_CALLBACK (curves_curve_type_callback),
                              tool);
  gtk_box_pack_start (GTK_BOX (hbox), combo, TRUE, TRUE, 0);
  gtk_widget_show (combo);
Sven Neumann's avatar
Sven Neumann committed
541 542

  gtk_label_set_mnemonic_widget (GTK_LABEL (label), combo);
543 544

  gimp_curves_tool_update_channel (tool);
Elliot Lee's avatar
Elliot Lee committed
545 546
}

547
static void
548
gimp_curves_tool_reset (GimpFilterTool *filter_tool)
549
{
550
  GimpCurvesConfig     *config = GIMP_CURVES_CONFIG (filter_tool->config);
551
  GimpCurvesConfig     *default_config;
552
  GimpHistogramChannel  channel;
553

554
  default_config = GIMP_CURVES_CONFIG (filter_tool->default_config);
555

556 557 558 559
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
560 561
      if (default_config)
        {
562
          GimpCurveType curve_type = config->curve[channel]->curve_type;
563

564
          g_object_freeze_notify (G_OBJECT (config->curve[channel]));
565 566

          gimp_config_copy (GIMP_CONFIG (default_config->curve[channel]),
567
                            GIMP_CONFIG (config->curve[channel]),
568 569
                            0);

570
          g_object_set (config->curve[channel],
571 572 573
                        "curve-type", curve_type,
                        NULL);

574
          g_object_thaw_notify (G_OBJECT (config->curve[channel]));
575 576 577
        }
      else
        {
578
          gimp_curve_reset (config->curve[channel], FALSE);
579
        }
580
    }
581 582
}

583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608
static void
gimp_curves_tool_config_notify (GimpFilterTool   *filter_tool,
                                GimpConfig       *config,
                                const GParamSpec *pspec)
{
  GimpCurvesTool   *curves_tool   = GIMP_CURVES_TOOL (filter_tool);
  GimpCurvesConfig *curves_config = GIMP_CURVES_CONFIG (config);
  GimpCurve        *curve         = curves_config->curve[curves_config->channel];

  GIMP_FILTER_TOOL_CLASS (parent_class)->config_notify (filter_tool,
                                                        config, pspec);

  if (! curves_tool->xrange)
    return;

  if (! strcmp (pspec->name, "channel"))
    {
      gimp_curves_tool_update_channel (GIMP_CURVES_TOOL (filter_tool));
    }
  else if (! strcmp (pspec->name, "curve"))
    {
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (curves_tool->curve_type),
                                     curve->curve_type);
    }
}

609
static gboolean
610 611 612
gimp_curves_tool_settings_import (GimpFilterTool  *filter_tool,
                                  GInputStream    *input,
                                  GError         **error)
613
{
614
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (filter_tool->config);
615 616
  gchar             header[64];
  gsize             bytes_read;
617

618 619 620
  if (! g_input_stream_read_all (input, header, sizeof (header),
                                 &bytes_read, NULL, error) ||
      bytes_read != sizeof (header))
621
    {
622
      g_prefix_error (error, _("Could not read header: "));
623 624 625
      return FALSE;
    }

626
  g_seekable_seek (G_SEEKABLE (input), 0, G_SEEK_SET, NULL, NULL);
627

628
  if (g_str_has_prefix (header, "# GIMP Curves File\n"))
629
    return gimp_curves_config_load_cruft (config, input, error);
630

631 632 633
  return GIMP_FILTER_TOOL_CLASS (parent_class)->settings_import (filter_tool,
                                                                 input,
                                                                 error);
634 635 636
}

static gboolean
637 638 639
gimp_curves_tool_settings_export (GimpFilterTool  *filter_tool,
                                  GOutputStream   *output,
                                  GError         **error)
640
{
641 642
  GimpCurvesTool   *tool   = GIMP_CURVES_TOOL (filter_tool);
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (filter_tool->config);
643

644
  if (tool->export_old_format)
645
    return gimp_curves_config_save_cruft (config, output, error);
646

647 648 649
  return GIMP_FILTER_TOOL_CLASS (parent_class)->settings_export (filter_tool,
                                                                 output,
                                                                 error);
650 651
}

652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681
static void
gimp_curves_tool_color_picked (GimpFilterTool *filter_tool,
                               gpointer        identifier,
                               gdouble         x,
                               gdouble         y,
                               const Babl     *sample_format,
                               const GimpRGB  *color)
{
  GimpCurvesTool   *tool   = GIMP_CURVES_TOOL (filter_tool);
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (filter_tool->config);
  GimpDrawable     *drawable;

  drawable = GIMP_FILTER_TOOL (tool)->drawable;

  tool->picked_color[GIMP_HISTOGRAM_RED]   = color->r;
  tool->picked_color[GIMP_HISTOGRAM_GREEN] = color->g;
  tool->picked_color[GIMP_HISTOGRAM_BLUE]  = color->b;

  if (gimp_drawable_has_alpha (drawable))
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = color->a;
  else
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = -1;

  tool->picked_color[GIMP_HISTOGRAM_VALUE] = MAX (MAX (color->r, color->g),
                                                  color->b);

  gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
                            tool->picked_color[config->channel]);
}

682 683 684 685 686 687 688
static void
gimp_curves_tool_export_setup (GimpSettingsBox      *settings_box,
                               GtkFileChooserDialog *dialog,
                               gboolean              export,
                               GimpCurvesTool       *tool)
{
  GtkWidget *button;
689

690 691 692 693 694 695 696 697
  if (! export)
    return;

  button = gtk_check_button_new_with_mnemonic (_("Use _old curves file format"));
  gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button),
                                tool->export_old_format);
  gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (dialog), button);
  gtk_widget_show (button);
698

699 700 701
  g_signal_connect (button, "toggled",
                    G_CALLBACK (gimp_toggle_button_update),
                    &tool->export_old_format);
702
}
703

Elliot Lee's avatar
Elliot Lee committed
704
static void
705
gimp_curves_tool_update_channel (GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
706
{
707 708 709
  GimpFilterTool       *filter_tool = GIMP_FILTER_TOOL (tool);
  GimpCurvesConfig     *config      = GIMP_CURVES_CONFIG (filter_tool->config);
  GimpCurve            *curve       = config->curve[config->channel];
710
  GimpHistogramChannel  channel;
711

712 713
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                 config->channel);
714

715
  switch (config->channel)
716
    {
717 718 719 720 721 722 723
      guchar r[256];
      guchar g[256];
      guchar b[256];

    case GIMP_HISTOGRAM_VALUE:
    case GIMP_HISTOGRAM_ALPHA:
    case GIMP_HISTOGRAM_RGB:
724
    case GIMP_HISTOGRAM_LUMINANCE:
725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752
      gimp_curve_get_uchar (curve, sizeof (r), r);

      gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
                                  r, r, r);
      break;

    case GIMP_HISTOGRAM_RED:
    case GIMP_HISTOGRAM_GREEN:
    case GIMP_HISTOGRAM_BLUE:
      gimp_curve_get_uchar (config->curve[GIMP_HISTOGRAM_RED],
                            sizeof (r), r);
      gimp_curve_get_uchar (config->curve[GIMP_HISTOGRAM_GREEN],
                            sizeof (g), g);
      gimp_curve_get_uchar (config->curve[GIMP_HISTOGRAM_BLUE],
                            sizeof (b), b);

      gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
                                  r, g, b);
      break;
    }

  gimp_histogram_view_set_channel (GIMP_HISTOGRAM_VIEW (tool->graph),
                                   config->channel);
  gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
                            tool->picked_color[config->channel]);

  gimp_color_bar_set_channel (GIMP_COLOR_BAR (tool->yrange),
                              config->channel);
753

754
  gimp_curve_view_remove_all_backgrounds (GIMP_CURVE_VIEW (tool->graph));
755

756 757 758 759
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
760 761 762 763 764
      GimpRGB  curve_color;
      gboolean has_color;

      has_color = curves_get_channel_color (tool->graph, channel, &curve_color);

765
      if (channel == config->channel)
766
        {
767
          gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph), curve,
768
                                     has_color ? &curve_color : NULL);
769
        }
770 771 772 773
      else
        {
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[channel],
774
                                          has_color ? &curve_color : NULL);
775 776
        }
    }
777

778 779 780
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                 curve->curve_type);
}
Elliot Lee's avatar
Elliot Lee committed
781

782 783 784 785
static void
curves_channel_callback (GtkWidget      *widget,
                         GimpCurvesTool *tool)
{
786 787
  GimpFilterTool   *filter_tool = GIMP_FILTER_TOOL (tool);
  GimpCurvesConfig *config      = GIMP_CURVES_CONFIG (filter_tool->config);
788 789 790 791 792 793 794 795
  gint              value;

  if (gimp_int_combo_box_get_active (GIMP_INT_COMBO_BOX (widget), &value) &&
      config->channel != value)
    {
      g_object_set (config,
                    "channel", value,
                    NULL);
796
    }
Elliot Lee's avatar
Elliot Lee committed
797 798 799
}

static void
800
curves_channel_reset_callback (GtkWidget      *widget,
801
                               GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
802
{
803 804
  GimpFilterTool   *filter_tool = GIMP_FILTER_TOOL (tool);
  GimpCurvesConfig *config      = GIMP_CURVES_CONFIG (filter_tool->config);
805 806

  gimp_curve_reset (config->curve[config->channel], FALSE);
Elliot Lee's avatar
Elliot Lee committed
807 808
}

809
static gboolean
810 811
curves_menu_sensitivity (gint      value,
                         gpointer  data)
812
{
813
  GimpDrawable         *drawable = GIMP_FILTER_TOOL (data)->drawable;
814
  GimpHistogramChannel  channel  = value;
Sven Neumann's avatar
Sven Neumann committed
815

816 817 818
  if (!drawable)
    return FALSE;

819 820 821 822
  switch (channel)
    {
    case GIMP_HISTOGRAM_VALUE:
      return TRUE;
823

824 825 826
    case GIMP_HISTOGRAM_RED:
    case GIMP_HISTOGRAM_GREEN:
    case GIMP_HISTOGRAM_BLUE:
827
      return gimp_drawable_is_rgb (drawable);
828

829
    case GIMP_HISTOGRAM_ALPHA:
830
      return gimp_drawable_has_alpha (drawable);
831 832 833

    case GIMP_HISTOGRAM_RGB:
      return FALSE;
834 835 836

    case GIMP_HISTOGRAM_LUMINANCE:
      return FALSE;
837
    }
838

839 840 841
  return FALSE;
}

Elliot Lee's avatar
Elliot Lee committed
842
static void
843
curves_curve_type_callback (GtkWidget      *widget,
844
                            GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
845
{
846
  gint value;
Elliot Lee's avatar
Elliot Lee committed
847

848 849
  if (gimp_int_combo_box_get_active (GIMP_INT_COMBO_BOX (widget), &value))
    {
850 851 852
      GimpFilterTool   *filter_tool = GIMP_FILTER_TOOL (tool);
      GimpCurvesConfig *config      = GIMP_CURVES_CONFIG (filter_tool->config);
      GimpCurveType     curve_type  = value;
Elliot Lee's avatar
Elliot Lee committed
853

854
      if (config->curve[config->channel]->curve_type != curve_type)
855
        gimp_curve_set_curve_type (config->curve[config->channel], curve_type);
856
    }
Elliot Lee's avatar
Elliot Lee committed
857
}
858

859 860 861 862
static gboolean
curves_get_channel_color (GtkWidget            *widget,
                          GimpHistogramChannel  channel,
                          GimpRGB              *color)