gimpcurvestool.c 39.2 KB
Newer Older
Elliot Lee's avatar
Elliot Lee committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/* The GIMP -- an image manipulation program
 * Copyright (C) 1995 Spencer Kimball and Peter Mattis
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (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
 * along with this program; if not, write to the Free Software
16
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
Elliot Lee's avatar
Elliot Lee committed
17
 */
Sven Neumann's avatar
Sven Neumann committed
18

19 20 21
#include "config.h"

#include <stdio.h>
22 23
#include <stdlib.h>
#include <string.h>
24
#include <errno.h>
25

26 27 28 29 30
#ifdef __GNUC__
#warning GTK_DISABLE_DEPRECATED
#endif
#undef GTK_DISABLE_DEPRECATED

Sven Neumann's avatar
Sven Neumann committed
31 32
#include <gtk/gtk.h>

33
#include "libgimpmath/gimpmath.h"
34
#include "libgimpbase/gimpbase.h"
35
#include "libgimpcolor/gimpcolor.h"
36
#include "libgimpwidgets/gimpwidgets.h"
37

38
#include "tools-types.h"
Michael Natterer's avatar
Michael Natterer committed
39

40
#include "base/curves.h"
Michael Natterer's avatar
Michael Natterer committed
41 42 43 44 45
#include "base/gimphistogram.h"
#include "base/gimplut.h"

#include "core/gimpdrawable.h"
#include "core/gimpimage.h"
46
#include "core/gimpimagemap.h"
47
#include "core/gimptoolinfo.h"
Michael Natterer's avatar
Michael Natterer committed
48 49

#include "widgets/gimpcursor.h"
50
#include "widgets/gimpenummenu.h"
Michael Natterer's avatar
Michael Natterer committed
51

52 53
#include "display/gimpdisplay.h"

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

58
#include "gimp-intl.h"
59

60

61 62 63 64
#define XRANGE_TOP     (1 << 0)
#define XRANGE_BOTTOM  (1 << 1)
#define YRANGE         (1 << 2)
#define ALL            (XRANGE_TOP | XRANGE_BOTTOM | YRANGE)
Elliot Lee's avatar
Elliot Lee committed
65

66
/*  NB: take care when changing these values: make sure the curve[] array in
67
 *  base/curves.h is large enough.
68
 */
69 70 71 72 73 74
#define GRAPH_WIDTH    256
#define GRAPH_HEIGHT   256
#define XRANGE_WIDTH   256
#define XRANGE_HEIGHT   16
#define YRANGE_WIDTH    16
#define YRANGE_HEIGHT  256
Elliot Lee's avatar
Elliot Lee committed
75 76 77
#define RADIUS           3
#define MIN_DISTANCE     8

78 79 80 81 82
#define GRAPH_MASK  (GDK_EXPOSURE_MASK       | \
                     GDK_LEAVE_NOTIFY_MASK   | \
		     GDK_POINTER_MOTION_MASK | \
		     GDK_BUTTON_PRESS_MASK   | \
		     GDK_BUTTON_RELEASE_MASK)
Elliot Lee's avatar
Elliot Lee committed
83 84


Michael Natterer's avatar
Michael Natterer committed
85 86 87
/*  local function prototypes  */

static void   gimp_curves_tool_class_init     (GimpCurvesToolClass *klass);
88 89 90 91 92 93 94 95 96 97 98 99
static void   gimp_curves_tool_init           (GimpCurvesTool      *c_tool);

static void   gimp_curves_tool_finalize       (GObject          *object);

static void   gimp_curves_tool_initialize     (GimpTool         *tool,
					       GimpDisplay      *gdisp);
static void   gimp_curves_tool_button_release (GimpTool         *tool,
                                               GimpCoords       *coords,
                                               guint32           time,
					       GdkModifierType   state,
					       GimpDisplay      *gdisp);

100 101 102 103
static void   gimp_curves_tool_color_picked   (GimpColorTool    *color_tool,
					       GimpImageType     sample_type,
					       GimpRGB          *color,
					       gint              color_index);
104 105 106 107 108 109 110 111 112 113 114 115 116
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);

static void   curves_add_point                (GimpCurvesTool   *c_tool,
					       gint              x,
					       gint              y,
					       gint              cchan);

static void   curves_update                   (GimpCurvesTool   *c_tool,
                                               gint              update);

static void   curves_channel_callback         (GtkWidget        *widget,
117
                                               GimpCurvesTool   *c_tool);
118
static void   curves_channel_reset_callback   (GtkWidget        *widget,
119
                                               GimpCurvesTool   *c_tool);
120 121 122

static gboolean curves_set_sensitive_callback (gpointer          item_data,
                                               GimpCurvesTool   *c_tool);
123
static void   curves_curve_type_callback      (GtkWidget        *widget,
124
                                               GimpCurvesTool   *c_tool);
