gimpcurvestool.c 31.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
#include "base/curves.h"
Michael Natterer's avatar
Michael Natterer committed
34 35 36
#include "base/gimphistogram.h"
#include "base/gimplut.h"

37 38 39
#include "gegl/gimpcurvesconfig.h"
#include "gegl/gimpoperationcurves.h"

40
#include "core/gimp.h"
41
#include "core/gimpcurve.h"
Sven Neumann's avatar
Sven Neumann committed
42
#include "core/gimpcurve-map.h"
Michael Natterer's avatar
Michael Natterer committed
43
#include "core/gimpdrawable.h"
44
#include "core/gimpdrawable-histogram.h"
45
#include "core/gimperror.h"
Michael Natterer's avatar
Michael Natterer committed
46 47
#include "core/gimpimage.h"

48
#include "widgets/gimpcolorbar.h"
49
#include "widgets/gimphelp-ids.h"
50
#include "widgets/gimpcurveview.h"
Michael Natterer's avatar
Michael Natterer committed
51

52 53
#include "display/gimpdisplay.h"

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

57
#include "gimp-intl.h"
58

59

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


Michael Natterer's avatar
Michael Natterer committed
65 66
/*  local function prototypes  */

67 68 69 70 71 72
static void       gimp_curves_tool_finalize       (GObject              *object);

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

static void       gimp_curves_tool_color_picked   (GimpColorTool        *color_tool,
                                                   GimpColorPickState    pick_state,
                                                   GimpImageType         sample_type,
                                                   GimpRGB              *color,
                                                   gint                  color_index);
92 93
static GeglNode * gimp_curves_tool_get_operation  (GimpImageMapTool     *image_map_tool,
                                                   GObject             **config);
94 95 96
static void       gimp_curves_tool_map            (GimpImageMapTool     *image_map_tool);
static void       gimp_curves_tool_dialog         (GimpImageMapTool     *image_map_tool);
static void       gimp_curves_tool_reset          (GimpImageMapTool     *image_map_tool);
97
static gboolean   gimp_curves_tool_settings_import(GimpImageMapTool     *image_map_tool,
98
                                                   const gchar          *filename,
99
                                                   GError              **error);
100
static gboolean   gimp_curves_tool_settings_export(GimpImageMapTool     *image_map_tool,
101 102
                                                   const gchar          *filename,
                                                   GError              **error);
103

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

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

Michael Natterer's avatar
Michael Natterer committed
123

124
G_DEFINE_TYPE (GimpCurvesTool, gimp_curves_tool, GIMP_TYPE_IMAGE_MAP_TOOL)
125 126

#define parent_class gimp_curves_tool_parent_class
127

Michael Natterer's avatar
Michael Natterer committed
128

129
/*  public functions  */
Michael Natterer's avatar
Michael Natterer committed
130 131

void
Nate Summers's avatar
Nate Summers committed
132
gimp_curves_tool_register (GimpToolRegisterCallback  callback,
133
                           gpointer                  data)
Michael Natterer's avatar
Michael Natterer committed
134
{
Nate Summers's avatar
Nate Summers committed
135
  (* callback) (GIMP_TYPE_CURVES_TOOL,
136
                GIMP_TYPE_HISTOGRAM_OPTIONS,
137
                gimp_color_options_gui,
138
                0,
139
                "gimp-curves-tool",
140
                _("Curves"),
141
                _("Curves Tool: Adjust color curves"),
142
                N_("_Curves..."), NULL,
143
                NULL, GIMP_HELP_TOOL_CURVES,
Nate Summers's avatar
Nate Summers committed
144
                GIMP_STOCK_TOOL_CURVES,
145
                data);
Michael Natterer's avatar
Michael Natterer committed
146 147
}

148 149 150

/*  private functions  */

Michael Natterer's avatar
Michael Natterer committed
151 152 153
static void
gimp_curves_tool_class_init (GimpCurvesToolClass *klass)
{
154 155 156 157 158
  GObjectClass          *object_class     = G_OBJECT_CLASS (klass);
  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);

