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

static void       gimp_curves_tool_color_picked   (GimpColorTool        *color_tool,
                                                   GimpColorPickState    pick_state,
85 86
                                                   gdouble               x,
                                                   gdouble               y,
87
                                                   const Babl           *sample_format,
88
                                                   const GimpRGB        *color,
89
                                                   gint                  color_index);
90
static GeglNode * gimp_curves_tool_get_operation  (GimpImageMapTool     *image_map_tool,
91 92
                                                   GObject             **config,
                                                   gchar               **undo_desc);
93 94
static void       gimp_curves_tool_dialog         (GimpImageMapTool     *image_map_tool);
static void       gimp_curves_tool_reset          (GimpImageMapTool     *image_map_tool);
95
static gboolean   gimp_curves_tool_settings_import(GimpImageMapTool     *image_map_tool,
96
                                                   const gchar          *filename,
97
                                                   GError              **error);
98
static gboolean   gimp_curves_tool_settings_export(GimpImageMapTool     *image_map_tool,
99 100
                                                   const gchar          *filename,
                                                   GError              **error);
101

102 103 104 105
static void       gimp_curves_tool_export_setup   (GimpSettingsBox      *settings_box,
                                                   GtkFileChooserDialog *dialog,
                                                   gboolean              export,
                                                   GimpCurvesTool       *tool);
106 107
static void       gimp_curves_tool_config_notify  (GObject              *object,
                                                   GParamSpec           *pspec,
108
                                                   GimpCurvesTool       *tool);
109

110 111 112 113 114 115 116 117 118 119
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);
120

Michael Natterer's avatar
Michael Natterer committed
121

122
G_DEFINE_TYPE (GimpCurvesTool, gimp_curves_tool, GIMP_TYPE_IMAGE_MAP_TOOL)
123 124

#define parent_class gimp_curves_tool_parent_class
125

126 127 128 129 130 131 132 133 134
static GimpRGB channel_colors[GIMP_HISTOGRAM_RGB] =
{
  { 0.0, 0.0, 0.0, 1.0 },
  { 1.0, 0.0, 0.0, 1.0 },
  { 0.0, 1.0, 0.0, 1.0 },
  { 0.0, 0.0, 1.0, 1.0 },
  { 0.5, 0.5, 0.5, 1.0 }
};

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 162 163 164
  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);

165 166 167 168
  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;
169

170
  color_tool_class->picked           = gimp_curves_tool_color_picked;
171

172
  im_tool_class->dialog_desc         = _("Adjust Color Curves");
173 174 175
  im_tool_class->settings_name       = "curves";
  im_tool_class->import_dialog_title = _("Import Curves");
  im_tool_class->export_dialog_title = _("Export Curves");
176

177 178 179 180 181
  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
182 183 184
}

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

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

193
static gboolean
194 195 196
gimp_curves_tool_initialize (GimpTool     *tool,
                             GimpDisplay  *display,
                             GError      **error)
