borderaverage.c 13.7 KB
Newer Older
1
/* borderaverage 0.01 - image processing plug-in for the Gimp.
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
 *
 * Copyright (C) 1998 Philipp Klaus (webmaster@access.ch)
 *
 *
 * 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.
 */
20

21
#include "config.h"
22 23 24 25 26 27 28

#include <libgimp/gimp.h>
#include <libgimp/gimpui.h>

#include "libgimp/stdplugins-intl.h"


29 30 31 32
#define PLUG_IN_PROC   "plug-in-borderaverage"
#define PLUG_IN_BINARY "borderaverage"


33 34 35
/* Declare local functions.
 */
static void      query  (void);
36
static void      run    (const gchar      *name,
37 38 39 40
                         gint              nparams,
                         const GimpParam  *param,
                         gint             *nreturn_vals,
                         GimpParam       **return_vals);
41 42


David Odin's avatar
David Odin committed
43 44
static void      borderaverage        (GimpDrawable *drawable,
                                       GimpRGB      *result);
45

David Odin's avatar
David Odin committed
46 47
static gboolean  borderaverage_dialog (gint32        image_ID,
                                       GimpDrawable *drawable);
48

David Odin's avatar
David Odin committed
49 50 51 52 53 54 55
static void      add_new_color        (gint          bytes,
                                       const guchar *buffer,
                                       gint         *cube,
                                       gint          bucket_expo);

static void      thickness_callback   (GtkWidget    *widget,
                                       gpointer      data);
56

Sven Neumann's avatar
Sven Neumann committed
57
GimpPlugInInfo PLUG_IN_INFO =
58
{
59 60 61 62
  NULL,  /* init  */
  NULL,  /* quit  */
  query, /* query */
  run,   /* run   */
63 64
};

65
static gint  borderaverage_thickness       = 3;
66
static gint  borderaverage_bucket_exponent = 4;
67

68 69
struct borderaverage_data
{
70 71
  gint  thickness;
  gint  bucket_exponent;
72
}
73 74

static borderaverage_data =
75 76 77 78
{
  3,
  4
};
79 80 81 82

MAIN ()

static void
83
query (void)
84
{
Sven Neumann's avatar
Sven Neumann committed
85
  static GimpParamDef args[] =
86
  {
87
    { GIMP_PDB_INT32,    "run-mode",        "Interactive, non-interactive" },
88 89 90
    { GIMP_PDB_IMAGE,    "image",           "Input image (unused)" },
    { GIMP_PDB_DRAWABLE, "drawable",        "Input drawable" },
    { GIMP_PDB_INT32,    "thickness",       "Border size to take in count" },
91
    { GIMP_PDB_INT32,    "bucket-exponent", "Bits for bucket size (default=4: 16 Levels)" },
92
  };
93
  static GimpParamDef return_vals[] =
94
  {
95
    { GIMP_PDB_COLOR,    "borderaverage",   "Sends the average color of the specified border to the Toolbox foreground." },
96
  };
97

98
  gimp_install_procedure (PLUG_IN_PROC,
99 100 101 102 103
                          "Borderaverage",
                          "",
                          "Philipp Klaus",
                          "Internet Access AG",
                          "1998",
104
                          N_("_Border Average..."),
105 106 107
                          "RGB*",
                          GIMP_PLUGIN,
                          G_N_ELEMENTS (args),
108
                          G_N_ELEMENTS (return_vals),
109
                          args, return_vals);
110

111
  gimp_plugin_menu_register (PLUG_IN_PROC, "<Image>/Colors/Info");
112 113 114
}

static void
115 116 117 118 119
run (const gchar      *name,
     gint              nparams,
     const GimpParam  *param,
     gint             *nreturn_vals,
     GimpParam       **return_vals)