159
  object_class->finalize             = gimp_curves_tool_finalize;
160

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

166
  color_tool_class->picked           = gimp_curves_tool_color_picked;
167

168 169 170 171
  im_tool_class->shell_desc          = _("Adjust Color Curves");
  im_tool_class->settings_name       = "curves";
  im_tool_class->import_dialog_title = _("Import Curves");
  im_tool_class->export_dialog_title = _("Export Curves");
172

173 174 175 176 177 178
  im_tool_class->get_operation       = gimp_curves_tool_get_operation;
  im_tool_class->map                 = gimp_curves_tool_map;
  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
179 180 181
}

static void
182
gimp_curves_tool_init (GimpCurvesTool *tool)
Michael Natterer's avatar
Michael Natterer committed
183
{
184 185
  GimpImageMapTool *im_tool = GIMP_IMAGE_MAP_TOOL (tool);
  gint              i;
186

187
  tool->lut = gimp_lut_new ();
188

189 190
  for (i = 0; i < G_N_ELEMENTS (tool->picked_color); i++)
    tool->picked_color[i] = -1.0;
191 192 193

  im_tool->apply_func = (GimpImageMapApplyFunc) gimp_lut_process;
  im_tool->apply_data = tool->lut;
Michael Natterer's avatar
Michael Natterer committed
194 195 196
}

static void
197
gimp_curves_tool_finalize (GObject *object)
Michael Natterer's avatar
Michael Natterer committed
198
{
199
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (object);
200

201 202
  gimp_lut_free (tool->lut);

203
  G_OBJECT_CLASS (parent_class)->finalize (object);
Michael Natterer's avatar
Michael Natterer committed
204 205
}

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

  if (! drawable)
217
    return FALSE;
218

219
  if (gimp_drawable_is_indexed (drawable))
Michael Natterer's avatar
Michael Natterer committed
220
    {
221
      g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
222
			   _("Curves does not operate on indexed layers."));
223
      return FALSE;
224
    }
Michael Natterer's avatar
Michael Natterer committed
225

226
  gimp_config_reset (GIMP_CONFIG (c_tool->config));
227

228 229 230 231
  if (! GIMP_TOOL_CLASS (parent_class)->initialize (tool, display, error))
    {
      return FALSE;
    }
232

233 234
  /*  always pick colors  */
  gimp_color_tool_enable (GIMP_COLOR_TOOL (tool),
235
                          GIMP_COLOR_TOOL_GET_OPTIONS (tool));
236

237
  gimp_int_combo_box_set_sensitivity (GIMP_INT_COMBO_BOX (c_tool->channel_menu),
238
                                      curves_menu_sensitivity, drawable, NULL);
239

240 241
  histogram = gimp_histogram_new ();
  gimp_drawable_calculate_histogram (drawable, histogram);
242
  gimp_histogram_view_set_background (GIMP_HISTOGRAM_VIEW (c_tool->graph),
243 244
                                      histogram);
  gimp_histogram_unref (histogram);
245

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

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

260
  if (state & GDK_SHIFT_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 & GDK_CONTROL_MASK)
Michael Natterer's avatar
Michael Natterer committed
275
    {
276
      gint i;
277

278 279
      for (i = 0; i < 5; i++)
        {
280
          GimpCurve *curve = config->curve[i];
281
          gdouble    value = c_tool->picked_color[i];
282
          gint       closest;
283

284
          closest = gimp_curve_get_closest_point (curve, value);
285

286
          gimp_curve_view_set_selected (GIMP_CURVE_VIEW (c_tool->graph),
287
                                        closest);
288

289
          gimp_curve_set_point (curve, closest,
290
                                value, gimp_curve_map_value (curve, value));
291
        }
Michael Natterer's avatar
Michael Natterer committed
292 293
    }

294
  /*  chain up to halt the tool */