Michael Natterer's avatar
Michael Natterer committed
197
{
198
  GimpCurvesTool *c_tool   = GIMP_CURVES_TOOL (tool);
199 200
  GimpImage      *image    = gimp_display_get_image (display);
  GimpDrawable   *drawable = gimp_image_get_active_drawable (image);
201
  GimpHistogram  *histogram;
202

203 204 205 206
  if (! GIMP_TOOL_CLASS (parent_class)->initialize (tool, display, error))
    {
      return FALSE;
    }
207

208 209
  /*  always pick colors  */
  gimp_color_tool_enable (GIMP_COLOR_TOOL (tool),
210
                          GIMP_COLOR_TOOL_GET_OPTIONS (tool));
211

212
  gimp_int_combo_box_set_sensitivity (GIMP_INT_COMBO_BOX (c_tool->channel_menu),
213
                                      curves_menu_sensitivity, drawable, NULL);
214

215
  histogram = gimp_histogram_new (TRUE);
216
  gimp_drawable_calculate_histogram (drawable, histogram);
217
  gimp_histogram_view_set_background (GIMP_HISTOGRAM_VIEW (c_tool->graph),
218
                                      histogram);
219
  g_object_unref (histogram);
220

221 222 223 224 225 226 227 228 229 230 231
  if (gimp_drawable_get_precision (drawable) == GIMP_PRECISION_U8)
    {
      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
    {
      gimp_curve_view_set_range_x (GIMP_CURVE_VIEW (c_tool->graph), 0.0, 1.0);
      gimp_curve_view_set_range_y (GIMP_CURVE_VIEW (c_tool->graph), 0.0, 1.0);
    }

232
  return TRUE;
Michael Natterer's avatar
Michael Natterer committed
233 234 235
}

static void
236
gimp_curves_tool_button_release (GimpTool              *tool,
237
                                 const GimpCoords      *coords,
238 239 240 241
                                 guint32                time,
                                 GdkModifierType        state,
                                 GimpButtonReleaseType  release_type,
                                 GimpDisplay           *display)
Michael Natterer's avatar
Michael Natterer committed
242
{
243 244
  GimpCurvesTool   *c_tool = GIMP_CURVES_TOOL (tool);
  GimpCurvesConfig *config = c_tool->config;
Michael Natterer's avatar
Michael Natterer committed
245

246
  if (state & GDK_SHIFT_MASK)
Michael Natterer's avatar
Michael Natterer committed
247
    {
248
      GimpCurve *curve = config->curve[config->channel];
249
      gdouble    value = c_tool->picked_color[config->channel];
250
      gint       closest;
251

252
      closest = gimp_curve_get_closest_point (curve, value);
253

254
      gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
255
                                    closest);
256

257
      gimp_curve_set_point (curve, closest,
258
                            value, gimp_curve_map_value (curve, value));
Michael Natterer's avatar
Michael Natterer committed
259
    }
260
  else if (state & gimp_get_toggle_behavior_mask ())
Michael Natterer's avatar
Michael Natterer committed
261
    {
262
      gint i;
263

264 265
      for (i = 0; i < 5; i++)
        {
266
          GimpCurve *curve = config->curve[i];
267
          gdouble    value = c_tool->picked_color[i];
268
          gint       closest;
269

270
          closest = gimp_curve_get_closest_point (curve, value);
271

272
          gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
273
                                        closest);
274

275
          gimp_curve_set_point (curve, closest,
276
                                value, gimp_curve_map_value (curve, value));
277
        }
Michael Natterer's avatar
Michael Natterer committed
278 279
    }

280
  /*  chain up to halt the tool */
281 282
  GIMP_TOOL_CLASS (parent_class)->button_release (tool, coords, time, state,
                                                  release_type, display);
Michael Natterer's avatar
Michael Natterer committed
283 284
}

285
static gboolean
286 287
gimp_curves_tool_key_press (GimpTool    *tool,
                            GdkEventKey *kevent,
288
                            GimpDisplay *display)
289
{
290 291
  GimpCurvesTool *c_tool = GIMP_CURVES_TOOL (tool);

292 293 294 295 296
  if (tool->display && c_tool->graph)
    {
      if (gtk_widget_event (c_tool->graph, (GdkEvent *) kevent))
        return TRUE;
    }
297 298

  return GIMP_TOOL_CLASS (parent_class)->key_press (tool, kevent, display);
299 300 301
}

static void
302 303 304 305 306
gimp_curves_tool_oper_update (GimpTool         *tool,
                              const GimpCoords *coords,
                              GdkModifierType   state,
                              gboolean          proximity,
                              GimpDisplay      *display)
307
{
308 309
  GimpColorPickMode  mode;
  const gchar       *status;
310

311
  GIMP_TOOL_CLASS (parent_class)->oper_update (tool, coords, state, proximity,
312
                                               display);
313

314
  gimp_tool_pop_status (tool, display);
315 316 317 318

  if (state & GDK_SHIFT_MASK)
    {
      mode   = GIMP_COLOR_PICK_MODE_PALETTE;
319
      status = _("Click to add a control point");
320
    }
321
  else if (state & gimp_get_toggle_behavior_mask ())
322 323
    {
      mode   = GIMP_COLOR_PICK_MODE_PALETTE;
324
      status = _("Click to add control points to all channels");
325
    }
326 327 328 329 330
  else
    {
      mode   = GIMP_COLOR_PICK_MODE_NONE;
      status = _("Click to locate on curve (try Shift, Ctrl)");
    }
331 332 333

  GIMP_COLOR_TOOL (tool)->pick_mode = mode;

334
  if (proximity)
335
    gimp_tool_push_status (tool, display, "%s", status);
336 337
}