125
static void   curves_load_callback            (GtkWidget        *widget,
126
                                               GimpCurvesTool   *c_tool);
127
static void   curves_save_callback            (GtkWidget        *widget,
128
                                               GimpCurvesTool   *c_tool);
129 130 131
static gint   curves_graph_events             (GtkWidget        *widget,
                                               GdkEvent         *event,
                                               GimpCurvesTool   *c_tool);
132 133 134
static void   curves_graph_expose             (GtkWidget        *widget,
                                               GdkRectangle     *area,
                                               GimpCurvesTool   *c_tool);
135 136 137 138 139 140 141 142 143

static void       file_dialog_create          (GimpCurvesTool   *c_tool);
static void       file_dialog_ok_callback     (GimpCurvesTool   *c_tool);

static gboolean   curves_read_from_file       (GimpCurvesTool   *c_tool,
                                               FILE             *file);
static void       curves_write_to_file        (GimpCurvesTool   *c_tool,
                                               FILE             *file);

Michael Natterer's avatar
Michael Natterer committed
144

145 146
static GimpImageMapToolClass *parent_class = NULL;

Michael Natterer's avatar
Michael Natterer committed
147

148
/*  public functions  */
Michael Natterer's avatar
Michael Natterer committed
149 150

void
Nate Summers's avatar
Nate Summers committed
151
gimp_curves_tool_register (GimpToolRegisterCallback  callback,
152
                           gpointer                  data)
Michael Natterer's avatar
Michael Natterer committed
153
{
Nate Summers's avatar
Nate Summers committed
154
  (* callback) (GIMP_TYPE_CURVES_TOOL,
155 156
                GIMP_TYPE_COLOR_OPTIONS,
                gimp_color_options_gui,
157
                0,
158
                "gimp-curves-tool",
159 160
                _("Curves"),
                _("Adjust color curves"),
161
                N_("/Tools/Color Tools/_Curves..."), NULL,
162
                NULL, "tools/curves.html",
Nate Summers's avatar
Nate Summers committed
163
                GIMP_STOCK_TOOL_CURVES,
164
                data);
Michael Natterer's avatar
Michael Natterer committed
165 166
}

167
GType
Michael Natterer's avatar
Michael Natterer committed
168 169
gimp_curves_tool_get_type (void)
{
170
  static GType tool_type = 0;
Michael Natterer's avatar
Michael Natterer committed
171 172 173

  if (! tool_type)
    {
174
      static const GTypeInfo tool_info =
Michael Natterer's avatar
Michael Natterer committed
175 176
      {
        sizeof (GimpCurvesToolClass),
177 178 179 180 181 182 183 184
	(GBaseInitFunc) NULL,
	(GBaseFinalizeFunc) NULL,
	(GClassInitFunc) gimp_curves_tool_class_init,
	NULL,           /* class_finalize */
	NULL,           /* class_data     */
	sizeof (GimpCurvesTool),
	0,              /* n_preallocs    */
	(GInstanceInitFunc) gimp_curves_tool_init,
Michael Natterer's avatar
Michael Natterer committed
185 186
      };

187 188 189
      tool_type = g_type_register_static (GIMP_TYPE_IMAGE_MAP_TOOL,
					  "GimpCurvesTool", 
                                          &tool_info, 0);
Michael Natterer's avatar
Michael Natterer committed
190 191 192 193 194
    }

  return tool_type;
}

195 196 197

/*  private functions  */

Michael Natterer's avatar
Michael Natterer committed
198 199 200
static void
gimp_curves_tool_class_init (GimpCurvesToolClass *klass)
{
201 202
  GObjectClass          *object_class;
  GimpToolClass         *tool_class;
203
  GimpColorToolClass    *color_tool_class;
204
  GimpImageMapToolClass *image_map_tool_class;
Michael Natterer's avatar
Michael Natterer committed
205

206 207
  object_class         = G_OBJECT_CLASS (klass);
  tool_class           = GIMP_TOOL_CLASS (klass);
208
  color_tool_class     = GIMP_COLOR_TOOL_CLASS (klass);
209
  image_map_tool_class = GIMP_IMAGE_MAP_TOOL_CLASS (klass);
Michael Natterer's avatar
Michael Natterer committed
210

211
  parent_class = g_type_class_peek_parent (klass);
Michael Natterer's avatar
Michael Natterer committed
212

213
  object_class->finalize       = gimp_curves_tool_finalize;
214

215 216 217 218
  tool_class->initialize       = gimp_curves_tool_initialize;
  tool_class->button_release   = gimp_curves_tool_button_release;

  color_tool_class->picked     = gimp_curves_tool_color_picked;
219 220 221 222

  image_map_tool_class->map    = gimp_curves_tool_map;
  image_map_tool_class->dialog = gimp_curves_tool_dialog;
  image_map_tool_class->reset  = gimp_curves_tool_reset;
Michael Natterer's avatar
Michael Natterer committed
223 224 225
}