295 296
  GIMP_TOOL_CLASS (parent_class)->button_release (tool, coords, time, state,
                                                  release_type, display);
Michael Natterer's avatar
Michael Natterer committed
297 298
}

299
static gboolean
300 301
gimp_curves_tool_key_press (GimpTool    *tool,
                            GdkEventKey *kevent,
302
                            GimpDisplay *display)
303
{
304 305
  GimpCurvesTool *c_tool = GIMP_CURVES_TOOL (tool);

306 307 308 309
  if (gtk_widget_event (c_tool->graph, (GdkEvent *) kevent))
    return TRUE;

  return GIMP_TOOL_CLASS (parent_class)->key_press (tool, kevent, display);
310 311 312
}

static void
313 314 315 316 317
gimp_curves_tool_oper_update (GimpTool         *tool,
                              const GimpCoords *coords,
                              GdkModifierType   state,
                              gboolean          proximity,
                              GimpDisplay      *display)
318
{
319 320
  GimpColorPickMode  mode;
  const gchar       *status;
321

322
  GIMP_TOOL_CLASS (parent_class)->oper_update (tool, coords, state, proximity,
323
                                               display);
324

325
  gimp_tool_pop_status (tool, display);
326 327 328 329

  if (state & GDK_SHIFT_MASK)
    {
      mode   = GIMP_COLOR_PICK_MODE_PALETTE;
330
      status = _("Click to add a control point");
331 332 333 334
    }
  else if (state & GDK_CONTROL_MASK)
    {
      mode   = GIMP_COLOR_PICK_MODE_PALETTE;
335
      status = _("Click to add control points to all channels");
336
    }
337 338 339 340 341
  else
    {
      mode   = GIMP_COLOR_PICK_MODE_NONE;
      status = _("Click to locate on curve (try Shift, Ctrl)");
    }
342 343 344

  GIMP_COLOR_TOOL (tool)->pick_mode = mode;

345
  if (proximity)
346
    gimp_tool_push_status (tool, display, "%s", status);
347 348
}

Michael Natterer's avatar
Michael Natterer committed
349
static void
350 351
gimp_curves_tool_color_picked (GimpColorTool      *color_tool,
                               GimpColorPickState  pick_state,
352 353 354
                               GimpImageType       sample_type,
                               GimpRGB            *color,
                               gint                color_index)
Michael Natterer's avatar
Michael Natterer committed
355
{
356 357
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (color_tool);
  GimpDrawable   *drawable;
Michael Natterer's avatar
Michael Natterer committed
358

359
  drawable = GIMP_IMAGE_MAP_TOOL (tool)->drawable;
360

361 362 363
  tool->picked_color[GIMP_HISTOGRAM_RED]   = color->r;
  tool->picked_color[GIMP_HISTOGRAM_GREEN] = color->g;
  tool->picked_color[GIMP_HISTOGRAM_BLUE]  = color->b;
364

365
  if (gimp_drawable_has_alpha (drawable))
366
    tool->picked_color[GIMP_HISTOGRAM_ALPHA] = color->a;
367

368 369
  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
370

371
  gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
372
                            tool->picked_color[tool->config->channel]);
373 374 375
}

static GeglNode *
376 377
gimp_curves_tool_get_operation (GimpImageMapTool  *image_map_tool,
                                GObject          **config)
378 379 380 381 382
{
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);
  GeglNode       *node;

  node = g_object_new (GEGL_TYPE_NODE,
383
                       "operation", "gimp:curves",
384 385 386 387
                       NULL);

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

388 389 390 391 392
  *config = G_OBJECT (tool->config);

  g_signal_connect_object (tool->config, "notify",
                           G_CALLBACK (gimp_curves_tool_config_notify),
                           tool, 0);
393 394 395 396 397 398

  gegl_node_set (node,
                 "config", tool->config,
                 NULL);

  return node;
399
}
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
400

