mirror of https://github.com/GNOME/gimp.git
482 lines
12 KiB
C
482 lines
12 KiB
C
/* The GIMP -- an image manipulation program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
* Copyright (C) 1999 Adrian Likins and Tor Lillqvist
|
|
*
|
|
* 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"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
#include "apptypes.h"
|
|
#include "appenv.h"
|
|
#include "brush_header.h"
|
|
#include "pattern_header.h"
|
|
#include "patterns.h"
|
|
#include "gimpbrush.h"
|
|
#include "gimpbrushpipe.h"
|
|
#include "gimpbrushpipeP.h"
|
|
#include "paint_core.h"
|
|
#include "gimprc.h"
|
|
|
|
#include "libgimp/gimpintl.h"
|
|
#include "libgimp/gimpmath.h"
|
|
#include "libgimp/parasiteio.h"
|
|
|
|
static GimpBrushClass* gimp_brush_class;
|
|
static GtkObjectClass* gimp_object_class;
|
|
|
|
static GimpBrush *gimp_brush_pixmap_select_brush (PaintCore *paint_core);
|
|
static gboolean gimp_brush_pixmap_want_null_motion (PaintCore *paint_core);
|
|
|
|
static void
|
|
gimp_brush_pixmap_destroy (GtkObject *object)
|
|
{
|
|
GimpBrushPixmap *pixmap;
|
|
|
|
g_return_if_fail (GIMP_IS_BRUSH_PIXMAP (object));
|
|
|
|
pixmap = GIMP_BRUSH_PIXMAP (object);
|
|
|
|
temp_buf_free (pixmap->pixmap_mask);
|
|
|
|
(* GTK_OBJECT_CLASS (gimp_object_class)->destroy) (object);
|
|
}
|
|
|
|
static void
|
|
gimp_brush_pixmap_class_init (GimpBrushPixmapClass *klass)
|
|
{
|
|
GtkObjectClass *object_class;
|
|
GimpBrushClass *brush_class;
|
|
|
|
object_class = GTK_OBJECT_CLASS (klass);
|
|
brush_class = GIMP_BRUSH_CLASS (klass);
|
|
|
|
gimp_brush_class = gtk_type_class (gimp_brush_get_type ());
|
|
|
|
object_class->destroy = gimp_brush_pixmap_destroy;
|
|
brush_class->select_brush = gimp_brush_pixmap_select_brush;
|
|
brush_class->want_null_motion = gimp_brush_pixmap_want_null_motion;
|
|
}
|
|
|
|
void
|
|
gimp_brush_pixmap_init (GimpBrushPixmap *brush)
|
|
{
|
|
brush->pixmap_mask = NULL;
|
|
brush->pipe = NULL;
|
|
}
|
|
|
|
GtkType
|
|
gimp_brush_pixmap_get_type (void)
|
|
{
|
|
static GtkType type = 0;
|
|
|
|
if (!type)
|
|
{
|
|
GtkTypeInfo info =
|
|
{
|
|
"GimpBrushPixmap",
|
|
sizeof (GimpBrushPixmap),
|
|
sizeof (GimpBrushPixmapClass),
|
|
(GtkClassInitFunc) gimp_brush_pixmap_class_init,
|
|
(GtkObjectInitFunc) gimp_brush_pixmap_init,
|
|
/* reserved_1 */ NULL,
|
|
/* reserved_2 */ NULL,
|
|
(GtkClassInitFunc) NULL
|
|
};
|
|
|
|
type = gtk_type_unique (GIMP_TYPE_BRUSH, &info);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
static GimpBrush *
|
|
gimp_brush_pixmap_select_brush (PaintCore *paint_core)
|
|
{
|
|
GimpBrushPipe *pipe;
|
|
int i, brushix, ix;
|
|
double angle;
|
|
|
|
g_return_val_if_fail (GIMP_IS_BRUSH_PIXMAP (paint_core->brush), NULL);
|
|
|
|
pipe = GIMP_BRUSH_PIXMAP (paint_core->brush)->pipe;
|
|
|
|
if (pipe->nbrushes == 1)
|
|
return GIMP_BRUSH (pipe->current);
|
|
|
|
brushix = 0;
|
|
for (i = 0; i < pipe->dimension; i++)
|
|
{
|
|
switch (pipe->select[i])
|
|
{
|
|
case PIPE_SELECT_INCREMENTAL:
|
|
ix = (pipe->index[i] + 1) % pipe->rank[i];
|
|
break;
|
|
case PIPE_SELECT_ANGULAR:
|
|
angle = atan2 (paint_core->cury - paint_core->lasty,
|
|
paint_core->curx - paint_core->lastx);
|
|
/* Offset angle to be compatible with PSP tubes */
|
|
angle += G_PI_2;
|
|
/* Map it to the [0..2*G_PI) interval */
|
|
if (angle < 0)
|
|
angle += 2.*G_PI;
|
|
else if (angle > 2.*G_PI)
|
|
angle -= 2.*G_PI;
|
|
ix = RINT (angle / (2.*G_PI) * pipe->rank[i]);
|
|
break;
|
|
case PIPE_SELECT_RANDOM:
|
|
/* This probably isn't the right way */
|
|
ix = rand () % pipe->rank[i];
|
|
break;
|
|
case PIPE_SELECT_PRESSURE:
|
|
ix = RINT (paint_core->curpressure * (pipe->rank[i] - 1));
|
|
break;
|
|
case PIPE_SELECT_TILT_X:
|
|
ix = RINT (paint_core->curxtilt / 2.0 * pipe->rank[i]) + pipe->rank[i]/2;
|
|
break;
|
|
case PIPE_SELECT_TILT_Y:
|
|
ix = RINT (paint_core->curytilt / 2.0 * pipe->rank[i]) + pipe->rank[i]/2;
|
|
break;
|
|
case PIPE_SELECT_CONSTANT:
|
|
default:
|
|
ix = pipe->index[i];
|
|
break;
|
|
}
|
|
pipe->index[i] = BOUNDS (ix, 0, pipe->rank[i]-1);
|
|
brushix += pipe->stride[i] * pipe->index[i];
|
|
/* g_print ("ix at %d: %d, brushix: %d\n", i, ix, brushix); */
|
|
}
|
|
|
|
/* Make sure is inside bounds */
|
|
brushix = BOUNDS (brushix, 0, pipe->nbrushes-1);
|
|
|
|
pipe->current = pipe->brushes[brushix];
|
|
|
|
return GIMP_BRUSH (pipe->current);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_brush_pixmap_want_null_motion (PaintCore *paint_core)
|
|
{
|
|
GimpBrushPipe *pipe;
|
|
int i;
|
|
|
|
g_return_val_if_fail (GIMP_IS_BRUSH_PIXMAP (paint_core->brush), TRUE);
|
|
|
|
pipe = GIMP_BRUSH_PIXMAP (paint_core->brush)->pipe;
|
|
|
|
if (pipe->nbrushes == 1)
|
|
return TRUE;
|
|
|
|
for (i = 0; i < pipe->dimension; i++)
|
|
if (pipe->select[i] == PIPE_SELECT_ANGULAR)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_brush_pipe_destroy(GtkObject *object)
|
|
{
|
|
GimpBrushPipe *pipe;
|
|
int i;
|
|
|
|
g_return_if_fail (object != NULL);
|
|
g_return_if_fail (GIMP_IS_BRUSH_PIPE (object));
|
|
|
|
pipe = GIMP_BRUSH_PIPE (object);
|
|
|
|
g_free (pipe->rank);
|
|
g_free (pipe->stride);
|
|
|
|
for (i = 1; i < pipe->nbrushes; i++)
|
|
gimp_object_destroy (pipe->brushes[i]);
|
|
|
|
g_free (pipe->brushes);
|
|
g_free (pipe->select);
|
|
g_free (pipe->index);
|
|
|
|
if (GTK_OBJECT_CLASS (gimp_object_class)->destroy)
|
|
(* GTK_OBJECT_CLASS (gimp_object_class)->destroy) (object);
|
|
}
|
|
|
|
static void
|
|
gimp_brush_pipe_class_init (GimpBrushPipeClass *klass)
|
|
{
|
|
GtkObjectClass *object_class;
|
|
|
|
object_class = GTK_OBJECT_CLASS (klass);
|
|
|
|
gimp_object_class = gtk_type_class (GIMP_TYPE_OBJECT);
|
|
object_class->destroy = gimp_brush_pipe_destroy;
|
|
}
|
|
|
|
void
|
|
gimp_brush_pipe_init (GimpBrushPipe *pipe)
|
|
{
|
|
pipe->dimension = 0;
|
|
pipe->rank = NULL;
|
|
pipe->nbrushes = 0;
|
|
pipe->select = NULL;
|
|
pipe->index = NULL;
|
|
}
|
|
|
|
GtkType
|
|
gimp_brush_pipe_get_type (void)
|
|
{
|
|
static GtkType type=0;
|
|
if (!type){
|
|
GtkTypeInfo info={
|
|
"GimpBrushPipe",
|
|
sizeof (GimpBrushPipe),
|
|
sizeof (GimpBrushPipeClass),
|
|
(GtkClassInitFunc) gimp_brush_pipe_class_init,
|
|
(GtkObjectInitFunc) gimp_brush_pipe_init,
|
|
/* reserved_1 */ NULL,
|
|
/* reserved_2 */ NULL,
|
|
(GtkClassInitFunc) NULL};
|
|
type = gtk_type_unique (GIMP_TYPE_BRUSH_PIXMAP, &info);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GimpBrushPipe *
|
|
gimp_brush_pipe_load (char *filename)
|
|
{
|
|
GimpBrushPipe *pipe;
|
|
GPattern *pattern;
|
|
PixPipeParams params;
|
|
FILE *fp;
|
|
gchar buf[1024];
|
|
gchar *name;
|
|
int i;
|
|
int num_of_brushes;
|
|
int totalcells;
|
|
gchar *paramstring;
|
|
|
|
if ((fp = fopen (filename, "rb")) == NULL)
|
|
return NULL;
|
|
|
|
/* The file format starts with a painfully simple text header
|
|
* and we use a painfully simple way to read it
|
|
*/
|
|
if (fgets (buf, 1024, fp) == NULL)
|
|
{
|
|
fclose (fp);
|
|
return NULL;
|
|
}
|
|
buf[strlen (buf) - 1] = 0;
|
|
|
|
pipe = GIMP_BRUSH_PIPE (gimp_type_new (GIMP_TYPE_BRUSH_PIPE));
|
|
name = g_strdup (buf);
|
|
|
|
/* get the number of brushes */
|
|
if (fgets (buf, 1024, fp) == NULL)
|
|
{
|
|
fclose (fp);
|
|
gimp_object_destroy (pipe);
|
|
return NULL;
|
|
}
|
|
num_of_brushes = strtol(buf, ¶mstring, 10);
|
|
if (num_of_brushes < 1)
|
|
{
|
|
g_message (_("pixmap brush pipe should have at least one brush"));
|
|
fclose (fp);
|
|
gimp_object_destroy (pipe);
|
|
return NULL;
|
|
}
|
|
|
|
while (*paramstring && isspace(*paramstring))
|
|
paramstring++;
|
|
|
|
if (*paramstring)
|
|
{
|
|
pixpipeparams_init (¶ms);
|
|
pixpipeparams_parse (paramstring, ¶ms);
|
|
pipe->dimension = params.dim;
|
|
pipe->rank = g_new (int, pipe->dimension);
|
|
pipe->select = g_new (PipeSelectModes, pipe->dimension);
|
|
pipe->index = g_new (int, pipe->dimension);
|
|
/* placement is not used at all ?? */
|
|
if (params.free_placement_string)
|
|
g_free (params.placement);
|
|
for (i = 0; i < pipe->dimension; i++)
|
|
{
|
|
pipe->rank[i] = params.rank[i];
|
|
if (strcmp (params.selection[i], "incremental") == 0)
|
|
pipe->select[i] = PIPE_SELECT_INCREMENTAL;
|
|
else if (strcmp (params.selection[i], "angular") == 0)
|
|
pipe->select[i] = PIPE_SELECT_ANGULAR;
|
|
else if (strcmp (params.selection[i], "velocity") == 0)
|
|
pipe->select[i] = PIPE_SELECT_VELOCITY;
|
|
else if (strcmp (params.selection[i], "random") == 0)
|
|
pipe->select[i] = PIPE_SELECT_RANDOM;
|
|
else if (strcmp (params.selection[i], "pressure") == 0)
|
|
pipe->select[i] = PIPE_SELECT_PRESSURE;
|
|
else if (strcmp (params.selection[i], "xtilt") == 0)
|
|
pipe->select[i] = PIPE_SELECT_TILT_X;
|
|
else if (strcmp (params.selection[i], "ytilt") == 0)
|
|
pipe->select[i] = PIPE_SELECT_TILT_Y;
|
|
else
|
|
pipe->select[i] = PIPE_SELECT_CONSTANT;
|
|
if (params.free_selection_string)
|
|
g_free (params.selection[i]);
|
|
pipe->index[i] = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pipe->dimension = 1;
|
|
pipe->rank = g_new (int, 1);
|
|
pipe->rank[0] = num_of_brushes;
|
|
pipe->select = g_new (PipeSelectModes, 1);
|
|
pipe->select[0] = PIPE_SELECT_INCREMENTAL;
|
|
pipe->index = g_new (int, 1);
|
|
pipe->index[0] = 0;
|
|
}
|
|
|
|
totalcells = 1; /* Not all necessarily present, maybe */
|
|
for (i = 0; i < pipe->dimension; i++)
|
|
totalcells *= pipe->rank[i];
|
|
pipe->stride = g_new (int, pipe->dimension);
|
|
for (i = 0; i < pipe->dimension; i++)
|
|
{
|
|
if (i == 0)
|
|
pipe->stride[i] = totalcells / pipe->rank[i];
|
|
else
|
|
pipe->stride[i] = pipe->stride[i-1] / pipe->rank[i];
|
|
}
|
|
g_assert (pipe->stride[pipe->dimension-1] == 1);
|
|
|
|
pattern = (GPattern*) g_malloc0 (sizeof (GPattern));
|
|
|
|
pipe->brushes =
|
|
(GimpBrushPixmap **) g_new0 (GimpBrushPixmap *, num_of_brushes);
|
|
|
|
/* First pixmap brush in the list is the pipe itself */
|
|
pipe->brushes[0] = GIMP_BRUSH_PIXMAP (pipe);
|
|
|
|
/* Current pixmap brush is the first one. */
|
|
pipe->current = pipe->brushes[0];
|
|
|
|
while (pipe->nbrushes < num_of_brushes)
|
|
{
|
|
if (pipe->nbrushes > 0)
|
|
{
|
|
pipe->brushes[pipe->nbrushes] =
|
|
GIMP_BRUSH_PIXMAP (gimp_type_new (GIMP_TYPE_BRUSH_PIXMAP));
|
|
g_free (GIMP_BRUSH (pipe->brushes[pipe->nbrushes])->name);
|
|
GIMP_BRUSH (pipe->brushes[pipe->nbrushes])->name = NULL;
|
|
}
|
|
|
|
pipe->brushes[pipe->nbrushes]->pipe = pipe;
|
|
|
|
/* load the brush */
|
|
if (!gimp_brush_load_brush (GIMP_BRUSH (pipe->brushes[pipe->nbrushes]),
|
|
fp, filename)
|
|
|| !pattern_load (pattern, fp, filename))
|
|
{
|
|
g_message (_("failed to load one of the pixmap brushes in the pipe"));
|
|
fclose (fp);
|
|
pattern_free (pattern);
|
|
gimp_object_destroy (pipe);
|
|
return NULL;
|
|
}
|
|
|
|
if (pipe->nbrushes == 0)
|
|
{
|
|
/* Replace name with the whole pipe's name */
|
|
g_free (GIMP_BRUSH (pipe)->name);
|
|
GIMP_BRUSH (pipe)->name = name;
|
|
}
|
|
pipe->brushes[pipe->nbrushes]->pixmap_mask = pattern->mask;
|
|
g_free (pattern->name);
|
|
pipe->nbrushes++;
|
|
}
|
|
|
|
/* Clean up */
|
|
fclose (fp);
|
|
|
|
g_free (pattern);
|
|
return pipe;
|
|
}
|
|
|
|
GimpBrushPipe *
|
|
gimp_brush_pixmap_load (char *filename)
|
|
{
|
|
GimpBrushPipe *pipe;
|
|
GPattern *pattern;
|
|
FILE *fp;
|
|
|
|
if ((fp = fopen (filename, "rb")) == NULL)
|
|
return NULL;
|
|
|
|
pipe = GIMP_BRUSH_PIPE (gimp_type_new (GIMP_TYPE_BRUSH_PIPE));
|
|
|
|
/* A (single) pixmap brush is a pixmap pipe brush with just one pixmap */
|
|
pipe->dimension = 1;
|
|
pipe->rank = g_new (int, 1);
|
|
pipe->rank[0] = 1;
|
|
pipe->select = g_new (PipeSelectModes, 1);
|
|
pipe->select[0] = PIPE_SELECT_INCREMENTAL;
|
|
pipe->index = g_new (int, 1);
|
|
pipe->index[0] = 0;
|
|
|
|
pattern = (GPattern *) g_malloc0 (sizeof (GPattern));
|
|
|
|
pipe->brushes = (GimpBrushPixmap **) g_new (GimpBrushPixmap *, 1);
|
|
|
|
pipe->brushes[0] = GIMP_BRUSH_PIXMAP (pipe);
|
|
pipe->current = pipe->brushes[0];
|
|
|
|
pipe->brushes[0]->pipe = pipe;
|
|
|
|
/* load the brush */
|
|
if (!gimp_brush_load_brush (GIMP_BRUSH (pipe->brushes[0]),
|
|
fp, filename)
|
|
|| !pattern_load (pattern, fp, filename))
|
|
{
|
|
g_message (_("failed to load pixmap brush"));
|
|
fclose (fp);
|
|
pattern_free (pattern);
|
|
gimp_object_destroy (pipe);
|
|
return NULL;
|
|
}
|
|
|
|
pipe->brushes[0]->pixmap_mask = pattern->mask;
|
|
|
|
pipe->nbrushes = 1;
|
|
/* Clean up */
|
|
fclose (fp);
|
|
|
|
g_free (pattern->name);
|
|
g_free (pattern);
|
|
return pipe;
|
|
}
|
|
|
|
TempBuf *
|
|
gimp_brush_pixmap_pixmap (GimpBrushPixmap *brush)
|
|
{
|
|
g_return_val_if_fail (brush != NULL, NULL);
|
|
g_return_val_if_fail (GIMP_IS_BRUSH_PIXMAP (brush), NULL);
|
|
|
|
return brush->pixmap_mask;
|
|
}
|