static void
226
gimp_curves_tool_init (GimpCurvesTool *c_tool)
Michael Natterer's avatar
Michael Natterer committed
227
{
228 229 230 231 232
  GimpImageMapTool *image_map_tool;
  gint              i;

  image_map_tool = GIMP_IMAGE_MAP_TOOL (c_tool);

233 234
  image_map_tool->shell_identifier = "gimp-curves-tool-dialog";
  image_map_tool->shell_desc       = _("Adjust Color Curves");
235

236 237 238
  c_tool->curves  = g_new0 (Curves, 1);
  c_tool->lut     = gimp_lut_new ();
  c_tool->channel = GIMP_HISTOGRAM_VALUE;
239 240 241

  curves_init (c_tool->curves);

242
  for (i = 0; i < G_N_ELEMENTS (c_tool->col_value); i++)
243
    c_tool->col_value[i] = -1;
244 245 246

  c_tool->cursor_x = -1;
  c_tool->cursor_y = -1;
Michael Natterer's avatar
Michael Natterer committed
247 248 249
}

static void
250
gimp_curves_tool_finalize (GObject *object)
Michael Natterer's avatar
Michael Natterer committed
251
{
252
  GimpCurvesTool *c_tool;
Michael Natterer's avatar
Michael Natterer committed
253

254 255 256
  c_tool = GIMP_CURVES_TOOL (object);

  if (c_tool->curves)
Michael Natterer's avatar
Michael Natterer committed
257
    {
258
      g_free (c_tool->curves);
259
      c_tool->curves = NULL;
Michael Natterer's avatar
Michael Natterer committed
260
    }
261
  if (c_tool->lut)
Michael Natterer's avatar
Michael Natterer committed
262
    {
263 264
      gimp_lut_free (c_tool->lut);
      c_tool->lut = NULL;
Michael Natterer's avatar
Michael Natterer committed
265
    }
266
  if (c_tool->cursor_layout)
Michael Natterer's avatar
Michael Natterer committed
267
    {
268 269
      g_object_unref (c_tool->cursor_layout);
      c_tool->cursor_layout = NULL;
Michael Natterer's avatar
Michael Natterer committed
270
    }
271 272 273 274 275
  if (c_tool->xpos_layout)
    {
      g_object_unref (c_tool->xpos_layout);
      c_tool->xpos_layout = NULL;
    }
Michael Natterer's avatar
Michael Natterer committed
276

277
  G_OBJECT_CLASS (parent_class)->finalize (object);
Michael Natterer's avatar
Michael Natterer committed
278 279 280
}

static void
281 282
gimp_curves_tool_initialize (GimpTool    *tool,
			     GimpDisplay *gdisp)
Michael Natterer's avatar
Michael Natterer committed
283
{
284 285 286 287
  GimpCurvesTool *c_tool;
  GimpDrawable   *drawable;

  c_tool = GIMP_CURVES_TOOL (tool);
288 289

  if (gimp_drawable_is_indexed (gimp_image_active_drawable (gdisp->gimage)))
Michael Natterer's avatar
Michael Natterer committed
290
    {
291 292 293
      g_message (_("Curves for indexed drawables cannot be adjusted."));
      return;
    }
Michael Natterer's avatar
Michael Natterer committed
294

295
  drawable = gimp_image_active_drawable (gdisp->gimage);
Michael Natterer's avatar
Michael Natterer committed
296

297
  curves_init (c_tool->curves);
Michael Natterer's avatar
Michael Natterer committed
298

299 300 301 302 303 304 305 306
  c_tool->color   = gimp_drawable_is_rgb (drawable);
  c_tool->channel = GIMP_HISTOGRAM_VALUE;

  c_tool->grab_point = -1;
  c_tool->last       = 0;

  GIMP_TOOL_CLASS (parent_class)->initialize (tool, gdisp);

307 308 309 310
  /*  always pick colors  */
  gimp_color_tool_enable (GIMP_COLOR_TOOL (tool),
                          GIMP_COLOR_OPTIONS (tool->tool_info->tool_options));

311 312 313 314 315 316
  /* set the sensitivity of the channel menu based on the drawable type */
  gimp_option_menu_set_sensitive (GTK_OPTION_MENU (c_tool->channel_menu),
                                  (GimpOptionMenuSensitivityCallback) curves_set_sensitive_callback,
                                  c_tool);

  /* set the current selection */
317 318
  gimp_option_menu_set_history (GTK_OPTION_MENU (c_tool->channel_menu),
                                GINT_TO_POINTER (c_tool->channel));
Michael Natterer's avatar
Michael Natterer committed
319

320
  curves_update (c_tool, ALL);
Michael Natterer's avatar
Michael Natterer committed
321 322 323
}

static void
324 325 326 327 328
gimp_curves_tool_button_release (GimpTool        *tool,
                                 GimpCoords      *coords,
                                 guint32          time,
				 GdkModifierType  state,
				 GimpDisplay     *gdisp)
