gimpcurvestool.c 31.5 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
static void       gimp_curves_tool_constructed    (GObject              *object);

67 68 69 70
static gboolean   gimp_curves_tool_initialize     (GimpTool             *tool,
                                                   GimpDisplay          *display,
                                                   GError              **error);
static void       gimp_curves_tool_button_release (GimpTool             *tool,
71
                                                   const GimpCoords     *coords,
72 73 74 75 76 77 78 79
                                                   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,
80
                                                   const GimpCoords     *coords,
81 82 83 84 85 86
                                                   GdkModifierType       state,
                                                   gboolean              proximity,
                                                   GimpDisplay          *display);

static void       gimp_curves_tool_color_picked   (GimpColorTool        *color_tool,
                                                   GimpColorPickState    pick_state,
87 88
                                                   gdouble               x,
                                                   gdouble               y,
89
                                                   const Babl           *sample_format,
90 91
                                                   gpointer              pixel,
                                                   const GimpRGB        *color);
92 93 94 95 96 97 98 99 100 101

static gchar    * gimp_curves_tool_get_operation  (GimpImageMapTool     *im_tool,
                                                   gchar               **title,
                                                   gchar               **description,
                                                   gchar               **undo_desc,
                                                   gchar               **icon_name,
                                                   gchar               **help_id);
static void       gimp_curves_tool_dialog         (GimpImageMapTool     *im_tool);
static void       gimp_curves_tool_reset          (GimpImageMapTool     *im_tool);
static gboolean   gimp_curves_tool_settings_import(GimpImageMapTool     *im_tool,
102
                                                   GInputStream         *input,
103
                                                   GError              **error);
104
static gboolean   gimp_curves_tool_settings_export(GimpImageMapTool     *im_tool,
105
                                                   GOutputStream        *output,
106
                                                   GError              **error);
107

108 109 110 111
static void       gimp_curves_tool_export_setup   (GimpSettingsBox      *settings_box,
                                                   GtkFileChooserDialog *dialog,
                                                   gboolean              export,
                                                   GimpCurvesTool       *tool);
112
static void       gimp_curves_tool_update_channel (GimpCurvesTool       *tool);
113 114
static void       gimp_curves_tool_config_notify  (GObject              *object,
                                                   GParamSpec           *pspec,
115
                                                   GimpCurvesTool       *tool);
116

117 118 119 120 121 122 123 124 125 126
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);
127

128 129
static const GimpRGB * curves_get_channel_color   (GimpHistogramChannel  channel);

Michael Natterer's avatar
Michael Natterer committed
130

131
G_DEFINE_TYPE (GimpCurvesTool, gimp_curves_tool, GIMP_TYPE_IMAGE_MAP_TOOL)
132 133

#define parent_class gimp_curves_tool_parent_class
134

Michael Natterer's avatar
Michael Natterer committed
135

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

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

155 156 157

/*  private functions  */

Michael Natterer's avatar
Michael Natterer committed
158 159 160
static void
gimp_curves_tool_class_init (GimpCurvesToolClass *klass)
{
161
  GObjectClass          *object_class     = G_OBJECT_CLASS (klass);
162 163 164 165
  GimpToolClass         *tool_class       = GIMP_TOOL_CLASS (klass);
  GimpColorToolClass    *color_tool_class = GIMP_COLOR_TOOL_CLASS (klass);
  GimpImageMapToolClass *im_tool_class    = GIMP_IMAGE_MAP_TOOL_CLASS (klass);

166 167
  object_class->constructed          = gimp_curves_tool_constructed;

168 169 170 171
  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;
172

173
  color_tool_class->picked           = gimp_curves_tool_color_picked;
174

175 176 177
  im_tool_class->settings_name       = "curves";
  im_tool_class->import_dialog_title = _("Import Curves");
  im_tool_class->export_dialog_title = _("Export Curves");
178

179 180 181 182 183
  im_tool_class->get_operation       = gimp_curves_tool_get_operation;
  im_tool_class->dialog              = gimp_curves_tool_dialog;
  im_tool_class->reset               = gimp_curves_tool_reset;
  im_tool_class->settings_import     = gimp_curves_tool_settings_import;
  im_tool_class->settings_export     = gimp_curves_tool_settings_export;
Michael Natterer's avatar
Michael Natterer committed
184 185 186
}