120
{
121 122
  static GimpParam   values[3];
  GimpDrawable      *drawable;
123
  gint32             image_ID;
124 125
  GimpPDBStatusType  status = GIMP_PDB_SUCCESS;
  GimpRGB            result_color;
126
  GimpRunMode        run_mode;
127

128
  INIT_I18N ();
129

130
  run_mode = param[0].data.d_int32;
131
  image_ID = param[1].data.d_int32;
132

133
  /*    Get the specified drawable      */
134 135 136 137
  drawable = gimp_drawable_get (param[2].data.d_drawable);

  switch (run_mode)
    {
Sven Neumann's avatar
Sven Neumann committed
138
    case GIMP_RUN_INTERACTIVE:
139
      gimp_get_data (PLUG_IN_PROC, &borderaverage_data);
140
      borderaverage_thickness       = borderaverage_data.thickness;
141
      borderaverage_bucket_exponent = borderaverage_data.bucket_exponent;
142
      if (! borderaverage_dialog (image_ID, drawable))
143
        status = GIMP_PDB_EXECUTION_ERROR;
144 145
      break;

Sven Neumann's avatar
Sven Neumann committed
146
    case GIMP_RUN_NONINTERACTIVE:
147
      if (nparams != 5)
148
        status = GIMP_PDB_CALLING_ERROR;
Sven Neumann's avatar
Sven Neumann committed
149
      if (status == GIMP_PDB_SUCCESS)
150 151 152 153
        {
          borderaverage_thickness       = param[3].data.d_int32;
          borderaverage_bucket_exponent = param[4].data.d_int32;
        }
154 155
      break;

Sven Neumann's avatar
Sven Neumann committed
156
    case GIMP_RUN_WITH_LAST_VALS:
157
      gimp_get_data (PLUG_IN_PROC, &borderaverage_data);
158
      borderaverage_thickness       = borderaverage_data.thickness;
159 160 161 162 163 164 165
      borderaverage_bucket_exponent = borderaverage_data.bucket_exponent;
      break;

    default:
      break;
    }

Sven Neumann's avatar
Sven Neumann committed
166
  if (status == GIMP_PDB_SUCCESS)
167
    {
168
      /*  Make sure that the drawable is RGB color  */
169
      if (gimp_drawable_is_rgb (drawable->drawable_id))
170 171 172 173 174 175
        {
          gimp_progress_init ( _("Border Average..."));
          borderaverage (drawable, &result_color);

          if (run_mode != GIMP_RUN_NONINTERACTIVE)
            {
176
              gimp_context_set_foreground (&result_color);
177 178 179 180 181
            }
          if (run_mode == GIMP_RUN_INTERACTIVE)
            {
              borderaverage_data.thickness       = borderaverage_thickness;
              borderaverage_data.bucket_exponent = borderaverage_bucket_exponent;
182
              gimp_set_data (PLUG_IN_PROC,
183 184 185
                             &borderaverage_data, sizeof (borderaverage_data));
            }
        }
186
      else
187 188 189
        {
          status = GIMP_PDB_EXECUTION_ERROR;
        }
190 191
    }
  *nreturn_vals = 3;
192
  *return_vals  = values;
193

194
  values[0].type          = GIMP_PDB_STATUS;
195
  values[0].data.d_status = status;
196

197
  values[1].type         = GIMP_PDB_COLOR;
198
  values[1].data.d_color = result_color;
199

200
  gimp_drawable_detach (drawable);
201 202
}

203 204 205 206 207 208 209 210
typedef struct {
  gint x1, x2, y1, y2;
  gint bucket_expo;
  gint *cube;
} BorderAverageParam_t;

static void
borderaverage_func (gint x,
David Odin's avatar
David Odin committed
211 212 213 214
                    gint y,
                    const guchar *src,
                    gint bpp,
                    gpointer data)
215 216 217 218 219 220 221 222 223 224 225 226
{
  BorderAverageParam_t *param = (BorderAverageParam_t*) data;

  if (x <  param->x1 + borderaverage_thickness ||
      x >= param->x2 - borderaverage_thickness ||
      y <  param->y1 + borderaverage_thickness ||
      y >= param->y2 - borderaverage_thickness)
    {
      add_new_color (bpp, src, param->cube, param->bucket_expo);
    }
}

227
static void
228
borderaverage (GimpDrawable *drawable,
229
               GimpRGB      *result)