Michael Natterer's avatar
Michael Natterer committed
329
{
330 331
  GimpCurvesTool *c_tool;
  GimpDrawable   *drawable;
Michael Natterer's avatar
Michael Natterer committed
332

333 334 335
  c_tool = GIMP_CURVES_TOOL (tool);

  drawable = gimp_image_active_drawable (gdisp->gimage);
Michael Natterer's avatar
Michael Natterer committed
336

337
  if (state & GDK_SHIFT_MASK)
Michael Natterer's avatar
Michael Natterer committed
338
    {
339
      curves_add_point (c_tool, coords->x, coords->y, c_tool->channel);
340
      curves_calculate_curve (c_tool->curves, c_tool->channel);
Michael Natterer's avatar
Michael Natterer committed
341
    }
342
  else if (state & GDK_CONTROL_MASK)
Michael Natterer's avatar
Michael Natterer committed
343
    {
344
      gint i;
345

346 347 348 349 350
      for (i = 0; i < 5; i++)
        {
          curves_add_point (c_tool, coords->x, coords->y, i);
          curves_calculate_curve (c_tool->curves, c_tool->channel);
        }
Michael Natterer's avatar
Michael Natterer committed
351 352
    }

353 354 355
  /*  chain up to halt the tool */
  GIMP_TOOL_CLASS (parent_class)->button_release (tool,
                                                  coords, time, state, gdisp);
Michael Natterer's avatar
Michael Natterer committed
356 357 358
}

static void
359 360 361 362
gimp_curves_tool_color_picked (GimpColorTool *color_tool,
			       GimpImageType  sample_type,
			       GimpRGB       *color,
			       gint           color_index)
Michael Natterer's avatar
Michael Natterer committed
363
{
364 365
  GimpCurvesTool *c_tool;
  GimpDrawable   *drawable;
366
  guchar          r, g, b, a;
Michael Natterer's avatar
Michael Natterer committed
367

368 369
  c_tool = GIMP_CURVES_TOOL (color_tool);
  drawable = GIMP_IMAGE_MAP_TOOL (c_tool)->drawable;
370

371
  gimp_rgba_get_uchar (color, &r, &g, &b, &a);
372

373 374 375 376 377 378
  c_tool->col_value[GIMP_HISTOGRAM_RED]   = r;
  c_tool->col_value[GIMP_HISTOGRAM_GREEN] = g;
  c_tool->col_value[GIMP_HISTOGRAM_BLUE]  = b;
  
  if (gimp_drawable_has_alpha (drawable))
    c_tool->col_value[GIMP_HISTOGRAM_ALPHA] = a;
379
      
380 381
  if (gimp_drawable_is_indexed (drawable))
    c_tool->col_value[GIMP_HISTOGRAM_ALPHA] = color_index;
382
      
383
  c_tool->col_value[GIMP_HISTOGRAM_VALUE] = MAX (MAX (r, g), b);
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
384

385
  gtk_widget_queue_draw (c_tool->graph);
386
}
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
387

388
static void
389 390 391 392
curves_add_point (GimpCurvesTool *c_tool,
		  gint            x,
		  gint            y,
		  gint            cchan)
393 394
{
  /* Add point onto the curve */
395 396 397 398
  gint closest_point = 0;
  gint distance;
  gint curvex;
  gint i;
399

400
  switch (c_tool->curves->curve_type[cchan])
401
    {
402
    case GIMP_CURVE_SMOOTH:
403
      curvex   = c_tool->col_value[cchan];
404
      distance = G_MAXINT;
405

406 407
      for (i = 0; i < 17; i++)
	{
408 409
	  if (c_tool->curves->points[cchan][i][0] != -1)
	    if (abs (curvex - c_tool->curves->points[cchan][i][0]) < distance)
410
	      {
411
		distance = abs (curvex - c_tool->curves->points[cchan][i][0]);
412 413 414 415 416 417 418
		closest_point = i;
	      }
	}
      
      if (distance > MIN_DISTANCE)
	closest_point = (curvex + 8) / 16;
      
419 420
      c_tool->curves->points[cchan][closest_point][0] = curvex;
      c_tool->curves->points[cchan][closest_point][1] = c_tool->curves->curve[cchan][curvex];
421 422
      break;
      
423
    case GIMP_CURVE_FREE:
424
      c_tool->curves->curve[cchan][x] = 255 - y;
425
      break;
426
    }
Elliot Lee's avatar
Elliot Lee committed
427 428
}

