ftp.nice.ch/pub/next/unix/graphics/imtools.2.0.N.bs.tar.gz#/imtools/src/impaste.c

This is impaste.c in view mode; [Download] [Up]

/**
 **	$Header: /import/dev-vis/image/imtools/v2.0/imtools/src/RCS/impaste.c,v 1.1 91/10/03 13:19:59 nadeau Exp $
 **	Copyright (c) 1989, 1990  San Diego Supercomputer Center (SDSC)
 **		San Diego, California, USA
 **
 **	Users and possessors of this source code are hereby granted a
 **	nonexclusive, royalty-free copyright and design patent license to
 **	use this code in individual software.  License is not granted for
 **	commercial resale, in whole or in part, without prior written
 **	permission from SDSC.  This source is provided "AS IS" without express
 **	or implied warranty of any kind.
 **
 **	For further information contact:
 **		E-Mail:		info@sds.sdsc.edu
 **
 **		Surface Mail:	Information Center
 **				San Diego Supercomputer Center
 **				P.O. Box 85608
 **				San Diego, CA  92138-5608
 **				(619) 534-5000
 **/

#define HEADER	"    $Header: /import/dev-vis/image/imtools/v2.0/imtools/src/RCS/impaste.c,v 1.1 91/10/03 13:19:59 nadeau Exp $"

/**
 **  FILE
 **	impaste.c	-  Paste an image into another image
 **
 **  PROJECT
 **	IM		-  Image Manipulation Tools
 **
 **  DESCRIPTION
 **	impaste copies an image on top of another image, without regard for
 **	alpha planes.
 **
 **  PUBLIC CONTENTS
 **			d =defined constant
 **			f =function
 **			m =defined macro
 **			t =typedef/struct/union
 **			v =variable
 **			? =other
 **
 **	main		f  main program
 **
 **  PRIVATE CONTENTS
 **	toolCommand	v  tool-specific tool info
 **	toolHelp	v  tool-specific help
 **	toolOptions	v  tool-specific options
 **	toolEquivs	v  tool-specific equivalent keywords
 **
 **	toolInFilename	v  the name of the input file (could be 'stdin')
 **	toolOutFilename	v  the name of the output file (could be 'stdout')
 **	toolBackFilenamev  the name of the background file (could be 'stdin')
 **
 **	toolInFormat	v  the name of the input file's format (could be '\0')
 **	toolOutFormat	v  the name of the output file's format (could be '\0')
 **	toolBackFormat	v  the name of the background file's format (could be..)
 **
 **	toolInTable	v  a table for the storage of data read in
 **	toolBackTable	v  a table for the storage of background data read in
 **	toolFlagsTable	v  a table for the storage of read/write flags
 **
 **	toolPosX	v  X image position
 **	toolPosY	v  Y image position
 **	toolCenterX	v  Flag X direction centering
 **	toolCenterY	v  Flag Y direction centering
 **	toolTileX	v  X tile count
 **	toolTileY	v  Y tile count
 **	toolDirX	v  X tile direction
 **	toolDirY	v  Y tile direction
 **
 **	toolInit	f  initialize things for the tool
 **
 **  HISTORY
 **	$Log:	impaste.c,v $
 **	Revision 1.1  91/10/03  13:19:59  nadeau
 **	Initial revision
 **	
 **
 **/

#include "imtools.h"


extern void toolInit( );		/* Initialize things		*/





/*
 *  GLOBALS
 *	toolCommand		-  tool-specific tool info
 *	toolHelp		-  tool-specific help
 *	toolOptions		-  tool-specific options
 *	toolEquivs		-  tool-specific equivalent keywords
 *
 *  DESCRIPTION
 *	toolCommand describes the command to the arg package.
 *
 *	toolHelp is the tool-specific help for the tool.  It will be
 *	concatenated with the generic image tools help message and
 *	added to the toolCommand structure as the help string to be
 *	printed after the option listing.
 *
 *	toolOptions is the tool-specific option list.  It will be merged
 *	with the generic image tools options, and the list of image file
 *	format names supported by the library.  This larger option list
 *	is then used as the list of options supported by the tool.
 *
 *	toolEquivs is the tool-specific option equivalent keyword list.
 *	It will be merged with the generic image tools equivs.  This large
 *	equivs list is then used as the list of equivs supported by the tool.
 */

private ArgCommand toolCommand =
{
	/* command name */		"impaste",

	/* major version # */		IMTOOLSMAJOR,
	/* minor version # */		IMTOOLSMINOR,
	/* subminor version # */	IMTOOLSSUBMINOR,

	/* -help pre-option list information				*/
"%command pastes an input image onto a background image and saves the\n\
result into a new file.  Input, background, and output files may have\n\
different image file formats.\n\
",
	/* -help post-option list information				*/
	NULL,				/* filled in later on		*/

	/* -fullhelp pre-option list information			*/
	NULL,				/* Use same message as for -help*/
	/* -fullhelp post-option list information			*/
	NULL,				/* filled in later on		*/

	ARGFNONE,			/* No special flags		*/
	"[options...] infilename backfilename [outfilename]",
	"[options...] infilename backfilename [outfilename]",
	"SDSC Image Tools, October 1991.",
	"Copyright (c) 1989-1991  San Diego Supercomputer Center (SDSC), CA, USA",
	NULL,				/* filled in later on		*/
	NULL,				/* filled in later on		*/
};