Michael Natterer's avatar
Michael Natterer committed
338
static void
339 340
gimp_curves_tool_color_picked (GimpColorTool      *color_tool,
                               GimpColorPickState  pick_state,
341 342
                               gdouble             x,
                               gdouble             y,
343
                               const Babl         *sample_format,
344
                               const GimpRGB      *color,
345
                               gint                color_index)
Michael Natterer's avatar
Michael Natterer committed
346
{
347 348
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (color_tool);
  GimpDrawable   *drawable;
Michael Natterer's avatar
Michael Natterer committed
349

350
  drawable = GIMP_IMAGE_MAP_TOOL (tool)->drawable;
351

352 353 354
  tool->picked_color[GIMP_HISTOGRAM_RED]   = color->r;
  tool->picked_color[GIMP_HISTOGRAM_GREEN] = color->g;
  tool->picked_color[GIMP_HISTOGRAM_BLUE]  = color->b;
355

356
  if (gimp_drawable_has_alpha (drawable))
357
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = color->a;
358

359 360
  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
361

362
  gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
363
                            tool->picked_color[tool->config->channel]);
364 365 366
}

static GeglNode *
367
gimp_curves_tool_get_operation (GimpImageMapTool  *image_map_tool,
368 369
                                GObject          **config,
                                gchar            **undo_desc)
370 371 372 373 374
{
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);

  tool->config = g_object_new (GIMP_TYPE_CURVES_CONFIG, NULL);

375 376 377
  g_signal_connect_object (tool->config, "notify",
                           G_CALLBACK (gimp_curves_tool_config_notify),
                           tool, 0);
378

379
  *config = G_OBJECT (tool->config);
380

381 382 383 384
  return gegl_node_new_child (NULL,
                              "operation", "gimp:curves",
                              "config",    tool->config,
                              NULL);
385
}
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
386

387

388 389 390
/*******************/
/*  Curves dialog  */
/*******************/
Elliot Lee's avatar
Elliot Lee committed
391

392 393
static void
gimp_curves_tool_dialog (GimpImageMapTool *image_map_tool)
Elliot Lee's avatar
Elliot Lee committed
394
{
395 396 397 398
  GimpCurvesTool   *tool         = GIMP_CURVES_TOOL (image_map_tool);
  GimpToolOptions  *tool_options = GIMP_TOOL_GET_OPTIONS (image_map_tool);
  GimpCurvesConfig *config       = tool->config;
  GtkListStore     *store;
399 400
  GtkSizeGroup     *label_group;
  GtkWidget        *main_vbox;
401 402 403 404 405 406 407 408
  GtkWidget        *vbox;
  GtkWidget        *hbox;
  GtkWidget        *hbox2;
  GtkWidget        *label;
  GtkWidget        *frame;
  GtkWidget        *table;
  GtkWidget        *button;
  GtkWidget        *bar;
409
  GtkWidget        *combo;
410

411 412 413 414
  g_signal_connect (image_map_tool->settings_box, "file-dialog-setup",
                    G_CALLBACK (gimp_curves_tool_export_setup),
                    image_map_tool);

415 416
  main_vbox   = gimp_image_map_tool_dialog_get_vbox (image_map_tool);
  label_group = gimp_image_map_tool_dialog_get_label_group (image_map_tool);
Elliot Lee's avatar
Elliot Lee committed
417

418
  /*  The combo box for selecting channels  */
Michael Natterer's avatar
Michael Natterer committed
419
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
420
  gtk_box_pack_start (GTK_BOX (main_vbox), hbox, FALSE, FALSE, 0);
421 422
  gtk_widget_show (hbox);

423
  label = gtk_label_new_with_mnemonic (_("Cha_nnel:"));
424 425 426
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);

427 428
  gtk_size_group_add_widget (label_group, label);

429 430 431
  store = gimp_enum_store_new_with_range (GIMP_TYPE_HISTOGRAM_CHANNEL,
                                          GIMP_HISTOGRAM_VALUE,
                                          GIMP_HISTOGRAM_ALPHA);