429 430
static void
gimp_curves_tool_map (GimpImageMapTool *image_map_tool)
Elliot Lee's avatar
Elliot Lee committed
431
{
432
  GimpCurvesTool *c_tool;
433

434
  c_tool = GIMP_CURVES_TOOL (image_map_tool);
435

436 437 438 439
  gimp_lut_setup (c_tool->lut, 
		  (GimpLutFunc) curves_lut_func,
                  c_tool->curves,
		  gimp_drawable_bytes (image_map_tool->drawable));
440

441 442 443
  gimp_image_map_apply (image_map_tool->image_map,
                        (GimpImageMapApplyFunc) gimp_lut_process_2,
                        c_tool->lut);
Elliot Lee's avatar
Elliot Lee committed
444 445
}

446

447 448 449
/*******************/
/*  Curves dialog  */
/*******************/
Elliot Lee's avatar
Elliot Lee committed
450

451 452
static void
gimp_curves_tool_dialog (GimpImageMapTool *image_map_tool)
Elliot Lee's avatar
Elliot Lee committed
453
{
454 455
  GimpCurvesTool *c_tool;
  GtkWidget      *hbox;
456
  GtkWidget      *vbox;
457 458
  GtkWidget      *hbbox;
  GtkWidget      *frame;
459
  GtkWidget      *menu;
460 461
  GtkWidget      *table;
  GtkWidget      *button;
462

463
  c_tool = GIMP_CURVES_TOOL (image_map_tool);
BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
464

465 466 467 468 469 470 471 472 473
  hbox = gtk_hbox_new (FALSE, 0);
  gtk_box_pack_start (GTK_BOX (image_map_tool->main_vbox), hbox,
                      FALSE, FALSE, 0);
  gtk_widget_show (hbox);

  vbox = gtk_vbox_new (FALSE, 4);
  gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, FALSE, 0);
  gtk_widget_show (vbox);

474 475 476
  table = gtk_table_new (2, 2, FALSE);
  gtk_table_set_col_spacings (GTK_TABLE (table), 4);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
477
  gtk_box_pack_start (GTK_BOX (vbox), table, FALSE, FALSE, 0);
478
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
479

480 481
  /*  The option menu for selecting channels  */
  hbox = gtk_hbox_new (FALSE, 4);
482 483 484 485 486 487 488 489 490
  menu = gimp_enum_option_menu_new (GIMP_TYPE_HISTOGRAM_CHANNEL,
                                    G_CALLBACK (curves_channel_callback),
                                    c_tool);
  gimp_enum_option_menu_set_stock_prefix (GTK_OPTION_MENU (menu),
                                          "gimp-channel");
  gtk_box_pack_start (GTK_BOX (hbox), menu, FALSE, FALSE, 0);
  gtk_widget_show (menu);

  c_tool->channel_menu = menu;
491

492
  button = gtk_button_new_with_mnemonic (_("R_eset Channel"));
493 494
  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
  gtk_widget_show (button);
495

496
  g_signal_connect (button, "clicked",
497 498
                    G_CALLBACK (curves_channel_reset_callback),
                    c_tool);
499

500 501 502
  gimp_table_attach_aligned (GTK_TABLE (table), 0, 0,
                             _("Modify Curves for Channel:"), 1.0, 0.5,
                             hbox, 1, FALSE);
Elliot Lee's avatar
Elliot Lee committed
503 504

  /*  The table for the yrange and the graph  */
505 506 507 508
  hbox = gtk_hbox_new (FALSE, 0);
  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
  gtk_widget_show (hbox);

Elliot Lee's avatar
Elliot Lee committed
509
  table = gtk_table_new (2, 2, FALSE);
510 511
  gtk_table_set_col_spacings (GTK_TABLE (table), 2);
  gtk_table_set_row_spacings (GTK_TABLE (table), 2);
512
  gtk_box_pack_start (GTK_BOX (hbox), table, TRUE, FALSE, 0);
Elliot Lee's avatar
Elliot Lee committed
513 514 515

  /*  The range drawing area  */
  frame = gtk_frame_new (NULL);
516
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
517 518 519 520
  gtk_table_attach (GTK_TABLE (table), frame, 0, 1, 0, 1,
		    GTK_EXPAND, GTK_EXPAND, 0, 0);
  gtk_widget_show (frame);

521 522 523 524 525
  c_tool->yrange = gtk_preview_new (GTK_PREVIEW_COLOR);
  gtk_preview_size (GTK_PREVIEW (c_tool->yrange), YRANGE_WIDTH, YRANGE_HEIGHT);
  gtk_container_add (GTK_CONTAINER (frame), c_tool->yrange);
  gtk_widget_show (c_tool->yrange);

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

534 535 536
  c_tool->graph = gtk_drawing_area_new ();
  gtk_widget_set_size_request (c_tool->graph,
                               GRAPH_WIDTH  + RADIUS * 2,
537
                               GRAPH_HEIGHT + RADIUS * 2);
538 539 540
  gtk_widget_set_events (c_tool->graph, GRAPH_MASK);
  gtk_container_add (GTK_CONTAINER (frame), c_tool->graph);
  gtk_widget_show (c_tool->graph);
541