401 402
static void
gimp_curves_tool_map (GimpImageMapTool *image_map_tool)
Elliot Lee's avatar
Elliot Lee committed
403
{
404 405
  GimpCurvesTool *tool     = GIMP_CURVES_TOOL (image_map_tool);
  GimpDrawable   *drawable = image_map_tool->drawable;
406 407
  Curves          curves;

408 409
  gimp_curves_config_to_cruft (tool->config, &curves,
                               gimp_drawable_is_rgb (drawable));
410

411
  gimp_lut_setup (tool->lut,
412
                  (GimpLutFunc) curves_lut_func,
413
                  &curves,
414
                  gimp_drawable_bytes (drawable));
Elliot Lee's avatar
Elliot Lee committed
415 416
}

417

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

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

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

445 446
  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
447

448
  /*  The combo box for selecting channels  */
449
  hbox = gtk_hbox_new (FALSE, 6);
450
  gtk_box_pack_start (GTK_BOX (main_vbox), hbox, FALSE, FALSE, 0);
451 452
  gtk_widget_show (hbox);

453
  label = gtk_label_new_with_mnemonic (_("Cha_nnel:"));
454 455 456
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);

457 458
  gtk_size_group_add_widget (label_group, label);

459 460 461
  store = gimp_enum_store_new_with_range (GIMP_TYPE_HISTOGRAM_CHANNEL,
                                          GIMP_HISTOGRAM_VALUE,
                                          GIMP_HISTOGRAM_ALPHA);
462 463
  tool->channel_menu =
    gimp_enum_combo_box_new_with_model (GIMP_ENUM_STORE (store));
464 465
  g_object_unref (store);

466 467 468
  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),
469
                                        "gimp-channel");
470 471
  gtk_box_pack_start (GTK_BOX (hbox), tool->channel_menu, FALSE, FALSE, 0);
  gtk_widget_show (tool->channel_menu);
472

473 474 475
  g_signal_connect (tool->channel_menu, "changed",
                    G_CALLBACK (curves_channel_callback),
                    tool);
476

477
  gtk_label_set_mnemonic_widget (GTK_LABEL (label), tool->channel_menu);
478

479
  button = gtk_button_new_with_mnemonic (_("R_eset Channel"));
480
  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
481
  gtk_widget_show (button);
482

483
  g_signal_connect (button, "clicked",
484
                    G_CALLBACK (curves_channel_reset_callback),
485
                    tool);
Elliot Lee's avatar
Elliot Lee committed
486

487 488 489 490 491 492
  /*  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);
493

494
  /*  The table for the color bars and the graph  */
Elliot Lee's avatar
Elliot Lee committed
495
  table = gtk_table_new (2, 2, FALSE);
496 497
  gtk_table_set_col_spacings (GTK_TABLE (table), 2);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
498
  gtk_box_pack_start (GTK_BOX (main_vbox), table, TRUE, TRUE, 0);
499 500

  /*  The left color bar  */
501 502
  vbox = gtk_vbox_new (FALSE, 0);
  gtk_table_attach (GTK_TABLE (table), vbox, 0, 1, 0, 1,
503
                    GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
504
  gtk_widget_show (vbox);
Elliot Lee's avatar
Elliot Lee committed
505 506

  frame = gtk_frame_new (NULL);
507
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
508
  gtk_box_pack_start (GTK_BOX (vbox), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
509 510
  gtk_widget_show (frame);

511 512 513 514
  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);
515

Elliot Lee's avatar
Elliot Lee committed
516 517
  /*  The curves graph  */
  frame = gtk_frame_new (NULL);
518
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
519
  gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 0, 1,
520
                    GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
521
  gtk_widget_show (frame);
Elliot Lee's avatar
Elliot Lee committed
522

523
  tool->graph = gimp_curve_view_new ();
524 525
  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);
526 527 528 529
  gtk_widget_set_size_request (tool->graph,
                               GRAPH_SIZE + RADIUS * 2,
                               GRAPH_SIZE + RADIUS * 2);
  g_object_set (tool->graph,
530 531 532
                "border-width", RADIUS,
                "subdivisions", 1,
                NULL);
