gimp/app/text_tool_cmds.c

793 lines
18 KiB
C
Raw Normal View History

/* The GIMP -- an image manipulation program
* Copyright (C) 1995-1999 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.
*/
/* NOTE: This file is autogenerated by pdbgen.pl. */
#include "procedural_db.h"
gave parasite undo a MISC_UNDO class for now so it compiles * app/gimpdrawable.c: gave parasite undo a MISC_UNDO class for now so it compiles * app/tools_cmds.c: fix crop invoker to give correct args to crop_image * app/color_cmds.c: s/GRAY/GRAY_LUT/g; * app/brush_select.[ch]: removed PDB procs, export brush_active_dialogs, brush_select_dialog, s/active_dialogs/brush_active_dialogs/ * app/gimage_cmds.[ch] * app/channel_ops.[ch]: removed channel ops PDB procs, moved duplicate function from gimage_cmds to channel_ops, export offset and duplicate * app/gimpbrushlist.[ch]: removed PDB procs * app/gradient.[ch]: removed PDB procs, * app/gradient_header.h: exported G_SAMPLE, GradSelect, num_gradients, grad_active_dialogs, gradient_select_dialog * app/gradient_select.c: removed PDB procs, s/active_dialogs/grad_active_dialogs/ * app/patterns.[ch]: removed PDB procs * app/pattern_select.[ch]: removed PDB procs, s/active_dialogs/pattern_active_dialogs/ * app/procedural_db.c: removed PDB procs and supporting functions * app/procedrual_db.h: fiddled with enums * app/channel_cmds.[ch] * app/drawable_cmds.[ch] * app/parasite_cmds.[ch]: pdbgenned now, removed header files * app/gimpparasite.c: minor cleanup * app/internal_procs.c: use pdbgen stuff * app/tools_cmds.c * app/text_tool_cmds.c: updated from pdbgen * app/brushes_cmds.c * app/brush_select_cmds.c * app/gradient_cmds.c * app/gradient_select_cmds.c * app/patterns_cmds.c * app/pattern_select_cmds.c * app/procedural_db_cmds.c: new pdbgen files * app/Makefile.am: file shuffle (see above) -Yosh
1999-04-24 04:54:02 +08:00
#include <stdio.h>
Win32 portability changes: * config.h.win32, README.win32: Small changes. * tools/pdbgen/pdb/*.pdb: Include <string.h>. * app/*_cmds.c: Autogenerated files reflect above changes. * libgimp/makefile.msc app/makefile.msc: Various updates, including new object files. Gtk+ directory now should be called gtk+ (not gtk-plus). Use win32-specific gdk subdir. Glib directory now should be called just glib. * libgimp/gimp.def: Updates. * libgimp/gimpfeatures.h.win32: Made current with gimpfeatures.h.in. * libgimp/gimpfileselection.c: Define S_ISDIR and S_ISREG if necessary. * tools/pdbgen/pdb/fileops.pdb: Must have a statement (even an empty one) after a label. * app/fileops_cmds.c: Autogenerated file reflects above changes. * app/crop.c: Include <string.h>. * app/{app_procs,batch,fileops,datafiles,errorconsole,general, plug_in,temp_buf,tile_swap}.c: Test NATIVE_WIN32, not _MSC_VER. (NATIVE_WIN32 means we are using the Microsoft C runtime, even if we might be compiling with gcc.) * app/fileops.c: Don't include <process.h> here. * app/fileops.h: Do include <process.h> here. * app/gimpparasite.c: Include config.h, guard inclusion of <unistd.h>. (Is the inclusion of unistd.h in source files all over the place really necessary?) * app/ink.c: MSC doesn't handle conversion from unsigned __int64 to double, so cast to signed. * app/lut_funcs.c: Include config.h, and define rint() if necessary. * app/pixel_processor.c: Include config.h without "..", like in all the other places. Include <string.h> * app/text_tool.c: Guard the "POINTS" identifier that clashes with <windows.h>, sigh.
1999-05-05 05:32:17 +08:00
#include <string.h>
#include "appenv.h"
#include "drawable.h"
#include "layer.h"
#include "text_tool.h"
#include "libgimp/gimplimits.h"
static ProcRecord text_fontname_proc;
static ProcRecord text_get_extents_fontname_proc;
static ProcRecord text_proc;
static ProcRecord text_get_extents_proc;
void
register_text_tool_procs (void)
{
procedural_db_register (&text_fontname_proc);
procedural_db_register (&text_get_extents_fontname_proc);
procedural_db_register (&text_proc);
procedural_db_register (&text_get_extents_proc);
}
static gchar *
text_xlfd_insert_size (gchar *fontname,
gdouble size,
SizeType metric,
gdouble xresolution,
gdouble yresolution,
gboolean antialias)
{
gchar *newfont, *workfont;
gchar size_buffer[16];
gchar xres_buffer[16];
gchar yres_buffer[16];
gint pos = 0;
if (size <= 0)
return NULL;
if (xresolution < GIMP_MIN_RESOLUTION ||
yresolution < GIMP_MIN_RESOLUTION ||
metric == PIXELS)
{
xresolution = yresolution = 0.0;
}
else
{
xresolution = MIN (xresolution, GIMP_MAX_RESOLUTION);
yresolution = MIN (yresolution, GIMP_MAX_RESOLUTION);
}
if (antialias)
size *= SUPERSAMPLE;
/* Note: if metric == POINTS we either use the passed resolution (if
* statement below) or we (implicitly) use the resolution hidden
* in the fontname.
*/
if (metric == POINTS && xresolution != 0.0)
{
/* the xlfd uses decipoints */
size *= 10;
/* X allows only integer resolution values, so we do some
* ugly calculations (starting with yres because the size of
* a font is it's height)
*/
if (yresolution < 1.0)
{
size /= (1.0 / yresolution);
xresolution *= (1.0 / yresolution);
yresolution = 1.0;
}
/* res may be != (int) res
* (important only for very small resolutions)
*/
size *= yresolution / (double) (int) yresolution;
xresolution /= yresolution / (double) (int) yresolution;
/* finally, if xres became invalid by the above calculations */
xresolution = BOUNDS (xresolution, 1.0, GIMP_MAX_RESOLUTION);
}
sprintf (size_buffer, "%d", (int) size);
sprintf (xres_buffer, "%d", (int) xresolution);
sprintf (yres_buffer, "%d", (int) yresolution);
newfont = workfont = g_new (char, strlen (fontname) + 32);
while (*fontname)
{
*workfont++ = *fontname;
if (*fontname++ == '-')
{
pos++;
if ((pos == 7 && metric == PIXELS) ||
(pos == 8 && metric == POINTS))
{
int len = strlen (size_buffer);
memcpy (workfont, size_buffer, len);
workfont += len;
while (*fontname != '-')
fontname++;
}
else if (pos == 9 && metric == POINTS && xresolution != 0.0)
{
int len = strlen (xres_buffer);
memcpy (workfont, xres_buffer, len);
workfont += len;
while (*fontname != '-')
fontname++;
}
else if (pos == 10 && metric == POINTS && yresolution != 0.0)
{
int len = strlen (yres_buffer);
memcpy (workfont, yres_buffer, len);
workfont += len;
while (*fontname != '-')
fontname++;
}
}
}
*workfont = '\0';
return newfont;
}
static gchar *
text_xlfd_create (gchar *foundry,
gchar *family,
gchar *weight,
gchar *slant,
gchar *set_width,
gchar *spacing,
gchar *registry,
gchar *encoding)
{
/* create the fontname */
return g_strdup_printf ("-%s-%s-%s-%s-%s-*-*-*-*-*-%s-*-%s-%s",
foundry, family, weight, slant, set_width,
spacing, registry, encoding);
}
static Argument *
text_fontname_invoker (Argument *args)
{
gboolean success = TRUE;
Argument *return_args;
GimpImage *gimage;
GimpDrawable *drawable;
gdouble x;
gdouble y;
gchar *text;
gint32 border;
gboolean antialias;
gdouble size;
gint32 size_type;
gchar *fontname;
GimpLayer *text_layer = NULL;
gchar *real_fontname;
gimage = pdb_id_to_image (args[0].value.pdb_int);
if (gimage == NULL)
success = FALSE;
drawable = gimp_drawable_get_ID (args[1].value.pdb_int);
x = args[2].value.pdb_float;
y = args[3].value.pdb_float;
text = (gchar *) args[4].value.pdb_pointer;
if (text == NULL)
success = FALSE;
border = args[5].value.pdb_int;
if (border < -1)
success = FALSE;
antialias = args[6].value.pdb_int ? TRUE : FALSE;
size = args[7].value.pdb_float;
if (size <= 0.0)
success = FALSE;
size_type = args[8].value.pdb_int;
if (size_type < PIXELS || size_type > POINTS)
success = FALSE;
fontname = (gchar *) args[9].value.pdb_pointer;
if (fontname == NULL)
success = FALSE;
if (success)
{
/* default to pixels by passing 0.0 as resolution */
real_fontname = text_xlfd_insert_size (fontname, size, size_type,
0.0, 0.0, antialias);
text_layer = text_render (gimage, drawable, x, y, real_fontname, text,
border, antialias);
if (text_layer == NULL)
success = FALSE;
g_free (real_fontname);
}
return_args = procedural_db_return_args (&text_fontname_proc, success);
if (success)
return_args[1].value.pdb_int = drawable_ID (GIMP_DRAWABLE (text_layer));
return return_args;
}
static ProcArg text_fontname_inargs[] =
{
{
PDB_IMAGE,
"image",
"The image"
},
{
PDB_DRAWABLE,
"drawable",
"The affected drawable: (-1 for a new text layer)"
},
{
PDB_FLOAT,
"x",
"The x coordinate for the left of the text bounding box"
},
{
PDB_FLOAT,
"y",
"The y coordinate for the top of the text bounding box"
},
{
PDB_STRING,
"text",
"The text to generate"
},
{
PDB_INT32,
"border",
"The size of the border: -1 <= border"
},
{
PDB_INT32,
"antialias",
"Antialiasing (TRUE or FALSE)"
},
{
PDB_FLOAT,
"size",
"The size of text in either pixels or points"
},
{
PDB_INT32,
"size_type",
"The units of specified size: PIXELS (0) or POINTS (1)"
},
{
PDB_STRING,
"fontname",
"The fontname (conforming to the X Logical Font Description Conventions)"
}
};
static ProcArg text_fontname_outargs[] =
{
{
PDB_LAYER,
"text_layer",
"The new text layer"
}
};
static ProcRecord text_fontname_proc =
{
"gimp_text_fontname",
"Add text at the specified location as a floating selection or a new layer.",
"This tool requires font information as a fontname conforming to the 'X Logical Font Description Conventions'. You can specify the fontsize in units of pixels or points, and the appropriate metric is specified using the size_type argument. The x and y parameters together control the placement of the new text by specifying the upper left corner of the text bounding box. If the antialias parameter is non-zero, the generated text will blend more smoothly with underlying layers. This option requires more time and memory to compute than non-antialiased text; the resulting floating selection or layer, however, will require the same amount of memory with or without antialiasing. If the specified drawable parameter is valid, the text will be created as a floating selection attached to the drawable. If the drawable parameter is not valid (-1), the text will appear as a new layer. Finally, a border can be specified around the final rendered text. The border is measured in pixels.",
"Martin Edlman & Sven Neumann",
"Spencer Kimball & Peter Mattis",
"1998",
PDB_INTERNAL,
10,
text_fontname_inargs,
1,
text_fontname_outargs,
{ { text_fontname_invoker } }
};
static Argument *
text_get_extents_fontname_invoker (Argument *args)
{
gboolean success = TRUE;
Argument *return_args;
gchar *text;
gdouble size;
gint32 size_type;
gchar *fontname;
gint32 width;
gint32 height;
gint32 ascent;
gint32 descent;
gchar *real_fontname;
text = (gchar *) args[0].value.pdb_pointer;
if (text == NULL)
success = FALSE;
size = args[1].value.pdb_float;
if (size <= 0.0)
success = FALSE;
size_type = args[2].value.pdb_int;
if (size_type < PIXELS || size_type > POINTS)
success = FALSE;
fontname = (gchar *) args[3].value.pdb_pointer;
if (fontname == NULL)
success = FALSE;
if (success)
{
/* default to pixels by passing 0.0 as resolution */
real_fontname = text_xlfd_insert_size (fontname, size, size_type,
0.0, 0.0, FALSE);
success = text_get_extents (real_fontname, text,
&width, &height,
&ascent, &descent);
g_free (real_fontname);
}
return_args = procedural_db_return_args (&text_get_extents_fontname_proc, success);
if (success)
{
return_args[1].value.pdb_int = width;
return_args[2].value.pdb_int = height;
return_args[3].value.pdb_int = ascent;
return_args[4].value.pdb_int = descent;
}
return return_args;
}
static ProcArg text_get_extents_fontname_inargs[] =
{
{
PDB_STRING,
"text",
"The text to generate"
},
{
PDB_FLOAT,
"size",
"The size of text in either pixels or points"
},
{
PDB_INT32,
"size_type",
"The units of specified size: PIXELS (0) or POINTS (1)"
},
{
PDB_STRING,
"fontname",
"The fontname (conforming to the X Logical Font Description Conventions)"
}
};
static ProcArg text_get_extents_fontname_outargs[] =
{
{
PDB_INT32,
"width",
"The width of the specified font"
},
{
PDB_INT32,
"height",
"The height of the specified font"
},
{
PDB_INT32,
"ascent",
"The ascent of the specified font"
},
{
PDB_INT32,
"descent",
"The descent of the specified font"
}
};
static ProcRecord text_get_extents_fontname_proc =
{
"gimp_text_get_extents_fontname",
"Get extents of the bounding box for the specified text.",
"This tool returns the width and height of a bounding box for the specified text string with the specified font information. Ascent and descent for the specified font are returned as well.",
"Martin Edlman & Sven Neumann",
"Spencer Kimball & Peter Mattis",
"1998",
PDB_INTERNAL,
4,
text_get_extents_fontname_inargs,
4,
text_get_extents_fontname_outargs,
{ { text_get_extents_fontname_invoker } }
};
static Argument *
text_invoker (Argument *args)
{
gboolean success = TRUE;
int i;
Argument argv[10];
gchar *foundry;
gchar *family;
gchar *weight;
gchar *slant;
gchar *set_width;
gchar *spacing;
gchar *registry;
gchar *encoding;
foundry = (gchar *) args[9].value.pdb_pointer;
if (foundry == NULL)
success = FALSE;
family = (gchar *) args[10].value.pdb_pointer;
if (family == NULL)
success = FALSE;
weight = (gchar *) args[11].value.pdb_pointer;
if (weight == NULL)
success = FALSE;
slant = (gchar *) args[12].value.pdb_pointer;
if (slant == NULL)
success = FALSE;
set_width = (gchar *) args[13].value.pdb_pointer;
if (set_width == NULL)
success = FALSE;
spacing = (gchar *) args[14].value.pdb_pointer;
if (spacing == NULL)
success = FALSE;
registry = (gchar *) args[15].value.pdb_pointer;
if (registry == NULL)
success = FALSE;
encoding = (gchar *) args[16].value.pdb_pointer;
if (encoding == NULL)
success = FALSE;
if (!success)
return procedural_db_return_args (&text_proc, FALSE);
for (i = 0; i < 9; i++)
argv[i] = args[i];
argv[9].arg_type = PDB_STRING;
argv[9].value.pdb_pointer =
text_xlfd_create (foundry,
family,
weight,
slant,
set_width,
spacing,
registry,
encoding);
return text_fontname_invoker (argv);
}
static ProcArg text_inargs[] =
{
{
PDB_IMAGE,
"image",
"The image"
},
{
PDB_DRAWABLE,
"drawable",
"The affected drawable: (-1 for a new text layer)"
},
{
PDB_FLOAT,
"x",
"The x coordinate for the left of the text bounding box"
},
{
PDB_FLOAT,
"y",
"The y coordinate for the top of the text bounding box"
},
{
PDB_STRING,
"text",
"The text to generate"
},
{
PDB_INT32,
"border",
"The size of the border: -1 <= border"
},
{
PDB_INT32,
"antialias",
"Antialiasing (TRUE or FALSE)"
},
{
PDB_FLOAT,
"size",
"The size of text in either pixels or points"
},
{
PDB_INT32,
"size_type",
"The units of specified size: PIXELS (0) or POINTS (1)"
},
{
PDB_STRING,
"foundry",
"The font foundry, \"*\" for any"
},
{
PDB_STRING,
"family",
"The font family, \"*\" for any"
},
{
PDB_STRING,
"weight",
"The font weight, \"*\" for any"
},
{
PDB_STRING,
"slant",
"The font slant, \"*\" for any"
},
{
PDB_STRING,
"set_width",
"The font set-width, \"*\" for any"
},
{
PDB_STRING,
"spacing",
"The font spacing, \"*\" for any"
},
{
PDB_STRING,
"registry",
"The font registry, \"*\" for any"
},
{
PDB_STRING,
"encoding",
"The font encoding, \"*\" for any"
}
};
static ProcArg text_outargs[] =
{
{
PDB_LAYER,
"text_layer",
"The new text layer"
}
};
static ProcRecord text_proc =
{
"gimp_text",
"Add text at the specified location as a floating selection or a new layer.",
"This tool requires font information in the form of nine parameters: size, foundry, family, weight, slant, set_width, spacing, registry, encoding. The font size can either be specified in units of pixels or points, and the appropriate metric is specified using the size_type argument. The x and y parameters together control the placement of the new text by specifying the upper left corner of the text bounding box. If the antialias parameter is non-zero, the generated text will blend more smoothly with underlying layers. This option requires more time and memory to compute than non-antialiased text; the resulting floating selection or layer, however, will require the same amount of memory with or without antialiasing. If the specified drawable parameter is valid, the text will be created as a floating selection attached to the drawable. If the drawable parameter is not valid (-1), the text will appear as a new layer. Finally, a border can be specified around the final rendered text."
"The border is measured in pixels.",
"Martin Edlman",
"Spencer Kimball & Peter Mattis",
"1998",
PDB_INTERNAL,
17,
text_inargs,
1,
text_outargs,
{ { text_invoker } }
};
static Argument *
text_get_extents_invoker (Argument *args)
{
gboolean success = TRUE;
int i;
Argument argv[4];
gchar *foundry;
gchar *family;
gchar *weight;
gchar *slant;
gchar *set_width;
gchar *spacing;
gchar *registry;
gchar *encoding;
foundry = (gchar *) args[3].value.pdb_pointer;
if (foundry == NULL)
success = FALSE;
family = (gchar *) args[4].value.pdb_pointer;
if (family == NULL)
success = FALSE;
weight = (gchar *) args[5].value.pdb_pointer;
if (weight == NULL)
success = FALSE;
slant = (gchar *) args[6].value.pdb_pointer;
if (slant == NULL)
success = FALSE;
set_width = (gchar *) args[7].value.pdb_pointer;
if (set_width == NULL)
success = FALSE;
spacing = (gchar *) args[8].value.pdb_pointer;
if (spacing == NULL)
success = FALSE;
registry = (gchar *) args[9].value.pdb_pointer;
if (registry == NULL)
success = FALSE;
encoding = (gchar *) args[10].value.pdb_pointer;
if (encoding == NULL)
success = FALSE;
if (!success)
return procedural_db_return_args (&text_get_extents_proc, FALSE);
for (i = 0; i < 3; i++)
argv[i] = args[i];
argv[3].arg_type = PDB_STRING;
argv[3].value.pdb_pointer =
text_xlfd_create (foundry,
family,
weight,
slant,
set_width,
spacing,
registry,
encoding);
return text_get_extents_fontname_invoker (argv);
}
static ProcArg text_get_extents_inargs[] =
{
{
PDB_STRING,
"text",
"The text to generate"
},
{
PDB_FLOAT,
"size",
"The size of text in either pixels or points"
},
{
PDB_INT32,
"size_type",
"The units of specified size: PIXELS (0) or POINTS (1)"
},
{
PDB_STRING,
"foundry",
"The font foundry, \"*\" for any"
},
{
PDB_STRING,
"family",
"The font family, \"*\" for any"
},
{
PDB_STRING,
"weight",
"The font weight, \"*\" for any"
},
{
PDB_STRING,
"slant",
"The font slant, \"*\" for any"
},
{
PDB_STRING,
"set_width",
"The font set-width, \"*\" for any"
},
{
PDB_STRING,
"spacing",
"The font spacing, \"*\" for any"
},
{
PDB_STRING,
"registry",
"The font registry, \"*\" for any"
},
{
PDB_STRING,
"encoding",
"The font encoding, \"*\" for any"
}
};
static ProcArg text_get_extents_outargs[] =
{
{
PDB_INT32,
"width",
"The width of the specified font"
},
{
PDB_INT32,
"height",
"The height of the specified font"
},
{
PDB_INT32,
"ascent",
"The ascent of the specified font"
},
{
PDB_INT32,
"descent",
"The descent of the specified font"
}
};
static ProcRecord text_get_extents_proc =
{
"gimp_text_get_extents",
"Get extents of the bounding box for the specified text.",
"This tool returns the width and height of a bounding box for the specified text string with the specified font information. Ascent and descent for the specified font are returned as well.",
"Martin Edlman",
"Spencer Kimball & Peter Mattis",
"1998",
PDB_INTERNAL,
11,
text_get_extents_inargs,
4,
text_get_extents_outargs,
{ { text_get_extents_invoker } }
};