542
  g_signal_connect (c_tool->graph, "event",
543
		    G_CALLBACK (curves_graph_events),
544
		    c_tool);
Elliot Lee's avatar
Elliot Lee committed
545 546 547

  /*  The range drawing area  */
  frame = gtk_frame_new (NULL);
548
  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
Elliot Lee's avatar
Elliot Lee committed
549 550 551 552
  gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 1, 2,
		    GTK_EXPAND, GTK_EXPAND, 0, 0);
  gtk_widget_show (frame);

553 554 555 556
  c_tool->xrange = gtk_preview_new (GTK_PREVIEW_COLOR);
  gtk_preview_size (GTK_PREVIEW (c_tool->xrange), XRANGE_WIDTH, XRANGE_HEIGHT);
  gtk_container_add (GTK_CONTAINER (frame), c_tool->xrange);
  gtk_widget_show (c_tool->xrange);
557

558
  gtk_widget_show (table);
Elliot Lee's avatar
Elliot Lee committed
559

560 561 562 563 564
  hbox = gtk_hbox_new (FALSE, 6);
  gtk_box_pack_end (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
  gtk_widget_show (hbox);
  
  /*  Horizontal button box for load / save */
565
  frame = gtk_frame_new (_("All Channels"));
566
  gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, FALSE, 0);
567 568
  gtk_widget_show (frame);

569
  hbbox = gtk_hbutton_box_new ();
570
  gtk_container_set_border_width (GTK_CONTAINER (hbbox), 2);
571
  gtk_box_set_spacing (GTK_BOX (hbbox), 4);
572
  gtk_button_box_set_layout (GTK_BUTTON_BOX (hbbox), GTK_BUTTONBOX_SPREAD);
573
  gtk_container_add (GTK_CONTAINER (frame), hbbox);
574
  gtk_widget_show (hbbox);
575

576
  button = gtk_button_new_from_stock (GTK_STOCK_OPEN);
577 578
  GTK_WIDGET_SET_FLAGS (button, GTK_CAN_DEFAULT);
  gtk_box_pack_start (GTK_BOX (hbbox), button, FALSE, FALSE, 0);
579
  gimp_help_set_help_data (button, _("Read curves settings from file"), NULL);
580 581
  gtk_widget_show (button);

582
  g_signal_connect (button, "clicked",
583
		    G_CALLBACK (curves_load_callback),
584
		    c_tool);
585

586
  button = gtk_button_new_from_stock (GTK_STOCK_SAVE);
587 588
  GTK_WIDGET_SET_FLAGS (button, GTK_CAN_DEFAULT);
  gtk_box_pack_start (GTK_BOX (hbbox), button, FALSE, FALSE, 0);
589
  gimp_help_set_help_data (button, _("Save curves settings to file"), NULL);
590 591
  gtk_widget_show (button);

592
  g_signal_connect (button, "clicked",
593
		    G_CALLBACK (curves_save_callback),
594
		    c_tool);
595 596 597 598 599 600 601

  /*  The radio box for selecting the curve type  */
  frame = gtk_frame_new (_("Curve Type"));
  gtk_box_pack_end (GTK_BOX (hbox), frame, FALSE, FALSE, 0);
  gtk_widget_show (frame);

  hbox = gimp_enum_stock_box_new (GIMP_TYPE_CURVE_TYPE,
602
				  "gimp-curve", GTK_ICON_SIZE_MENU,
603 604 605 606 607 608 609 610 611
				  G_CALLBACK (curves_curve_type_callback),
				  c_tool,
				  &c_tool->curve_type);

  gtk_container_set_border_width (GTK_CONTAINER (hbox), 2);
  gtk_box_set_spacing (GTK_BOX (hbox), 4);

  gtk_container_add (GTK_CONTAINER (frame), hbox);
  gtk_widget_show (hbox);
Elliot Lee's avatar
Elliot Lee committed
612 613
}

614
static void
615
gimp_curves_tool_reset (GimpImageMapTool *image_map_tool)
616
{
617 618
  GimpCurvesTool       *c_tool;
  GimpHistogramChannel  channel;
619 620 621 622 623

  c_tool = GIMP_CURVES_TOOL (image_map_tool);

  c_tool->grab_point = -1;

624 625 626 627
  for (channel =  GIMP_HISTOGRAM_VALUE;
       channel <= GIMP_HISTOGRAM_ALPHA;
       channel++)
    curves_channel_reset (c_tool->curves, channel);
628

629 630
  curves_update (c_tool, XRANGE_TOP);
  gtk_widget_queue_draw (c_tool->graph);
631 632
}

633

634
/* TODO: preview alpha channel stuff correctly.  -- austin, 20/May/99 */
Elliot Lee's avatar
Elliot Lee committed
635
static void
636 637
curves_update (GimpCurvesTool *c_tool,
	       gint            update)
