gimp/app/tools/xinput_airbrush.c

3273 lines
76 KiB
C
Raw Normal View History

/* 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
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "config.h"
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include "airbrush_blob.h"
#include "drawable.h"
#include "draw_core.h"
#include "gdisplay.h"
#include "gimage_mask.h"
#include "gimprc.h"
#include "gimpui.h"
#include "paint_options.h"
#include "tools.h"
#include "undo.h"
#include "xinput_airbrush.h"
#include "libgimp/gimpintl.h"
#include "libgimp/gimpmath.h"
#include "tile.h" /* ick. */
#define SUBSAMPLE 8
#define DIST_SMOOTHER_BUFFER 10
#define TIME_SMOOTHER_BUFFER 10
/* the XinputAirbrush structures */
typedef struct _XinputAirbrushTool XinputAirbrushTool;
struct _XinputAirbrushTool
{
DrawCore * core; /* Core select object */
AirBrush * last_airbrush;
AirBrushBlob * last_airbrush_blob; /* airbrush_blob for last cursor position */
AirBlob * last_airblob;
int x1, y1; /* image space coordinate */
int x2, y2; /* image space coords */
/* circular distance history buffer */
gdouble dt_buffer[DIST_SMOOTHER_BUFFER];
gint dt_index;
/* circular timing history buffer */
guint32 ts_buffer[TIME_SMOOTHER_BUFFER];
gint ts_index;
/* Direction and center point */
double xcenter, ycenter;
double direction_abs;
double direction;
double c_direction;
double c_direction_abs;
gdouble last_time; /* previous time of a motion event */
gdouble lastx, lasty; /* previous position of a motion event */
gdouble last_lastx, last_lasty; /* The previous previous position of a motion event i.e pos of the blob */
gboolean init_velocity;
gboolean init_prepre;
double last_value;
};
typedef struct _XinputAirbrushOptions XinputAirbrushOptions;
struct _XinputAirbrushOptions
{
PaintOptions paint_options;
double flow;
double flow_d;
GtkObject *flow_w;
double sensitivity;
double sensitivity_d;
GtkObject *sensitivity_w;
double starttilt;
double starttilt_d;
GtkObject *starttilt_w;
double vel_sensitivity;
double vel_sensitivity_d;
GtkObject *vel_sensitivity_w;
double tilt_sensitivity;
double tilt_sensitivity_d;
GtkObject *tilt_sensitivity_w;
#ifdef GTK_HAVE_SIX_VALUATORS
double minheight;
double minheight_d;
GtkObject *minheight_w;
double maxheight;
double maxheight_d;
GtkObject *maxheight_w;
#else /* !GTK_HAVE_SIX_VALUATORS */
double height;
double height_d;
GtkObject *height_w;
#endif /* GTK_HAVE_SIX_VALUATORS */
};
/* the xinput_airbrush tool options */
static XinputAirbrushOptions *xinput_airbrush_options = NULL;
/* local variables */
/* undo blocks variables */
static TileManager * undo_tiles = NULL;
/* Tiles used to render the stroke at 1 byte/pp */
static TileManager * canvas_tiles = NULL;
/* Flat buffer that is used to used to render the dirty region
* for composition onto the destination drawable
*/
static TempBuf * canvas_buf = NULL;
/* local function prototypes */
static void xinput_airbrush_button_press (Tool *, GdkEventButton *, gpointer);
static void xinput_airbrush_button_release (Tool *, GdkEventButton *, gpointer);
static void xinput_airbrush_motion (Tool *, GdkEventMotion *, gpointer);
static void xinput_airbrush_cursor_update (Tool *, GdkEventMotion *, gpointer);
static void xinput_airbrush_control (Tool *, ToolAction, gpointer);
static void time_smoother_add (XinputAirbrushTool* xinput_airbrush_tool, guint32 value);
static gdouble time_smoother_result (XinputAirbrushTool* xinput_airbrush_tool);
static void time_smoother_init (XinputAirbrushTool* xinput_airbrush_tool, guint32 initval);
static void dist_smoother_add (XinputAirbrushTool* xinput_airbrush_tool, gdouble value);
static gdouble dist_smoother_result (XinputAirbrushTool* xinput_airbrush_tool);
static void dist_smoother_init (XinputAirbrushTool* xinput_airbrush_tool, gdouble initval);
static void xinput_airbrush_init (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
double x,
double y);
static void xinput_airbrush_finish (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
int tool_id);
static void xinput_airbrush_cleanup (void);
/*Mask functions*/
static void calc_angle (AirBrushBlob *airbrush_blob,
double xcenter,
double ycenter);
static void calc_width (AirBrushBlob *airbrush_blob);
static void render_airbrush_line (AirBrushBlob *airbrush_blob,
guchar *dest,
int y,
int width,
XinputAirbrushTool *xinput_airbrush_tool);
static void make_single_mask (AirBrushBlob *airbrush_blob,
XinputAirbrushTool *xinput_airbrush_tool,
MaskBuf *dest);
static void make_stroke(AirBlob *airblob,
XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
guint last_value,
guint present_value);
static void make_mask(AirLine *airline,
MaskBuf *brush_mask,
guint value);
static void print_mask(MaskBuf *dest);
/* Rendering functions */
static void xinput_airbrush_set_paint_area (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
int x,
int y,
int width,
int height);
static void xinput_airbrush_paste (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
MaskBuf *brush_mask,
int x,
int y,
int width,
int height);
static void xinput_airbrush_to_canvas_tiles (XinputAirbrushTool *xinput_airbrush_tool,
MaskBuf *brush_mask,
int brush_opacity);
static void xinput_airbrush_canvas_tiles_to_canvas_buf(XinputAirbrushTool *xinput_airbrush_tool);
static void xinput_airbrush_set_undo_tiles (GimpDrawable *drawable,
int x,
int y,
int w,
int h);
static void xinput_airbrush_set_canvas_tiles(int x,
int y,
int w,
int h);
/* Start of functions */
static void
xinput_airbrush_options_reset (void)
{
XinputAirbrushOptions *options = xinput_airbrush_options;
paint_options_reset ((PaintOptions *) options);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->flow_w),
options->flow_d);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->sensitivity_w),
options->sensitivity_d);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->starttilt_w),
options->starttilt_d);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->tilt_sensitivity_w),
options->tilt_sensitivity_d);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->vel_sensitivity_w),
options->vel_sensitivity_d);
#ifdef GTK_HAVE_SIX_VALUATORS
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->minheight_w),
options->minheight_d);
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->maxheight_w),
options->maxheight_d);
#else /* !GTK_HAVE_SIX_VALUATORS */
gtk_adjustment_set_value (GTK_ADJUSTMENT (options->height_w),
options->height_d);
#endif /* GTK_HAVE_SIX_VALUATORS */
}
static XinputAirbrushOptions *
xinput_airbrush_options_new (void)
{
XinputAirbrushOptions *options;
GtkWidget *table;
GtkWidget *abox;
GtkWidget *label;
GtkWidget *slider;
/* the new xinput_airbrush tool options structure */
options = (XinputAirbrushOptions *) g_malloc (sizeof (XinputAirbrushOptions));
paint_options_init ((PaintOptions *) options,
XINPUT_AIRBRUSH,
xinput_airbrush_options_reset);
options->flow = options->flow_d = 100;
options->sensitivity = options->sensitivity_d = 1.0;
options->starttilt = options->starttilt_d = 90;
options->vel_sensitivity = options->vel_sensitivity_d = 0.8;
options->tilt_sensitivity = options->tilt_sensitivity_d = 0.4;
#ifdef GTK_HAVE_SIX_VALUATORS
options->minheight = options->minheight_d = 25.0;
options->maxheight = options->maxheight_d = 50.0;
#else /* !GTK_HAVE_SIX_VALUATORS */
options->height = options->height_d = 45.0;
#endif /* GTK_HAVE_SIX_VALUATORS */
/*the main table*/
table = gtk_table_new (6, 2, FALSE);
gtk_box_pack_start (GTK_BOX (((ToolOptions *) options)->main_vbox), table,
FALSE, FALSE, 0);
/*flow slider*/
label = gtk_label_new (_("Flow Relation:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 0, 1,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->flow_w =
gtk_adjustment_new (options->flow_d, 50.0, 201.0, 1.0, 10.0, 1.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->flow_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 0, 1);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->flow_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->flow);
gtk_widget_show (slider);
gtk_scale_set_digits (GTK_SCALE (slider), 0);
/*flow sens slider*/
label = gtk_label_new (_("Flow Sensitivity:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 1, 2,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->sensitivity_w =
gtk_adjustment_new (options->sensitivity_d, 0.0, 1.0, 0.01, 0.1, 0.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->sensitivity_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 1, 2);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->sensitivity_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->sensitivity);
gtk_widget_show (slider);
/*base tilt slider*/
label = gtk_label_new (_("Base Tilt:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 2, 3,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->starttilt_w =
gtk_adjustment_new (options->starttilt_d, 30.0 , 91.0, 1.0, 1.0, 1.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->starttilt_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 2, 3);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->starttilt_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->starttilt);
gtk_widget_show (slider);
gtk_scale_set_digits (GTK_SCALE (slider), 0);
/*tilt sens slider*/
label = gtk_label_new (_("Tilt Sensitivity:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 3, 4,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
abox = gtk_alignment_new (0.5, 1.0, 1.0, 0.0);
gtk_table_attach (GTK_TABLE (table), abox, 1, 2, 3, 4,
GTK_EXPAND | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (abox);
options->tilt_sensitivity_w =
gtk_adjustment_new (options->tilt_sensitivity_d, 0.0, 1.0, 0.01, 0.1, 0.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->tilt_sensitivity_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_container_add (GTK_CONTAINER (abox), slider);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->tilt_sensitivity_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->tilt_sensitivity);
gtk_widget_show (slider);
/*velocity sens slider*/
label = gtk_label_new (_("Speed Sensitivity:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 5, 6,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
abox = gtk_alignment_new (0.5, 1.0, 1.0, 0.0);
gtk_table_attach (GTK_TABLE (table), abox, 1, 2, 5, 6,
GTK_EXPAND | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (abox);
options->vel_sensitivity_w =
gtk_adjustment_new (options->vel_sensitivity_d, 0.0, 1.0, 0.01, 0.1, 0.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->vel_sensitivity_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_container_add (GTK_CONTAINER (abox), slider);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->vel_sensitivity_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->vel_sensitivity);
gtk_widget_show (slider);
#ifdef GTK_HAVE_SIX_VALUATORS
/*min height slider*/
label = gtk_label_new (_("Min Height:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 7, 8,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->minheight_w =
gtk_adjustment_new (options->minheight_d, 25.0 , 41.0, 1.0, 1.0, 1.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->minheight_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 7, 8);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->minheight_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->minheight);
gtk_widget_show (slider);
gtk_scale_set_digits (GTK_SCALE (slider), 0);
/*max height slider*/
label = gtk_label_new (_("Max Height:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 9, 10,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->maxheight_w =
gtk_adjustment_new (options->maxheight_d, 41.0 , 81.0, 1.0, 1.0, 1.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->maxheight_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 9, 10);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->maxheight_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->maxheight);
gtk_widget_show (slider);
gtk_scale_set_digits (GTK_SCALE (slider), 0);
#else /* !GTK_HAVE_SIX_VALUATORS */
label = gtk_label_new (_("Height:"));
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 1.0);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 7, 8,
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
gtk_widget_show (label);
options->height_w =
gtk_adjustment_new (options->height_d, 25.0 , 81.0, 1.0, 1.0, 1.0);
slider = gtk_hscale_new (GTK_ADJUSTMENT (options->height_w));
gtk_scale_set_value_pos (GTK_SCALE (slider), GTK_POS_TOP);
gtk_table_attach_defaults (GTK_TABLE (table), slider, 1, 2, 7, 8);
gtk_range_set_update_policy (GTK_RANGE (slider), GTK_UPDATE_DELAYED);
gtk_signal_connect (GTK_OBJECT (options->height_w), "value_changed",
GTK_SIGNAL_FUNC (gimp_double_adjustment_update),
&options->height);
gtk_widget_show (slider);
gtk_scale_set_digits (GTK_SCALE (slider), 0);
#endif /* GTK_HAVE_SIX_VALUATORS */
gtk_widget_show_all (table);
return options;
}
static AirBlob *
1999-11-30 08:58:38 +08:00
#ifdef GTK_HAVE_SIX_VALUATORS
xinput_airbrush_pen_ellipse (XinputAirbrushTool *xinput_airbrush_tool,
gdouble x_center, gdouble y_center,
gdouble pressure, gdouble xtiltv,
gdouble ytiltv, gdouble wheel)
#else /* !GTK_HAVE_SIX_VALUATORS */
xinput_airbrush_pen_ellipse (XinputAirbrushTool *xinput_airbrush_tool,
gdouble x_center, gdouble y_center,
gdouble pressure, gdouble xtiltv,
gdouble ytiltv)
#endif /* GTK_HAVE_SIX_VALUATORS */
{
gdouble xtilt, ytilt;
double height;
#ifdef GTK_HAVE_SIX_VALUATORS
double inter_height;
double min_height;
double max_height;
#endif /* GTK_HAVE_SIX_VALUATORS */
double tanx, tany;
double tanytop, tanxright, tanybot, tanxleft;
double sprayangle;
double distx, disty;
double ytop, xright, ybot, xleft;
/* Virtual height over the drawing */
#ifdef GTK_HAVE_SIX_VALUATORS
min_height=xinput_airbrush_options->minheight;
max_height=xinput_airbrush_options->maxheight;
inter_height = CLAMP(wheel, 0.0, 1.0) * (max_height - min_height);
height = min_height + inter_height;
/*
printf("Height: %f\n", height);
*/
#else /* !GTK_HAVE_SIX_VALUATORS */
height = xinput_airbrush_options->height;
#endif /* GTK_HAVE_SIX_VALUATORS */
/*
Fix av tilt negative when it should be positive
Also a clamp of to big tilts compared to the height
i.e you will not get big brushes
*/
xtilt = xtiltv * -1.0 * (1 - ((0.5 * (height - 15))/100));
ytilt = ytiltv * -1.0 * (1 - ((0.5 * (height - 15))/100));
/* The airflow controls the spray angel
High airflow renders in a big spray angel
but if the inkflow is low the blob will be
very thin.
*/
sprayangle = G_PI/6 * (xinput_airbrush_options->flow/100.0) * MAX(pressure, 0.1);
sprayangle = MAX(sprayangle, 0.0);
/*printf("Angle: %f\n", sprayangle);*/
/*Tan of x and y tilt plus spray angles x r/l and y t/b tan*/
tanx = tan(xtilt * G_PI / 2.0);
tany = tan(ytilt * G_PI / 2.0);
tanytop = tan((ytilt * G_PI / 2.0) + (sprayangle/2));
tanxright = tan((xtilt * G_PI / 2.0) + (sprayangle/2));
tanybot = tan((ytilt * G_PI / 2.0) - (sprayangle/2));
tanxleft = tan((xtilt * G_PI / 2.0) - (sprayangle/2));
/* Offset from cursor due to tilt in x and y depening on the hight*/
distx = tanx * height;
disty = tany * height;
/* Offset from center of blob in all for axies */
ytop = fabs(fabs(tanytop * height) - fabs(disty));
xright = fabs(fabs(tanxright * height) - fabs(distx));
ybot = fabs(fabs(tanybot * height) - fabs(disty));
xleft = fabs(fabs(tanxleft * height) - fabs(distx));
x_center = x_center + distx;
y_center = y_center + disty;
xinput_airbrush_tool->xcenter=x_center;
xinput_airbrush_tool->ycenter=y_center;
xinput_airbrush_tool->direction_abs=atan2(ytiltv, xtiltv) + G_PI;
xinput_airbrush_tool->direction=atan(ytiltv/xtiltv);
xinput_airbrush_tool->c_direction=atan(ytiltv/xtilt);
xinput_airbrush_tool->c_direction_abs=atan2(ytiltv, xtilt) + G_PI;
return create_air_blob(x_center * SUBSAMPLE, y_center * SUBSAMPLE,
0., ytop * SUBSAMPLE, xright * SUBSAMPLE, 0.,
0., ybot * SUBSAMPLE, xleft * SUBSAMPLE, 0.,
(xinput_airbrush_tool->c_direction_abs - G_PI),
xinput_airbrush_tool->c_direction);
}
static void
xinput_airbrush_button_press (Tool *tool,
GdkEventButton *bevent,
gpointer gdisp_ptr)
{
gdouble x, y;
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
GimpDisplay *gdisp;
XinputAirbrushTool *xinput_airbrush_tool;
GimpDrawable *drawable;
AirBlob *airblob;
AirLine *airline;
MaskBuf *brush_mask;
int width, height;
int x_min, y_min;
guint value;
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
gdisp = (GimpDisplay *) gdisp_ptr;
xinput_airbrush_tool = (XinputAirbrushTool *) tool->private;
/* Keep the coordinates of the target */
gdisplay_untransform_coords_f (gdisp, bevent->x, bevent->y,
&x, &y, TRUE);
drawable = gimage_active_drawable (gdisp->gimage);
xinput_airbrush_init (xinput_airbrush_tool, drawable, x, y);
tool->state = ACTIVE;
tool->gdisp_ptr = gdisp_ptr;
tool->paused_count = 0;
/* pause the current selection and grab the pointer */
gdisplays_selection_visibility (gdisp->gimage, SelectionPause);
/* add motion memory if you press mod1 first ^ perfectmouse */
if (((bevent->state & GDK_MOD1_MASK) != 0) != (perfectmouse != 0))
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_BUTTON1_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
else
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK | GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
tool->gdisp_ptr = gdisp_ptr;
tool->state = ACTIVE;
/* Get the AirBlob from xinput_airbrush_pen_ellipse */
#ifdef GTK_HAVE_SIX_VALUATORS
airblob = xinput_airbrush_pen_ellipse (xinput_airbrush_tool, x, y,
bevent->pressure, bevent->xtilt,
bevent->ytilt, bevent->wheel);
#else /* !GTK_HAVE_SIX_VALUATORS */
airblob = xinput_airbrush_pen_ellipse (xinput_airbrush_tool, x, y,
bevent->pressure, bevent->xtilt,
bevent->ytilt);
#endif /* GTK_HAVE_SIX_VALUATORS */
/* Make the AirBlob mask */
value = 256 * bevent->pressure;
airline = create_air_line(airblob);
x_min = airline->min_x;
y_min = airline->min_y;
width = airline->width;
height = airline->height;
brush_mask = mask_buf_new(width, height);
make_mask(airline, brush_mask, value);
/*print_mask(brush_mask);*/
/*Paint the mask on the drawable*/
xinput_airbrush_paste (xinput_airbrush_tool, drawable, brush_mask, x_min, y_min, width, height);
/*Prepare for next stroke*/
xinput_airbrush_tool->last_airblob = airblob;
time_smoother_init (xinput_airbrush_tool, bevent->time);
xinput_airbrush_tool->last_time = bevent->time;
dist_smoother_init (xinput_airbrush_tool, 0.0);
xinput_airbrush_tool->init_velocity = TRUE;
xinput_airbrush_tool->init_prepre = FALSE;
xinput_airbrush_tool->lastx = xinput_airbrush_tool->xcenter;
xinput_airbrush_tool->lasty = xinput_airbrush_tool->ycenter;
xinput_airbrush_tool->last_value = value;
/*Free the maks_buf and airline*/
mask_buf_free(brush_mask);
g_free(airline);
gdisplay_flush_now (gdisp);
}
static void
xinput_airbrush_button_release (Tool *tool,
GdkEventButton *bevent,
gpointer gdisp_ptr)
{
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
GimpDisplay * gdisp;
GImage * gimage;
XinputAirbrushTool * xinput_airbrush_tool;
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
gdisp = (GimpDisplay *) gdisp_ptr;
gimage = gdisp->gimage;
xinput_airbrush_tool = (XinputAirbrushTool *) tool->private;
/* resume the current selection and ungrab the pointer */
gdisplays_selection_visibility (gdisp->gimage, SelectionResume);
gdk_pointer_ungrab (bevent->time);
gdk_flush ();
/* Set tool state to inactive -- no longer painting */
tool->state = INACTIVE;
xinput_airbrush_finish (xinput_airbrush_tool, gimage_active_drawable (gdisp->gimage), tool->ID);
gdisplays_flush ();
}
static void
dist_smoother_init (XinputAirbrushTool* xinput_airbrush_tool, gdouble initval)
{
gint i;
xinput_airbrush_tool->dt_index = 0;
for (i=0; i<DIST_SMOOTHER_BUFFER; i++)
{
xinput_airbrush_tool->dt_buffer[i] = initval;
}
}
static gdouble
dist_smoother_result (XinputAirbrushTool* xinput_airbrush_tool)
{
gint i;
gdouble result = 0.0;
for (i=0; i<DIST_SMOOTHER_BUFFER; i++)
{
result += xinput_airbrush_tool->dt_buffer[i];
}
return (result / (gdouble)DIST_SMOOTHER_BUFFER);
}
static void
dist_smoother_add (XinputAirbrushTool* xinput_airbrush_tool, gdouble value)
{
xinput_airbrush_tool->dt_buffer[xinput_airbrush_tool->dt_index] = value;
if ((++xinput_airbrush_tool->dt_index) == DIST_SMOOTHER_BUFFER)
xinput_airbrush_tool->dt_index = 0;
}
static void
time_smoother_init (XinputAirbrushTool* xinput_airbrush_tool, guint32 initval)
{
gint i;
xinput_airbrush_tool->ts_index = 0;
for (i=0; i<TIME_SMOOTHER_BUFFER; i++)
{
xinput_airbrush_tool->ts_buffer[i] = initval;
}
}
static gdouble
time_smoother_result (XinputAirbrushTool* xinput_airbrush_tool)
{
gint i;
guint64 result = 0;
for (i=0; i<TIME_SMOOTHER_BUFFER; i++)
{
result += xinput_airbrush_tool->ts_buffer[i];
}
#ifdef _MSC_VER
return (gdouble) (gint64) (result / TIME_SMOOTHER_BUFFER);
#else
return (result / TIME_SMOOTHER_BUFFER);
#endif
}
static void
time_smoother_add (XinputAirbrushTool* xinput_airbrush_tool, guint32 value)
{
xinput_airbrush_tool->ts_buffer[xinput_airbrush_tool->ts_index] = value;
if ((++xinput_airbrush_tool->ts_index) == TIME_SMOOTHER_BUFFER)
xinput_airbrush_tool->ts_index = 0;
}
static void
xinput_airbrush_motion (Tool *tool,
GdkEventMotion *mevent,
gpointer gdisp_ptr)
{
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
GimpDisplay *gdisp;
XinputAirbrushTool *xinput_airbrush_tool;
GimpDrawable *drawable;
AirBlob *airblob;
double x, y;
double pressure;
double velocity;
double dist;
gdouble lasttime, thistime;
guint last_value, present_value;
double a, b, c, A;
gboolean turn_around;
2000-02-07 23:49:54 +08:00
#ifdef GTK_HAVE_SIX_VALUATORS
double min_height;
double max_height;
double inter_height;
2000-02-07 23:49:54 +08:00
#endif /* GTK_HAVE_SIX_VALUATORS */
double height;
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
gdisp = (GimpDisplay *) gdisp_ptr;
xinput_airbrush_tool = (XinputAirbrushTool *) tool->private;
gdisplay_untransform_coords_f (gdisp, mevent->x, mevent->y, &x, &y, TRUE);
drawable = gimage_active_drawable (gdisp->gimage);
pressure = mevent->pressure;
#ifdef GTK_HAVE_SIX_VALUATORS
airblob = xinput_airbrush_pen_ellipse (xinput_airbrush_tool, x, y, pressure, mevent->xtilt, mevent->ytilt, mevent->wheel);
#else /* !GTK_HAVE_SIX_VALUATORS */
airblob = xinput_airbrush_pen_ellipse (xinput_airbrush_tool, x, y, pressure, mevent->xtilt, mevent->ytilt);
#endif /* GTK_HAVE_SIX_VALUATORS */
x = xinput_airbrush_tool->xcenter;
y = xinput_airbrush_tool->ycenter;
lasttime = xinput_airbrush_tool->last_time;
time_smoother_add (xinput_airbrush_tool, mevent->time);
thistime = xinput_airbrush_tool->last_time =
time_smoother_result(xinput_airbrush_tool);
/* The time resolution on X-based GDK motion events is
bloody awful, hence the use of the smoothing function.
Sadly this also means that there is always the chance of
having an indeterminite velocity since this event and
the previous several may still appear to issue at the same
instant. -ADM */
if (thistime == lasttime)
thistime = lasttime + 1;
if (xinput_airbrush_tool->init_velocity)
{
dist_smoother_init (xinput_airbrush_tool, dist = sqrt((xinput_airbrush_tool->lastx-x)*(xinput_airbrush_tool->lastx-x) +
(xinput_airbrush_tool->lasty-y)*(xinput_airbrush_tool->lasty-y)));
xinput_airbrush_tool->init_velocity = FALSE;
}
else
{
dist_smoother_add (xinput_airbrush_tool, sqrt((xinput_airbrush_tool->lastx-x)*(xinput_airbrush_tool->lastx-x) +
(xinput_airbrush_tool->lasty-y)*(xinput_airbrush_tool->lasty-y)));
dist = dist_smoother_result(xinput_airbrush_tool);
}
turn_around = FALSE;
if (xinput_airbrush_tool->init_prepre)
{
a = hypot(xinput_airbrush_tool->last_lastx - x, xinput_airbrush_tool->last_lasty - y);
b = hypot(xinput_airbrush_tool->lastx - x, xinput_airbrush_tool->lasty - y);
c = hypot(xinput_airbrush_tool->last_lastx - xinput_airbrush_tool->lastx, xinput_airbrush_tool->last_lasty - xinput_airbrush_tool->lasty);
/* Maybe fix the fact that a, b or c can be zero i.e that we had a perfectly strait turn around or continue*/
A = acos((b*b + c*c - a*a)/2*b*c);
/* 1.65806 RAD == 95deg*/
if ( (A >= 0.0) && (A < 1.65806) )
{
turn_around = TRUE;
/* printf("TURN_AROUND\n");*/
}
}
velocity = 10.0 * sqrt((dist) / (double)(thistime - lasttime));
/* Normal Speed is 2.0, Break point for zero ink is 18.
Slow speed is between 2.0 and 0.5 */
/*printf("Speed: %f\n", velocity);*/
#ifdef GTK_HAVE_SIX_VALUATORS
min_height=xinput_airbrush_options->minheight;
max_height=xinput_airbrush_options->maxheight;
inter_height = CLAMP(mevent->wheel, 0.0, 1.0) * (max_height - min_height);
height = min_height + inter_height;
/*
printf("Height: %f\n", height);
*/
#else /* !GTK_HAVE_SIX_VALUATORS */
height = xinput_airbrush_options->height;
#endif /* GTK_HAVE_SIX_VALUATORS */
last_value = xinput_airbrush_tool->last_value;
present_value = 256 * pressure * (1 - ((height - 25)/100));
make_stroke(airblob, xinput_airbrush_tool, drawable, last_value, present_value);
xinput_airbrush_tool->last_lastx = xinput_airbrush_tool->lastx;
xinput_airbrush_tool->last_lasty = xinput_airbrush_tool->lasty;
xinput_airbrush_tool->lastx = x;
xinput_airbrush_tool->lasty = y;
xinput_airbrush_tool->last_value = present_value;
g_free(xinput_airbrush_tool->last_airblob);
xinput_airbrush_tool->last_airblob = airblob;
xinput_airbrush_tool->init_velocity = TRUE;
xinput_airbrush_tool->init_prepre = TRUE;
gdisplay_flush_now (gdisp);
}
static void
xinput_airbrush_cursor_update (Tool *tool,
GdkEventMotion *mevent,
gpointer gdisp_ptr)
{
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
GimpDisplay *gdisp;
Layer *layer;
GdkCursorType ctype = GDK_TOP_LEFT_ARROW;
int x, y;
Cleanup weekend... 2001-10-29 Michael Natterer <mitch@gimp.org> Cleanup weekend... * app/app_procs.c: pass "no_interface" to gimp_new(). * app/core/gimp.[ch]: added "gboolean no_interface" and the load_procs and save_procs GSLists. * app/core/gimptoolinfo.[ch]: added a "Gimp" pointer to the GimpToolInfo object so more functions find their context without accessing the global "the_gimp" variable. * app/display/display-types.h: removed the GDisplay -> GimpDisplay typedef. * app/display/gimpdisplay.c: look at gimp->no_interface, don't include "appenv.h". * app/file/file-open.[ch] * app/file/file-save.[ch]: don't use "the_gimp" any more. Instead, pass around lots of "Gimp" pointers. Removed the global load_procs and save_procs variables here. Use access() to find out whether a file is readable/writable, removed the manual voodoo and it's Win32 wrappers. Added an optional (can be NULL) "PlunInProcDef" parameter to file_save(), removed file_save_with_proc(). * app/gui/menus.c: Use the unused "gpointer data" parameter of the GtkItemFactory callbacks to pass a "Gimp" pointer to all of them. This reduces the usage of the global "the_gimp" hack to zero in app/gui/... yeah. * app/gui/channels-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palettes-commands.c * app/gui/select-commands.c * app/gui/test-commands.c * app/gui/tools-commands.c * app/gui/view-commands.c: use the passed "Gimp" pointer. * app/gui/color-area.[ch] * app/gui/convert-dialog.c * app/gui/dialogs-constructors.c * app/gui/file-new-dialog.[ch] * app/gui/file-open-dialog.[ch] * app/gui/file-save-dialog.[ch] * app/gui/gui.c * app/gui/info-window.[ch] * app/gui/module-browser.[ch] * app/gui/palette-editor.c * app/gui/palette-import-dialog.[ch] * app/gui/paths-dialog.c * app/gui/preferences-dialog.[ch] * app/gui/resize-dialog.[ch] * app/gui/tool-options-dialog.[ch] * app/gui/toolbox.c: pass around lots more "Gimp" and "GimpContext" pointers and don't use "the_gimp" any more. * app/tools/gimptool.h: added a pointer to the corresponding GimpToolInfo object (which in turn has a pointer to a Gimp). * app/tools/tool_manager.[ch]: set the pointer after creating the tool object. Removed tool_manager_get_info_by_tool() as there is a tool->tool_info pointer now. * app/tools/gimpbezierselecttool.c * app/tools/gimpblendtool.c * app/tools/gimpbrightnesscontrasttool.c * app/tools/gimpclonetool.c * app/tools/gimpcolorbalancetool.c * app/tools/gimpcolorpickertool.c * app/tools/gimpconvolvetool.c * app/tools/gimpcroptool.c * app/tools/gimpcurvestool.c * app/tools/gimpdodgeburntool.c * app/tools/gimpdrawtool.c * app/tools/gimpeditselectiontool.c * app/tools/gimpellipseselecttool.c * app/tools/gimperasertool.c * app/tools/gimpfliptool.c * app/tools/gimphistogramtool.c * app/tools/gimphuesaturationtool.c * app/tools/gimpinktool.c * app/tools/gimplevelstool.c * app/tools/gimpmagnifytool.c * app/tools/gimpmeasuretool.c * app/tools/gimpmovetool.c * app/tools/gimppainttool.c * app/tools/gimppathtool.c * app/tools/gimpperspectivetool.c * app/tools/gimpposterizetool.c * app/tools/gimprectselecttool.c * app/tools/gimprotatetool.c * app/tools/gimpscaletool.c * app/tools/gimpsheartool.c * app/tools/gimptexttool.c * app/tools/gimpthresholdtool.c * app/tools/path_tool.c * app/tools/xinput_airbrush.c: s/GDisplay/GimpDisplay/g. Use tool->tool_info and tool_info->gimp in some places to get rid of using "the_gimp". Removing the remaining ones involves changing the tool options system and is scheduled next... * app/widgets/gimpdnd.c * app/widgets/gimpdocumentview.c: pass a "Gimp" pointer to all file_open_*() functions. * app/gdisplay_color.[ch] * app/gdisplay_color_ui.[ch] * app/image_map.[ch] * app/nav_window.[ch] * app/path.c * app/path_bezier.c * app/path_transform.h * app/qmask.[ch]: s/GDisplay/GimpDisplay/g * tools/pdbgen/pdb/fileops.pdb: load_procs and save_procs are members of the "Gimp" object now. * tools/pdbgen/pdb/plug_in.pdb: use gimp->no_interface, don't include "appenv.h". * app/pdb/fileops_cmds.c * app/pdb/plug_in_cmds.c: regenerated.
2001-10-29 19:47:11 +08:00
gdisp = (GimpDisplay *) gdisp_ptr;
gdisplay_untransform_coords (gdisp, mevent->x, mevent->y, &x, &y, FALSE, FALSE);
if ((layer = gimage_get_active_layer (gdisp->gimage)))
{
int off_x, off_y;
drawable_offsets (GIMP_DRAWABLE(layer), &off_x, &off_y);
if (x >= off_x && y >= off_y &&
x < (off_x + drawable_width (GIMP_DRAWABLE(layer))) &&
y < (off_y + drawable_height (GIMP_DRAWABLE(layer))))
{
/* One more test--is there a selected region?
* if so, is cursor inside?
*/
if (gimage_mask_is_empty (gdisp->gimage))
ctype = GDK_PENCIL;
else if (gimage_mask_value (gdisp->gimage, x, y))
ctype = GDK_PENCIL;
}
}
gdisplay_install_tool_cursor (gdisp, ctype);
}
static void
xinput_airbrush_control (Tool *tool,
ToolAction action,
gpointer gdisp_ptr)
{
XinputAirbrushTool *xinput_airbrush_tool;
xinput_airbrush_tool = (XinputAirbrushTool *) tool->private;
switch (action)
{
case PAUSE:
draw_core_pause (xinput_airbrush_tool->core, tool);
break;
case RESUME:
draw_core_resume (xinput_airbrush_tool->core, tool);
break;
case HALT:
xinput_airbrush_cleanup ();
break;
default:
break;
}
}
static void
xinput_airbrush_init (XinputAirbrushTool *xinput_airbrush_tool, GimpDrawable *drawable,
double x, double y)
{
/* free the block structures */
if (undo_tiles)
tile_manager_destroy (undo_tiles);
if (canvas_tiles)
tile_manager_destroy (canvas_tiles);
/* Allocate the undo structure */
undo_tiles = tile_manager_new (drawable_width (drawable),
drawable_height (drawable),
drawable_bytes (drawable));
/* Allocate the canvas blocks structure */
canvas_tiles = tile_manager_new (drawable_width (drawable),
drawable_height (drawable), 1);
/* Get the initial undo extents */
xinput_airbrush_tool->x1 = xinput_airbrush_tool->x2 = x;
xinput_airbrush_tool->y1 = xinput_airbrush_tool->y2 = y;
}
static void
xinput_airbrush_finish (XinputAirbrushTool *xinput_airbrush_tool, GimpDrawable *drawable, int tool_id)
{
/* push an undo */
drawable_apply_image (drawable, xinput_airbrush_tool->x1, xinput_airbrush_tool->y1,
xinput_airbrush_tool->x2, xinput_airbrush_tool->y2, undo_tiles, TRUE);
undo_tiles = NULL;
/* invalidate the drawable--have to do it here, because
* it is not done during the actual painting.
*/
gimp_drawable_invalidate_preview (drawable, TRUE);
}
static void
xinput_airbrush_cleanup (void)
{
/* CLEANUP */
/* If the undo tiles exist, nuke them */
if (undo_tiles)
{
tile_manager_destroy (undo_tiles);
undo_tiles = NULL;
}
/* If the canvas blocks exist, nuke them */
if (canvas_tiles)
{
tile_manager_destroy (canvas_tiles);
canvas_tiles = NULL;
}
/* Free the temporary buffer if it exist */
if (canvas_buf)
temp_buf_free (canvas_buf);
canvas_buf = NULL;
}
/*********************************
* Rendering functions *
*********************************/
/* Some of this stuff should probably be combined with the
* code it was copied from in paint_core.c; but I wanted
* to learn this stuff, so I've kept it simple.
*/
static inline int
number_of_steps(int x0, int y0, int x1, int y1)
{
int dx, dy;
dx = abs(x0 - x1);
dy = abs(y0 - y1);
if (dy > dx)
{
return dy + 1;
}
else
{
return dx + 1;
}
}
static void
make_stroke(AirBlob *airblob,
XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
guint last_value,
guint present_value)
{
int steps;
int i, j, k;
int dx, dy;
int x, y;
int x0, x1, y0, y1;
double x0d, x1d, y0d, y1d;
int x_min, x_max, y_min, y_max, width, height;
int number, ystart, xstart;
guint ivalue;
double lv, pv, mv;
double dist;
int slopeterm;
AirBlob *last_airblob;
AirBlob *trans_blob;
AirLine *air_line;
MaskBuf *brush_mask;
MaskBuf **bufs;
guchar *source;
guchar *dest;
int *xpoints;
int *ypoints;
gboolean something;
something = FALSE;
x0 = xinput_airbrush_tool->lastx;
x1 = xinput_airbrush_tool->xcenter;
y0 = xinput_airbrush_tool->lasty;
y1 = xinput_airbrush_tool->ycenter;
x0d = xinput_airbrush_tool->lastx;
x1d = xinput_airbrush_tool->xcenter;
y0d = xinput_airbrush_tool->lasty;
y1d = xinput_airbrush_tool->ycenter;
last_airblob = xinput_airbrush_tool->last_airblob;
steps = number_of_steps(x0, y0, x1, y1);
/*
printf("Stoke Steps: %d\n", steps);
*/
ypoints = g_new(int, steps);
xpoints = g_new(int, steps);
bufs = g_new (MaskBuf*, steps);
dx = abs(x0 - x1);
dy = abs(y0 - y1);
lv = last_value;
pv = present_value;
mv = pv - lv;
/*
Yes I know this is bulky code :-),
--> see comment on make_mask.
But I wanted to keep it simple while I
implemented the tool :-). I will make it
more effective later on. And yes I could
look for the three special cases.
*/
/*
There is also a possiblity of a bug in the
scanline function that will explain the
"jumping lines".
*/
if (dy > dx)
{
/* Step in the y direction*/
if (y0 < y1)
{
/*
Step from y0 to y1
*/
if(x1 < x0)
{
/*
We have to x--
*/
x = x0;
y = y0;
number = 0;
j = 1;
slopeterm = dx;
while(y < y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x--;
y++;
}
else
{
slopeterm += dx;
y++;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
else
{
/*
We have to x++
*/
x = x0;
y = y0;
number = 0;
slopeterm = dx;
j = 1;
while(y < y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x++;
y++;
}
else
{
slopeterm += dx;
y++;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
}
else
{
/* Step from y0 to y1 with neg steps */
if(x1 < x0)
{
/*
We have to x--
*/
x = x0;
y = y0;
number = 0;
slopeterm = dx;
j = 1;
while(y > y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x--;
y--;
}
else
{
slopeterm += dx;
y--;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
else
{
/*
We have to x++
*/
x = x0;
y = y0;
number = 0;
slopeterm = dx;
j = 1;
while(y > y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x++;
y--;
}
else
{
slopeterm += dx;
y--;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
}
}
else
{
/* Step in the X direction*/
if (x0 < x1)
{
/* Step from x0 to x1 */
if(y1 < y0)
{
/*
We have to y--
*/
x = x0;
y = y0;
number = 0;
slopeterm = dy;
j = 1;
while(x < x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y--;
x++;
}
else
{
slopeterm += dy;
x++;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
else
{
/*
We have to y++
*/
x = x0;
y = y0;
number = 0;
slopeterm = dy;
j = 1;
while(x < x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y++;
x++;
}
else
{
slopeterm += dy;
x++;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
}
else
{
/* Step from x0 to x1 neg direction */
if(y1 < y0)
{
/*
We have to y--
*/
x = x0;
y = y0;
number = 0;
slopeterm = dy;
j = 1;
while(x > x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y--;
x--;
}
else
{
slopeterm += dy;
x--;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
else
{
/*
We have to y++
*/
x = x0;
y = y0;
number = 0;
slopeterm = dy;
j = 1;
while(x > x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y++;
x--;
}
else
{
slopeterm += dy;
x--;
}
j++;
dist = 1.0 - (double)j/(double)steps;
ivalue = lv + mv * dist;
trans_blob = trans_air_blob(last_airblob, airblob, dist, x, y);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], ivalue);
number++;
something = TRUE;
}
}
}
}
if(!something)
{
/*
printf("Nothing \n");
*/
number = 0;
dist = 1.0;
trans_blob = trans_air_blob(last_airblob, airblob, 0.5 , x1, y1);
air_line = create_air_line(trans_blob);
x_min = air_line->min_x;
y_min = air_line->min_y;
width = air_line->width;
height = air_line->height;
bufs[number] = mask_buf_new(width, height);
ypoints[number] = y_min;
xpoints[number] = x_min;
make_mask(air_line, bufs[number], pv);
/*
print_mask(bufs[number]);
*/
brush_mask = mask_buf_new(width, height);
dest = brush_mask->data;
source = bufs[number]->data;
for(k = 0; k < bufs[number]->height; k++)
{
for(j = 0; j < bufs[number]->width; j++ )
{
if(((int)dest[k * width + j] + (int)source[k * bufs[number]->width + j]) > 255)
{
dest[k * width + j] = 255;
}
else
{
dest[k * width + j] += source[k * bufs[number]->width + j];
}
}
}
xinput_airbrush_paste (xinput_airbrush_tool, drawable, brush_mask, x_min, y_min, width, height);
/*
print_mask(brush_mask);
*/
mask_buf_free(brush_mask);
mask_buf_free(bufs[number]);
g_free(bufs);
/*
printf("Nothing again\n");
*/
}
if(something)
{
y_min = y_max = ypoints[0];
height = bufs[0]->height;
x_min = x_max= xpoints[0];
width = bufs[0]->width;
/*printf("xmin %d, ymin %d, width %d, weight %d\n", x_min, y_min, width, height);*/
for (i = 1; i < steps -1; i++)
{
/*printf("xmin %d, ymin %d, width %d, weight %d\n", xpoints[i], ypoints[i], bufs[i]->width, bufs[i]->height);*/
x_min = MIN(xpoints[i], x_min);
y_min = MIN(ypoints[i], y_min);
if ((x_max + width) < (xpoints[i] + bufs[i]->width))
{
x_max = xpoints[i];
width = bufs[i]->width;
}
if ((y_max + height) < (ypoints[i] + bufs[i]->height))
{
y_max = ypoints[i];
height = bufs[i]->height;
}
}
width = x_max - x_min + width;
height = y_max - y_min + height;
/*printf("Xmin %d, Ymin %d, Width %d, Height %d\n", x_min, y_min, width, height);
printf("\n");*/
brush_mask = mask_buf_new(width, height);
dest = brush_mask->data;
for (i = 0; i < steps - 1; i++)
{
ystart = ypoints[i] - y_min;
xstart = xpoints[i] - x_min;
/*printf("xstart %d, ystart %d\n", xstart, ystart);*/
y = width * ystart;
source = bufs[i]->data;
for(k = 0; k < bufs[i]->height; k++)
{
for(j = 0; j < bufs[i]->width; j++ )
{
if(((int)dest[y + k * width + xstart + j] + (int)source[k * bufs[i]->width + j]) > 255)
{
dest[y + k * width + xstart + j] = 255;
}
else
{
dest[y + k * width + xstart + j] += source[k * bufs[i]->width + j];
}
}
}
}
/*printf("\n\n");*/
for (i = steps - 2; i >= 0; i--)
{
mask_buf_free(bufs[i]);
}
g_free(bufs);
xinput_airbrush_paste (xinput_airbrush_tool, drawable, brush_mask, x_min, y_min, width, height);
mask_buf_free(brush_mask);
}
else
{
/*
printf("Hmm something was FALSE\n");
*/
}
}
/**************************************************
***************************************************
*/
static void
make_mask (AirLine *airline,
MaskBuf *dest,
guint value)
{
int steps;
int total_steps;
int i;
int j;
int dx, dy;
int x, y;
int x0, x1, y0, y1;
unsigned char *s;
guint midvalue;
double ivalue;
int rowlength;
int slopeterm;
gboolean nothing;
rowlength = dest->width * dest->bytes;
nothing = TRUE;
total_steps=0;
for (i=0; i < airline->nlines ; i++)
{
steps = number_of_steps(airline->xcenter, airline->ycenter,
airline->line[i].x, airline->line[i].y);
total_steps += steps;
}
/*
printf("Total Steps: %d\n", total_steps);
*/
for (i=0; i < airline->nlines ; i++)
{
steps = number_of_steps(airline->xcenter, airline->ycenter,
airline->line[i].x, airline->line[i].y);
x0 = airline->xcenter - airline->min_x ;
x1 = airline->line[i].x - airline->min_x ;
y0 = airline->ycenter - airline->min_y ;
y1 = airline->line[i].y -airline->min_y ;
dx = abs(x0 - x1);
dy = abs(y0 - y1);
/*
Yes I know this is bulky code :-),
you could insted set the x direction
to 1 or -1, and y to - or + rowside.
Then set the first X and Y and of you
go.
But I wanted to keep it simple while I
implemented the tool :-). I will make it
more effective later on. And yes I could
look for the three special cases.
*/
midvalue = value;
if (dy > dx)
{
/* Step in the y direction*/
if (y0 < y1)
{
/*
Step from y0 to y1
*/
if(x1 < x0)
{
/*
We have to x--
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dx;
j = 1;
while(y < y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x--;
y++;
}
else
{
slopeterm += dx;
y++;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
else
{
/*
We have to x++
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dx;
j = 1;
while(y < y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x++;
y++;
}
else
{
slopeterm += dx;
y++;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
}
else
{
/* Step from y0 to y1 with neg steps */
if(x1 < x0)
{
/*
We have to x--
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dx;
j = 1;
while(y > y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x--;
y--;
}
else
{
slopeterm += dx;
y--;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
else
{
/*
We have to x++
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dx;
j = 1;
while(y > y1)
{
if (slopeterm > dy)
{
slopeterm -= dy;
x++;
y--;
}
else
{
slopeterm += dx;
y--;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
}
}
else
{
/* Step in the X direction*/
if (x0 < x1)
{
/* Step from x0 to x1 */
if(y1 < y0)
{
/*
We have to y--
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dy;
j = 1;
while(x < x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y--;
x++;
}
else
{
slopeterm += dy;
x++;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
else
{
/*
We have to y++
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dy;
j = 1;
while(x < x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y++;
x++;
}
else
{
slopeterm += dy;
x++;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
}
else
{
/* Step from x0 to x1 neg direction */
if(y1 < y0)
{
/*
We have to y--
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dy;
j = 1;
while(x > x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y--;
x--;
}
else
{
slopeterm += dy;
x--;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
else
{
/*
We have to y++
*/
s = dest->data;
x = x0;
y = y0;
s += y * rowlength + x;
if (((int)*s + (int)midvalue) > 255)
{
*s = 255;
}
else
{
*s = midvalue + *s;
}
slopeterm = dy;
j = 1;
while(x > x1)
{
if (slopeterm > dx)
{
slopeterm -= dx;
y++;
x--;
}
else
{
slopeterm += dy;
x--;
}
j++;
s = dest->data;
s += y * rowlength + x;
ivalue = value * (1.0 - (double)j/(double)steps);
if(((int)*s + ivalue) > 255)
{
*s = 255;
}
else
{
*s = *s + ivalue;
}
nothing = FALSE;
}
}
}
}
}
if (nothing)
{
printf("Hmm I retured a nothing brush\n");
}
}
static void
xinput_airbrush_set_paint_area (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
int x, int y, int width, int height)
{
int iwidth, iheight;
int x1, y1, x2, y2;
int bytes;
int dwidth, dheight;
bytes = drawable_has_alpha (drawable) ?
drawable_bytes (drawable) : drawable_bytes (drawable) + 1;
dwidth = drawable_width (drawable);
dheight = drawable_height (drawable);
x1 = CLAMP (x, 0, dwidth);
y1 = CLAMP (y, 0, dheight);
x2 = CLAMP ((x + width), 0, dwidth);
y2 = CLAMP ((y + height), 0, dheight);
iwidth = MIN((x2 - x1), width);
iheight = MIN((y2 - y1), height);
/* configure the canvas buffer */
if ((x2 - x1) && (y2 - y1))
canvas_buf = temp_buf_resize (canvas_buf, bytes, x1, y1,
iwidth, iheight);
}
static void
render_airbrush_line (AirBrushBlob *airbrush_blob, guchar *dest,
int y, int width, XinputAirbrushTool *xinput_airbrush_tool)
{
int i, j, k, l, m, g;
int left, right;
int brush_width;
double x_dest;
double xdist, ydist, dist, angle;
double i_value;
guchar value;
if (airbrush_blob->height <= 6)
{
return;
}
left = 0;
right = 0;
j = y * SUBSAMPLE;
g = y * SUBSAMPLE;
for (i=0; i<SUBSAMPLE; i++, j++)
{
if (j >= airbrush_blob->height)
{
if ( i == 0)
{
return;
}
else
{
left = airbrush_blob->data[g].left - airbrush_blob->min_x;
right = airbrush_blob->data[g].right - airbrush_blob->min_x;
break;
}
}
if ( i == SUBSAMPLE/2)
{
left = airbrush_blob->data[j].left - airbrush_blob->min_x;
right = airbrush_blob->data[j].right - airbrush_blob->min_x;
break;
}
}
brush_width = right - left;
if (brush_width <= SUBSAMPLE*2)
{
return;
}
dest = dest + (left/SUBSAMPLE);
for (i=4; i <= brush_width; i = i + SUBSAMPLE)
{
x_dest = (left + i)/SUBSAMPLE;
xdist = (airbrush_blob->min_x/SUBSAMPLE) + x_dest - xinput_airbrush_tool->xcenter;
ydist = xinput_airbrush_tool->ycenter - (airbrush_blob->y/SUBSAMPLE) - y;
angle = atan2(ydist, xdist) + G_PI;
dist = hypot(xdist, ydist);
for (j=1; j< (airbrush_blob->height - 1) ; j++)
{
k = j - 1;
l = j + 1;
m = 0;
if ( (fabs(airbrush_blob->data[k].angle_right_abs - airbrush_blob->data[l].angle_right_abs) >
fabs(airbrush_blob->data[k].angle_right_abs - angle)) &&
(fabs(airbrush_blob->data[k].angle_right_abs - airbrush_blob->data[l].angle_right_abs) >
fabs(airbrush_blob->data[l].angle_right_abs - angle))
)
{
i_value = MIN(1., (dist*SUBSAMPLE)/(airbrush_blob->data[j].dist_right));
value = 255 * (1 - i_value);
*dest = value;
dest++;
m = 1;
break;
}
else if ( (fabs(airbrush_blob->data[k].angle_left_abs - airbrush_blob->data[l].angle_left_abs) >
fabs(airbrush_blob->data[k].angle_left_abs - angle)) &&
(fabs(airbrush_blob->data[k].angle_left_abs - airbrush_blob->data[l].angle_left_abs) >
fabs(airbrush_blob->data[l].angle_left_abs - angle))
)
{
i_value = MIN(1., (dist*SUBSAMPLE)/(airbrush_blob->data[j].dist_left));
value = 255 * (1 - i_value);
*dest = value;
dest++;
m = 1;
break;
}
else if ((j == (airbrush_blob->height - 2)) && ( m == 0))
{
*dest = 0;
dest++;
}
}
}
}
static void
calc_width (AirBrushBlob *airbrush_blob)
{
int i;
int min_x, max_x;
i = (airbrush_blob->height)/2;
min_x = airbrush_blob->data[i].left;
max_x = airbrush_blob->data[i].right;
for (i=0; i< (airbrush_blob->height) ; i++)
{
if ( airbrush_blob->data[i].left <= airbrush_blob->data[i].right)
{
min_x = MIN(airbrush_blob->data[i].left, min_x);
}
if ( airbrush_blob->data[i].left <= airbrush_blob->data[i].right)
{
max_x = MAX(airbrush_blob->data[i].right, max_x );
}
}
airbrush_blob->width = max_x - min_x;
airbrush_blob->min_x = min_x;
airbrush_blob->max_x = max_x;
}
static void
calc_angle (AirBrushBlob *airbrush_blob, double xcenter, double ycenter)
{
int i;
double y_center, x_center;
double y_dist, x_dist;
double left_ang, right_ang;
double left_ang_abs, right_ang_abs;
int min_x, max_x;
x_center = xcenter * SUBSAMPLE;
y_center = ycenter * SUBSAMPLE - airbrush_blob->y;
i = (airbrush_blob->height)/2;
min_x = airbrush_blob->data[i].left;
max_x = airbrush_blob->data[i].right;
for (i=0; i< (airbrush_blob->height) ; i++)
{
y_dist = y_center - i;
x_dist = airbrush_blob->data[i].left - x_center;
left_ang_abs = atan2(y_dist, x_dist) + G_PI;
left_ang = atan(y_dist/x_dist);
airbrush_blob->data[i].angle_left = left_ang;
airbrush_blob->data[i].angle_left_abs = left_ang_abs;
airbrush_blob->data[i].dist_left = hypot(x_dist, y_dist);
if ( airbrush_blob->data[i].left <= airbrush_blob->data[i].right)
{
min_x = MIN(airbrush_blob->data[i].left, min_x);
}
x_dist = airbrush_blob->data[i].right - x_center;
right_ang_abs = atan2(y_dist, x_dist) + G_PI;
right_ang = atan(y_dist/x_dist);
airbrush_blob->data[i].angle_right = right_ang;
airbrush_blob->data[i].angle_right_abs = right_ang_abs;
airbrush_blob->data[i].dist_right = hypot(x_dist, y_dist);
if ( airbrush_blob->data[i].left <= airbrush_blob->data[i].right)
{
max_x = MAX(airbrush_blob->data[i].right, max_x );
}
}
airbrush_blob->width = max_x - min_x;
airbrush_blob->min_x = min_x;
airbrush_blob->max_x = max_x;
}
static void
make_single_mask(AirBrushBlob *airbrush_blob, XinputAirbrushTool *xinput_airbrush_tool, MaskBuf *dest)
{
unsigned char * s;
int h;
int i;
h = dest->height;
s = dest->data;
for (i=0; i<h; i++)
{
render_airbrush_line(airbrush_blob, s,
i, dest->width, xinput_airbrush_tool);
s += (dest->width * dest->bytes);
}
}
static void
print_line (guchar *dest,
int width)
{
int i;
for (i=0; i < width; i++)
{
printf("%d ", *dest);
dest++;
}
}
static void
print_mask(MaskBuf *dest)
{
unsigned char * s;
int h;
int i;
h = dest->height;
s = dest->data;
printf("\nBrush_Mask\n\n");
for (i=0; i<h; i++)
{
print_line( s,
dest->width);
s += (dest->width * dest->bytes);
printf("\n");
}
printf("\n");
}
static inline MaskBuf *
blur_mask_1(MaskBuf *mask)
{
MaskBuf *retur;
int y, x;
unsigned char *source;
unsigned char *dest;
int height, width;
retur = mask_buf_new(mask->width, mask->height);
source = mask->data;
dest = retur->data;
width = mask->width;
height = mask->height;
for (y = 1; y < (height -1); y++)
{
for(x = 1; x < (width -1); x++)
{
dest[(y * retur->width + x)] = (
2 * source[((y-1) * width) + x - 1] +
2 * source[((y-1) * width) + x] +
2 * source[((y-1) * width) + x + 1] +
2 * source[(y * width) + x - 1] +
source[(y * width) + x] +
2 * source[(y * width) + x + 1] +
2 * source[((y+1) * width) + x - 1] +
2 * source[((y+1) * width) + x] +
source[((y+1) * width) + x + 1])/16;
}
}
mask_buf_free(mask);
return retur;
}
static inline MaskBuf *
blur_mask_2(MaskBuf *mask)
{
MaskBuf *retur;
int y, x;
unsigned char *source;
unsigned char *dest;
int height, width;
retur = mask_buf_new(mask->width, mask->height);
source = mask->data;
dest = retur->data;
width = mask->width;
height = mask->height;
for (y = 2; y < (height -2); y++)
{
for(x = 2; x < (width -2); x++)
{
dest[(y * retur->width + x)] = (
2 * source[((y-2) * width) + x - 2] +
2 * source[((y-2) * width) + x - 1] +
2 * source[((y-2) * width) + x] +
2 * source[((y-2) * width) + x + 1] +
2 * source[((y-2) * width) + x + 2] +
2 * source[((y-1) * width) + x - 2] +
4 * source[((y-1) * width) + x - 1] +
4 * source[((y-1) * width) + x] +
4 * source[((y-1) * width) + x + 1] +
2 * source[((y-1) * width) + x + 2] +
2 * source[(y * width) + x - 2] +
4 * source[(y * width) + x - 1] +
1 * source[(y * width) + x] +
4 * source[(y * width) + x + 1] +
1 * source[(y * width) + x + 2] +
2 * source[((y+1) * width) + x - 2] +
4 * source[((y+1) * width) + x - 1] +
4 * source[((y+1) * width) + x] +
4 * source[((y+1) * width) + x + 1] +
2 * source[((y+1) * width) + x + 2] +
2 * source[((y+2) * width) + x - 2] +
2 * source[((y+2) * width) + x - 1] +
2 * source[((y+2) * width) + x] +
2 * source[((y+2) * width) + x + 1] +
2 * source[((y+2) * width) + x + 2]
)/64;
}
}
mask_buf_free(mask);
return retur;
}
static inline MaskBuf *
blur_mask_3(MaskBuf *mask)
{
MaskBuf *retur;
int y, x, y_3, y_2, y_1, y1, y2, y3, x_3, x_2,x_1, x1, x2, x3;
unsigned char *source;
unsigned char *dest;
int height, width, max_height, max_width;
retur = mask_buf_new(mask->width, mask->height);
source = mask->data;
dest = retur->data;
width = mask->width;
height = mask->height;
max_height = width * height - 3 * width;
max_width = width - 3;
y_3 = 0;
y_2 = width;
y_1 = 2 * width;
y = 3 * width;
y1 = 4 * width;
y2 = 5 * width;
y3 = 6 * width;
for (; y < max_height; y_3 += width, y_2 += width, y_1 += width, y += width, y1 += width, y2 += width, y3 += width)
{
for( x_3 = 0, x_2 = 1, x_1 = 2, x = 3, x1 = 4, x2 = 5 , x3 = 6; x < max_width; x_3++, x_2++, x_1++, x++, x1++, x2++, x3++)
{
dest[y + x] = (
1 * source[y_3 + x_3] +
1 * source[y_3 + x_2] +
1 * source[y_3 + x_1] +
1 * source[y_3 + x] +
1 * source[y_3 + x1] +
1 * source[y_3 + x2] +
1 * source[y_3 + x3] +
2 * source[y_2 + x_3] +
2 * source[y_2 + x_2] +
2 * source[y_2 + x_1] +
2 * source[y_2 + x] +
2 * source[y_2 + x1] +
2 * source[y_2 + x2] +
2 * source[y_2 + x3] +
2 * source[y_1 + x_3] +
2 * source[y_1 + x_2] +
8 * source[y_1 + x_1] +
8 * source[y_1 + x] +
8 * source[y_1 + x1] +
2 * source[y_1 + x2] +
2 * source[y_1 + x3] +
1 * source[y + x_3] +
2 * source[y + x_2] +
8 * source[y + x_1] +
1 * source[y + x] +
8 * source[y + x1] +
2 * source[y + x2] +
1 * source[y + x3] +
2 * source[y1 + x_3] +
2 * source[y1 + x_2] +
8 * source[y1 + x_1] +
8 * source[y1 + x] +
8 * source[y1 + x1] +
2 * source[y1 + x2] +
2 * source[y1 + x3] +
2 * source[y2 + x_3] +
2 * source[y2 + x_2] +
2 * source[y2 + x_1] +
1 * source[y2 + x] +
2 * source[y2 + x1] +
2 * source[y2 + x2] +
2 * source[y2 + x3] +
1 * source[y3 + x_3] +
1 * source[y3 + x_2] +
1 * source[y3 + x_1] +
1 * source[y3 + x] +
1 * source[y3 + x1] +
1 * source[y3 + x2] +
1 * source[y3 + x3]
)/128;
}
}
mask_buf_free(mask);
return retur;
}
MaskBuf *
blur_mask_4(MaskBuf *mask)
{
MaskBuf *retur;
int y, x, x_4, x_3, x_2,x_1, x1, x2, x3, x4,x5;
unsigned char *source;
unsigned char *dest;
int height, width, max_height, max_width;
retur = mask_buf_new(mask->width, mask->height);
source = mask->data;
dest = retur->data;
width = mask->width;
height = mask->height;
max_height = width * height - 3 * width;
max_width = width - 5;
y = width;
for (y=0 ; y < max_height; y += width)
{
for( x_4 = 0, x_3 = 1, x_2 = 2, x_1 = 3, x = 4, x1 = 5, x2 = 6 , x3 = 7, x4 = 8, x5 = 9; x < max_width; x_3++, x_2++, x_1++, x++, x1++, x2++, x3++)
{
dest[y + x] = (
4 * source[y + x_4] +
4 * source[y + x_3] +
8 * source[y + x_2] +
16 * source[y + x_1] +
1 * source[y + x] +
16 * source[y + x1] +
8 * source[y + x2] +
4 * source[y + x3] +
2 * source[y + x4] +
1 * source[y + x5]
)/64;
}
}
mask_buf_free(mask);
return retur;
}
static void
xinput_airbrush_paste (XinputAirbrushTool *xinput_airbrush_tool,
GimpDrawable *drawable,
MaskBuf *brush_mask_in,
int x,
int y,
int width,
int height)
{
GImage *gimage;
PixelRegion srcPR;
int offx, offy;
int size;
unsigned char col[MAX_CHANNELS];
MaskBuf *brush_mask1;
MaskBuf *brush_mask3;
MaskBuf *brush_mask2;
MaskBuf *brush_mask;
int i;
if (! (gimage = drawable_gimage (drawable)))
return;
size = height * width;
brush_mask1 = mask_buf_new(brush_mask_in->width, brush_mask_in->height);
brush_mask1 = temp_buf_copy(brush_mask_in, brush_mask1);
/* printf("Size: %d\n", size); */
if (size <= 40)
{
brush_mask1 = blur_mask_1(brush_mask1);
}
else if ((size >= 40) && (size <= 90))
{
for(i = 0 ; i < 2; i++)
{
brush_mask2 = blur_mask_1(brush_mask1);
brush_mask3 = blur_mask_1(brush_mask2);
brush_mask1 = blur_mask_1(brush_mask3);
}
}
else if ((size >= 91) && (size <= 1000))
{
for(i = 0 ; i < 2; i++)
{
brush_mask2 = blur_mask_3(brush_mask1);
brush_mask3 = blur_mask_3(brush_mask2);
brush_mask1 = blur_mask_3(brush_mask3);
}
}
else if ((size >= 1001) && (size <= 5000))
{
for(i = 0 ; i < 4; i++)
{
brush_mask2 = blur_mask_4(brush_mask1);
brush_mask3 = blur_mask_4(brush_mask2);
brush_mask1 = blur_mask_3(brush_mask3);
}
}
else if (size >= 5001)
{
for(i = 0 ; i < 5; i++)
{
brush_mask2 = blur_mask_3(brush_mask1);
brush_mask3 = blur_mask_4(brush_mask2);
brush_mask1 = blur_mask_3(brush_mask3);
}
}
brush_mask = brush_mask1;
/* Get the the buffer */
xinput_airbrush_set_paint_area (xinput_airbrush_tool, drawable, x - 1, y - 1, width, height);
/* check to make sure there is actually a canvas to draw on */
if (!canvas_buf)
return;
gimage_get_foreground (gimage, drawable, col);
/* set the alpha channel */
col[canvas_buf->bytes - 1] = OPAQUE_OPACITY;
/* color the pixels */
color_pixels (temp_buf_data (canvas_buf), col,
canvas_buf->width * canvas_buf->height, canvas_buf->bytes);
/* set undo blocks */
xinput_airbrush_set_undo_tiles (drawable,
canvas_buf->x, canvas_buf->y,
canvas_buf->width, canvas_buf->height);
/* initialize any invalid canvas tiles */
xinput_airbrush_set_canvas_tiles (canvas_buf->x, canvas_buf->y,
canvas_buf->width, canvas_buf->height);
/*DON'T FORGETT THE 100 VALUE, I.E. THE BRUSH OPACITY WHICH SHOULD BE 255*/
xinput_airbrush_to_canvas_tiles (xinput_airbrush_tool, brush_mask, 255);
xinput_airbrush_canvas_tiles_to_canvas_buf(xinput_airbrush_tool);
/* initialize canvas buf source pixel regions */
srcPR.bytes = canvas_buf->bytes;
srcPR.x = 0;
srcPR.y = 0;
srcPR.w = canvas_buf->width;
srcPR.h = canvas_buf->height;
srcPR.rowstride = canvas_buf->width * canvas_buf->bytes;
srcPR.data = temp_buf_data (canvas_buf);
/* apply the paint area to the gimage */
gimage_apply_image (gimage, drawable, &srcPR,
FALSE,
(int) (gimp_context_get_opacity (NULL) * 255),
gimp_context_get_paint_mode (NULL),
undo_tiles, /* specify an alternative src1 */
canvas_buf->x, canvas_buf->y);
/* Update the undo extents */
xinput_airbrush_tool->x1 = MIN (xinput_airbrush_tool->x1, canvas_buf->x);
xinput_airbrush_tool->y1 = MIN (xinput_airbrush_tool->y1, canvas_buf->y);
xinput_airbrush_tool->x2 = MAX (xinput_airbrush_tool->x2, (canvas_buf->x + canvas_buf->width));
xinput_airbrush_tool->y2 = MAX (xinput_airbrush_tool->y2, (canvas_buf->y + canvas_buf->height));
2001-11-01 05:20:09 +08:00
/* Update the gimage--it is important to call gimp_image_update
* instead of drawable_update because we don't want the drawable
* preview to be constantly invalidated
*/
drawable_offsets (drawable, &offx, &offy);
2001-11-01 05:20:09 +08:00
gimp_image_update (gimage,
canvas_buf->x + offx,
canvas_buf->y + offy,
canvas_buf->width,
canvas_buf->height);
mask_buf_free(brush_mask);
}
static void
xinput_airbrush_to_canvas_tiles (XinputAirbrushTool *xinput_airbrush_tool,
MaskBuf *brush_mask,
int brush_opacity)
{
PixelRegion srcPR, maskPR;
/* combine the brush mask and the canvas tiles */
pixel_region_init (&srcPR, canvas_tiles,
canvas_buf->x, canvas_buf->y,
canvas_buf->width, canvas_buf->height, TRUE);
maskPR.bytes = 1;
maskPR.x = 0;
maskPR.y = 0;
maskPR.w = srcPR.w;
maskPR.h = srcPR.h;
maskPR.rowstride = maskPR.bytes * brush_mask->width;
maskPR.data = mask_buf_data (brush_mask);
/* combine the mask to the canvas tiles */
combine_mask_and_region (&srcPR, &maskPR, brush_opacity);
}
static void
xinput_airbrush_canvas_tiles_to_canvas_buf(XinputAirbrushTool *xinput_airbrush_tool)
{
PixelRegion srcPR, maskPR;
/* combine the canvas tiles and the canvas buf */
srcPR.bytes = canvas_buf->bytes;
srcPR.x = 0;
srcPR.y = 0;
srcPR.w = canvas_buf->width;
srcPR.h = canvas_buf->height;
srcPR.rowstride = canvas_buf->width * canvas_buf->bytes;
srcPR.data = temp_buf_data (canvas_buf);
pixel_region_init (&maskPR, canvas_tiles,
canvas_buf->x, canvas_buf->y,
canvas_buf->width, canvas_buf->height, FALSE);
/* apply the canvas tiles to the canvas buf */
apply_mask_to_region (&srcPR, &maskPR, OPAQUE_OPACITY);
}
static void
xinput_airbrush_set_undo_tiles (GimpDrawable *drawable,
int x, int y, int w, int h)
{
int i, j;
Tile *src_tile;
Tile *dest_tile;
for (i = y; i < (y + h); i += (TILE_HEIGHT - (i % TILE_HEIGHT)))
{
for (j = x; j < (x + w); j += (TILE_WIDTH - (j % TILE_WIDTH)))
{
dest_tile = tile_manager_get_tile (undo_tiles, j, i, FALSE, FALSE);
if (tile_is_valid (dest_tile) == FALSE)
{
src_tile = tile_manager_get_tile (drawable_data (drawable), j, i, TRUE, FALSE);
tile_manager_map_tile (undo_tiles, j, i, src_tile);
tile_release (src_tile, FALSE);
}
}
}
}
static void
xinput_airbrush_set_canvas_tiles (int x, int y, int w, int h)
{
int i, j;
Tile *tile;
for (i = y; i < (y + h); i += (TILE_HEIGHT - (i % TILE_HEIGHT)))
{
for (j = x; j < (x + w); j += (TILE_WIDTH - (j % TILE_WIDTH)))
{
tile = tile_manager_get_tile (canvas_tiles, j, i, FALSE, FALSE);
if (tile_is_valid (tile) == FALSE)
{
tile = tile_manager_get_tile (canvas_tiles, j, i, TRUE, TRUE);
memset (tile_data_pointer (tile, 0, 0),
0,
tile_size (tile));
tile_release (tile, TRUE);
}
}
}
}
/**************************/
/* Global xinput_airbrush functions */
/**************************/
void
xinput_airbrush_no_draw (Tool *tool)
{
return;
}
Tool *
tools_new_xinput_airbrush (void)
{
Tool * tool;
XinputAirbrushTool * private;
/* The tool options */
if (! xinput_airbrush_options)
{
xinput_airbrush_options = xinput_airbrush_options_new ();
tools_register (XINPUT_AIRBRUSH, (ToolOptions *) xinput_airbrush_options);
/* press all default buttons */
xinput_airbrush_options_reset ();
}
tool = tools_new_tool (XINPUT_AIRBRUSH);
allocate the tool's paint_core with g_new0. This prevents us from having 2000-06-22 Matt Wilson <msw@redhat.com> * app/paint_core.c (paint_core_new): allocate the tool's paint_core with g_new0. This prevents us from having cruft in unused tools. Systems with sensitive FPUs (Alpha) will raise exception in the paint_core_cursor_update if paint_core->last{x,y} are messy. * app/bezier_select.c (tools_new_bezier_select) * app/blend.c (blend_options_new) * app/brightness_contrast.c (tools_new_brightness_contrast) * app/bucket_fill.c (tools_new_bucket_fill) * app/by_color_select.c (tools_new_by_color_select) * app/color_balance.c (tools_new_color_balance) * app/color_panel.c (color_panel_new) * app/color_picker.c (tools_new_color_picker) * app/crop.c (tools_new_crop) * app/curves.c (tools_new_curves) * app/ellipse_select.c (tools_new_ellipse_select) * app/free_select.c (tools_new_free_select) * app/fuzzy_select.c (tools_new_fuzzy_select) * app/histogram_tool.c (tools_new_histogram_tool) * app/hue_saturation.c (tools_new_hue_saturation) * app/ink.c (tools_new_ink) * app/iscissors.c (tools_new_iscissors) * app/levels.c (tools_new_levels) * app/magnify.c (tools_new_magnify) * app/measure.c (tools_new_measure_tool) * app/move.c (tools_new_move_tool) * app/path_tool.c (tools_new_path_tool) * app/posterize.c (tools_new_posterize) * app/rect_select.c (tools_new_rect_select) * app/resize.c (resize_widget_new) * app/threshold.c (tools_new_threshold) * app/transform_core.c (transform_core_new) * app/xinput_airbrush.c (tools_new_xinput_airbrush): likewise (it can only help and it really isn't slow.) * app/color_area.c: #include <string.h> for memcpy declaration * app/gimphelp.c: #include <string.h> for strlen declaration
2000-06-23 08:14:07 +08:00
private = g_new0 (XinputAirbrushTool, 1);
private->core = draw_core_new (xinput_airbrush_no_draw);
private->last_airbrush_blob = NULL;
tool->private = private;
tool->button_press_func = xinput_airbrush_button_press;
tool->button_release_func = xinput_airbrush_button_release;
tool->motion_func = xinput_airbrush_motion;
tool->cursor_update_func = xinput_airbrush_cursor_update;
tool->control_func = xinput_airbrush_control;
return tool;
}
void
tools_free_xinput_airbrush(Tool *tool)
{
XinputAirbrushTool * xinput_airbrush_tool;
xinput_airbrush_tool = (XinputAirbrushTool *) tool->private;
/* Make sure the selection core is not visible */
if (tool->state == ACTIVE && xinput_airbrush_tool->core)
draw_core_stop (xinput_airbrush_tool->core, tool);
/* Free the selection core */
if (xinput_airbrush_tool->core)
draw_core_free (xinput_airbrush_tool->core);
/* Free the last airbrush_blob, if any */
if (xinput_airbrush_tool->last_airbrush_blob)
g_free (xinput_airbrush_tool->last_airbrush_blob);
/* Cleanup memory */
xinput_airbrush_cleanup ();
/* Free the paint core */
g_free (xinput_airbrush_tool);
}