static void
187
gimp_curves_tool_init (GimpCurvesTool *tool)
Michael Natterer's avatar
Michael Natterer committed
188
{
189
  gint i;
190

191 192
  for (i = 0; i < G_N_ELEMENTS (tool->picked_color); i++)
    tool->picked_color[i] = -1.0;
Michael Natterer's avatar
Michael Natterer committed
193 194
}

195 196 197 198 199
static void
gimp_curves_tool_constructed (GObject *object)
{
  G_OBJECT_CLASS (parent_class)->constructed (object);

200 201 202 203
  g_signal_connect_object (GIMP_IMAGE_MAP_TOOL (object)->config, "notify",
                           G_CALLBACK (gimp_curves_tool_config_notify),
                           object, 0);

204 205 206 207 208 209
  /*  always pick colors  */
  gimp_color_tool_enable (GIMP_COLOR_TOOL (object),
                          GIMP_COLOR_TOOL_GET_OPTIONS (object));

}

210
static gboolean
211 212 213
gimp_curves_tool_initialize (GimpTool     *tool,
                             GimpDisplay  *display,
                             GError      **error)
Michael Natterer's avatar
Michael Natterer committed
214
{
215
  GimpCurvesTool *c_tool   = GIMP_CURVES_TOOL (tool);
216 217
  GimpImage      *image    = gimp_display_get_image (display);
  GimpDrawable   *drawable = gimp_image_get_active_drawable (image);
218
  GimpHistogram  *histogram;
219

220 221 222 223
  if (! GIMP_TOOL_CLASS (parent_class)->initialize (tool, display, error))
    {
      return FALSE;
    }
224

225
  gimp_int_combo_box_set_sensitivity (GIMP_INT_COMBO_BOX (c_tool->channel_menu),
226
                                      curves_menu_sensitivity, drawable, NULL);
227

228
  histogram = gimp_histogram_new (TRUE);
229
  gimp_drawable_calculate_histogram (drawable, histogram);
230
  gimp_histogram_view_set_background (GIMP_HISTOGRAM_VIEW (c_tool->graph),
231
                                      histogram);
232
  g_object_unref (histogram);
233

234
  if (gimp_drawable_get_component_type (drawable) == GIMP_COMPONENT_TYPE_U8)
235 236 237 238 239 240
    {
      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
    {
241 242
      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);
243 244
    }

245
  return TRUE;
Michael Natterer's avatar
Michael Natterer committed
246 247 248
}

static void
249
gimp_curves_tool_button_release (GimpTool              *tool,
250
                                 const GimpCoords      *coords,
251 252 253 254
                                 guint32                time,
                                 GdkModifierType        state,
                                 GimpButtonReleaseType  release_type,
                                 GimpDisplay           *display)
Michael Natterer's avatar
Michael Natterer committed
255
{
256 257 258
  GimpCurvesTool   *c_tool  = GIMP_CURVES_TOOL (tool);
  GimpImageMapTool *im_tool = GIMP_IMAGE_MAP_TOOL (tool);
  GimpCurvesConfig *config  = GIMP_CURVES_CONFIG (im_tool->config);
Michael Natterer's avatar
Michael Natterer committed
259

260
  if (state & gimp_get_extend_selection_mask ())
Michael Natterer's avatar
Michael Natterer committed
261
    {
262
      GimpCurve *curve = config->curve[config->channel];
263
      gdouble    value = c_tool->picked_color[config->channel];
264
      gint       closest;
265

266
      closest = gimp_curve_get_closest_point (curve, value);
267

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

271
      gimp_curve_set_point (curve, closest,
272
                            value, gimp_curve_map_value (curve, value));
Michael Natterer's avatar
Michael Natterer committed
273
    }
274
  else if (state & gimp_get_toggle_behavior_mask ())
Michael Natterer's avatar
Michael Natterer committed
275
    {
276
      GimpHistogramChannel channel;
277

278 279 280
      for (channel = GIMP_HISTOGRAM_VALUE;
           channel <= GIMP_HISTOGRAM_ALPHA;
           channel++)
281
        {
282 283
          GimpCurve *curve = config->curve[channel];
          gdouble    value = c_tool->picked_color[channel];
284
          gint       closest;
285

286 287 288
          if (value != -1)
            {
              closest = gimp_curve_get_closest_point (curve, value);
289

290 291
              gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
                                            closest);
292

293 294 295
              gimp_curve_set_point (curve, closest,
                                    value, gimp_curve_map_value (curve, value));
            }
296
        }
Michael Natterer's avatar
Michael Natterer committed
297 298
    }