private char *toolHelp = "\n\
Typical Pasting Invocations:\n\
    Paste a logo at (10,20) on a background and save it in a new HDF file:\n\
        %command logo.pix -xpos 10 -ypos 20 back.pix result.hdf\n\
    Paste a logo as before, but modify the background file in place:\n\
        %command logo.pix -xpos 10 -ypos 20 back.pix\n\
\n\
Typical Tiling Invocations:\n\
    Tile a pattern across the whole background:\n\
        %command pat.pix -xpos 0 -ypos 0 -xrepeat 0 -yrepeat 0 back.pix\n\
    Tile a pattern 3 times in X and none in Y starting at (-10,-20):\n\
        %command pat.pix -xrepeat 3 -xpos -10 -ypos -20 back.pix\n\
";

private char *toolFullHelp = "\n\
Files:\n\
    -infile selects the file whose images are to be pasted.\n\
\n\
    -backfile selects the file whose images are to act as backgrounds.\n\
\n\
    -outfile selects the file into which to write the resulting images.\n\
\n\
    The background and output files may be the same file.  If no output file\n\
    is given, the background file is used both as the source for background\n\
    images, and as the destination for completed images.\n\
\n\
Pasting:\n\
    Each image in the input file is pasted onto the corresponding image in\n\
    the background file and output to the output file.  If the background\n\
    file only contains a single image, the same background image is used for\n\
    each paste operation.\n\
\n\
    -xposition and -yposition select the location in the background image at\n\
    which the upper-left corner of the paste image is pasted.  (0,0) is the\n\
    upper-left corner of the background.  Positions outside the background\n\
    are allowed.  Paste images are clipped to the borders of the background\n\
    images.\n\
\n\
    -xposition and -yposition default to a position that centers the pasted\n\
    image within the background image.\n\
\n\
    Input images are converted to the same depth as the background before\n\
    pasting.  Pasting occurs independently of any alpha planes in any image.\n\
\n\
Tiling (Repeating):\n\
    -xrepeat and -yrepeat select the number of times to repeat the input image\n\
    in the X and Y directions.  A value of 0 repeats until the edge of the\n\
    background image is reached.  Both repeat counts default to 1.\n\
\n\
    -xdirection and -ydirection select the repeat direction in X and Y.\n\
    X directions may be 'left' or 'right' (default).  Y directions may be\n\
    'up' or 'down' (default).\n\
";

private char *toolNote = "\n\
Additional Help:\n\
    This is an abbreviated help listing.  For a full listing of options,\n\
    including a list of image file formats supported, type:\n\
        %command -fullhelp\n\
";


#define TOOLNOPTIONS	10
private ArgOption toolOptions[TOOLNOPTIONS] =
{
	{ "infile", "image_filename", "Specify an input image file name",
	  ARGFREQUIRED | ARGFIMPKEYWORD, 1, 1, ARGTSTRING },

	{ "backfile", "image_filename", "Specify a background image file name",
	  ARGFREQUIRED | ARGFIMPKEYWORD, 1, 1, ARGTSTRING },

	{ "outfile", "image_filename", "Specify an output image file name",
	  ARGFIMPKEYWORD, 1, 1, ARGTSTRING },

	{ "xposition", "x", "Specify left edge of the paste area",
	  ARGFNONE, 1, 1, ARGTINT },

	{ "yposition", "y", "Specify upper edge of the paste area",
	  ARGFNONE, 1, 1, ARGTINT },

	{ "xrepeat", "nx", "Select # of times to repeat in X",
	  ARGFNONE, 1, 1, ARGTINT },

	{ "yrepeat", "ny", "Select # of times to repeat in Y",
	  ARGFNONE, 1, 1, ARGTINT },

	{ "xdirection", "xdir", "Select X repeat direction",
	  ARGFNONE, 1, 1, ARGTSTRING },

	{ "ydirection", "ydir", "Select Y repeat direction",
	  ARGFNONE, 1, 1, ARGTSTRING },

	{ "verbose", NULL, "Be verbose",
	  ARGFFULLHELP, 0, 0, ARGTNONE },
};

#define TOOLNEQUIVS	0
#if TOOLNEQUIVS == 0
private ArgEquiv *toolEquivs;
#else
private ArgEquiv toolEquivs[TOOLNEQUIVS] =
{
};
#endif