Elliot Lee's avatar
Elliot Lee committed
638
{
639 640
  GimpHistogramChannel sel_channel;
  gint                 i, j;
641
  
642
  if (c_tool->color)
643
    {
644
      sel_channel = c_tool->channel;
645
    }
646 647
  else 
    {
648
      if (c_tool->channel == 2)
649 650 651 652
        sel_channel = GIMP_HISTOGRAM_ALPHA;
      else
        sel_channel = GIMP_HISTOGRAM_VALUE;
    }
653

Elliot Lee's avatar
Elliot Lee committed
654 655
  if (update & XRANGE_TOP)
    {
656
      guchar buf[XRANGE_WIDTH * 3];
657

658
      switch (sel_channel)
659
	{
Michael Natterer's avatar
Michael Natterer committed
660 661
	case GIMP_HISTOGRAM_VALUE:
	case GIMP_HISTOGRAM_ALPHA:
662
	  for (i = 0; i < XRANGE_HEIGHT / 2; i++)
663
	    {
664
	      for (j = 0; j < XRANGE_WIDTH ; j++)
665
		{
666 667 668
		  buf[j * 3 + 0] = c_tool->curves->curve[sel_channel][j];
		  buf[j * 3 + 1] = c_tool->curves->curve[sel_channel][j];
		  buf[j * 3 + 2] = c_tool->curves->curve[sel_channel][j];
669
		}
670
	      gtk_preview_draw_row (GTK_PREVIEW (c_tool->xrange),
671
				    buf, 0, i, XRANGE_WIDTH);
672
	    }
673 674
	  break;

Michael Natterer's avatar
Michael Natterer committed
675 676 677
	case GIMP_HISTOGRAM_RED:
	case GIMP_HISTOGRAM_GREEN:
	case GIMP_HISTOGRAM_BLUE:
678
	  {
679
	    for (i = 0; i < XRANGE_HEIGHT / 2; i++)
680
	      {
681 682
		for (j = 0; j < XRANGE_WIDTH; j++)
		  {
683 684 685
		    buf[j * 3 + 0] = c_tool->curves->curve[GIMP_HISTOGRAM_RED][j];
		    buf[j * 3 + 1] = c_tool->curves->curve[GIMP_HISTOGRAM_GREEN][j];
		    buf[j * 3 + 2] = c_tool->curves->curve[GIMP_HISTOGRAM_BLUE][j];
686
		  }
687
		gtk_preview_draw_row (GTK_PREVIEW (c_tool->xrange),
688
				      buf, 0, i, XRANGE_WIDTH);
689
	      }
690
	    break;
691 692
	  }

693
	default:
694
	  g_warning ("unknown channel type %d, can't happen!?!?",
695
		     c_tool->channel);
696
	  break;
697
	}
Elliot Lee's avatar
Elliot Lee committed
698

699 700 701
      gtk_widget_queue_draw_area (c_tool->xrange,
                                  0, 0,
                                  XRANGE_WIDTH, XRANGE_HEIGHT / 2);
Elliot Lee's avatar
Elliot Lee committed
702
    }
703

Elliot Lee's avatar
Elliot Lee committed
704 705
  if (update & XRANGE_BOTTOM)
    {
706
      guchar buf[XRANGE_WIDTH * 3];
Elliot Lee's avatar
Elliot Lee committed
707 708

      for (i = 0; i < XRANGE_WIDTH; i++)
709 710 711 712 713
        {
          buf[i * 3 + 0] = i;
          buf[i * 3 + 1] = i;
          buf[i * 3 + 2] = i;
        }
Elliot Lee's avatar
Elliot Lee committed
714 715

      for (i = XRANGE_HEIGHT / 2; i < XRANGE_HEIGHT; i++)
716 717
	gtk_preview_draw_row (GTK_PREVIEW (c_tool->xrange),
                              buf, 0, i, XRANGE_WIDTH);
Elliot Lee's avatar
Elliot Lee committed
718

719 720 721
      gtk_widget_queue_draw_area (c_tool->xrange,
                                  0, XRANGE_HEIGHT / 2,
                                  XRANGE_WIDTH, XRANGE_HEIGHT / 2);
Elliot Lee's avatar
Elliot Lee committed
722
    }
723

Elliot Lee's avatar
Elliot Lee committed
724 725
  if (update & YRANGE)
    {
726 727
      guchar buf[YRANGE_WIDTH * 3];
      guchar pix[3];
Elliot Lee's avatar
Elliot Lee committed
728 729 730

      for (i = 0; i < YRANGE_HEIGHT; i++)
	{
731
	  switch (sel_channel)
732
	    {
Michael Natterer's avatar
Michael Natterer committed
733 734
	    case GIMP_HISTOGRAM_VALUE:
	    case GIMP_HISTOGRAM_ALPHA:
735 736 737
	      pix[0] = pix[1] = pix[2] = (255 - i);
	      break;

Michael Natterer's avatar
Michael Natterer committed
738 739 740
	    case GIMP_HISTOGRAM_RED:
	    case GIMP_HISTOGRAM_GREEN:
	    case GIMP_HISTOGRAM_BLUE:
741
	      pix[0] = pix[1] = pix[2] = 0;
742
	      pix[sel_channel - 1] = (255 - i);
743 744
	      break;

745
	    default:
746
	      g_warning ("unknown channel type %d, can't happen!?!?",
747
			 c_tool->channel);
748
	      break;
749
	    }
750 751

	  for (j = 0; j < YRANGE_WIDTH * 3; j++)
752
	    buf[j] = pix[j%3];
Elliot Lee's avatar
Elliot Lee committed
753

754
	  gtk_preview_draw_row (GTK_PREVIEW (c_tool->yrange),
755
				buf, 0, i, YRANGE_WIDTH);
Elliot Lee's avatar
Elliot Lee committed
756 757
	}

758
      gtk_widget_queue_draw (c_tool->yrange);
Elliot Lee's avatar
Elliot Lee committed
759 760 761 762
    }
}