299
  /*  chain up to halt the tool */
300 301
  GIMP_TOOL_CLASS (parent_class)->button_release (tool, coords, time, state,
                                                  release_type, display);
Michael Natterer's avatar
Michael Natterer committed
302 303
}

304
static gboolean
305 306
gimp_curves_tool_key_press (GimpTool    *tool,
                            GdkEventKey *kevent,
307
                            GimpDisplay *display)
308
{
309 310
  GimpCurvesTool *c_tool = GIMP_CURVES_TOOL (tool);

311 312 313 314 315
  if (tool->display && c_tool->graph)
    {
      if (gtk_widget_event (c_tool->graph, (GdkEvent *) kevent))
        return TRUE;
    }
316 317

  return GIMP_TOOL_CLASS (parent_class)->key_press (tool, kevent, display);
318 319 320
}

static void
321 322 323 324 325
gimp_curves_tool_oper_update (GimpTool         *tool,
                              const GimpCoords *coords,
                              GdkModifierType   state,
                              gboolean          proximity,
                              GimpDisplay      *display)
326
{
327 328
  if (gimp_image_map_tool_on_guide (GIMP_IMAGE_MAP_TOOL (tool),
                                    coords, display))
329
    {
330 331
      GIMP_TOOL_CLASS (parent_class)->oper_update (tool, coords, state, proximity,
                                                   display);
332
    }
333 334
  else
    {
335 336 337 338
      GimpColorPickMode  mode;
      gchar             *status      = NULL;
      GdkModifierType    extend_mask = gimp_get_extend_selection_mask ();
      GdkModifierType    toggle_mask = gimp_get_toggle_behavior_mask ();
339

340
      gimp_tool_pop_status (tool, display);
341

342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360
      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);
        }
361

362 363 364 365 366 367 368
      GIMP_COLOR_TOOL (tool)->pick_mode = mode;

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

      g_free (status);
    }
369 370
}

Michael Natterer's avatar
Michael Natterer committed
371
static void
372 373
gimp_curves_tool_color_picked (GimpColorTool      *color_tool,
                               GimpColorPickState  pick_state,
374 375
                               gdouble             x,
                               gdouble             y,
376
                               const Babl         *sample_format,
377 378
                               gpointer            pixel,
                               const GimpRGB      *color)
Michael Natterer's avatar
Michael Natterer committed
379
{
380 381 382 383
  GimpCurvesTool   *tool    = GIMP_CURVES_TOOL (color_tool);
  GimpImageMapTool *im_tool = GIMP_IMAGE_MAP_TOOL (color_tool);
  GimpCurvesConfig *config  = GIMP_CURVES_CONFIG (im_tool->config);
  GimpDrawable     *drawable;
Michael Natterer's avatar
Michael Natterer committed
384

385
  drawable = GIMP_IMAGE_MAP_TOOL (tool)->drawable;
386

387 388 389
  tool->picked_color[GIMP_HISTOGRAM_RED]   = color->r;
  tool->picked_color[GIMP_HISTOGRAM_GREEN] = color->g;
  tool->picked_color[GIMP_HISTOGRAM_BLUE]  = color->b;
390

391
  if (gimp_drawable_has_alpha (drawable))
392
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = color->a;
393 394
  else
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = -1;
395

396 397
  tool->picked_color[GIMP_HISTOGRAM_VALUE] = MAX (MAX (color->r, color->g),
                                                  color->b);
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
398

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

403 404 405 406 407 408 409
static gchar *
gimp_curves_tool_get_operation (GimpImageMapTool  *im_tool,
                                gchar            **title,
                                gchar            **description,
                                gchar            **undo_desc,
                                gchar            **icon_name,
                                gchar            **help_id)
410
{
411
  *description = g_strdup (_("Adjust Color Curves"));
412

413
  return g_strdup ("gimp:curves");
414
}
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
415

416

417 418 419
/*******************/
/*  Curves dialog  */
/*******************/
Elliot Lee's avatar
Elliot Lee committed
420

421
static void
422
gimp_curves_tool_dialog (GimpImageMapTool *im_tool)
Elliot Lee's avatar
Elliot Lee committed
423
{
424 425 426
  GimpCurvesTool   *tool         = GIMP_CURVES_TOOL (im_tool);
  GimpToolOptions  *tool_options = GIMP_TOOL_GET_OPTIONS (im_tool);
  GimpCurvesConfig *config       = GIMP_CURVES_CONFIG (im_tool->config);
427
  GtkListStore     *store;
428
  GtkWidget        *main_vbox;
429
  GtkWidget        *frame_vbox;
430 431 432 433
  GtkWidget        *vbox;
  GtkWidget        *hbox;
  GtkWidget        *hbox2;
  GtkWidget        *label;
434
  GtkWidget        *main_frame;
435 436 437 438
  GtkWidget        *frame;
  GtkWidget        *table;
  GtkWidget        *button;
  GtkWidget        *bar;
439
  GtkWidget        *combo;
440

441
  g_signal_connect (im_tool->settings_box, "file-dialog-setup",
442
                    G_CALLBACK (gimp_curves_tool_export_setup),
443
                    im_tool);
444

445
  main_vbox = gimp_image_map_tool_dialog_get_vbox (im_tool);
Elliot Lee's avatar
Elliot Lee committed
446

447
  /*  The combo box for selecting channels  */
448 449 450 451
  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
452
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
453
  gtk_frame_set_label_widget (GTK_FRAME (main_frame), hbox);
454 455
  gtk_widget_show (hbox);

456
  label = gtk_label_new_with_mnemonic (_("Cha_nnel:"));
457 458 459
  gimp_label_set_attributes (GTK_LABEL (label),
                             PANGO_ATTR_WEIGHT, PANGO_WEIGHT_BOLD,
                             -1);
460 461 462
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);

463 464 465
  store = gimp_enum_store_new_with_range (GIMP_TYPE_HISTOGRAM_CHANNEL,
                                          GIMP_HISTOGRAM_VALUE,
                                          GIMP_HISTOGRAM_ALPHA);
466 467
  tool->channel_menu =
    gimp_enum_combo_box_new_with_model (GIMP_ENUM_STORE (store));
468 469
  g_object_unref (store);

470 471
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                 config->channel);
472 473
  gimp_enum_combo_box_set_icon_prefix (GIMP_ENUM_COMBO_BOX (tool->channel_menu),
                                       "gimp-channel");