/*
 *  GLOBALS
 *	toolInFilename	-  the name of the input file (could be 'stdin')
 *	toolOutFilename	-  the name of the output file (could be 'stdout')
 *
 *	toolInFormat	-  the name of the input file's format (could be NULL)
 *	toolOutFormat	-  the name of the output file's format (could be NULL)
 *
 *	toolInTable	-  a table for the storage of data read in
 *	toolBackTable	-  a table for the storage of background data read in
 *	toolFlagsTable	-  a table for the storage of read/write flags
 *
 *	toolPosX	-  X image position
 *	toolPosY	-  Y image position
 *	toolCenterX	-  Flag X direction centering
 *	toolCenterY	-  Flag Y direction centering
 *	toolTileX	-  X tile count
 *	toolTileY	-  Y tile count
 *	toolDirX	-  X tile direction
 *	toolDirY	-  Y tile direction
 *
 *  DESCRIPTION
 *	Data held by these variables is used throughout the tool.
 */

private char      toolInFilename[1024];	/* Input file name		*/
private char      toolInFormat[1024];	/* Input file's format name	*/

private char      toolOutFilename[1024];/* Output file name		*/
private char      toolOutFormat[1024];	/* Output file's format name	*/

private char      toolBackFilename[1024];/* Background file name	*/
private char      toolBackFormat[1024];	/* Background file's format name*/

private TagTable *toolInTable;		/* Input Data tag table		*/
private TagTable *toolBackTable;	/* Background data tag table	*/
private TagTable *toolFlagsTable;	/* Flags tag table		*/

private int	  toolPosX;		/* X Position			*/
private int	  toolPosY;		/* Y Position			*/
private boolean   toolCenterX;		/* Center in X			*/
private boolean   toolCenterY;		/* Center in Y			*/
private int	  toolTileX;		/* X Repeats			*/
private int	  toolTileY;		/* Y Repeats			*/

private int	  toolDirX;		/* X Direction			*/
private int	  toolDirY;		/* Y Direction			*/

#define TOOL_DIRLEFT	0
#define TOOL_DIRUP	0
#define TOOL_DIRRIGHT	1
#define TOOL_DIRDOWN	1





/*
 *  FUNCTION
 *	main	-  main program
 *
 *  DESCRIPTION
 *	Control things:
 *		-  Initialize things (parse arguments and set up tables).
 *		-  Read in the input file (put data into data table).
 *		-  Copy part of each image.
 *		-  Replace the data table images with their smaller copy pieces.
 *		-  Write out the output file (get data from data table).
 *	That's about it.
 *
 *  NOTES
 *	This 'main' is pretty much the same for each of the image tools.
 *	Differences between tools include:
 *		-  the number of input files read in
 *		-  the number of output files written out
 *		-  the actions taken on the data between read and write
 */

#define ImageType(f)		((f)&IMVFBIMAGEMASK)
#define IsImage(f)		(((f)&IMVFBIMAGEMASK) != 0)
#define SameImageType(f1,f2)	(ImageType(f1) == ImageType(f2))