533 534
  gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph),
                             config->curve[config->channel]);
535 536
  gtk_container_add (GTK_CONTAINER (frame), tool->graph);
  gtk_widget_show (tool->graph);
537

538
  gimp_histogram_options_connect_view (GIMP_HISTOGRAM_OPTIONS (tool_options),
539 540 541 542 543
                                       GIMP_HISTOGRAM_VIEW (tool->graph));

  /*  The bottom color bar  */
  hbox2 = gtk_hbox_new (FALSE, 0);
  gtk_table_attach (GTK_TABLE (table), hbox2, 1, 2, 1, 2,
544
                    GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
545
  gtk_widget_show (hbox2);
546

Elliot Lee's avatar
Elliot Lee committed
547
  frame = gtk_frame_new (NULL);
548
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
549
  gtk_box_pack_start (GTK_BOX (hbox2), frame, TRUE, TRUE, RADIUS);
Elliot Lee's avatar
Elliot Lee committed
550 551
  gtk_widget_show (frame);

552 553 554
  vbox = gtk_vbox_new (TRUE, 0);
  gtk_container_add (GTK_CONTAINER (frame), vbox);
  gtk_widget_show (vbox);
555

556 557
  tool->xrange = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
  gtk_widget_set_size_request (tool->xrange, -1, BAR_SIZE / 2);
558
  gtk_box_pack_start (GTK_BOX (vbox), tool->xrange, TRUE, TRUE, 0);
559
  gtk_widget_show (tool->xrange);
560

561
  bar = gimp_color_bar_new (GTK_ORIENTATION_HORIZONTAL);
562
  gtk_box_pack_start (GTK_BOX (vbox), bar, TRUE, TRUE, 0);
563 564
  gtk_widget_show (bar);

565
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
566

567
  hbox = gtk_hbox_new (FALSE, 6);
568
  gtk_box_pack_end (GTK_BOX (main_vbox), hbox, FALSE, FALSE, 0);
569
  gtk_widget_show (hbox);
570

571 572 573
  label = gtk_label_new_with_mnemonic (_("Curve _type:"));
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);
574

575 576 577 578 579 580 581 582
  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
583 584

  gtk_label_set_mnemonic_widget (GTK_LABEL (label), combo);
Elliot Lee's avatar
Elliot Lee committed
585 586
}

587
static void
588
gimp_curves_tool_reset (GimpImageMapTool *image_map_tool)
589
{
590
  GimpCurvesTool       *tool = GIMP_CURVES_TOOL (image_map_tool);
591
  GimpCurvesConfig     *default_config;
592
  GimpHistogramChannel  channel;
593

594 595
  default_config = GIMP_CURVES_CONFIG (image_map_tool->default_config);

596 597 598 599
  for (channel = GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    {
600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619
      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);
        }
620
    }
621 622
}

623
static gboolean
624 625 626
gimp_curves_tool_settings_import (GimpImageMapTool  *image_map_tool,
                                  const gchar       *filename,
                                  GError           **error)
627
{
628
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);
629
  FILE           *file;
630
  gchar           header[64];
631 632 633 634 635 636 637 638 639 640 641 642

  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;
    }

643 644 645 646 647 648 649 650 651 652 653 654
  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"))
    {
655 656
      gboolean success;

657 658 659 660 661 662 663 664
      rewind (file);

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

      fclose (file);

      return success;
    }
665 666

  fclose (file);
667

668 669 670
  return GIMP_IMAGE_MAP_TOOL_CLASS (parent_class)->settings_import (image_map_tool,
                                                                    filename,
                                                                    error);
671 672 673
}

static gboolean
674 675 676
gimp_curves_tool_settings_export (GimpImageMapTool  *image_map_tool,
                                  const gchar       *filename,
                                  GError           **error)
677
{
678
  GimpCurvesTool *tool = GIMP_CURVES_TOOL (image_map_tool);
679

680
  if (tool->export_old_format)
681
    {
682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700
      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;
701 702
    }

703 704 705 706 707 708 709 710 711 712 713 714
  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;
715

716 717 718 719 720 721 722 723
  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);