474 475
  gtk_box_pack_start (GTK_BOX (hbox), tool->channel_menu, FALSE, FALSE, 0);
  gtk_widget_show (tool->channel_menu);
476

477 478 479
  g_signal_connect (tool->channel_menu, "changed",
                    G_CALLBACK (curves_channel_callback),
                    tool);
480

481
  gtk_label_set_mnemonic_widget (GTK_LABEL (label), tool->channel_menu);
482

483
  button = gtk_button_new_with_mnemonic (_("R_eset Channel"));
484
  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
485
  gtk_widget_show (button);
486

487
  g_signal_connect (button, "clicked",
488
                    G_CALLBACK (curves_channel_reset_callback),
489
                    tool);
Elliot Lee's avatar
Elliot Lee committed
490

491
  /*  The histogram scale radio buttons  */
492 493 494
  hbox2 = gimp_prop_enum_icon_box_new (G_OBJECT (tool_options),
                                       "histogram-scale", "gimp-histogram",
                                       0, 0);
495 496
  gtk_box_pack_end (GTK_BOX (hbox), hbox2, FALSE, FALSE, 0);
  gtk_widget_show (hbox2);
497

498 499 500 501
  frame_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 4);
  gtk_container_add (GTK_CONTAINER (main_frame), frame_vbox);
  gtk_widget_show (frame_vbox);

502
  /*  The table for the color bars and the graph  */
Elliot Lee's avatar
Elliot Lee committed
503
  table = gtk_table_new (2, 2, FALSE);
504 505
  gtk_table_set_col_spacings (GTK_TABLE (table), 2);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
506
  gtk_box_pack_start (GTK_BOX (frame_vbox), table, TRUE, TRUE, 0);
507 508

  /*  The left color bar  */
Michael Natterer's avatar
Michael Natterer committed
509
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
510
  gtk_table_attach (GTK_TABLE (table), vbox, 0, 1, 0, 1,
511
                    GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
512
  gtk_widget_show (vbox);
Elliot Lee's avatar
Elliot Lee committed
513 514

  frame = gtk_frame_new (NULL);
515
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
516
  gtk_box_pack_start (GTK_BOX (vbox), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
517 518
  gtk_widget_show (frame);

519 520 521 522
  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);
523

Elliot Lee's avatar
Elliot Lee committed
524 525
  /*  The curves graph  */
  frame = gtk_frame_new (NULL);
526
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
527
  gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 0, 1,
528
                    GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
529
  gtk_widget_show (frame);
Elliot Lee's avatar
Elliot Lee committed
530

531
  tool->graph = gimp_curve_view_new ();
532 533
  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);