main( argc, argv )
	int argc;			/* Argument count		*/
	char *argv[];			/* Argument vector		*/
{
	int         nInVfb;		/* Number of images in input file*/
	int         nBackVfb;		/* Number of images in background file*/
	int         i;			/* Counter			*/
	int	    n;			/* Counter			*/
	int	    rep;		/* Repeat count temp		*/
	int	    nPix;		/* Pixel count temp		*/
	char	   *tag;		/* Tag name			*/

	TagEntry   *dataEntry;		/* Entry from data table	*/

	ImVfb      *pasteVfb;		/* Paste image			*/
	ImVfb	   *newVfb;		/* New paste image		*/
	ImVfb      *backVfb;		/* Background image		*/
	ImClt	   *backClt;		/* Background image's CLT	*/
	int	    backFields;		/* Background fields		*/
	int         pasteFields;	/* Paste fields			*/
	int	    copyFields;		/* What fields to copy		*/

	int	    regionW;		/* Region width			*/
	int	    regionH;		/* Region height		*/
	int	    regionX;		/* Region left edge		*/
	int	    regionY;		/* Region top edge		*/
	int	    xrep, yrep;		/* X and Y repeat counters	*/
	int	    nxrep, nyrep;	/* # of X and Y repeats		*/
	int	    x, y;		/* Position counters		*/
	int	    dx, dy;		/* Delta X and Y for tiling	*/
	int	    xat, yat;		/* Where to paste it at		*/


	/*
	 *  Initialize things:
	 *	-  Prepare the arg parsing data, then parse the command-line.
	 *	-  Prepare the flags table based upon command-line args.
	 *	-  Determine the file formats for input and output files.
	 */
	toolInit( argc, argv );


	/*
	 *  Read in the input file.
	 *	-  Open the file (or stdin) and read data into the data table.
	 */
	ImToolsFileRead( toolInFilename, toolInFormat, toolFlagsTable,
		toolInTable );


	/*
	 *  Read in the background file.
	 *	-  Open the file (or stdin) and read data into the data table.
	 */
	ImToolsFileRead( toolBackFilename, toolBackFormat, toolFlagsTable,
		toolBackTable );


	/*
	 *  Check for errors
	 *	-  no input images
	 *	-  no background images
	 */
	nInVfb = TagTableQNEntry( toolInTable, "image vfb" );
	if ( nInVfb == 0 )
	{
		fprintf( stderr, "%s: Input file contains no images!\n",
			ImToolsProgram );
		exit( 1 );
	}
	nBackVfb = TagTableQNEntry( toolBackTable, "image vfb" );
	if ( nBackVfb == 0 )
	{
		fprintf( stderr, "%s: Background file contains no images!\n",
			ImToolsProgram );
		exit( 1 );
	}


	/*
	 *  Prep the background data table.
	 *
	 *  We want one background image for each input image.
	 *	-  if there are multiple input images, but only one background
	 *	   image, duplicate the background image a bunch of times.
	 *	-  if there are more input images than background images, take
	 *	   the last background image and add duplicates of it.
	 *	-  if there are more background images than input images,
	 *	   that's OK.
	 */
	if ( nBackVfb < nInVfb )
	{
		dataEntry = TagTableQDirect( toolBackTable, "image vfb",
			nBackVfb - 1 );
		TagEntryQValue( dataEntry, &backVfb );

		for ( i = nBackVfb; i < nInVfb; i++ )
		{
			backVfb = ImVfbDup( backVfb );
			backClt = ImVfbQClt( backVfb );
			if ( backClt != IMCLTNULL )
				TagTableAppend( toolBackTable,
					TagEntryAlloc( "image clt",
					POINTER, &backClt ) );
			TagTableAppend( toolBackTable,
				TagEntryAlloc( "image vfb",
				POINTER, &backVfb ) );
		}
		nBackVfb = nInVfb;
	}


	/*
	 *  Paste the images
	 *	-  For each input image, paste it onto the equivalent
	 *	   background image and store the result back in-place in
	 *	   the background data table.
	 */
	for ( i = 0; i < nInVfb; i++ )
	{
		/*
		 *  Get the next input and background images out of the tables.
		 */
		dataEntry = TagTableQDirect( toolInTable, "image vfb", i );
		TagEntryQValue( dataEntry, &pasteVfb );
		dataEntry = TagTableQDirect( toolBackTable, "image vfb", i );
		TagEntryQValue( dataEntry, &backVfb );


		/*
		 *  Compute the region.
		 */
		if ( toolCenterX )
			toolPosX = (ImVfbQWidth( backVfb ) - ImVfbQWidth( pasteVfb )) / 2;
		else if ( toolPosX > ImVfbQWidth( backVfb ) )
		{
			fprintf( stderr, "%s: Paste position is out of bounds for background %d of %d.\n",
				ImToolsProgram, i+1, nInVfb );
			fprintf( stderr, "%s: No paste done.\n",
				ImToolsProgram );
			continue;
		}

		if ( toolCenterY )
			toolPosY = (ImVfbQHeight( backVfb ) - ImVfbQHeight( pasteVfb )) / 2;
		else if ( toolPosY > ImVfbQHeight( backVfb ) )
		{
			fprintf( stderr, "%s: Paste position is out of bounds for background %d of %d.\n",
				ImToolsProgram, i+1, nInVfb );
			fprintf( stderr, "%s: No paste done.\n",
				ImToolsProgram );
			continue;
		}


		regionW = ImVfbQWidth( pasteVfb );
		regionH = ImVfbQHeight( pasteVfb );


		/*
		 *  Make sure we have something in common between the paste
		 *  and background VFBs.  The rule is that the output
		 *  results will have the same fields as the background image.
		 *
		 *  Lots of cases exist here:
		 *	1.  The paste and background VFBs have the same fields.
		 *		-  Leave the paste VFB as is.  The paste
		 *		   operation will use the full field set.
		 *
		 *	2.  The paste and background VFBs have some number
		 *	    of fields in common, but not all in common.
		 *	    Two sub-cases exist:
		 *
		 *		a.  Either or both of the background or paste
		 *		    VFBs have no image fields (rgb, index16,
		 *		    index8, or mono).
		 *			-  Leave the paste VFB as is.  The
		 *			   paste operation will use the
		 *			   intersection of the field set.
		 *
		 *		b.  Both background and paste VFBs have
		 *		    image fields.  Two sub-sub-cases exist:
		 *			i.  They are the same image fields.
		 *				-  Leave the paste VFB as is.
		 *				   The paste operation will use
		 *				   the intersection of the
		 *				   field set.
		 *			ii. They are different.
		 *				-  Convert the paste VFB to
		 *				   the same image field type
		 *				   as the background image.
		 *				   The paste operation will use
		 *				   the intersection of the
		 *				   converted paste image's
		 *				   fields and the background
		 *				   image's fields.
		 *
		 *	3.  The paste and background VFBs have no fields
		 *	    in common.  Two sub-cases exist:
		 *
		 *		a.  Either or both of the background or paste
		 *		    VFBs have no image fields (rgb, index16,
		 *		    index8, or mono).
		 *			-  No paste operation is done.
		 *
		 *		b.  Both background and paste VFBs have
		 *		    image fields, but they are different.
		 *			-  Convert the paste VFB to
		 *			   the same image field type
		 *			   as the background VFB.
		 *			   The paste operation will use
		 *			   the intersection of the
		 *			   converted paste VFB's
		 *			   fields and the background
		 *			   VFB's fields.
		 *
		 *  All of this can be reduced to the following test:
		 *
		 *	if both the background and the paste VFBs contain
		 *		image fields, but they aren't the same fields,
		 *			-  Convert the paste VFB's image field
		 *			   data to the same type as the
		 *			   background VFB.
		 *	else if the two VFBs have nothing in common,
		 *			-  Skip the paste.
		 */
		backFields  = ImVfbQFields( backVfb );
		pasteFields = ImVfbQFields( pasteVfb );
		copyFields  = backFields & pasteFields;

		if ( (IsImage( backFields) && IsImage( pasteFields )) &&
			!SameImageType( backFields, pasteFields ) )
		{
			/*
			 *  Convert to the right VFB image type:
			 *	-  Make a new VFB capable of storing the
			 *	   fields common to background and paste VFBs,
			 *	   plus the image type of the background.
			 *	-  Copy the common fields from the paste VFB
			 *	   into the new VFB.
			 *	-  Convert the paste VFB's image fields into
			 *	   that used by the background and new VFBs.
			 */
			pasteFields = copyFields | ImageType( backFields );
			newVfb = ImVfbAlloc( regionW, regionH, pasteFields );

			if ( ImVfbCopy( pasteVfb,	/* Copy this VFB*/
				0, 0,			/* Start at top	*/
				regionW, regionH,	/* This much	*/
				pasteFields,		/* These planes	*/
				newVfb,			/* Into this VFB*/
				0, 0 ) == IMVFBNULL )	/* At the top	*/
			{
                                ImPError( ImToolsProgram );
                                fprintf( stderr, "%s: Couldn't create common paste region for background %d of %d.\n", 
					ImToolsProgram, i+1, nInVfb );
				fprintf( stderr, "%s: No paste done.\n",
                                        ImToolsProgram );
                                continue;
			}

			copyFields = pasteFields;

			switch ( ImageType( backFields ) )
			{
			case IMVFBMONO:
				ImVfbToMono( pasteVfb, 127, newVfb );
				break;

			case IMVFBINDEX8:
				ImVfbToIndex8( pasteVfb, newVfb );
				break;

			case IMVFBINDEX16:
				ImVfbToIndex16( pasteVfb, newVfb );
				break;

			case IMVFBRGB:
				ImVfbToRgb( pasteVfb, newVfb );
				break;
			}
			ImVfbFree( pasteVfb );
			pasteVfb = newVfb;
		}
		else if ( copyFields == 0 )
		{
			/* No fields in common!				*/
			fprintf( stderr, "%s: Paste image %d of %d has no per-pixel info in common with\n",
				ImToolsProgram, i, nInVfb );
			fprintf( stderr, "%s: background image %d of %d.  Pasting would have no effect.\n",
				ImToolsProgram, i, nInVfb );
			fprintf( stderr, "%s: No paste done.\n",
				ImToolsProgram );
			continue;
		}


		/*
		 *  Check the tiling repeat counts:
		 *	-  Compute the maximum repeat count possible.
		 *	-  If a 0 repeat count was given (do as many as
		 *	   possible), use the maximum.
		 *	-  If a given repeat count is greater than the max,
		 *	   use the max.
		 */
		if ( toolDirX == TOOL_DIRRIGHT )
			nPix = ImVfbQWidth( backVfb ) - toolPosX;
		else
			nPix = toolPosX;
		rep = nPix / regionW;
		if ( (nPix % regionW) != 0 )
			++rep;			/* Partial paste	*/
		xrep = toolTileX;
		if ( xrep == 0 || xrep > rep )
			xrep = rep;

		if ( toolDirY == TOOL_DIRDOWN )
			nPix = ImVfbQHeight( backVfb ) - toolPosY;
		else
			nPix = toolPosY;
		rep = nPix / regionH;
		if ( (nPix % regionH) != 0 )
			++rep;			/* Partial paste	*/
		yrep = toolTileY;
		if ( yrep == 0 || yrep > rep )
			yrep = rep;


		if ( ImToolsVerbose )
		{
			if ( xrep == 1 && yrep == 1 )
			{
				fprintf( stderr, "%s: Pasting image %d of %d at (%d,%d)\n",
					ImToolsProgram, i + 1, nInVfb,
					toolPosX, toolPosY );
			}
			else
			{
				fprintf( stderr, "%s: Tiling image %d of %d at (%d,%d)\n",
					ImToolsProgram, i + 1, nInVfb,
					toolPosX, toolPosY );
				fprintf( stderr, "%s: Image is repeated %d times in X and %d times in Y\n",
					ImToolsProgram, xrep, yrep );
				fprintf( stderr, "%s: Tiling progresses ",
					ImToolsProgram );
				if ( toolDirX == TOOL_DIRRIGHT )
					fprintf( stderr, "to the right and " );
				else
					fprintf( stderr, "to the left and " );
				if ( toolDirY == TOOL_DIRDOWN )
					fprintf( stderr, "down\n" );
				else
					fprintf( stderr, "up\n" );
			}
		}


		/*
		 *  Tile!
		 *	-  Walk the tile region, with the direction depending
		 *	   upon that selected by command-line arguments.
		 *	-  For each tile position, clip the region to the
		 *	   edges of the background, and copy the paste image
		 *	   into it.
		 */
		if ( toolDirY == TOOL_DIRDOWN )
			dy = regionH;
		else
			dy = -regionH;
		if ( toolDirX == TOOL_DIRRIGHT )
			dx = regionW;
		else
			dx = -regionW;

		y = toolPosY;
		for ( nyrep = 0; nyrep < yrep; nyrep++, y += dy )
		{
			regionY = 0;
			regionH = ImVfbQHeight( pasteVfb );
			yat = y;
			if ( y < 0 )
			{
				/* Clip top off of image.		*/
				regionY = -y;
				regionH -= regionY;
				yat = 0;
			}

			nPix = ImVfbQHeight( backVfb ) - y;
			if ( regionH > nPix )
			{
				/* Clip bottom off of image.		*/
				regionH = nPix;
			}

			if ( regionH <= 0 )
				break;		/* Nothing more to do*/


			x = toolPosX;
			for ( nxrep = 0; nxrep < xrep; nxrep++, x += dx )
			{
				regionX = 0;
				regionW = ImVfbQWidth( pasteVfb );
				xat = x;
				if ( x < 0 )
				{
					/* Clip left off of image.	*/
					regionX = -x;
					regionW -= regionX;
					xat = 0;
				}

				nPix = ImVfbQWidth( backVfb ) - x;
				if ( regionW > nPix )
					regionW = nPix;

				if ( regionW <= 0 )
					break;	/* Nothing more to do	*/


				/*
				 *  Copy the region into the background VFB.
				 */
				if ( ImVfbCopy( pasteVfb,      /* Use this VFB*/
						regionX, regionY,/* Start here*/
						regionW, regionH,  /* This big*/
						copyFields,	  /* This deep*/
						backVfb,      /* Put into this*/
						xat,      	/* Put it here*/
						yat ) == IMVFBNULL )
				{
					if ( xrep == 1 && yrep == 1 )
					{
	                                	ImPError( ImToolsProgram );
                                		fprintf( stderr, "%s: Couldn't paste image %d of %d.\n",
                                        		ImToolsProgram, i + 1, nInVfb );
                                		fprintf( stderr, "%s: No paste done.\n",
                                        		ImToolsProgram );
                                		continue;
					}
					else
					{
	                                	ImPError( ImToolsProgram );
                                		fprintf( stderr, "%s: Couldn't tile image %d of %d.\n",
                                        		ImToolsProgram, i + 1, nInVfb );
                                		fprintf( stderr, "%s: No tiling done.\n",
                                        		ImToolsProgram );
						nyrep = yrep;			
                                		break;

					}
				}
			}
		}
	}


	/*
	 *  Write out the output file.
	 *	-  Open the file (or stdout) and write the data in the data
	 *	   table.  Upon failure, remove the bad output file.
	 */
	ImToolsFileWrite( toolOutFilename, toolOutFormat, toolFlagsTable,
		toolBackTable );


	exit( 0 );
}





