gimpoperationcolorize.c 4.59 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/* GIMP - The GNU Image Manipulation Program
 * Copyright (C) 1995 Spencer Kimball and Peter Mattis
 *
 * gimpoperationcolorize.c
 * Copyright (C) 2007 Michael Natterer <mitch@gimp.org>
 *
 * 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
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 */

#include "config.h"

24
#include <gegl.h>
25 26 27 28 29

#include "libgimpcolor/gimpcolor.h"

#include "gegl-types.h"

30
#include "gimpcolorizeconfig.h"
31 32 33
#include "gimpoperationcolorize.h"


34 35 36 37 38
static gboolean gimp_operation_colorize_process (GeglOperation       *operation,
                                                 void                *in_buf,
                                                 void                *out_buf,
                                                 glong                samples,
                                                 const GeglRectangle *roi);
39 40 41


G_DEFINE_TYPE (GimpOperationColorize, gimp_operation_colorize,
42
               GIMP_TYPE_OPERATION_POINT_FILTER)
43 44 45 46 47

#define parent_class gimp_operation_colorize_parent_class


static void
48
gimp_operation_colorize_class_init (GimpOperationColorizeClass *klass)
49 50 51 52 53
{
  GObjectClass                  *object_class    = G_OBJECT_CLASS (klass);
  GeglOperationClass            *operation_class = GEGL_OPERATION_CLASS (klass);
  GeglOperationPointFilterClass *point_class     = GEGL_OPERATION_POINT_FILTER_CLASS (klass);

54 55
  object_class->set_property   = gimp_operation_point_filter_set_property;
  object_class->get_property   = gimp_operation_point_filter_get_property;
56

57
  operation_class->name        = "gimp:colorize";
58 59
  operation_class->categories  = "color";
  operation_class->description = "GIMP Colorize operation";
60

61
  point_class->process         = gimp_operation_colorize_process;
62

63 64
  g_object_class_install_property (object_class,
                                   GIMP_OPERATION_POINT_FILTER_PROP_CONFIG,
65 66 67 68
                                   g_param_spec_object ("config",
                                                        "Config",
                                                        "The config object",
                                                        GIMP_TYPE_COLORIZE_CONFIG,
69 70
                                                        G_PARAM_READWRITE |
                                                        G_PARAM_CONSTRUCT));
71 72 73 74 75 76 77 78
}

static void
gimp_operation_colorize_init (GimpOperationColorize *self)
{
}

static gboolean
79 80 81 82 83
gimp_operation_colorize_process (GeglOperation       *operation,
                                 void                *in_buf,
                                 void                *out_buf,
                                 glong                samples,
                                 const GeglRectangle *roi)
84
{
85 86 87 88 89 90 91 92
  GimpOperationPointFilter *point  = GIMP_OPERATION_POINT_FILTER (operation);
  GimpColorizeConfig       *config = GIMP_COLORIZE_CONFIG (point->config);
  gfloat                   *src    = in_buf;
  gfloat                   *dest   = out_buf;
  GimpHSL                   hsl;

  if (! config)
    return FALSE;
93

94 95
  hsl.h = config->hue;
  hsl.s = config->saturation;
96

97
  while (samples--)
98 99
    {
      GimpRGB rgb;
100 101 102
      gfloat  lum = GIMP_RGB_LUMINANCE (src[RED],
                                        src[GREEN],
                                        src[BLUE]);
103

104
      if (config->lightness > 0)
105
        {
106
          lum = lum * (1.0 - config->lightness);
107

108
          lum += 1.0 - (1.0 - config->lightness);
109
        }
110
      else if (config->lightness < 0)
111
        {
112
          lum = lum * (config->lightness + 1.0);
113 114 115 116 117 118
        }

      hsl.l = lum;

      gimp_hsl_to_rgb (&hsl, &rgb);

119 120 121 122 123
      /*  the code in base/colorize.c would multiply r,b,g with lum,
       *  but this is a bug since it should multiply with 255. We
       *  don't repeat this bug here (this is the reason why the gegl
       *  colorize is brighter than the legacy one).
       */
124 125 126 127
      dest[RED]   = rgb.r; /* * lum; */
      dest[GREEN] = rgb.g; /* * lum; */
      dest[BLUE]  = rgb.b; /* * lum */;
      dest[ALPHA] = src[ALPHA];
128 129 130 131 132 133 134

      src  += 4;
      dest += 4;
    }

  return TRUE;
}