534 535 536 537
  gtk_widget_set_size_request (tool->graph,
                               GRAPH_SIZE + RADIUS * 2,
                               GRAPH_SIZE + RADIUS * 2);
  g_object_set (tool->graph,
538 539 540
                "border-width", RADIUS,
                "subdivisions", 1,
                NULL);
541 542
  gtk_container_add (GTK_CONTAINER (frame), tool->graph);
  gtk_widget_show (tool->graph);
543

544
  gimp_histogram_options_connect_view (GIMP_HISTOGRAM_OPTIONS (tool_options),
545 546 547
                                       GIMP_HISTOGRAM_VIEW (tool->graph));

  /*  The bottom color bar  */
Michael Natterer's avatar
Michael Natterer committed
548
  hbox2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
549
  gtk_table_attach (GTK_TABLE (table), hbox2, 1, 2, 1, 2,
550
                    GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
551
  gtk_widget_show (hbox2);
552

Elliot Lee's avatar
Elliot Lee committed
553
  frame = gtk_frame_new (NULL);
554
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
555
  gtk_box_pack_start (GTK_BOX (hbox2), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
556 557
  gtk_widget_show (frame);

Michael Natterer's avatar
Michael Natterer committed
558 559
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
  gtk_box_set_homogeneous (GTK_BOX (vbox), TRUE);
560 561
  gtk_container_add (GTK_CONTAINER (frame), vbox);
  gtk_widget_show (vbox);
562

563 564
  tool->xrange = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
  gtk_widget_set_size_request (tool->xrange, -1, BAR_SIZE / 2);
565
  gtk_box_pack_start (GTK_BOX (vbox), tool->xrange, TRUE, TRUE, 0);
566
  gtk_widget_show (tool->xrange);
567

568
  bar = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
569
  gtk_box_pack_start (GTK_BOX (vbox), bar, TRUE, TRUE, 0);
570 571
  gtk_widget_show (bar);

572
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
573

Michael Natterer's avatar
Michael Natterer committed
574
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
575
  gtk_box_pack_end (GTK_BOX (frame_vbox), hbox, FALSE, FALSE, 0);
576
  gtk_widget_show (hbox);
577

578 579 580
  label = gtk_label_new_with_mnemonic (_("Curve _type:"));
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);
581

582
  tool->curve_type = combo = gimp_enum_combo_box_new (GIMP_TYPE_CURVE_TYPE);
583 584
  gimp_enum_combo_box_set_icon_prefix (GIMP_ENUM_COMBO_BOX (combo),
                                       "gimp-curve");
585 586 587 588 589
  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
590 591

  gtk_label_set_mnemonic_widget (GTK_LABEL (label), combo);
592 593

  gimp_curves_tool_update_channel (tool);
Elliot Lee's avatar
Elliot Lee committed
594 595
}

596
static void
597
gimp_curves_tool_reset (GimpImageMapTool *im_tool)
598
{
599
  GimpCurvesConfig     *config = GIMP_CURVES_CONFIG (im_tool->config);
600
  GimpCurvesConfig     *default_config;
601
  GimpHistogramChannel  channel;
602

603
  default_config = GIMP_CURVES_CONFIG (im_tool->default_config);
604

605 606 607 608
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
609 610
      if (default_config)
        {
611
          GimpCurveType curve_type = config->curve[channel]->curve_type;
612

613
          g_object_freeze_notify (G_OBJECT (config->curve[channel]));
614 615

          gimp_config_copy (GIMP_CONFIG (default_config->curve[channel]),
616
                            GIMP_CONFIG (config->curve[channel]),
617 618
                            0);

619
          g_object_set (config->curve[channel],
620 621 622
                        "curve-type", curve_type,
                        NULL);

623
          g_object_thaw_notify (G_OBJECT (config->curve[channel]));
624 625 626
        }
      else
        {
627
          gimp_curve_reset (config->curve[channel], FALSE);
628
        }
629
    }
630 631
}

632
static gboolean
633
gimp_curves_tool_settings_import (GimpImageMapTool  *im_tool,
634
                                  GInputStream      *input,
635
                                  GError           **error)