432 433
  tool->channel_menu =
    gimp_enum_combo_box_new_with_model (GIMP_ENUM_STORE (store));
434 435
  g_object_unref (store);

436 437 438
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                 config->channel);
  gimp_enum_combo_box_set_stock_prefix (GIMP_ENUM_COMBO_BOX (tool->channel_menu),
439
                                        "gimp-channel");
440 441
  gtk_box_pack_start (GTK_BOX (hbox), tool->channel_menu, FALSE, FALSE, 0);
  gtk_widget_show (tool->channel_menu);
442

443 444 445
  g_signal_connect (tool->channel_menu, "changed",
                    G_CALLBACK (curves_channel_callback),
                    tool);
446

447
  gtk_label_set_mnemonic_widget (GTK_LABEL (label), tool->channel_menu);
448

449
  button = gtk_button_new_with_mnemonic (_("R_eset Channel"));
450
  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
451
  gtk_widget_show (button);
452

453
  g_signal_connect (button, "clicked",
454
                    G_CALLBACK (curves_channel_reset_callback),
455
                    tool);
Elliot Lee's avatar
Elliot Lee committed
456

457 458 459 460 461 462
  /*  The histogram scale radio buttons  */
  hbox2 = gimp_prop_enum_stock_box_new (G_OBJECT (tool_options),
                                        "histogram-scale", "gimp-histogram",
                                        0, 0);
  gtk_box_pack_end (GTK_BOX (hbox), hbox2, FALSE, FALSE, 0);
  gtk_widget_show (hbox2);
463

464
  /*  The table for the color bars and the graph  */
Elliot Lee's avatar
Elliot Lee committed
465
  table = gtk_table_new (2, 2, FALSE);
466 467
  gtk_table_set_col_spacings (GTK_TABLE (table), 2);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
468
  gtk_box_pack_start (GTK_BOX (main_vbox), table, TRUE, TRUE, 0);
469 470

  /*  The left color bar  */
Michael Natterer's avatar
Michael Natterer committed
471
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
472
  gtk_table_attach (GTK_TABLE (table), vbox, 0, 1, 0, 1,
473
                    GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
474
  gtk_widget_show (vbox);
Elliot Lee's avatar
Elliot Lee committed
475 476

  frame = gtk_frame_new (NULL);
477
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
478
  gtk_box_pack_start (GTK_BOX (vbox), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
479 480
  gtk_widget_show (frame);

481 482 483 484
  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);
485

Elliot Lee's avatar
Elliot Lee committed
486 487
  /*  The curves graph  */
  frame = gtk_frame_new (NULL);
488
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
489
  gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 0, 1,
490
                    GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
491
  gtk_widget_show (frame);
Elliot Lee's avatar
Elliot Lee committed
492

493
  tool->graph = gimp_curve_view_new ();
494 495
  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);
496 497 498 499
  gtk_widget_set_size_request (tool->graph,
                               GRAPH_SIZE + RADIUS * 2,
                               GRAPH_SIZE + RADIUS * 2);
  g_object_set (tool->graph,
500 501 502
                "border-width", RADIUS,
                "subdivisions", 1,
                NULL);
503
  gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph),
504 505
                             config->curve[config->channel],
                             &channel_colors[config->channel]);
506 507
  gtk_container_add (GTK_CONTAINER (frame), tool->graph);
  gtk_widget_show (tool->graph);
508

509
  gimp_histogram_options_connect_view (GIMP_HISTOGRAM_OPTIONS (tool_options),
510 511 512
                                       GIMP_HISTOGRAM_VIEW (tool->graph));

  /*  The bottom color bar  */
Michael Natterer's avatar
Michael Natterer committed
513
  hbox2 = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