static void
763 764
curves_channel_callback (GtkWidget      *widget,
			 GimpCurvesTool *c_tool)
Elliot Lee's avatar
Elliot Lee committed
765
{
766
  gimp_menu_item_update (widget, &c_tool->channel);
Elliot Lee's avatar
Elliot Lee committed
767

768
  if (! c_tool->color)
Elliot Lee's avatar
Elliot Lee committed
769
    {
770 771 772 773
      if (c_tool->channel > 1)
        c_tool->channel = 2;
      else 
        c_tool->channel = 1;
Elliot Lee's avatar
Elliot Lee committed
774 775
    }

776 777
  gimp_radio_group_set_active (GTK_RADIO_BUTTON (c_tool->curve_type),
			       GINT_TO_POINTER (c_tool->curves->curve_type[c_tool->channel]));
778

779 780
  curves_update (c_tool, XRANGE_TOP | YRANGE);
  gtk_widget_queue_draw (c_tool->graph);
Elliot Lee's avatar
Elliot Lee committed
781 782 783
}

static void
784 785
curves_channel_reset_callback (GtkWidget      *widget,
                               GimpCurvesTool *c_tool)
Elliot Lee's avatar
Elliot Lee committed
786
{
787
  c_tool->grab_point = -1;
Elliot Lee's avatar
Elliot Lee committed
788

789
  curves_channel_reset (c_tool->curves, c_tool->channel);
Elliot Lee's avatar
Elliot Lee committed
790

791 792
  curves_update (c_tool, XRANGE_TOP);
  gtk_widget_queue_draw (c_tool->graph);
Elliot Lee's avatar
Elliot Lee committed
793

794
  gimp_image_map_tool_preview (GIMP_IMAGE_MAP_TOOL (c_tool));
Elliot Lee's avatar
Elliot Lee committed
795 796
}

797
static gboolean
798 799
curves_set_sensitive_callback (gpointer        item_data,
                               GimpCurvesTool *c_tool)
800 801
{
  GimpHistogramChannel  channel = GPOINTER_TO_INT (item_data);
802

803 804 805 806 807 808 809
  switch (channel)
    {
    case GIMP_HISTOGRAM_VALUE:
      return TRUE;
    case GIMP_HISTOGRAM_RED:
    case GIMP_HISTOGRAM_GREEN:
    case GIMP_HISTOGRAM_BLUE:
810
      return c_tool->color;
811
    case GIMP_HISTOGRAM_ALPHA:
812
      return gimp_drawable_has_alpha (GIMP_IMAGE_MAP_TOOL (c_tool)->drawable);
813
    }
814

815 816 817
  return FALSE;
}

Elliot Lee's avatar
Elliot Lee committed
818
static void
819 820
curves_curve_type_callback (GtkWidget      *widget,
			    GimpCurvesTool *c_tool)
Elliot Lee's avatar
Elliot Lee committed
821
{
822
  GimpCurveType curve_type;
Elliot Lee's avatar
Elliot Lee committed
823

824
  gimp_radio_button_update (widget, &curve_type);
Elliot Lee's avatar
Elliot Lee committed
825

826
  if (c_tool->curves->curve_type[c_tool->channel] != curve_type)
Elliot Lee's avatar
Elliot Lee committed
827
    {
828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844
      c_tool->curves->curve_type[c_tool->channel] = curve_type;

      if (curve_type == GIMP_CURVE_SMOOTH)
        {
          gint   i;
          gint32 index;

          /*  pick representative points from the curve
           *  and make them control points
           */
          for (i = 0; i <= 8; i++)
            {
              index = CLAMP0255 (i * 32);
              c_tool->curves->points[c_tool->channel][i * 2][0] = index;
              c_tool->curves->points[c_tool->channel][i * 2][1] = c_tool->curves->curve[c_tool->channel][index];
            }
        }
845