636
{
637 638 639
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (im_tool->config);
  gchar             header[64];
  gsize             bytes_read;
640

641 642 643
  if (! g_input_stream_read_all (input, header, sizeof (header),
                                 &bytes_read, NULL, error) ||
      bytes_read != sizeof (header))
644
    {
645
      g_prefix_error (error, _("Could not read header: "));
646 647 648
      return FALSE;
    }

649
  g_seekable_seek (G_SEEKABLE (input), 0, G_SEEK_SET, NULL, NULL);
650

651
  if (g_str_has_prefix (header, "# GIMP Curves File\n"))
652
    return gimp_curves_config_load_cruft (config, input, error);
653

654
  return GIMP_IMAGE_MAP_TOOL_CLASS (parent_class)->settings_import (im_tool,
655
                                                                    input,
656
                                                                    error);
657 658 659
}

static gboolean
660
gimp_curves_tool_settings_export (GimpImageMapTool  *im_tool,
661
                                  GOutputStream     *output,
662
                                  GError           **error)
663
{
664 665
  GimpCurvesTool   *tool   = GIMP_CURVES_TOOL (im_tool);
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (im_tool->config);
666

667
  if (tool->export_old_format)
668
    return gimp_curves_config_save_cruft (config, output, error);
669

670
  return GIMP_IMAGE_MAP_TOOL_CLASS (parent_class)->settings_export (im_tool,
671
                                                                    output,
672 673 674 675 676 677 678 679 680 681
                                                                    error);
}

static void
gimp_curves_tool_export_setup (GimpSettingsBox      *settings_box,
                               GtkFileChooserDialog *dialog,
                               gboolean              export,
                               GimpCurvesTool       *tool)
{
  GtkWidget *button;
682

683 684 685 686 687 688 689 690
  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);
691

692 693 694
  g_signal_connect (button, "toggled",
                    G_CALLBACK (gimp_toggle_button_update),
                    &tool->export_old_format);
695
}
696

Elliot Lee's avatar
Elliot Lee committed
697
static void
698
gimp_curves_tool_update_channel (GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
699
{
700 701 702
  GimpImageMapTool     *im_tool = GIMP_IMAGE_MAP_TOOL (tool);
  GimpCurvesConfig     *config  = GIMP_CURVES_CONFIG (im_tool->config);
  GimpCurve            *curve   = config->curve[config->channel];
703
  GimpHistogramChannel  channel;
704

705 706
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                 config->channel);
707

708
  switch (config->channel)
709
    {
710 711 712 713 714 715 716
      guchar r[256];
      guchar g[256];
      guchar b[256];

    case GIMP_HISTOGRAM_VALUE:
    case GIMP_HISTOGRAM_ALPHA:
    case GIMP_HISTOGRAM_RGB:
717
    case GIMP_HISTOGRAM_LUMINANCE:
718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745
      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);
746

747
  gimp_curve_view_remove_all_backgrounds (GIMP_CURVE_VIEW (tool->graph));
748

749 750 751 752 753
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
      if (channel == config->channel)
754
        {
755 756
          gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph), curve,
                                     curves_get_channel_color (channel));
757
        }
758 759 760 761 762 763 764
      else
        {
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[channel],
                                          curves_get_channel_color (channel));
        }
    }
765

766 767 768
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                 curve->curve_type);
}
Elliot Lee's avatar
Elliot Lee committed
769

770 771 772 773 774 775 776
static void
gimp_curves_tool_config_notify (GObject        *object,
                                GParamSpec     *pspec,
                                GimpCurvesTool *tool)
{
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (object);
  GimpCurve        *curve  = config->curve[config->channel];
777

778 779
  if (! tool->xrange)
    return;
780

781 782 783
  if (! strcmp (pspec->name, "channel"))
    {
      gimp_curves_tool_update_channel (GIMP_CURVES_TOOL (tool));
784 785 786
    }
  else if (! strcmp (pspec->name, "curve"))
    {
787 788
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                     curve->curve_type);
789 790
    }
}
791

792 793 794 795
static void
curves_channel_callback (GtkWidget      *widget,
                         GimpCurvesTool *tool)
{
796 797
  GimpImageMapTool *im_tool = GIMP_IMAGE_MAP_TOOL (tool);
  GimpCurvesConfig *config  = GIMP_CURVES_CONFIG (im_tool->config);
798 799 800 801 802 803 804 805
  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);
806
    }
Elliot Lee's avatar
Elliot Lee committed
807 808 809
}

static void
810
curves_channel_reset_callback (GtkWidget      *widget,
811
                               GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
812
{