514
  gtk_table_attach (GTK_TABLE (table), hbox2, 1, 2, 1, 2,
515
                    GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
516
  gtk_widget_show (hbox2);
517

Elliot Lee's avatar
Elliot Lee committed
518
  frame = gtk_frame_new (NULL);
519
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
520
  gtk_box_pack_start (GTK_BOX (hbox2), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
521 522
  gtk_widget_show (frame);

Michael Natterer's avatar
Michael Natterer committed
523 524
  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
  gtk_box_set_homogeneous (GTK_BOX (vbox), TRUE);
525 526
  gtk_container_add (GTK_CONTAINER (frame), vbox);
  gtk_widget_show (vbox);
527

528 529
  tool->xrange = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
  gtk_widget_set_size_request (tool->xrange, -1, BAR_SIZE / 2);
530
  gtk_box_pack_start (GTK_BOX (vbox), tool->xrange, TRUE, TRUE, 0);
531
  gtk_widget_show (tool->xrange);
532

533
  bar = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
534
  gtk_box_pack_start (GTK_BOX (vbox), bar, TRUE, TRUE, 0);
535 536
  gtk_widget_show (bar);

537
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
538

Michael Natterer's avatar
Michael Natterer committed
539
  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
540
  gtk_box_pack_end (GTK_BOX (main_vbox), hbox, FALSE, FALSE, 0);
541
  gtk_widget_show (hbox);
542

543 544 545
  label = gtk_label_new_with_mnemonic (_("Curve _type:"));
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);
546

547 548 549 550 551 552 553 554
  tool->curve_type = combo = gimp_enum_combo_box_new (GIMP_TYPE_CURVE_TYPE);
  gimp_enum_combo_box_set_stock_prefix (GIMP_ENUM_COMBO_BOX (combo),
                                        "gimp-curve");
  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
555 556

  gtk_label_set_mnemonic_widget (GTK_LABEL (label), combo);
Elliot Lee's avatar
Elliot Lee committed
557 558
}

559
static void
560
gimp_curves_tool_reset (GimpImageMapTool *image_map_tool)
561
{
562
  GimpCurvesTool       *tool = GIMP_CURVES_TOOL (image_map_tool);
563
  GimpCurvesConfig     *default_config;
564
  GimpHistogramChannel  channel;
565

566 567
  default_config = GIMP_CURVES_CONFIG (image_map_tool->default_config);

568 569 570 571
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
      if (default_config)
        {
          GimpCurveType curve_type = tool->config->curve[channel]->curve_type;

          g_object_freeze_notify (G_OBJECT (tool->config->curve[channel]));

          gimp_config_copy (GIMP_CONFIG (default_config->curve[channel]),
                            GIMP_CONFIG (tool->config->curve[channel]),
                            0);

          g_object_set (tool->config->curve[channel],
                        "curve-type", curve_type,
                        NULL);

          g_object_thaw_notify (G_OBJECT (tool->config->curve[channel]));
        }
      else
        {
          gimp_curve_reset (tool->config->curve[channel], FALSE);
        }
592
    }
593 594
}

595
static gboolean
596 597 598
gimp_curves_tool_settings_import (GimpImageMapTool  *image_map_tool,
                                  const gchar       *filename,
                                  GError           **error)
599
{
600
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);
601
  FILE           *file;
602
  gchar           header[64];
603 604 605 606 607 608 609 610 611 612 613 614

  file = g_fopen (filename, "rt");

  if (! file)
    {
      g_set_error (error, G_FILE_ERROR, g_file_error_from_errno (errno),
                   _("Could not open '%s' for reading: %s"),
                   gimp_filename_to_utf8 (filename),
                   g_strerror (errno));
      return FALSE;
    }

615 616 617 618 619 620 621 622 623 624 625 626
  if (! fgets (header, sizeof (header), file))
    {
      g_set_error (error, G_FILE_ERROR, g_file_error_from_errno (errno),
                   _("Could not read header from '%s': %s"),
                   gimp_filename_to_utf8 (filename),
                   g_strerror (errno));
      fclose (file);
      return FALSE;
    }

  if (g_str_has_prefix (header, "# GIMP Curves File\n"))
    {
627 628
      gboolean success;

629 630 631 632 633 634 635 636
      rewind (file);

      success = gimp_curves_config_load_cruft (tool->config, file, error);

      fclose (file);

      return success;
    }
637 638

  fclose (file);
639

640 641 642
  return GIMP_IMAGE_MAP_TOOL_CLASS (parent_class)->settings_import (image_map_tool,
                                                                    filename,
                                                                    error);
643 644 645
}

static gboolean
646 647 648
gimp_curves_tool_settings_export (GimpImageMapTool  *image_map_tool,
                                  const gchar       *filename,
                                  GError           **error)