230
{
231 232 233 234 235 236 237 238 239 240
  gint    width;
  gint    height;
  gint    x1, x2, y1, y2;
  gint    bytes;
  gint    max;
  guchar  r, g, b;
  guchar *buffer;
  gint    bucket_num, bucket_expo, bucket_rexpo;
  gint   *cube;
  gint    i, j, k; /* index variables */
241
  GimpRgnIterator *iter;
242

243
  BorderAverageParam_t     param;
244 245 246 247

  /* allocate and clear the cube before */
  bucket_expo = borderaverage_bucket_exponent;
  bucket_rexpo = 8 - bucket_expo;
248
  param.cube = cube = g_new (gint, 1 << (bucket_rexpo * 3));
249
  bucket_num = 1 << bucket_rexpo;
250

251 252 253
  for (i = 0; i < bucket_num; i++)
    {
      for (j = 0; j < bucket_num; j++)
254 255 256 257 258 259
        {
          for (k = 0; k < bucket_num; k++)
            {
              cube[(i << (bucket_rexpo << 1)) + (j << bucket_rexpo) + k] = 0;
            }
        }
260 261
    }

262
  gimp_drawable_mask_bounds (drawable->drawable_id, &x1, &y1, &x2, &y2);
263 264 265 266 267
  param.x1 = x1;
  param.y1 = y1;
  param.x2 = x2;
  param.y2 = y2;
  param.bucket_expo = bucket_expo;
268 269 270 271 272 273 274 275

  /*  Get the size of the input image. (This will/must be the same
   *  as the size of the output image.
   */
  width = drawable->width;
  height = drawable->height;
  bytes = drawable->bpp;

276 277
  gimp_tile_cache_ntiles (2 * ((x2 - x1) / gimp_tile_width () + 1));

278 279 280
  /*  allocate row buffer  */
  buffer = g_new (guchar, (x2 - x1) * bytes);

281
  iter = gimp_rgn_iterator_new (drawable, 0);
282 283
  gimp_rgn_iterator_src (iter, borderaverage_func, &param);
  gimp_rgn_iterator_free (iter);
284

285
  max = 0; r = 0; g = 0; b = 0;
286

287 288 289 290
  /* get max of cube */
  for (i = 0; i < bucket_num; i++)
    {
      for (j = 0; j < bucket_num; j++)
291 292 293 294 295 296 297 298 299 300 301 302 303 304
        {
          for (k = 0; k < bucket_num; k++)
            {
              if (cube[(i << (bucket_rexpo << 1)) +
                      (j << bucket_rexpo) + k] > max)
                {
                  max = cube[(i << (bucket_rexpo << 1)) +
                            (j << bucket_rexpo) + k];
                  r = (i<<bucket_expo) + (1<<(bucket_expo - 1));
                  g = (j<<bucket_expo) + (1<<(bucket_expo - 1));
                  b = (k<<bucket_expo) + (1<<(bucket_expo - 1));
                }
            }
        }
305
    }
306

307
  /* return the color */
308
  gimp_rgb_set_uchar (result, r, g, b);
309

310 311 312
  g_free (buffer);
  g_free (cube);
}
313

314
static void
David Odin's avatar
David Odin committed
315
add_new_color (gint          bytes,
316
               const guchar *buffer,
David Odin's avatar
David Odin committed
317
               gint         *cube,
318
               gint          bucket_expo)
319
{
320
  guchar r, g, b;
321
  gint   bucket_rexpo;
322 323

  bucket_rexpo = 8 - bucket_expo;
324 325 326
  r = buffer[0] >> bucket_expo;
  g = (bytes > 1) ? buffer[1] >> bucket_expo : 0;
  b = (bytes > 2) ? buffer[2] >> bucket_expo : 0;
327 328 329
  cube[(r << (bucket_rexpo << 1)) + (g << bucket_rexpo) + b]++;
}

330
static gboolean
David Odin's avatar
David Odin committed
331 332
borderaverage_dialog (gint32        image_ID,
                      GimpDrawable *drawable)
333
{
David Odin's avatar
David Odin committed
334
  GtkWidget    *dialog;
335
  GtkWidget    *frame;
David Odin's avatar
David Odin committed
336
  GtkWidget    *main_vbox;
337 338
  GtkWidget    *hbox;
  GtkWidget    *label;
339 340
  GtkWidget    *size_entry;
  GimpUnit      unit;
341 342 343
  GtkWidget    *combo;
  GtkSizeGroup *group;
  gboolean      run;
David Odin's avatar
David Odin committed
344
  gdouble       xres, yres;
345

346 347 348
  const gchar *labels[] =
    { "1", "2", "4", "8", "16", "32", "64", "128", "256" };

349
  gimp_ui_init (PLUG_IN_BINARY, FALSE);
350

351
  dialog = gimp_dialog_new (_("Borderaverage"), PLUG_IN_BINARY,
David Odin's avatar
David Odin committed
352
                            NULL, 0,
353
                            gimp_standard_help_func, PLUG_IN_PROC,
354

David Odin's avatar
David Odin committed
355 356
                            GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
                            GTK_STOCK_OK,     GTK_RESPONSE_OK,
Sven Neumann's avatar
Sven Neumann committed
357

David Odin's avatar
David Odin committed
358
                            NULL);
359

360
  gtk_dialog_set_alternative_button_order (GTK_DIALOG (dialog),
361 362 363
                                           GTK_RESPONSE_OK,
                                           GTK_RESPONSE_CANCEL,
                                           -1);
364

365
  gimp_window_set_transient (GTK_WINDOW (dialog));
366

David Odin's avatar
David Odin committed
367 368 369 370
  main_vbox = gtk_vbox_new (FALSE, 12);
  gtk_container_set_border_width (GTK_CONTAINER (main_vbox), 12);
  gtk_container_add (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox), main_vbox);
  gtk_widget_show (main_vbox);
