1997-11-25 06:05:25 +08:00
|
|
|
/* 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
|
1998-04-13 13:44:11 +08:00
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
1997-11-25 06:05:25 +08:00
|
|
|
*/
|
app/appenv.h New file. Includes <math.h>. Move G_PI, RINT(), ROUND() etc
1999-09-01 Tor Lillqvist <tml@iki.fi>
* app/appenv.h
* libgimp/gimpmath.h: New file. Includes <math.h>. Move G_PI,
RINT(), ROUND() etc from app/appenv.h here, so plug-ins can
use them, too. Remove some commented-out old stuff in appenv.h.
* libgimp/gimp.h: Include gimpmath.h.
* libgimp/gimp.c (gimp_main): Win32: Don't install signal
handlers, we can't do anything useful in the handler ourselves
anyway (it would be nice to print out a backtrace, but that seems
pretty hard to do, even if not impossible). Let Windows inform the
user about the crash. If the plug-in was compiled with MSVC, and
the user also has it, she is offered a chance to start the
debugger automatically anyway.
* app/*several*.c: Include gimpmath.h for G_PI etc. Don't include
<math.h>, as gimpmath.h includes it.
* plug-ins/*/*many*.c: Include config.h. Don't include <math.h>.
Remove all the duplicated definitions of G_PI and rint(). Use
RINT() instead of rint().
* app/app_procs.[ch]: app_exit() takes a gboolean.
* app/batch.c
* app/commands.c
* app/interface.c: Call app_exit() with FALSE or TRUE.
* app/main.c (on_error): Call gimp_fatal_error. (main): Don't
install any signal handler on Win32 here, either.
* app/errors.c (gimp_fatal_error, gimp_terminate): Win32: Format
the message and call MessageBox with it. g_on_error_query doesn't
do anything useful on Win32, and printf'ing a message to stdout or
stderr doesn't do anything, either, in a windowing application.
1999-09-02 04:30:56 +08:00
|
|
|
#include "config.h"
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
app/appenv.h New file. Includes <math.h>. Move G_PI, RINT(), ROUND() etc
1999-09-01 Tor Lillqvist <tml@iki.fi>
* app/appenv.h
* libgimp/gimpmath.h: New file. Includes <math.h>. Move G_PI,
RINT(), ROUND() etc from app/appenv.h here, so plug-ins can
use them, too. Remove some commented-out old stuff in appenv.h.
* libgimp/gimp.h: Include gimpmath.h.
* libgimp/gimp.c (gimp_main): Win32: Don't install signal
handlers, we can't do anything useful in the handler ourselves
anyway (it would be nice to print out a backtrace, but that seems
pretty hard to do, even if not impossible). Let Windows inform the
user about the crash. If the plug-in was compiled with MSVC, and
the user also has it, she is offered a chance to start the
debugger automatically anyway.
* app/*several*.c: Include gimpmath.h for G_PI etc. Don't include
<math.h>, as gimpmath.h includes it.
* plug-ins/*/*many*.c: Include config.h. Don't include <math.h>.
Remove all the duplicated definitions of G_PI and rint(). Use
RINT() instead of rint().
* app/app_procs.[ch]: app_exit() takes a gboolean.
* app/batch.c
* app/commands.c
* app/interface.c: Call app_exit() with FALSE or TRUE.
* app/main.c (on_error): Call gimp_fatal_error. (main): Don't
install any signal handler on Win32 here, either.
* app/errors.c (gimp_fatal_error, gimp_terminate): Win32: Format
the message and call MessageBox with it. g_on_error_query doesn't
do anything useful on Win32, and printf'ing a message to stdout or
stderr doesn't do anything, either, in a windowing application.
1999-09-02 04:30:56 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
#include "appenv.h"
|
|
|
|
#include "actionarea.h"
|
|
|
|
#include "buildmenu.h"
|
|
|
|
#include "colormaps.h"
|
|
|
|
#include "cursorutil.h"
|
|
|
|
#include "drawable.h"
|
|
|
|
#include "general.h"
|
|
|
|
#include "gdisplay.h"
|
added gimphistogram*, histogramwidget*, removed histogram.[ch]
* app/Makefile.am: added gimphistogram*, histogramwidget*,
removed histogram.[ch]
* app/histogram.[ch]: removed. replaced with histogramwidget.[ch].
* app/{gimphistogramP.h, gimphistogram.h, gimphistogram.c}: new
functions that calculate histograms in parallel and perform
calculations on them.
* app/histogramwidget.[ch]: Same as old histogram.[ch], only it is
now a real widget, and it uses GimpHistograms instead of arrays of
values.
* app/curves.c: #include gimphistogram.h instead of histogram.h.
* app/equalize.c: use GimpHistogram and GimpLut.
* app/gimpbrush.c, app/gimpimage.c, app/gimpset.c: use
GTK_RUN_FIRST in calls to gimp_signal_new.
* app/histogram_tool.c, app/levels.c, app/threshold.c: modified to
use the new HistogramWidget.
* app/paint_funcs.c: removed some unused variables.
* app/preferences_dialog.c: only display the num-processor field
if we are configured --with-mp
* plug-ins/gee/gee.c: removed a couple of c++ style comments.
1999-03-01 13:11:19 +08:00
|
|
|
#include "gimphistogram.h"
|
1997-11-25 06:05:25 +08:00
|
|
|
#include "interface.h"
|
|
|
|
#include "curves.h"
|
1999-02-16 16:53:54 +08:00
|
|
|
#include "gimplut.h"
|
1997-11-25 06:05:25 +08:00
|
|
|
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
#include "libgimp/gimpintl.h"
|
app/appenv.h New file. Includes <math.h>. Move G_PI, RINT(), ROUND() etc
1999-09-01 Tor Lillqvist <tml@iki.fi>
* app/appenv.h
* libgimp/gimpmath.h: New file. Includes <math.h>. Move G_PI,
RINT(), ROUND() etc from app/appenv.h here, so plug-ins can
use them, too. Remove some commented-out old stuff in appenv.h.
* libgimp/gimp.h: Include gimpmath.h.
* libgimp/gimp.c (gimp_main): Win32: Don't install signal
handlers, we can't do anything useful in the handler ourselves
anyway (it would be nice to print out a backtrace, but that seems
pretty hard to do, even if not impossible). Let Windows inform the
user about the crash. If the plug-in was compiled with MSVC, and
the user also has it, she is offered a chance to start the
debugger automatically anyway.
* app/*several*.c: Include gimpmath.h for G_PI etc. Don't include
<math.h>, as gimpmath.h includes it.
* plug-ins/*/*many*.c: Include config.h. Don't include <math.h>.
Remove all the duplicated definitions of G_PI and rint(). Use
RINT() instead of rint().
* app/app_procs.[ch]: app_exit() takes a gboolean.
* app/batch.c
* app/commands.c
* app/interface.c: Call app_exit() with FALSE or TRUE.
* app/main.c (on_error): Call gimp_fatal_error. (main): Don't
install any signal handler on Win32 here, either.
* app/errors.c (gimp_fatal_error, gimp_terminate): Win32: Format
the message and call MessageBox with it. g_on_error_query doesn't
do anything useful on Win32, and printf'ing a message to stdout or
stderr doesn't do anything, either, in a windowing application.
1999-09-02 04:30:56 +08:00
|
|
|
#include "libgimp/gimpmath.h"
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
#define GRAPH 0x1
|
|
|
|
#define XRANGE_TOP 0x2
|
|
|
|
#define XRANGE_BOTTOM 0x4
|
|
|
|
#define YRANGE 0x8
|
|
|
|
#define DRAW 0x10
|
|
|
|
#define ALL 0xFF
|
|
|
|
|
1999-06-23 23:24:46 +08:00
|
|
|
/* NB: take care when changing these values: make sure the curve[] array in
|
|
|
|
* curves.h is large enough.
|
|
|
|
*/
|
1997-11-25 06:05:25 +08:00
|
|
|
#define GRAPH_WIDTH 256
|
|
|
|
#define GRAPH_HEIGHT 256
|
|
|
|
#define XRANGE_WIDTH 256
|
|
|
|
#define XRANGE_HEIGHT 16
|
|
|
|
#define YRANGE_WIDTH 16
|
|
|
|
#define YRANGE_HEIGHT 256
|
|
|
|
#define RADIUS 3
|
|
|
|
#define MIN_DISTANCE 8
|
|
|
|
|
|
|
|
#define RANGE_MASK GDK_EXPOSURE_MASK | \
|
|
|
|
GDK_ENTER_NOTIFY_MASK
|
|
|
|
|
|
|
|
#define GRAPH_MASK GDK_EXPOSURE_MASK | \
|
|
|
|
GDK_POINTER_MOTION_MASK | \
|
|
|
|
GDK_POINTER_MOTION_HINT_MASK | \
|
|
|
|
GDK_ENTER_NOTIFY_MASK | \
|
|
|
|
GDK_BUTTON_PRESS_MASK | \
|
|
|
|
GDK_BUTTON_RELEASE_MASK | \
|
|
|
|
GDK_BUTTON1_MOTION_MASK
|
|
|
|
|
1999-04-09 06:25:54 +08:00
|
|
|
/* the curves structures */
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-04-09 06:25:54 +08:00
|
|
|
typedef struct _Curves Curves;
|
1997-11-25 06:05:25 +08:00
|
|
|
struct _Curves
|
|
|
|
{
|
|
|
|
int x, y; /* coords for last mouse click */
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef double CRMatrix[4][4];
|
|
|
|
|
1999-04-13 01:55:06 +08:00
|
|
|
|
1999-04-09 06:25:54 +08:00
|
|
|
/* the curves tool options */
|
1999-04-13 01:55:06 +08:00
|
|
|
static ToolOptions * curves_options = NULL;
|
1999-04-09 06:25:54 +08:00
|
|
|
|
|
|
|
/* the curves dialog */
|
1999-04-13 01:55:06 +08:00
|
|
|
static CurvesDialog * curves_dialog = NULL;
|
1999-04-09 06:25:54 +08:00
|
|
|
|
|
|
|
static CRMatrix CR_basis =
|
|
|
|
{
|
|
|
|
{ -0.5, 1.5, -1.5, 0.5 },
|
|
|
|
{ 1.0, -2.5, 2.0, -0.5 },
|
|
|
|
{ -0.5, 0.0, 0.5, 0.0 },
|
|
|
|
{ 0.0, 1.0, 0.0, 0.0 },
|
|
|
|
};
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-04-09 06:25:54 +08:00
|
|
|
/* curves action functions */
|
1999-06-22 06:12:07 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
static void curves_button_press (Tool *, GdkEventButton *, gpointer);
|
|
|
|
static void curves_button_release (Tool *, GdkEventButton *, gpointer);
|
|
|
|
static void curves_motion (Tool *, GdkEventMotion *, gpointer);
|
1999-06-22 06:12:07 +08:00
|
|
|
static void curves_control (Tool *, ToolAction, gpointer);
|
|
|
|
|
1999-06-23 23:24:46 +08:00
|
|
|
static CurvesDialog * curves_new_dialog (void);
|
|
|
|
|
|
|
|
static void curves_update (CurvesDialog *, int);
|
|
|
|
static void curves_plot_curve (CurvesDialog *, int, int, int, int);
|
|
|
|
static void curves_preview (CurvesDialog *);
|
|
|
|
static void curves_value_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_red_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_green_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_blue_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_alpha_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_smooth_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_free_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_reset_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_ok_callback (GtkWidget *, gpointer);
|
|
|
|
static void curves_cancel_callback (GtkWidget *, gpointer);
|
|
|
|
static gint curves_delete_callback (GtkWidget *, GdkEvent *, gpointer);
|
|
|
|
static void curves_preview_update (GtkWidget *, gpointer);
|
|
|
|
static gint curves_xrange_events (GtkWidget *, GdkEvent *, CurvesDialog *);
|
|
|
|
static gint curves_yrange_events (GtkWidget *, GdkEvent *, CurvesDialog *);
|
|
|
|
static gint curves_graph_events (GtkWidget *, GdkEvent *, CurvesDialog *);
|
|
|
|
static void curves_CR_compose (CRMatrix, CRMatrix, CRMatrix);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-04-09 06:25:54 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
/* curves machinery */
|
|
|
|
|
moved a bunch of PDB stuff here
* app/color_cmds.c: moved a bunch of PDB stuff here
* app/color_balance.[ch]: removed PDB proc, exported TransferMode
enum, ColorBalanceDialog, color_balance_create_lookup_tables, and
color_balance
* app/curves.[ch]: removed PDB procs, exported SMOOTH and GFREE
#defines, CurvesDialog, curves_lut_func and curves_calculate_curve
* app/desaturate.[ch]: removed PDB proc, exported desaturate
* app/equalize.[ch]: removed PDB proc, exported equalize
* app/histogram_tool.[ch]: removed PDB proc, exported HISTOGRAM_WIDTH
and HISTOGRAM_HEIGHT #defines, HistogramToolDialog,
histogram_tool_histogram_range
* app/hue_saturation.[ch]: removed PDB proc, exported HueRange enum,
HueSaturationDialog, hue_saturation_calculate_transfers,
hue_saturation
* app/invert.[ch]: remove PDB proc, export invert
* app/threshold.[ch]: remove PDB proc, export ThresholdDialog and
threshold_2
* internal_procs.c: changes for pdbgen
* app/gimprc.c: removed leftover declaration
* app/image_map.h: add #include "gimpdrawableF.h"
* app/lut_funcs.h: add ALPHA_LUT to ChannelLutType
-Yosh
1999-04-21 13:39:57 +08:00
|
|
|
float
|
|
|
|
curves_lut_func (CurvesDialog *cd,
|
|
|
|
int nchannels, int channel, float value)
|
1997-11-25 06:05:25 +08:00
|
|
|
{
|
1999-02-16 16:53:54 +08:00
|
|
|
float f;
|
|
|
|
int index;
|
|
|
|
double inten;
|
|
|
|
int j;
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-02-16 16:53:54 +08:00
|
|
|
if (nchannels == 1)
|
|
|
|
j = 0;
|
|
|
|
else
|
|
|
|
j = channel + 1;
|
|
|
|
inten = value;
|
moved a bunch of PDB stuff here
* app/color_cmds.c: moved a bunch of PDB stuff here
* app/color_balance.[ch]: removed PDB proc, exported TransferMode
enum, ColorBalanceDialog, color_balance_create_lookup_tables, and
color_balance
* app/curves.[ch]: removed PDB procs, exported SMOOTH and GFREE
#defines, CurvesDialog, curves_lut_func and curves_calculate_curve
* app/desaturate.[ch]: removed PDB proc, exported desaturate
* app/equalize.[ch]: removed PDB proc, exported equalize
* app/histogram_tool.[ch]: removed PDB proc, exported HISTOGRAM_WIDTH
and HISTOGRAM_HEIGHT #defines, HistogramToolDialog,
histogram_tool_histogram_range
* app/hue_saturation.[ch]: removed PDB proc, exported HueRange enum,
HueSaturationDialog, hue_saturation_calculate_transfers,
hue_saturation
* app/invert.[ch]: remove PDB proc, export invert
* app/threshold.[ch]: remove PDB proc, export ThresholdDialog and
threshold_2
* internal_procs.c: changes for pdbgen
* app/gimprc.c: removed leftover declaration
* app/image_map.h: add #include "gimpdrawableF.h"
* app/lut_funcs.h: add ALPHA_LUT to ChannelLutType
-Yosh
1999-04-21 13:39:57 +08:00
|
|
|
/* For color images this runs through the loop with j = channel +1
|
1999-02-16 16:53:54 +08:00
|
|
|
the first time and j = 0 the second time */
|
|
|
|
/* For bw images this runs through the loop with j = 0 the first and
|
|
|
|
only time */
|
|
|
|
for (; j >= 0; j -= (channel + 1))
|
|
|
|
{
|
|
|
|
/* don't apply the overall curve to the alpha channel */
|
|
|
|
if (j == 0 && (nchannels == 2 || nchannels == 4)
|
|
|
|
&& channel == nchannels -1)
|
|
|
|
return inten;
|
|
|
|
|
|
|
|
if (inten < 0.0)
|
|
|
|
inten = cd->curve[j][0]/255.0;
|
|
|
|
else if (inten >= 1.0)
|
|
|
|
inten = cd->curve[j][255]/255.0;
|
|
|
|
else /* interpolate the curve */
|
1997-11-25 06:05:25 +08:00
|
|
|
{
|
1999-02-16 16:53:54 +08:00
|
|
|
index = floor(inten * 255.0);
|
|
|
|
f = inten*255.0 - index;
|
|
|
|
inten = ((1.0 - f) * cd->curve[j][index ] +
|
|
|
|
( f) * cd->curve[j][index + 1] ) / 255.0;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
1999-02-16 16:53:54 +08:00
|
|
|
}
|
|
|
|
return inten;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
1999-04-08 06:02:26 +08:00
|
|
|
static void
|
1999-06-22 06:12:07 +08:00
|
|
|
curves_colour_update (Tool *tool,
|
|
|
|
GDisplay *gdisp,
|
|
|
|
GimpDrawable *drawable,
|
|
|
|
gint x,
|
|
|
|
gint y)
|
1999-04-08 06:02:26 +08:00
|
|
|
{
|
|
|
|
unsigned char *color;
|
|
|
|
int offx, offy;
|
|
|
|
int has_alpha;
|
|
|
|
int is_indexed;
|
|
|
|
int sample_type;
|
|
|
|
int maxval;
|
|
|
|
|
|
|
|
if(!tool || tool->state != ACTIVE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
drawable_offsets (drawable, &offx, &offy);
|
|
|
|
|
|
|
|
x -= offx;
|
|
|
|
y -= offy;
|
|
|
|
|
|
|
|
if (!(color = image_map_get_color_at(curves_dialog->image_map, x, y)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
sample_type = gimp_drawable_type(drawable);
|
|
|
|
is_indexed = gimp_drawable_indexed (drawable);
|
|
|
|
has_alpha = TYPE_HAS_ALPHA(sample_type);
|
|
|
|
|
|
|
|
curves_dialog->col_value[HISTOGRAM_RED] = color[RED_PIX];
|
|
|
|
curves_dialog->col_value[HISTOGRAM_GREEN] = color[GREEN_PIX];
|
|
|
|
curves_dialog->col_value[HISTOGRAM_BLUE] = color[BLUE_PIX];
|
|
|
|
|
|
|
|
if (has_alpha)
|
|
|
|
{
|
|
|
|
curves_dialog->col_value [HISTOGRAM_ALPHA] = color[3];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_indexed)
|
|
|
|
curves_dialog->col_value [HISTOGRAM_ALPHA] = color[4];
|
|
|
|
|
|
|
|
maxval = MAXIMUM(color[RED_PIX],color[GREEN_PIX]);
|
|
|
|
curves_dialog->col_value[HISTOGRAM_VALUE] = MAXIMUM(maxval,color[BLUE_PIX]);
|
|
|
|
|
|
|
|
g_free(color);
|
1999-05-14 06:53:40 +08:00
|
|
|
}
|
1999-04-08 06:02:26 +08:00
|
|
|
|
1999-05-14 06:53:40 +08:00
|
|
|
static void
|
|
|
|
curves_add_point(GimpDrawable * drawable,gint x, gint y,gint cchan)
|
|
|
|
{
|
|
|
|
/* Add point onto the curve */
|
|
|
|
int closest_point = 0;
|
|
|
|
int distance;
|
|
|
|
int curvex;
|
|
|
|
int i;
|
|
|
|
|
1999-09-04 05:58:41 +08:00
|
|
|
switch (curves_dialog->curve_type)
|
1999-05-14 06:53:40 +08:00
|
|
|
{
|
1999-09-04 05:58:41 +08:00
|
|
|
case SMOOTH:
|
|
|
|
curvex = curves_dialog->col_value[cchan];
|
|
|
|
distance = G_MAXINT;
|
|
|
|
for (i = 0; i < 17; i++)
|
|
|
|
{
|
|
|
|
if (curves_dialog->points[cchan][i][0] != -1)
|
|
|
|
if (abs (curvex - curves_dialog->points[cchan][i][0]) < distance)
|
|
|
|
{
|
|
|
|
distance = abs (curvex - curves_dialog->points[cchan][i][0]);
|
|
|
|
closest_point = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (distance > MIN_DISTANCE)
|
|
|
|
closest_point = (curvex + 8) / 16;
|
|
|
|
|
|
|
|
curves_dialog->points[cchan][closest_point][0] = curvex;
|
|
|
|
curves_dialog->points[cchan][closest_point][1] = curves_dialog->curve[cchan][curvex];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GFREE:
|
|
|
|
curves_dialog->curve[cchan][x] = 255 - y;
|
|
|
|
break;
|
1999-05-14 06:53:40 +08:00
|
|
|
}
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
/* curves action functions */
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_button_press (Tool *tool,
|
|
|
|
GdkEventButton *bevent,
|
|
|
|
gpointer gdisp_ptr)
|
|
|
|
{
|
|
|
|
GDisplay *gdisp;
|
|
|
|
gint x, y;
|
|
|
|
GimpDrawable * drawable;
|
|
|
|
|
|
|
|
gdisp = gdisp_ptr;
|
|
|
|
drawable = gimage_active_drawable (gdisp->gimage);
|
|
|
|
|
|
|
|
tool->gdisp_ptr = gdisp;
|
|
|
|
|
|
|
|
if (drawable != tool->drawable)
|
|
|
|
{
|
|
|
|
active_tool->preserve = TRUE;
|
|
|
|
image_map_abort (curves_dialog->image_map);
|
|
|
|
active_tool->preserve = FALSE;
|
|
|
|
|
|
|
|
tool->drawable = drawable;
|
|
|
|
|
|
|
|
curves_dialog->drawable = drawable;
|
|
|
|
curves_dialog->color = drawable_color (drawable);
|
|
|
|
curves_dialog->image_map = image_map_create (gdisp, drawable);
|
|
|
|
}
|
|
|
|
|
|
|
|
tool->state = ACTIVE;
|
|
|
|
|
|
|
|
gdisplay_untransform_coords (gdisp, bevent->x, bevent->y, &x, &y,
|
|
|
|
FALSE, FALSE);
|
|
|
|
curves_colour_update (tool, gdisp, drawable, x, y);
|
|
|
|
curves_update (curves_dialog, GRAPH | DRAW);
|
|
|
|
}
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
static void
|
|
|
|
curves_button_release (Tool *tool,
|
|
|
|
GdkEventButton *bevent,
|
|
|
|
gpointer gdisp_ptr)
|
|
|
|
{
|
1999-04-08 06:02:26 +08:00
|
|
|
gint x, y;
|
|
|
|
GimpDrawable * drawable;
|
|
|
|
GDisplay *gdisp;
|
|
|
|
|
|
|
|
gdisp = (GDisplay *) gdisp_ptr;
|
|
|
|
|
|
|
|
if(!curves_dialog ||
|
|
|
|
!gdisp ||
|
|
|
|
!(drawable = gimage_active_drawable (gdisp->gimage)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
gdisplay_untransform_coords (gdisp, bevent->x, bevent->y, &x, &y, FALSE, FALSE);
|
|
|
|
curves_colour_update(tool,gdisp,drawable,x,y);
|
1999-05-14 06:53:40 +08:00
|
|
|
|
|
|
|
if(bevent->state & GDK_SHIFT_MASK)
|
|
|
|
{
|
|
|
|
curves_add_point(drawable,x,y,curves_dialog->channel);
|
|
|
|
curves_calculate_curve(curves_dialog);
|
|
|
|
}
|
|
|
|
else if(bevent->state & GDK_CONTROL_MASK)
|
|
|
|
{
|
|
|
|
curves_add_point(drawable,x,y,HISTOGRAM_VALUE);
|
|
|
|
curves_add_point(drawable,x,y,HISTOGRAM_RED);
|
|
|
|
curves_add_point(drawable,x,y,HISTOGRAM_GREEN);
|
|
|
|
curves_add_point(drawable,x,y,HISTOGRAM_BLUE);
|
|
|
|
curves_add_point(drawable,x,y,HISTOGRAM_ALPHA);
|
|
|
|
curves_calculate_curve(curves_dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
curves_update (curves_dialog, GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_motion (Tool *tool,
|
|
|
|
GdkEventMotion *mevent,
|
|
|
|
gpointer gdisp_ptr)
|
|
|
|
{
|
1999-04-08 06:02:26 +08:00
|
|
|
GDisplay *gdisp;
|
|
|
|
gint x, y;
|
|
|
|
GimpDrawable * drawable;
|
|
|
|
|
|
|
|
gdisp = (GDisplay *) gdisp_ptr;
|
|
|
|
|
|
|
|
if(!curves_dialog ||
|
|
|
|
!gdisp ||
|
|
|
|
!(drawable = gimage_active_drawable (gdisp->gimage)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
gdisplay_untransform_coords (gdisp, mevent->x, mevent->y, &x, &y, FALSE, FALSE);
|
|
|
|
curves_colour_update(tool,gdisp,drawable,x,y);
|
1999-05-14 06:53:40 +08:00
|
|
|
curves_update (curves_dialog, GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1999-06-22 06:12:07 +08:00
|
|
|
curves_control (Tool *tool,
|
|
|
|
ToolAction action,
|
|
|
|
gpointer gdisp_ptr)
|
1997-11-25 06:05:25 +08:00
|
|
|
{
|
|
|
|
switch (action)
|
|
|
|
{
|
1999-06-23 23:24:46 +08:00
|
|
|
case PAUSE:
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
1999-06-22 06:12:07 +08:00
|
|
|
|
1999-06-23 23:24:46 +08:00
|
|
|
case RESUME:
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
1999-06-22 06:12:07 +08:00
|
|
|
|
1999-06-23 23:24:46 +08:00
|
|
|
case HALT:
|
1997-11-25 06:05:25 +08:00
|
|
|
if (curves_dialog)
|
1999-06-23 23:24:46 +08:00
|
|
|
curves_cancel_callback (NULL, (gpointer) curves_dialog);
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
1999-06-22 06:12:07 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Tool *
|
|
|
|
tools_new_curves ()
|
|
|
|
{
|
|
|
|
Tool * tool;
|
|
|
|
Curves * private;
|
|
|
|
|
|
|
|
/* The tool options */
|
|
|
|
if (!curves_options)
|
1999-04-09 06:25:54 +08:00
|
|
|
{
|
1999-04-13 01:55:06 +08:00
|
|
|
curves_options = tool_options_new (_("Curves Options"));
|
|
|
|
tools_register (CURVES, curves_options);
|
1999-04-09 06:25:54 +08:00
|
|
|
}
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
tool = tools_new_tool (CURVES);
|
|
|
|
private = g_new (Curves, 1);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
tool->scroll_lock = TRUE; /* Disallow scrolling */
|
|
|
|
tool->preserve = FALSE; /* Don't preserve on drawable change */
|
1999-04-09 06:25:54 +08:00
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
tool->private = (void *) private;
|
1999-06-26 19:16:47 +08:00
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
tool->button_press_func = curves_button_press;
|
1997-11-25 06:05:25 +08:00
|
|
|
tool->button_release_func = curves_button_release;
|
1999-07-03 01:40:10 +08:00
|
|
|
tool->motion_func = curves_motion;
|
|
|
|
tool->control_func = curves_control;
|
1999-06-23 23:24:46 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
return tool;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tools_free_curves (Tool *tool)
|
|
|
|
{
|
|
|
|
Curves * _curves;
|
|
|
|
|
|
|
|
_curves = (Curves *) tool->private;
|
|
|
|
|
|
|
|
/* Close the color select dialog */
|
|
|
|
if (curves_dialog)
|
1998-04-02 12:51:44 +08:00
|
|
|
curves_cancel_callback (NULL, (gpointer) curves_dialog);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
g_free (_curves);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MenuItem channel_items[] =
|
|
|
|
{
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
{ N_("Value"), 0, 0, curves_value_callback, NULL, NULL, NULL },
|
|
|
|
{ N_("Red"), 0, 0, curves_red_callback, NULL, NULL, NULL },
|
|
|
|
{ N_("Green"), 0, 0, curves_green_callback, NULL, NULL, NULL },
|
|
|
|
{ N_("Blue"), 0, 0, curves_blue_callback, NULL, NULL, NULL },
|
|
|
|
{ N_("Alpha"), 0, 0, curves_alpha_callback, NULL, NULL, NULL },
|
1997-11-25 06:05:25 +08:00
|
|
|
{ NULL, 0, 0, NULL, NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
1998-06-22 04:17:21 +08:00
|
|
|
curves_initialize (GDisplay *gdisp)
|
1997-11-25 06:05:25 +08:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
if (drawable_indexed (gimage_active_drawable (gdisp->gimage)))
|
|
|
|
{
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
g_message (_("Curves for indexed drawables cannot be adjusted."));
|
1997-11-25 06:05:25 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The curves dialog */
|
|
|
|
if (!curves_dialog)
|
|
|
|
curves_dialog = curves_new_dialog ();
|
|
|
|
|
|
|
|
/* Initialize the values */
|
|
|
|
curves_dialog->channel = HISTOGRAM_VALUE;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
for (j = 0; j < 256; j++)
|
|
|
|
curves_dialog->curve[i][j] = j;
|
|
|
|
|
|
|
|
curves_dialog->grab_point = -1;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
{
|
|
|
|
for (j = 0; j < 17; j++)
|
|
|
|
{
|
|
|
|
curves_dialog->points[i][j][0] = -1;
|
|
|
|
curves_dialog->points[i][j][1] = -1;
|
|
|
|
}
|
|
|
|
curves_dialog->points[i][0][0] = 0;
|
|
|
|
curves_dialog->points[i][0][1] = 0;
|
|
|
|
curves_dialog->points[i][16][0] = 255;
|
|
|
|
curves_dialog->points[i][16][1] = 255;
|
|
|
|
}
|
|
|
|
|
1998-01-22 15:02:57 +08:00
|
|
|
curves_dialog->drawable = gimage_active_drawable (gdisp->gimage);
|
|
|
|
curves_dialog->color = drawable_color ( (curves_dialog->drawable));
|
1998-06-22 04:17:21 +08:00
|
|
|
curves_dialog->image_map = image_map_create (gdisp, curves_dialog->drawable);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
/* check for alpha channel */
|
1998-01-22 15:02:57 +08:00
|
|
|
if (drawable_has_alpha ( (curves_dialog->drawable)))
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_widget_set_sensitive( channel_items[4].widget, TRUE);
|
|
|
|
else
|
|
|
|
gtk_widget_set_sensitive( channel_items[4].widget, FALSE);
|
|
|
|
|
|
|
|
/* hide or show the channel menu based on image type */
|
|
|
|
if (curves_dialog->color)
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
gtk_widget_set_sensitive( channel_items[i].widget, TRUE);
|
|
|
|
else
|
|
|
|
for (i = 1; i < 4; i++)
|
|
|
|
gtk_widget_set_sensitive( channel_items[i].widget, FALSE);
|
|
|
|
|
|
|
|
/* set the current selection */
|
1999-06-26 19:16:47 +08:00
|
|
|
gtk_option_menu_set_history (GTK_OPTION_MENU (curves_dialog->channel_menu), 0);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1998-04-30 15:16:54 +08:00
|
|
|
if (!GTK_WIDGET_VISIBLE (curves_dialog->shell))
|
|
|
|
gtk_widget_show (curves_dialog->shell);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
curves_update (curves_dialog, GRAPH | DRAW);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
curves_free ()
|
|
|
|
{
|
|
|
|
if (curves_dialog)
|
|
|
|
{
|
|
|
|
if (curves_dialog->image_map)
|
|
|
|
{
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = TRUE;
|
1997-11-25 06:05:25 +08:00
|
|
|
image_map_abort (curves_dialog->image_map);
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = FALSE;
|
1999-06-23 23:24:46 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_dialog->image_map = NULL;
|
|
|
|
}
|
1999-06-23 23:24:46 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
if (curves_dialog->pixmap)
|
1997-12-18 14:30:11 +08:00
|
|
|
gdk_pixmap_unref (curves_dialog->pixmap);
|
1999-06-23 23:24:46 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_widget_destroy (curves_dialog->shell);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-07-03 01:40:10 +08:00
|
|
|
/*******************/
|
|
|
|
/* Curves dialog */
|
|
|
|
/*******************/
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1998-03-19 06:35:31 +08:00
|
|
|
static CurvesDialog *
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_new_dialog ()
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
GtkWidget *vbox;
|
|
|
|
GtkWidget *hbox;
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *frame;
|
|
|
|
GtkWidget *toggle;
|
|
|
|
GtkWidget *option_menu;
|
|
|
|
GtkWidget *channel_hbox;
|
|
|
|
GtkWidget *menu;
|
|
|
|
GtkWidget *table;
|
1998-04-28 09:01:34 +08:00
|
|
|
int i, j;
|
1997-11-25 06:05:25 +08:00
|
|
|
|
1999-06-23 23:24:46 +08:00
|
|
|
static ActionAreaItem action_items[] =
|
|
|
|
{
|
|
|
|
{ N_("Reset"), curves_reset_callback, NULL, NULL },
|
|
|
|
{ N_("OK"), curves_ok_callback, NULL, NULL },
|
|
|
|
{ N_("Cancel"), curves_cancel_callback, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
static MenuItem curve_type_items[] =
|
|
|
|
{
|
|
|
|
{ N_("Smooth"), 0, 0, curves_smooth_callback, NULL, NULL, NULL },
|
|
|
|
{ N_("Free"), 0, 0, curves_free_callback, NULL, NULL, NULL },
|
|
|
|
{ NULL, 0, 0, NULL, NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
cd = g_malloc (sizeof (CurvesDialog));
|
1999-09-04 05:58:41 +08:00
|
|
|
cd->cursor_ind_height = cd->cursor_ind_width = -1;
|
1997-11-25 06:05:25 +08:00
|
|
|
cd->preview = TRUE;
|
|
|
|
cd->curve_type = SMOOTH;
|
|
|
|
cd->pixmap = NULL;
|
1998-04-28 09:01:34 +08:00
|
|
|
cd->channel = HISTOGRAM_VALUE;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
for (j = 0; j < 256; j++)
|
|
|
|
cd->curve[i][j] = j;
|
1999-04-08 06:02:26 +08:00
|
|
|
|
|
|
|
for(i = 0; i < (sizeof(cd->col_value)/sizeof(cd->col_value[0])); i++)
|
|
|
|
cd->col_value[i] = 0;
|
|
|
|
|
1999-02-16 16:53:54 +08:00
|
|
|
cd->lut = gimp_lut_new();
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
channel_items [i].user_data = (gpointer) cd;
|
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
curve_type_items [i].user_data = (gpointer) cd;
|
|
|
|
|
|
|
|
/* The shell and main vbox */
|
|
|
|
cd->shell = gtk_dialog_new ();
|
1998-01-26 06:13:00 +08:00
|
|
|
gtk_window_set_wmclass (GTK_WINDOW (cd->shell), "curves", "Gimp");
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
gtk_window_set_title (GTK_WINDOW (cd->shell), _("Curves"));
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
gtk_signal_connect (GTK_OBJECT (cd->shell), "delete_event",
|
|
|
|
GTK_SIGNAL_FUNC (curves_delete_callback),
|
|
|
|
cd);
|
|
|
|
|
|
|
|
vbox = gtk_vbox_new (FALSE, 2);
|
1998-12-04 07:01:44 +08:00
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 2);
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (cd->shell)->vbox), vbox, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
/* The option menu for selecting channels */
|
|
|
|
channel_hbox = gtk_hbox_new (FALSE, 2);
|
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), channel_hbox, FALSE, FALSE, 0);
|
|
|
|
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
label = gtk_label_new (_("Modify Curves for Channel: "));
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (channel_hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
menu = build_menu (channel_items, NULL);
|
|
|
|
cd->channel_menu = gtk_option_menu_new ();
|
|
|
|
gtk_box_pack_start (GTK_BOX (channel_hbox), cd->channel_menu, FALSE, FALSE, 2);
|
|
|
|
|
|
|
|
gtk_widget_show (label);
|
|
|
|
gtk_widget_show (cd->channel_menu);
|
|
|
|
gtk_widget_show (channel_hbox);
|
|
|
|
gtk_option_menu_set_menu (GTK_OPTION_MENU (cd->channel_menu), menu);
|
|
|
|
|
|
|
|
/* The table for the yrange and the graph */
|
|
|
|
table = gtk_table_new (2, 2, FALSE);
|
1998-12-04 07:01:44 +08:00
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (table), 2);
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), table, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
/* The range drawing area */
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
|
|
|
|
gtk_table_attach (GTK_TABLE (table), frame, 0, 1, 0, 1,
|
|
|
|
GTK_EXPAND, GTK_EXPAND, 0, 0);
|
|
|
|
|
1999-05-20 23:43:56 +08:00
|
|
|
cd->yrange = gtk_preview_new (GTK_PREVIEW_COLOR);
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_preview_size (GTK_PREVIEW (cd->yrange), YRANGE_WIDTH, YRANGE_HEIGHT);
|
|
|
|
gtk_widget_set_events (cd->yrange, RANGE_MASK);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (cd->yrange), "event",
|
|
|
|
(GtkSignalFunc) curves_yrange_events,
|
|
|
|
cd);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), cd->yrange);
|
|
|
|
gtk_widget_show (cd->yrange);
|
|
|
|
gtk_widget_show (frame);
|
|
|
|
|
|
|
|
/* The curves graph */
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
|
|
|
|
gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 0, 1,
|
|
|
|
GTK_EXPAND | GTK_SHRINK | GTK_FILL,
|
|
|
|
GTK_FILL, 0, 0);
|
|
|
|
|
|
|
|
cd->graph = gtk_drawing_area_new ();
|
|
|
|
gtk_drawing_area_size (GTK_DRAWING_AREA (cd->graph),
|
|
|
|
GRAPH_WIDTH + RADIUS * 2,
|
|
|
|
GRAPH_HEIGHT + RADIUS * 2);
|
|
|
|
gtk_widget_set_events (cd->graph, GRAPH_MASK);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (cd->graph), "event",
|
|
|
|
(GtkSignalFunc) curves_graph_events,
|
|
|
|
cd);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), cd->graph);
|
|
|
|
gtk_widget_show (cd->graph);
|
|
|
|
gtk_widget_show (frame);
|
|
|
|
|
|
|
|
/* The range drawing area */
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
|
|
|
|
gtk_table_attach (GTK_TABLE (table), frame, 1, 2, 1, 2,
|
|
|
|
GTK_EXPAND, GTK_EXPAND, 0, 0);
|
|
|
|
|
1999-05-20 23:43:56 +08:00
|
|
|
cd->xrange = gtk_preview_new (GTK_PREVIEW_COLOR);
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_preview_size (GTK_PREVIEW (cd->xrange), XRANGE_WIDTH, XRANGE_HEIGHT);
|
|
|
|
gtk_widget_set_events (cd->xrange, RANGE_MASK);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (cd->xrange), "event",
|
|
|
|
(GtkSignalFunc) curves_xrange_events,
|
|
|
|
cd);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), cd->xrange);
|
|
|
|
gtk_widget_show (cd->xrange);
|
|
|
|
gtk_widget_show (frame);
|
|
|
|
gtk_widget_show (table);
|
|
|
|
|
|
|
|
/* Horizontal box for preview */
|
|
|
|
hbox = gtk_hbox_new (FALSE, 2);
|
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
/* The option menu for selecting the drawing method */
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
label = gtk_label_new (_("Curve Type: "));
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
menu = build_menu (curve_type_items, NULL);
|
|
|
|
option_menu = gtk_option_menu_new ();
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), option_menu, FALSE, FALSE, 2);
|
|
|
|
|
|
|
|
gtk_widget_show (label);
|
|
|
|
gtk_widget_show (option_menu);
|
|
|
|
gtk_widget_show (hbox);
|
|
|
|
gtk_option_menu_set_menu (GTK_OPTION_MENU (option_menu), menu);
|
|
|
|
|
|
|
|
/* The preview toggle */
|
Lots of ii8n stuff here and some additions to the de.po. Applied
Wed Oct 14 17:46:15 EDT 1998 Adrian Likins <adrian@gimp.org>
* app/*, po/de.po, de/POTFILES.in, libgimp/gimpintl.h:
Lots of ii8n stuff here and some additions to the de.po.
Applied gimp-egger-981005-1 ,gimp-egger-981006-1,
gimp-egger-981007-1, gimp-egger-981008-1,
gimp-egger-981009-1.patch, gimp-egger-981010-1.patch
* plug-in/guillotine/guillotine.c: added the coordinates
of the split images from the original image to the title.
ie foo.jpg (0,0) for the image in the topleft.
* plug-in/script-fu/scripts/neon-logo.scm,
perspective-shadow.scm, predator.scm,rendermap.scm,
ripply-anim.scm, select_to_image.scm,swirltile.scm,
xach-effect.scm: updated scripts to use new script-fu stuff
wooo boy! a big un!
in testing this, it looks like some of the po files are busted.
but the code stuff seems okay.
-adrian
1998-10-15 07:23:52 +08:00
|
|
|
toggle = gtk_check_button_new_with_label (_("Preview"));
|
1999-01-16 01:35:04 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), cd->preview);
|
1997-11-25 06:05:25 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), toggle, TRUE, FALSE, 0);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
|
|
|
|
(GtkSignalFunc) curves_preview_update,
|
|
|
|
cd);
|
|
|
|
|
|
|
|
gtk_widget_show (label);
|
|
|
|
gtk_widget_show (toggle);
|
|
|
|
gtk_widget_show (hbox);
|
|
|
|
|
|
|
|
/* The action area */
|
|
|
|
action_items[0].user_data = cd;
|
|
|
|
action_items[1].user_data = cd;
|
|
|
|
action_items[2].user_data = cd;
|
|
|
|
build_action_area (GTK_DIALOG (cd->shell), action_items, 3, 0);
|
|
|
|
|
|
|
|
gtk_widget_show (vbox);
|
|
|
|
|
|
|
|
return cd;
|
|
|
|
}
|
|
|
|
|
1999-09-04 05:58:41 +08:00
|
|
|
static void
|
|
|
|
curve_print_loc(CurvesDialog *cd,
|
|
|
|
gint xpos,
|
|
|
|
gint ypos)
|
|
|
|
{
|
|
|
|
char buf[32];
|
|
|
|
gint width;
|
|
|
|
gint ascent;
|
|
|
|
gint descent;
|
|
|
|
|
|
|
|
if(cd->cursor_ind_width < 0)
|
|
|
|
{
|
|
|
|
/* Calc max extents */
|
|
|
|
gdk_string_extents(cd->graph->style->font,
|
|
|
|
"x:888 y:888",
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&width,
|
|
|
|
&ascent,
|
|
|
|
&descent);
|
|
|
|
|
|
|
|
cd->cursor_ind_width = width;
|
|
|
|
cd->cursor_ind_height = ascent + descent;
|
|
|
|
cd->cursor_ind_ascent = ascent;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(xpos >= 0 && xpos <= 255 && ypos >=0 && ypos <= 255)
|
|
|
|
{
|
|
|
|
g_snprintf (buf, sizeof (buf), "x:%d y:%d",xpos,ypos);
|
|
|
|
|
|
|
|
gdk_draw_rectangle (cd->graph->window,
|
|
|
|
cd->graph->style->bg_gc[GTK_STATE_ACTIVE],
|
|
|
|
TRUE, RADIUS*2 + 2 , RADIUS*2 + 2,
|
|
|
|
cd->cursor_ind_width+4,
|
|
|
|
cd->cursor_ind_height+5);
|
|
|
|
|
|
|
|
gdk_draw_rectangle (cd->graph->window,
|
|
|
|
cd->graph->style->black_gc,
|
|
|
|
FALSE, RADIUS*2 + 2 , RADIUS*2 + 2,
|
|
|
|
cd->cursor_ind_width+3,
|
|
|
|
cd->cursor_ind_height+4);
|
|
|
|
|
|
|
|
gdk_draw_string (cd->graph->window,
|
|
|
|
cd->graph->style->font,
|
|
|
|
cd->graph->style->black_gc,
|
|
|
|
RADIUS*2 + 4,
|
|
|
|
RADIUS*2 + 5 + cd->cursor_ind_ascent,
|
|
|
|
buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-05-20 23:43:56 +08:00
|
|
|
/* TODO: preview alpha channel stuff correctly. -- austin, 20/May/99 */
|
1997-11-25 06:05:25 +08:00
|
|
|
static void
|
|
|
|
curves_update (CurvesDialog *cd,
|
|
|
|
int update)
|
|
|
|
{
|
|
|
|
GdkRectangle area;
|
|
|
|
int i, j;
|
1999-09-04 05:58:41 +08:00
|
|
|
char buf[32];
|
|
|
|
gint offset;
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
if (update & XRANGE_TOP)
|
|
|
|
{
|
1999-05-20 23:43:56 +08:00
|
|
|
unsigned char buf[XRANGE_WIDTH * 3];
|
|
|
|
|
|
|
|
switch (cd->channel) {
|
|
|
|
case HISTOGRAM_VALUE:
|
|
|
|
case HISTOGRAM_ALPHA:
|
|
|
|
for (i = 0; i < XRANGE_HEIGHT / 2; i++)
|
|
|
|
{
|
|
|
|
for (j = 0; j < XRANGE_WIDTH ; j++)
|
|
|
|
{
|
|
|
|
buf[j*3+0] = cd->curve[cd->channel][j];
|
|
|
|
buf[j*3+1] = cd->curve[cd->channel][j];
|
|
|
|
buf[j*3+2] = cd->curve[cd->channel][j];
|
|
|
|
}
|
|
|
|
gtk_preview_draw_row (GTK_PREVIEW (cd->xrange),
|
|
|
|
buf, 0, i, XRANGE_WIDTH);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HISTOGRAM_RED:
|
|
|
|
case HISTOGRAM_GREEN:
|
|
|
|
case HISTOGRAM_BLUE:
|
|
|
|
{
|
|
|
|
for (i = 0; i < XRANGE_HEIGHT / 2; i++)
|
|
|
|
{
|
|
|
|
for (j = 0; j < XRANGE_WIDTH; j++)
|
|
|
|
{
|
|
|
|
buf[j*3+0] = cd->curve[HISTOGRAM_RED][j];
|
|
|
|
buf[j*3+1] = cd->curve[HISTOGRAM_GREEN][j];
|
|
|
|
buf[j*3+2] = cd->curve[HISTOGRAM_BLUE][j];
|
|
|
|
}
|
|
|
|
gtk_preview_draw_row (GTK_PREVIEW (cd->xrange),
|
|
|
|
buf, 0, i, XRANGE_WIDTH);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_warning ("unknown channel type %d, can't happen!?!?",
|
|
|
|
cd->channel);
|
|
|
|
break;
|
|
|
|
} /* end switch */
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
if (update & DRAW)
|
|
|
|
{
|
|
|
|
area.x = 0;
|
|
|
|
area.y = 0;
|
|
|
|
area.width = XRANGE_WIDTH;
|
|
|
|
area.height = XRANGE_HEIGHT / 2;
|
|
|
|
gtk_widget_draw (cd->xrange, &area);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (update & XRANGE_BOTTOM)
|
|
|
|
{
|
1999-05-20 23:43:56 +08:00
|
|
|
unsigned char buf[XRANGE_WIDTH * 3];
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
for (i = 0; i < XRANGE_WIDTH; i++)
|
1999-05-20 23:43:56 +08:00
|
|
|
{
|
|
|
|
buf[i*3+0] = i;
|
|
|
|
buf[i*3+1] = i;
|
|
|
|
buf[i*3+2] = i;
|
|
|
|
}
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
for (i = XRANGE_HEIGHT / 2; i < XRANGE_HEIGHT; i++)
|
|
|
|
gtk_preview_draw_row (GTK_PREVIEW (cd->xrange), buf, 0, i, XRANGE_WIDTH);
|
|
|
|
|
|
|
|
if (update & DRAW)
|
|
|
|
{
|
|
|
|
area.x = 0;
|
|
|
|
area.y = XRANGE_HEIGHT / 2;
|
|
|
|
area.width = XRANGE_WIDTH;
|
|
|
|
area.height = XRANGE_HEIGHT / 2;
|
|
|
|
gtk_widget_draw (cd->xrange, &area);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (update & YRANGE)
|
|
|
|
{
|
1999-05-20 23:43:56 +08:00
|
|
|
unsigned char buf[YRANGE_WIDTH * 3];
|
|
|
|
unsigned char pix[3];
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
for (i = 0; i < YRANGE_HEIGHT; i++)
|
|
|
|
{
|
1999-05-20 23:43:56 +08:00
|
|
|
switch (cd->channel) {
|
|
|
|
case HISTOGRAM_VALUE:
|
|
|
|
case HISTOGRAM_ALPHA:
|
|
|
|
pix[0] = pix[1] = pix[2] = (255 - i);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HISTOGRAM_RED:
|
|
|
|
case HISTOGRAM_GREEN:
|
|
|
|
case HISTOGRAM_BLUE:
|
|
|
|
pix[0] = pix[1] = pix[2] = 0;
|
|
|
|
pix[cd->channel - 1] = (255 - i);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_warning ("unknown channel type %d, can't happen!?!?",
|
|
|
|
cd->channel);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (j = 0; j < YRANGE_WIDTH * 3; j++)
|
|
|
|
buf[j] = pix[j%3];
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
gtk_preview_draw_row (GTK_PREVIEW (cd->yrange), buf, 0, i, YRANGE_WIDTH);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (update & DRAW)
|
|
|
|
gtk_widget_draw (cd->yrange, NULL);
|
|
|
|
}
|
|
|
|
if ((update & GRAPH) && (update & DRAW) && cd->pixmap != NULL)
|
|
|
|
{
|
|
|
|
GdkPoint points[256];
|
|
|
|
|
|
|
|
/* Clear the pixmap */
|
|
|
|
gdk_draw_rectangle (cd->pixmap, cd->graph->style->bg_gc[GTK_STATE_NORMAL],
|
|
|
|
TRUE, 0, 0, GRAPH_WIDTH + RADIUS * 2, GRAPH_HEIGHT + RADIUS * 2);
|
|
|
|
|
|
|
|
/* Draw the grid lines */
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
{
|
|
|
|
gdk_draw_line (cd->pixmap, cd->graph->style->dark_gc[GTK_STATE_NORMAL],
|
|
|
|
RADIUS, i * (GRAPH_HEIGHT / 4) + RADIUS,
|
|
|
|
GRAPH_WIDTH + RADIUS, i * (GRAPH_HEIGHT / 4) + RADIUS);
|
|
|
|
gdk_draw_line (cd->pixmap, cd->graph->style->dark_gc[GTK_STATE_NORMAL],
|
|
|
|
i * (GRAPH_WIDTH / 4) + RADIUS, RADIUS,
|
|
|
|
i * (GRAPH_WIDTH / 4) + RADIUS, GRAPH_HEIGHT + RADIUS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Draw the curve */
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
points[i].x = i + RADIUS;
|
|
|
|
points[i].y = 255 - cd->curve[cd->channel][i] + RADIUS;
|
|
|
|
}
|
|
|
|
gdk_draw_points (cd->pixmap, cd->graph->style->black_gc, points, 256);
|
|
|
|
|
|
|
|
/* Draw the points */
|
|
|
|
if (cd->curve_type == SMOOTH)
|
|
|
|
for (i = 0; i < 17; i++)
|
|
|
|
{
|
|
|
|
if (cd->points[cd->channel][i][0] != -1)
|
|
|
|
gdk_draw_arc (cd->pixmap, cd->graph->style->black_gc, TRUE,
|
|
|
|
cd->points[cd->channel][i][0],
|
|
|
|
255 - cd->points[cd->channel][i][1],
|
|
|
|
RADIUS * 2, RADIUS * 2, 0, 23040);
|
|
|
|
}
|
|
|
|
|
1999-04-08 06:02:26 +08:00
|
|
|
/* draw the colour line */
|
|
|
|
gdk_draw_line(cd->pixmap, cd->graph->style->black_gc,
|
1999-05-14 06:53:40 +08:00
|
|
|
cd->col_value[cd->channel]+RADIUS,RADIUS,
|
|
|
|
cd->col_value[cd->channel]+RADIUS,GRAPH_HEIGHT + RADIUS);
|
1999-04-08 06:02:26 +08:00
|
|
|
|
1999-09-04 05:58:41 +08:00
|
|
|
/* and xpos indicator */
|
|
|
|
g_snprintf (buf, sizeof (buf), "x:%d",cd->col_value[cd->channel]);
|
|
|
|
|
|
|
|
if((cd->col_value[cd->channel]+RADIUS) < 127)
|
|
|
|
{
|
|
|
|
offset = RADIUS + 4;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
offset = -gdk_string_width(cd->graph->style->font,buf) - 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
gdk_draw_string (cd->pixmap,
|
|
|
|
cd->graph->style->font,
|
|
|
|
cd->graph->style->black_gc,
|
|
|
|
cd->col_value[cd->channel]+offset,
|
|
|
|
GRAPH_HEIGHT,
|
|
|
|
buf);
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
gdk_draw_pixmap (cd->graph->window, cd->graph->style->black_gc, cd->pixmap,
|
|
|
|
0, 0, 0, 0, GRAPH_WIDTH + RADIUS * 2, GRAPH_HEIGHT + RADIUS * 2);
|
1999-09-04 05:58:41 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_plot_curve (CurvesDialog *cd,
|
|
|
|
int p1,
|
|
|
|
int p2,
|
|
|
|
int p3,
|
|
|
|
int p4)
|
|
|
|
{
|
|
|
|
CRMatrix geometry;
|
|
|
|
CRMatrix tmp1, tmp2;
|
|
|
|
CRMatrix deltas;
|
|
|
|
double x, dx, dx2, dx3;
|
|
|
|
double y, dy, dy2, dy3;
|
|
|
|
double d, d2, d3;
|
|
|
|
int lastx, lasty;
|
1999-02-08 06:13:10 +08:00
|
|
|
gint32 newx, newy;
|
1997-11-25 06:05:25 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
/* construct the geometry matrix from the segment */
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
geometry[i][2] = 0;
|
|
|
|
geometry[i][3] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
geometry[0][i] = cd->points[cd->channel][p1][i];
|
|
|
|
geometry[1][i] = cd->points[cd->channel][p2][i];
|
|
|
|
geometry[2][i] = cd->points[cd->channel][p3][i];
|
|
|
|
geometry[3][i] = cd->points[cd->channel][p4][i];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* subdivide the curve 1000 times */
|
|
|
|
/* n can be adjusted to give a finer or coarser curve */
|
|
|
|
d = 1.0 / 1000;
|
|
|
|
d2 = d * d;
|
|
|
|
d3 = d * d * d;
|
|
|
|
|
|
|
|
/* construct a temporary matrix for determining the forward differencing deltas */
|
|
|
|
tmp2[0][0] = 0; tmp2[0][1] = 0; tmp2[0][2] = 0; tmp2[0][3] = 1;
|
|
|
|
tmp2[1][0] = d3; tmp2[1][1] = d2; tmp2[1][2] = d; tmp2[1][3] = 0;
|
|
|
|
tmp2[2][0] = 6*d3; tmp2[2][1] = 2*d2; tmp2[2][2] = 0; tmp2[2][3] = 0;
|
|
|
|
tmp2[3][0] = 6*d3; tmp2[3][1] = 0; tmp2[3][2] = 0; tmp2[3][3] = 0;
|
|
|
|
|
|
|
|
/* compose the basis and geometry matrices */
|
|
|
|
curves_CR_compose (CR_basis, geometry, tmp1);
|
|
|
|
|
|
|
|
/* compose the above results to get the deltas matrix */
|
|
|
|
curves_CR_compose (tmp2, tmp1, deltas);
|
|
|
|
|
|
|
|
/* extract the x deltas */
|
|
|
|
x = deltas[0][0];
|
|
|
|
dx = deltas[1][0];
|
|
|
|
dx2 = deltas[2][0];
|
|
|
|
dx3 = deltas[3][0];
|
|
|
|
|
|
|
|
/* extract the y deltas */
|
|
|
|
y = deltas[0][1];
|
|
|
|
dy = deltas[1][1];
|
|
|
|
dy2 = deltas[2][1];
|
|
|
|
dy3 = deltas[3][1];
|
|
|
|
|
|
|
|
lastx = BOUNDS (x, 0, 255);
|
|
|
|
lasty = BOUNDS (y, 0, 255);
|
|
|
|
|
|
|
|
cd->curve[cd->channel][lastx] = lasty;
|
|
|
|
|
|
|
|
/* loop over the curve */
|
|
|
|
for (i = 0; i < 1000; i++)
|
|
|
|
{
|
|
|
|
/* increment the x values */
|
|
|
|
x += dx;
|
|
|
|
dx += dx2;
|
|
|
|
dx2 += dx3;
|
|
|
|
|
|
|
|
/* increment the y values */
|
|
|
|
y += dy;
|
|
|
|
dy += dy2;
|
|
|
|
dy2 += dy3;
|
|
|
|
|
1998-07-25 02:52:03 +08:00
|
|
|
newx = CLAMP0255 (ROUND (x));
|
|
|
|
newy = CLAMP0255 (ROUND (y));
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
/* if this point is different than the last one...then draw it */
|
|
|
|
if ((lastx != newx) || (lasty != newy))
|
|
|
|
cd->curve[cd->channel][newx] = newy;
|
|
|
|
|
|
|
|
lastx = newx;
|
|
|
|
lasty = newy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
moved a bunch of PDB stuff here
* app/color_cmds.c: moved a bunch of PDB stuff here
* app/color_balance.[ch]: removed PDB proc, exported TransferMode
enum, ColorBalanceDialog, color_balance_create_lookup_tables, and
color_balance
* app/curves.[ch]: removed PDB procs, exported SMOOTH and GFREE
#defines, CurvesDialog, curves_lut_func and curves_calculate_curve
* app/desaturate.[ch]: removed PDB proc, exported desaturate
* app/equalize.[ch]: removed PDB proc, exported equalize
* app/histogram_tool.[ch]: removed PDB proc, exported HISTOGRAM_WIDTH
and HISTOGRAM_HEIGHT #defines, HistogramToolDialog,
histogram_tool_histogram_range
* app/hue_saturation.[ch]: removed PDB proc, exported HueRange enum,
HueSaturationDialog, hue_saturation_calculate_transfers,
hue_saturation
* app/invert.[ch]: remove PDB proc, export invert
* app/threshold.[ch]: remove PDB proc, export ThresholdDialog and
threshold_2
* internal_procs.c: changes for pdbgen
* app/gimprc.c: removed leftover declaration
* app/image_map.h: add #include "gimpdrawableF.h"
* app/lut_funcs.h: add ALPHA_LUT to ChannelLutType
-Yosh
1999-04-21 13:39:57 +08:00
|
|
|
void
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_calculate_curve (CurvesDialog *cd)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int points[17];
|
|
|
|
int num_pts;
|
|
|
|
int p1, p2, p3, p4;
|
|
|
|
|
|
|
|
switch (cd->curve_type)
|
|
|
|
{
|
|
|
|
case GFREE:
|
|
|
|
break;
|
|
|
|
case SMOOTH:
|
|
|
|
/* cycle through the curves */
|
|
|
|
num_pts = 0;
|
|
|
|
for (i = 0; i < 17; i++)
|
|
|
|
if (cd->points[cd->channel][i][0] != -1)
|
|
|
|
points[num_pts++] = i;
|
|
|
|
|
|
|
|
/* Initialize boundary curve points */
|
|
|
|
if (num_pts != 0)
|
|
|
|
{
|
|
|
|
for (i = 0; i < cd->points[cd->channel][points[0]][0]; i++)
|
|
|
|
cd->curve[cd->channel][i] = cd->points[cd->channel][points[0]][1];
|
|
|
|
for (i = cd->points[cd->channel][points[num_pts - 1]][0]; i < 256; i++)
|
|
|
|
cd->curve[cd->channel][i] = cd->points[cd->channel][points[num_pts - 1]][1];
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < num_pts - 1; i++)
|
|
|
|
{
|
|
|
|
p1 = (i == 0) ? points[i] : points[(i - 1)];
|
|
|
|
p2 = points[i];
|
|
|
|
p3 = points[(i + 1)];
|
|
|
|
p4 = (i == (num_pts - 2)) ? points[(num_pts - 1)] : points[(i + 2)];
|
|
|
|
|
|
|
|
curves_plot_curve (cd, p1, p2, p3, p4);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
1999-02-16 16:53:54 +08:00
|
|
|
gimp_lut_setup(cd->lut, (GimpLutFunc) curves_lut_func,
|
|
|
|
(void *) cd, gimp_drawable_bytes(cd->drawable));
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_preview (CurvesDialog *cd)
|
|
|
|
{
|
|
|
|
if (!cd->image_map)
|
1999-05-30 00:35:47 +08:00
|
|
|
g_message ("curves_preview(): No image map");
|
1998-03-31 15:23:50 +08:00
|
|
|
|
|
|
|
active_tool->preserve = TRUE; /* Going to dirty the display... */
|
|
|
|
|
1999-02-16 16:53:54 +08:00
|
|
|
image_map_apply (cd->image_map, (ImageMapApplyFunc)gimp_lut_process_2,
|
|
|
|
(void *) cd->lut);
|
1998-03-31 15:23:50 +08:00
|
|
|
|
|
|
|
active_tool->preserve = FALSE; /* All done */
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_value_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->channel != HISTOGRAM_VALUE)
|
|
|
|
{
|
|
|
|
cd->channel = HISTOGRAM_VALUE;
|
1999-05-20 23:43:56 +08:00
|
|
|
curves_update (cd, XRANGE_TOP | YRANGE | GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_red_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->channel != HISTOGRAM_RED)
|
|
|
|
{
|
|
|
|
cd->channel = HISTOGRAM_RED;
|
1999-05-20 23:43:56 +08:00
|
|
|
curves_update (cd, XRANGE_TOP | YRANGE | GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_green_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->channel != HISTOGRAM_GREEN)
|
|
|
|
{
|
|
|
|
cd->channel = HISTOGRAM_GREEN;
|
1999-05-20 23:43:56 +08:00
|
|
|
curves_update (cd, XRANGE_TOP | YRANGE | GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_blue_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->channel != HISTOGRAM_BLUE)
|
|
|
|
{
|
|
|
|
cd->channel = HISTOGRAM_BLUE;
|
1999-05-20 23:43:56 +08:00
|
|
|
curves_update (cd, XRANGE_TOP | YRANGE | GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_alpha_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->channel != HISTOGRAM_ALPHA)
|
|
|
|
{
|
|
|
|
cd->channel = HISTOGRAM_ALPHA;
|
1999-05-20 23:43:56 +08:00
|
|
|
curves_update (cd, XRANGE_TOP | YRANGE | GRAPH | DRAW);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_smooth_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
1999-02-08 06:13:10 +08:00
|
|
|
int i;
|
|
|
|
gint32 index;
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->curve_type != SMOOTH)
|
|
|
|
{
|
|
|
|
cd->curve_type = SMOOTH;
|
|
|
|
|
|
|
|
/* pick representative points from the curve and make them control points */
|
|
|
|
for (i = 0; i <= 8; i++)
|
|
|
|
{
|
1998-07-25 02:52:03 +08:00
|
|
|
index = CLAMP0255 (i * 32);
|
1997-11-25 06:05:25 +08:00
|
|
|
cd->points[cd->channel][i * 2][0] = index;
|
|
|
|
cd->points[cd->channel][i * 2][1] = cd->curve[cd->channel][index];
|
|
|
|
}
|
|
|
|
|
|
|
|
curves_calculate_curve (cd);
|
|
|
|
curves_update (cd, GRAPH | DRAW);
|
|
|
|
|
|
|
|
if (cd->preview)
|
|
|
|
curves_preview (cd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_free_callback (GtkWidget *w,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (cd->curve_type != GFREE)
|
|
|
|
{
|
|
|
|
cd->curve_type = GFREE;
|
1999-05-14 06:53:40 +08:00
|
|
|
curves_calculate_curve (cd);
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_update (cd, GRAPH | DRAW);
|
1999-05-14 06:53:40 +08:00
|
|
|
|
|
|
|
if (cd->preview)
|
|
|
|
curves_preview (cd);
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_reset_callback (GtkWidget *widget,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
/* Initialize the values */
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
cd->curve[cd->channel][i] = i;
|
|
|
|
|
|
|
|
cd->grab_point = -1;
|
|
|
|
for (i = 0; i < 17; i++)
|
|
|
|
{
|
|
|
|
cd->points[cd->channel][i][0] = -1;
|
|
|
|
cd->points[cd->channel][i][1] = -1;
|
|
|
|
}
|
|
|
|
cd->points[cd->channel][0][0] = 0;
|
|
|
|
cd->points[cd->channel][0][1] = 0;
|
|
|
|
cd->points[cd->channel][16][0] = 255;
|
|
|
|
cd->points[cd->channel][16][1] = 255;
|
|
|
|
|
1999-05-14 06:53:40 +08:00
|
|
|
curves_calculate_curve (cd);
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_update (cd, GRAPH | XRANGE_TOP | DRAW);
|
|
|
|
if (cd->preview)
|
|
|
|
curves_preview (cd);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_ok_callback (GtkWidget *widget,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
|
|
|
|
if (GTK_WIDGET_VISIBLE (cd->shell))
|
|
|
|
gtk_widget_hide (cd->shell);
|
|
|
|
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = TRUE; /* We're about to dirty... */
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
if (!cd->preview)
|
1999-02-16 16:53:54 +08:00
|
|
|
image_map_apply (cd->image_map, (ImageMapApplyFunc)gimp_lut_process_2,
|
|
|
|
(void *) cd->lut);
|
1997-11-25 06:05:25 +08:00
|
|
|
|
|
|
|
if (cd->image_map)
|
|
|
|
image_map_commit (cd->image_map);
|
|
|
|
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = FALSE;
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
cd->image_map = NULL;
|
1999-06-26 19:16:47 +08:00
|
|
|
|
|
|
|
active_tool->gdisp_ptr = NULL;
|
|
|
|
active_tool->drawable = NULL;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_cancel_callback (GtkWidget *widget,
|
|
|
|
gpointer client_data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) client_data;
|
|
|
|
if (GTK_WIDGET_VISIBLE (cd->shell))
|
|
|
|
gtk_widget_hide (cd->shell);
|
|
|
|
|
|
|
|
if (cd->image_map)
|
|
|
|
{
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = TRUE;
|
1997-11-25 06:05:25 +08:00
|
|
|
image_map_abort (cd->image_map);
|
1998-03-31 15:23:50 +08:00
|
|
|
active_tool->preserve = FALSE;
|
1999-06-23 23:24:46 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
gdisplays_flush ();
|
1999-06-26 19:16:47 +08:00
|
|
|
cd->image_map = NULL;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
1999-06-26 19:16:47 +08:00
|
|
|
|
|
|
|
active_tool->gdisp_ptr = NULL;
|
|
|
|
active_tool->drawable = NULL;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static gint
|
|
|
|
curves_delete_callback (GtkWidget *w,
|
|
|
|
GdkEvent *e,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
curves_cancel_callback (w, data);
|
|
|
|
|
1998-03-13 06:01:43 +08:00
|
|
|
return TRUE;
|
1997-11-25 06:05:25 +08:00
|
|
|
}
|
|
|
|
static void
|
|
|
|
curves_preview_update (GtkWidget *w,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
CurvesDialog *cd;
|
|
|
|
|
|
|
|
cd = (CurvesDialog *) data;
|
1998-03-31 15:23:50 +08:00
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
if (GTK_TOGGLE_BUTTON (w)->active)
|
|
|
|
{
|
|
|
|
cd->preview = TRUE;
|
|
|
|
curves_preview (cd);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
cd->preview = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gint
|
|
|
|
curves_graph_events (GtkWidget *widget,
|
|
|
|
GdkEvent *event,
|
|
|
|
CurvesDialog *cd)
|
|
|
|
{
|
|
|
|
static GdkCursorType cursor_type = GDK_TOP_LEFT_ARROW;
|
|
|
|
GdkCursorType new_type;
|
|
|
|
GdkEventButton *bevent;
|
|
|
|
GdkEventMotion *mevent;
|
|
|
|
int i;
|
|
|
|
int tx, ty;
|
|
|
|
int x, y;
|
|
|
|
int closest_point;
|
|
|
|
int distance;
|
|
|
|
int x1, x2, y1, y2;
|
|
|
|
|
|
|
|
new_type = GDK_X_CURSOR;
|
|
|
|
closest_point = 0;
|
|
|
|
|
|
|
|
/* get the pointer position */
|
|
|
|
gdk_window_get_pointer (cd->graph->window, &tx, &ty, NULL);
|
|
|
|
x = BOUNDS ((tx - RADIUS), 0, 255);
|
|
|
|
y = BOUNDS ((ty - RADIUS), 0, 255);
|
|
|
|
|
|
|
|
distance = G_MAXINT;
|
|
|
|
for (i = 0; i < 17; i++)
|
|
|
|
{
|
|
|
|
if (cd->points[cd->channel][i][0] != -1)
|
|
|
|
if (abs (x - cd->points[cd->channel][i][0]) < distance)
|
|
|
|
{
|
|
|
|
distance = abs (x - cd->points[cd->channel][i][0]);
|
|
|
|
closest_point = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (distance > MIN_DISTANCE)
|
|
|
|
closest_point = (x + 8) / 16;
|
|
|
|
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case GDK_EXPOSE:
|
|
|
|
if (cd->pixmap == NULL)
|
|
|
|
cd->pixmap = gdk_pixmap_new (cd->graph->window,
|
|
|
|
GRAPH_WIDTH + RADIUS * 2,
|
|
|
|
GRAPH_HEIGHT + RADIUS * 2, -1);
|
|
|
|
|
|
|
|
curves_update (cd, GRAPH | DRAW);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_BUTTON_PRESS:
|
|
|
|
bevent = (GdkEventButton *) event;
|
|
|
|
new_type = GDK_TCROSS;
|
|
|
|
|
|
|
|
switch (cd->curve_type)
|
|
|
|
{
|
|
|
|
case SMOOTH:
|
|
|
|
/* determine the leftmost and rightmost points */
|
|
|
|
cd->leftmost = -1;
|
|
|
|
for (i = closest_point - 1; i >= 0; i--)
|
|
|
|
if (cd->points[cd->channel][i][0] != -1)
|
|
|
|
{
|
|
|
|
cd->leftmost = cd->points[cd->channel][i][0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cd->rightmost = 256;
|
|
|
|
for (i = closest_point + 1; i < 17; i++)
|
|
|
|
if (cd->points[cd->channel][i][0] != -1)
|
|
|
|
{
|
|
|
|
cd->rightmost = cd->points[cd->channel][i][0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
cd->grab_point = closest_point;
|
|
|
|
cd->points[cd->channel][cd->grab_point][0] = x;
|
|
|
|
cd->points[cd->channel][cd->grab_point][1] = 255 - y;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GFREE:
|
|
|
|
cd->curve[cd->channel][x] = 255 - y;
|
|
|
|
cd->grab_point = x;
|
|
|
|
cd->last = y;
|
|
|
|
break;
|
|
|
|
}
|
1999-04-08 06:58:44 +08:00
|
|
|
|
|
|
|
curves_calculate_curve (cd);
|
1997-11-25 06:05:25 +08:00
|
|
|
curves_update (cd, GRAPH | XRANGE_TOP | DRAW);
|
1999-04-08 06:02:26 +08:00
|
|
|
gtk_grab_add(widget);
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_BUTTON_RELEASE:
|
|
|
|
new_type = GDK_FLEUR;
|
|
|
|
cd->grab_point = -1;
|
|
|
|
|
|
|
|
if (cd->preview)
|
|
|
|
curves_preview (cd);
|
1999-04-08 06:02:26 +08:00
|
|
|
|
|
|
|
gtk_grab_remove(widget);
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_MOTION_NOTIFY:
|
|
|
|
mevent = (GdkEventMotion *) event;
|
|
|
|
|
|
|
|
if (mevent->is_hint)
|
|
|
|
{
|
|
|
|
mevent->x = tx;
|
|
|
|
mevent->y = ty;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (cd->curve_type)
|
|
|
|
{
|
|
|
|
case SMOOTH:
|
|
|
|
/* If no point is grabbed... */
|
|
|
|
if (cd->grab_point == -1)
|
|
|
|
{
|
|
|
|
if (cd->points[cd->channel][closest_point][0] != -1)
|
|
|
|
new_type = GDK_FLEUR;
|
|
|
|
else
|
|
|
|
new_type = GDK_TCROSS;
|
|
|
|
}
|
|
|
|
/* Else, drag the grabbed point */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
new_type = GDK_TCROSS;
|
|
|
|
|
|
|
|
cd->points[cd->channel][cd->grab_point][0] = -1;
|
|
|
|
|
|
|
|
if (x > cd->leftmost && x < cd->rightmost)
|
|
|
|
{
|
|
|
|
closest_point = (x + 8) / 16;
|
|
|
|
if (cd->points[cd->channel][closest_point][0] == -1)
|
|
|
|
cd->grab_point = closest_point;
|
|
|
|
cd->points[cd->channel][cd->grab_point][0] = x;
|
|
|
|
cd->points[cd->channel][cd->grab_point][1] = 255 - y;
|
|
|
|
}
|
|
|
|
|
|
|
|
curves_calculate_curve (cd);
|
|
|
|
curves_update (cd, GRAPH | XRANGE_TOP | DRAW);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GFREE:
|
|
|
|
if (cd->grab_point != -1)
|
|
|
|
{
|
|
|
|
if (cd->grab_point > x)
|
|
|
|
{
|
|
|
|
x1 = x;
|
|
|
|
x2 = cd->grab_point;
|
|
|
|
y1 = y;
|
|
|
|
y2 = cd->last;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
x1 = cd->grab_point;
|
|
|
|
x2 = x;
|
|
|
|
y1 = cd->last;
|
|
|
|
y2 = y;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x2 != x1)
|
|
|
|
for (i = x1; i <= x2; i++)
|
|
|
|
cd->curve[cd->channel][i] = 255 - (y1 + ((y2 - y1) * (i - x1)) / (x2 - x1));
|
|
|
|
else
|
|
|
|
cd->curve[cd->channel][x] = 255 - y;
|
|
|
|
|
|
|
|
cd->grab_point = x;
|
|
|
|
cd->last = y;
|
|
|
|
|
|
|
|
curves_update (cd, GRAPH | XRANGE_TOP | DRAW);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mevent->state & GDK_BUTTON1_MASK)
|
|
|
|
new_type = GDK_TCROSS;
|
|
|
|
else
|
|
|
|
new_type = GDK_PENCIL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_type != cursor_type)
|
|
|
|
{
|
|
|
|
cursor_type = new_type;
|
|
|
|
change_win_cursor (cd->graph->window, cursor_type);
|
|
|
|
}
|
1999-09-04 05:58:41 +08:00
|
|
|
|
|
|
|
curve_print_loc(cd,x,255-y);
|
|
|
|
|
1997-11-25 06:05:25 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gint
|
|
|
|
curves_xrange_events (GtkWidget *widget,
|
|
|
|
GdkEvent *event,
|
|
|
|
CurvesDialog *cd)
|
|
|
|
{
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case GDK_EXPOSE:
|
|
|
|
curves_update (cd, XRANGE_TOP | XRANGE_BOTTOM);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_DELETE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gint
|
|
|
|
curves_yrange_events (GtkWidget *widget,
|
|
|
|
GdkEvent *event,
|
|
|
|
CurvesDialog *cd)
|
|
|
|
{
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case GDK_EXPOSE:
|
|
|
|
curves_update (cd, YRANGE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
curves_CR_compose (CRMatrix a,
|
|
|
|
CRMatrix b,
|
|
|
|
CRMatrix ab)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
for (j = 0; j < 4; j++)
|
|
|
|
{
|
|
|
|
ab[i][j] = (a[i][0] * b[0][j] +
|
|
|
|
a[i][1] * b[1][j] +
|
|
|
|
a[i][2] * b[2][j] +
|
|
|
|
a[i][3] * b[3][j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|