/*
 *  FUNCTION
 *	toolInit	-  initialize things for the tool
 *
 *  DESCRIPTION
 *	The tool's argument parsing data structures are set up to include:
 *		- the full help message (generic help + tool-specific help)
 *		- the full option list (generic options + tool-specific options)
 *		- the full equivs list (generic equivs + tool-specific equivs)
 *
 *	Command-line arguments are then parsed.  The results are used to
 *	set up the flags table (the generic -out* options).
 *
 *	Input, output and background file's names and formats are determined
 *	from the command-line arguments.
 *
 *  NOTES
 *	This function is included in most of the image tools and differs
 *	only in slight ways.  Typical differences include:
 *		-  the number of input and output file names found
 *		-  the number of input and output file formats found
 *		-  the number of command-line arg flags checked for
 */

private void				/* Returns nothing		*/
toolInit( argc, argv )
	int argc;			/* Argument count		*/
	char *argv[ ];			/* Argument vector		*/
{
	int	    i;			/* Counter			*/
	int	    noccur;		/* Number of option occurrences	*/
	int         nOpt;		/* Number of options		*/
	int         nEquiv;		/* Number of equivalences	*/
	ArgOption  *options1;		/* Argument options		*/
	ArgOption  *options;		/* Argument options		*/
	ArgEquiv   *equivs1;		/* Argument equivalent keywords	*/
	ArgEquiv   *equivs;		/* Argument equivalent keywords	*/

	char       *xdir, *ydir;	/* X and Y direction names	*/

	char       *tmp;		/* Temporary string holder	*/
	char       *tmpFormat;		/* Tmp format name		*/


	/*
	 *  Save the name of the program, as invoked.
	 */
	ImToolsProgram = argv[0];


	/*
	 *  Make data tables to hold the data.
	 */
	if ( (toolInTable = TagTableAlloc( )) == TAGTABLENULL )
	{
		TagPError( ImToolsProgram );
		exit( 1 );
	}
	if ( (toolBackTable = TagTableAlloc( )) == TAGTABLENULL )
	{
		TagPError( ImToolsProgram );
		exit( 1 );
	}


	/*
	 *  Use the standard Image Tools user registration and feedback forms.
	 */
	toolCommand.arg_register = ImToolsRegister;
	toolCommand.arg_feedback = ImToolsFeedback;


	/*
	 *  Allocate space for the total help string for the tool.  Copy the
	 *  tool-specific help in, then concatenate on the generic help text
	 *  used by most of the image tools.
	 */
	if ( (tmp = (char *)malloc( sizeof( char ) * (strlen( toolNote ) +
		strlen( toolHelp ) + 1) )) == NULL )
	{
		perror( ImToolsProgram );
		exit( 1 );
	}
	strcpy( tmp, toolHelp );
	strcat( tmp, toolNote );
	toolCommand.arg_help2 = tmp;

	if ( (tmp = (char *)malloc( sizeof( char ) * (strlen( ImToolsBaseHelp) +
		strlen( toolHelp ) + strlen( toolFullHelp ) + 1) )) == NULL )
	{
		perror( ImToolsProgram );
		exit( 1 );
	}
	strcpy( tmp, toolHelp );
	strcat( tmp, toolFullHelp );
	strcat( tmp, ImToolsBaseHelp );
	toolCommand.arg_fullhelp2 = tmp;


	/*
	 *  Build up an option list by merging the tool-specific options,
	 *  the standard (base) tool options, and those for the various
	 *  image file formats.
	 */
	nOpt = ImToolsMergeOptions( TOOLNOPTIONS, toolOptions,
		IMTOOLSNBASEOPTIONS, ImToolsBaseOptions, &options1 );
	if ( (nOpt = ImFileFormatOptions( nOpt, options1, &options )) == -1)
	{
		ImPError( ImToolsProgram );
		exit( 1 );
	}


	/*
	 *  Build up an equivalent keyword list by merging the tool-specific
	 *  equivalences, the standard (base) tool equivalences, and those
	 *  for the various image file formats.
	 */
	nEquiv = ImToolsMergeEquivs( TOOLNEQUIVS, toolEquivs,
		 IMTOOLSNBASEEQUIVS, ImToolsBaseEquivs, &equivs1 );
	if ( (nEquiv = ImFileFormatEquivs( nEquiv, equivs1, &equivs )) == -1)
	{
		ImPError( ImToolsProgram );
		exit( 1 );
	}


	/*
	 *  Parse the command line!
	 */
	nOpt = ArgParse( argc, argv, &toolCommand, nOpt, options,
		nEquiv, equivs );
	if ( ArgQNOccur( "verbose" ) != 0 )
		ImToolsVerbose = TRUE;


	/*
	 *  Get the image position and tiling selections.
	 */
	toolPosX = toolPosY = 0;
	toolCenterX = toolCenterY = TRUE;
	if ( ArgQNOccur( "xposition" ) != 0 )
	{
		toolPosX = ArgQValue( "xposition", 0, 0 )->arg_i;
		toolCenterX = FALSE;
	}
	if ( ArgQNOccur( "yposition" ) != 0 )
	{
		toolPosY = ArgQValue( "yposition", 0, 0 )->arg_i;
		toolCenterY = FALSE;
	}

	toolTileX = toolTileY = 1;
	if ( ArgQNOccur( "xrepeat" ) != 0 )
		toolTileX = ArgQValue( "xrepeat", 0, 0 )->arg_i;
	if ( ArgQNOccur( "yrepeat" ) != 0 )
		toolTileY = ArgQValue( "yrepeat", 0, 0 )->arg_i;
	if ( toolTileX < 0 || toolTileY < 0 )
	{
		fprintf( stderr, "%s:  tile repeats must be zero or positive\n",
			ImToolsProgram );
		exit( 1 );
	}

	toolDirX = TOOL_DIRRIGHT;
	toolDirY = TOOL_DIRDOWN;
	if ( ArgQNOccur( "xdirection" ) != 0 )
	{
		xdir = ArgQValue( "xdirection", 0, 0 )->arg_s;
		i = strlen( xdir );
		if ( strncmp( xdir, "left", i ) == 0 )
			toolDirX = TOOL_DIRLEFT;
		else if ( strncmp( xdir, "right", i ) == 0 )
			toolDirX = TOOL_DIRRIGHT;
		else
		{
			fprintf( stderr, "%s: X direction must select either 'left' or 'right'\n",
				ImToolsProgram );
			exit( 1 );
		}
	}
	if ( ArgQNOccur( "ydirection" ) != 0 )
	{
		ydir = ArgQValue( "ydirection", 0, 0 )->arg_s;
		i = strlen( ydir );
		if ( strncmp( ydir, "up", i ) == 0 )
			toolDirX = TOOL_DIRUP;
		else if ( strncmp( ydir, "down", i ) == 0 )
			toolDirX = TOOL_DIRDOWN;
		else
		{
			fprintf( stderr, "%s: Y direction must select either 'up' or 'down'\n",
				ImToolsProgram );
			exit( 1 );
		}
	}


	/*
	 *  Set up the flags table based upon command-line arguments.
	 *  This is primarily derived from the -out* directives part of the
	 *  standard image tool option set (see ImToolsBaseOptions above).
	 *  Also included are flags to direct error messages to stderr and
	 *  a flag giving the program's name for later use in error messages.
	 */
	toolFlagsTable = ImToolsBuildFlagsTable( );


	/*
	 *  Get the input file's name (-infile), and search backwards in the
	 *  command-line option list to find the last format selection (if
	 *  any).  Stop the search on the beginning of the command-line, or
	 *  on -outfile or -backfile.
	 */
	strcpy( toolInFilename, ArgQValue( "infile", 0, 0 )->arg_s );
	tmpFormat = NULL;
	for ( i = ArgQOccurOpt( "infile", 0 ) - 1; i >= 0; i-- )
	{
		tmp = ArgQOpt( i, &noccur );


		/*
		 *  Stop looking backward when we reach any other file name
		 *  argument.
		 */
		if ( strcmp( tmp, "outfile" ) == 0 )
			break;
		if ( strcmp( tmp, "backfile" ) == 0 )
			break;


		/*
		 *  Skip it if it isn't the name of a file format.
		 */
		if ( !ImToolsIsFormat( tmp ) )
			continue;


		if ( tmpFormat != NULL )
		{
			fprintf( stderr, "%s:  Only 1 file format selection may precede -infile.\n",
				ImToolsProgram );
			exit( 1 );
		}
		tmpFormat = tmp;
	}
	if ( tmpFormat == NULL )
		*toolInFormat = '\0';
	else
		strcpy( toolInFormat, tmpFormat );


	/*
	 *  Get the background file's name (-backfile), and search backwards
	 *  in the command-line option list to find the last format selection
	 *  (if any).  Stop the search on the beginning of the command-line, or
	 *  on -infile or -outfile.
	 */
	strcpy( toolBackFilename, ArgQValue( "backfile", 0, 0 )->arg_s );
	tmpFormat = NULL;
	for ( i = ArgQOccurOpt( "backfile", 0 ) - 1; i >= 0; i-- )
	{
		tmp = ArgQOpt( i, &noccur );


		/*
		 *  Stop looking backward when we reach any other file name
		 *  argument.
		 */
		if ( strcmp( tmp, "infile" ) == 0 )
			break;
		if ( strcmp( tmp, "outfile" ) == 0 )
			break;


		/*
		 *  Skip it if it isn't the name of a file format.
		 */
		if ( !ImToolsIsFormat( tmp ) )
			continue;


		if ( tmpFormat != NULL )
		{
			fprintf( stderr, "%s:  Only 1 file format selection may precede -backfile.\n",
				ImToolsProgram );
			exit( 1 );
		}
		tmpFormat = tmp;
	}
	if ( tmpFormat == NULL )
		*toolBackFormat = '\0';
	else
		strcpy( toolBackFormat, tmpFormat );


	/*
	 *  Get the output file's name (-outfile), and search backwards in the
	 *  command-line option list to find the last format selection (if
	 *  any).  Stop the search on the beginning of the command-line, or
	 *  on -infile or -backfile.
	 */
	if ( ArgQNOccur( "outfile" ) == 0 )
	{
		/* No output filename given.  Use the background file.	*/
		strcpy( toolOutFilename, toolBackFilename );
		strcpy( toolOutFormat,   toolBackFormat );
	}
	else
	{
		strcpy( toolOutFilename, ArgQValue( "outfile", 0, 0 )->arg_s );
		tmpFormat = NULL;
		for ( i = ArgQOccurOpt( "outfile", 0 ) - 1; i >= 0; i-- )
		{
			tmp = ArgQOpt( i, &noccur );


			/*
			 *  Stop looking backward when we reach any other
			 *  file name argument.
			 */
			if ( strcmp( tmp, "infile" ) == 0 )
				break;
			if ( strcmp( tmp, "backfile" ) == 0 )
				break;


			/*
			 *  Skip it if it isn't the name of a file format.
			 */
			if ( !ImToolsIsFormat( tmp ) )
				continue;


			if ( tmpFormat != NULL )
			{
				fprintf( stderr, "%s:  Only 1 file format selection may precede -outfile.\n",
					ImToolsProgram );
				exit( 1 );
			}
			tmpFormat = tmp;
		}
		if ( tmpFormat == NULL )
			*toolOutFormat = '\0';
		else
			strcpy( toolOutFormat, tmpFormat );
	}


	/*
	 *  Check that the input and background files are not both stdin.
	 */
	if ( strcmp( toolInFilename, "-" ) == 0 &&
		strcmp( toolBackFilename, "-" ) == 0)
	{
		fprintf( stderr, "%s: -infile and -backfile cannot both be stdin.\n",
			ImToolsProgram );
		exit( 1 );
	}
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.