371

372
  frame = gimp_frame_new (_("Border Size"));
David Odin's avatar
David Odin committed
373
  gtk_box_pack_start (GTK_BOX (main_vbox), frame, FALSE, FALSE, 0);
374
  gtk_widget_show (frame);
375

376
  hbox = gtk_hbox_new (FALSE, 6);
377 378 379
  gtk_container_add (GTK_CONTAINER (frame), hbox);
  gtk_widget_show (hbox);

380
  label = gtk_label_new_with_mnemonic (_("_Thickness:"));
381
  gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
382 383
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);
384

385 386 387 388
  group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
  gtk_size_group_add_widget (group, label);
  g_object_unref (group);

389 390 391
  /*  Get the image resolution and unit  */
  gimp_image_get_resolution (image_ID, &xres, &yres);
  unit = gimp_image_get_unit (image_ID);
392

393 394 395 396 397 398 399 400
  size_entry = gimp_size_entry_new (1, unit, "%a", TRUE, TRUE, FALSE, 4,
                                    GIMP_SIZE_ENTRY_UPDATE_SIZE);
  gtk_box_pack_start (GTK_BOX (hbox), size_entry, FALSE, FALSE, 0);

  gimp_size_entry_set_unit (GIMP_SIZE_ENTRY (size_entry), GIMP_UNIT_PIXEL);
  gimp_size_entry_set_resolution (GIMP_SIZE_ENTRY (size_entry), 0, xres, TRUE);

  /*  set the size (in pixels) that will be treated as 0% and 100%  */
David Odin's avatar
David Odin committed
401 402
  gimp_size_entry_set_size (GIMP_SIZE_ENTRY (size_entry), 0, 0.0,
                            drawable->width);
403 404 405 406 407 408 409

  gimp_size_entry_set_refval_boundaries (GIMP_SIZE_ENTRY (size_entry), 0,
                                         1.0, 256.0);
  gtk_table_set_col_spacing (GTK_TABLE (size_entry), 0, 4);
  gtk_table_set_col_spacing (GTK_TABLE (size_entry), 2, 12);
  gimp_size_entry_set_refval (GIMP_SIZE_ENTRY (size_entry), 0,
                              (gdouble) borderaverage_thickness);
410
  g_signal_connect (size_entry, "value-changed",
411 412 413
                    G_CALLBACK (thickness_callback),
                    NULL);
  gtk_widget_show (size_entry);
414

415
  frame = gimp_frame_new (_("Number of Colors"));
David Odin's avatar
David Odin committed
416
  gtk_box_pack_start (GTK_BOX (main_vbox), frame, FALSE, FALSE, 0);
417 418
  gtk_widget_show (frame);

419
  hbox = gtk_hbox_new (FALSE, 6);
420 421 422
  gtk_container_add (GTK_CONTAINER (frame), hbox);
  gtk_widget_show (hbox);

423
  label = gtk_label_new_with_mnemonic (_("_Bucket size:"));
424
  gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
425 426 427
  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
  gtk_widget_show (label);

428 429
  gtk_size_group_add_widget (group, label);

430 431 432 433 434 435 436 437 438 439 440
  combo = gimp_int_combo_box_new_array (G_N_ELEMENTS (labels), labels);
  gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (combo),
                                 borderaverage_bucket_exponent);

  g_signal_connect (combo, "changed",
                    G_CALLBACK (gimp_int_combo_box_get_active),
                    &borderaverage_bucket_exponent);

  gtk_box_pack_start (GTK_BOX (hbox), combo, FALSE, FALSE, 0);
  gtk_widget_show (combo);
  gtk_label_set_mnemonic_widget (GTK_LABEL (label), combo);
441

David Odin's avatar
David Odin committed
442
  gtk_widget_show (dialog);
443

David Odin's avatar
David Odin committed
444
  run = (gimp_dialog_run (GIMP_DIALOG (dialog)) == GTK_RESPONSE_OK);
445

David Odin's avatar
David Odin committed
446
  gtk_widget_destroy (dialog);
447

448
  return run;
449
}
450 451 452

static void
thickness_callback (GtkWidget *widget,
David Odin's avatar
David Odin committed
453
                    gpointer   data)
454
{
David Odin's avatar
David Odin committed
455
  borderaverage_thickness =
456 457
    gimp_size_entry_get_refval (GIMP_SIZE_ENTRY (widget), 0);
}