724

725 726 727
  g_signal_connect (button, "toggled",
                    G_CALLBACK (gimp_toggle_button_update),
                    &tool->export_old_format);
728
}
729

Elliot Lee's avatar
Elliot Lee committed
730
static void
731 732 733
gimp_curves_tool_config_notify (GObject        *object,
                                GParamSpec     *pspec,
                                GimpCurvesTool *tool)
Elliot Lee's avatar
Elliot Lee committed
734
{
735
  GimpCurvesConfig *config = GIMP_CURVES_CONFIG (object);
736
  GimpCurve        *curve  = config->curve[config->channel];
737

738
  if (! tool->xrange)
739 740
    return;

741
  if (! strcmp (pspec->name, "channel"))
742
    {
743 744 745 746
      GimpRGB red;
      GimpRGB green;
      GimpRGB blue;

747 748 749
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->channel_menu),
                                     config->channel);

750
      switch (config->channel)
751
        {
752 753 754 755
          guchar r[256];
          guchar g[256];
          guchar b[256];

756 757
        case GIMP_HISTOGRAM_VALUE:
        case GIMP_HISTOGRAM_ALPHA:
758
        case GIMP_HISTOGRAM_RGB:
759
          gimp_curve_get_uchar (curve, sizeof (r), r);
760

761
          gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
762
                                      r, r, r);
763
          break;
764

765 766 767
        case GIMP_HISTOGRAM_RED:
        case GIMP_HISTOGRAM_GREEN:
        case GIMP_HISTOGRAM_BLUE:
768 769 770 771 772 773
          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);
774

775
          gimp_color_bar_set_buffers (GIMP_COLOR_BAR (tool->xrange),
776
                                      r, g, b);
777 778
          break;
        }
779

780
      gimp_histogram_view_set_channel (GIMP_HISTOGRAM_VIEW (tool->graph),
781
                                       config->channel);
782
      gimp_curve_view_set_xpos (GIMP_CURVE_VIEW (tool->graph),
783
                                tool->picked_color[config->channel]);
784

785 786
      gimp_color_bar_set_channel (GIMP_COLOR_BAR (tool->yrange),
                                  config->channel);
Elliot Lee's avatar
Elliot Lee committed
787

788
      gimp_curve_view_remove_all_backgrounds (GIMP_CURVE_VIEW (tool->graph));
789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826

      gimp_rgb_set (&red,   1.0, 0.0, 0.0);
      gimp_rgb_set (&green, 0.0, 1.0, 0.0);
      gimp_rgb_set (&blue,  0.0, 0.0, 1.0);

      switch (config->channel)
        {
        case GIMP_HISTOGRAM_RED:
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_GREEN],
                                          &green);
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_BLUE],
                                          &blue);
          break;

        case GIMP_HISTOGRAM_GREEN:
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_RED],
                                          &red);
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_BLUE],
                                          &blue);
          break;

        case GIMP_HISTOGRAM_BLUE:
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_RED],
                                          &red);
          gimp_curve_view_add_background (GIMP_CURVE_VIEW (tool->graph),
                                          config->curve[GIMP_HISTOGRAM_GREEN],
                                          &green);
          break;

        default:
          break;
        }

827 828
      gimp_curve_view_set_curve (GIMP_CURVE_VIEW (tool->graph), curve);

829 830
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                     curve->curve_type);
831 832 833
    }
  else if (! strcmp (pspec->name, "curve"))
    {
834 835
      gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (tool->curve_type),
                                     curve->curve_type);
836 837 838 839
    }

  gimp_image_map_tool_preview (GIMP_IMAGE_MAP_TOOL (tool));
}
840

841 842 843 844 845 846 847 848 849 850 851 852 853
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);
854
    }
Elliot Lee's avatar
Elliot Lee committed
855 856 857
}

static void