649
{
650
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);
651

652
  if (tool->export_old_format)
653
    {
654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672
      FILE     *file;
      gboolean  success;

      file = g_fopen (filename, "wt");

      if (! file)
        {
          g_set_error (error, G_FILE_ERROR, g_file_error_from_errno (errno),
                       _("Could not open '%s' for writing: %s"),
                       gimp_filename_to_utf8 (filename),
                       g_strerror (errno));
          return FALSE;
        }

      success = gimp_curves_config_save_cruft (tool->config, file, error);

      fclose (file);

      return success;
673 674
    }

675 676 677 678 679 680 681 682 683 684 685 686
  return GIMP_IMAGE_MAP_TOOL_CLASS (parent_class)->settings_export (image_map_tool,
                                                                    filename,
                                                                    error);
}

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

688 689 690 691 692 693 694 695
  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);
696

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

Elliot Lee's avatar
Elliot Lee committed
702
static void
703 704 705
gimp_curves_tool_config_notify (GObject        *object,
                                GParamSpec     *pspec,
                                GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
706
{
707
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (object);
708
  GimpCurve        *curve  = config->curve[config->channel];
709

710
  if (! tool->xrange)
711 712
    return;

713
  if (! strcmp (pspec->name, "channel"))
714
    {
715
      GimpHistogramChannel channel;
716

717 718 719
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                     config->channel);

720
      switch (config->channel)
721
        {
722 723 724 725
          guchar r[256];
          guchar g[256];
          guchar b[256];

726 727
        case GIMP_HISTOGRAM_VALUE:
        case GIMP_HISTOGRAM_ALPHA:
728
        case GIMP_HISTOGRAM_RGB:
729
          gimp_curve_get_uchar (curve, sizeof (r), r);
730

731
          gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
732
                                      r, r, r);
733
          break;
734

735 736 737
        case GIMP_HISTOGRAM_RED:
        case GIMP_HISTOGRAM_GREEN:
        case GIMP_HISTOGRAM_BLUE:
738 739 740 741 742 743
          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);
744

745
          gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
746
                                      r, g, b);
747 748
          break;
        }
749

750
      gimp_histogram_view_set_channel (GIMP_HISTOGRAM_VIEW (tool->graph),
751
                                       config->channel);
752
      gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
753
                                tool->picked_color[config->channel]);
754

755 756
      gimp_color_bar_set_channel (GIMP_COLOR_BAR (tool->yrange),
                                  config->channel);
Elliot Lee's avatar
Elliot Lee committed
757

758
      gimp_curve_view_remove_all_backgrounds (GIMP_CURVE_VIEW (tool->graph));
759

760 761 762
      for (channel = GIMP_HISTOGRAM_VALUE;
           channel <= GIMP_HISTOGRAM_ALPHA;
           channel++)
763
        {
764 765 766 767 768 769 770 771 772 773 774
          if (channel == config->channel)
            {
              gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph), curve,
                                         &channel_colors[channel]);
            }
          else
            {
              gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                              config->curve[channel],
                                              &channel_colors[channel]);
            }
775 776
        }

777 778
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                     curve->curve_type);
779 780 781
    }
  else if (! strcmp (pspec->name, "curve"))
    {
782 783
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                     curve->curve_type);
784 785
    }
}
786

787 788 789 790 791 792 793 794 795 796 797 798 799
static void
curves_channel_callback (GtkWidget      *widget,
                         GimpCurvesTool *tool)
{
  GimpCurvesConfig *config = tool->config;
  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);
800
    }
Elliot Lee's avatar
Elliot Lee committed
801 802 803
}

static void
804
curves_channel_reset_callback (GtkWidget      *widget,
805
                               GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
806
{
807
  gimp_curve_reset (tool->config->curve[tool->config->channel], FALSE);
Elliot Lee's avatar
Elliot Lee committed
808 809
}

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

817 818 819 820
  switch (channel)
    {
    case GIMP_HISTOGRAM_VALUE:
      return TRUE;
821

822 823 824
    case GIMP_HISTOGRAM_RED:
    case GIMP_HISTOGRAM_GREEN:
    case GIMP_HISTOGRAM_BLUE:
825
      return gimp_drawable_is_rgb (drawable);
Michael Natterer's avatar