kopia lustrzana https://gitlab.com/sane-project/backends
				
				
				
			
		
			
				
	
	
		
			392 wiersze
		
	
	
		
			9.9 KiB
		
	
	
	
		
			Java
		
	
	
			
		
		
	
	
			392 wiersze
		
	
	
		
			9.9 KiB
		
	
	
	
		
			Java
		
	
	
| /* sane - Scanner Access Now Easy.
 | |
|    Copyright (C) 1997 Jeffrey S. Freedman
 | |
|    This file is part of the SANE package.
 | |
| 
 | |
|    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.
 | |
| 
 | |
|    As a special exception, the authors of SANE give permission for
 | |
|    additional uses of the libraries contained in this release of SANE.
 | |
| 
 | |
|    The exception is that, if you link a SANE library with other files
 | |
|    to produce an executable, this does not by itself cause the
 | |
|    resulting executable to be covered by the GNU General Public
 | |
|    License.  Your use of that executable is in no way restricted on
 | |
|    account of linking the SANE library code into it.
 | |
| 
 | |
|    This exception does not, however, invalidate any other reasons why
 | |
|    the executable file might be covered by the GNU General Public
 | |
|    License.
 | |
| 
 | |
|    If you submit changes to SANE to the maintainers to be included in
 | |
|    a subsequent release, you agree by submitting the changes that
 | |
|    those changes may be distributed with this exception intact.
 | |
| 
 | |
|    If you write modifications of your own for SANE, it is your choice
 | |
|    whether to permit this exception to apply to your modifications.
 | |
|    If you do not wish that, delete this exception notice.  */
 | |
| 
 | |
| /**
 | |
|  **	ScanIt.java - Do the actual scanning for SANE.
 | |
|  **
 | |
|  **	Written: 11/3/97 - JSF
 | |
|  **/
 | |
| 
 | |
| import java.util.Vector;
 | |
| import java.util.Enumeration;
 | |
| import java.awt.image.ImageProducer;
 | |
| import java.awt.image.ImageConsumer;
 | |
| import java.awt.image.ColorModel;
 | |
| import java.io.OutputStream;
 | |
| import java.io.PrintWriter;
 | |
| import java.io.BufferedWriter;
 | |
| import java.io.IOException;
 | |
| 
 | |
| /*
 | |
|  *	This class uses SANE to scan an image.
 | |
|  */
 | |
| public class ScanIt implements ImageProducer
 | |
|     {
 | |
| 					// # lines we incr. image height.
 | |
|     private static final int STRIP_HEIGHT = 256;
 | |
|     private Sane sane;
 | |
|     private int handle = 0;		// SANE device handle.
 | |
|     private Vector consumers = new Vector();
 | |
| 					// File to write to.
 | |
|     private OutputStream outputFile = null;
 | |
|     private SaneParameters parms = new SaneParameters();
 | |
|     private ColorModel cm;		// RGB color model.
 | |
|     private int width, height;		// Dimensions.
 | |
|     private int x, y;			// Position.
 | |
|     private int image[] = null;		// Image that we build as we scan.
 | |
|     private int offset;			// Offset within image in pixels if
 | |
| 					//   doing separate frames, bytes
 | |
| 					//   (3/word) if RBG.
 | |
| 
 | |
| 	/*
 | |
| 	 *	Tell consumers our status.  The scan is also terminated de-
 | |
| 	 *	pending on the status.
 | |
| 	 */
 | |
|     private void tellStatus(int s)
 | |
| 	{
 | |
| 	Enumeration next = consumers.elements();
 | |
| 	while (next.hasMoreElements())
 | |
| 		{
 | |
| 		ImageConsumer ic = (ImageConsumer) next.nextElement();
 | |
| 		ic.imageComplete(s);
 | |
| 		}
 | |
| 					// Done?  Stop scan.
 | |
| 	if (s == ImageConsumer.STATICIMAGEDONE ||
 | |
| 	    s == ImageConsumer.IMAGEERROR)
 | |
| 		sane.cancel(handle);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Tell consumers the image size.
 | |
| 	 */
 | |
|     private void tellDimensions(int w, int h)
 | |
| 	{
 | |
| 	System.out.println("tellDimensions:  " + w + ", " + h);
 | |
| 	Enumeration next = consumers.elements();
 | |
| 	while (next.hasMoreElements())
 | |
| 		{
 | |
| 		ImageConsumer ic = (ImageConsumer) next.nextElement();
 | |
| 		ic.setDimensions(w, h);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Send pixels to the clients.
 | |
| 	 */
 | |
|     private void tellPixels(int x, int y, int w, int h)
 | |
| 	{
 | |
| /*
 | |
| 	System.out.println("image length=" + image.length);
 | |
| 	System.out.println("width=" + width);
 | |
| 	System.out.println("tellPixels:  x="+x +" y="+y + " w="+w
 | |
| 					+ " h="+h);
 | |
|  */
 | |
| 	Enumeration next = consumers.elements();
 | |
| 	while (next.hasMoreElements())
 | |
| 		{
 | |
| 		ImageConsumer ic = (ImageConsumer) next.nextElement();
 | |
| 		ic.setPixels(x, y, w, h, cm, image, 0, width);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Construct.
 | |
| 	 */
 | |
|     public ScanIt(Sane s, int hndl)
 | |
| 	{
 | |
| 	sane = s;
 | |
| 	handle = hndl;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Add a consumer.
 | |
| 	 */
 | |
|     public synchronized void addConsumer(ImageConsumer ic)
 | |
| 	{
 | |
| 	if (consumers.contains(ic))
 | |
| 		return;			// Already here.
 | |
| 	consumers.addElement(ic);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Is a consumer in the list?
 | |
| 	 */
 | |
|     public synchronized boolean isConsumer(ImageConsumer ic)
 | |
| 	{ return consumers.contains(ic); }
 | |
| 
 | |
| 	/*
 | |
| 	 *	Remove consumer.
 | |
| 	 */
 | |
|     public synchronized void removeConsumer(ImageConsumer ic)
 | |
| 	{ consumers.removeElement(ic); }
 | |
| 
 | |
| 	/*
 | |
| 	 *	Add a consumer and start scanning.
 | |
| 	 */
 | |
|     public void startProduction(ImageConsumer ic)
 | |
| 	{
 | |
| 	System.out.println("In startProduction()");
 | |
| 	addConsumer(ic);
 | |
| 	scan();
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Set file to write to.
 | |
| 	 */
 | |
|     public void setOutputFile(OutputStream o)
 | |
| 	{ outputFile = o; }
 | |
| 
 | |
| 	/*
 | |
| 	 *	Ignore this:
 | |
| 	 */
 | |
|     public void requestTopDownLeftRightResend(ImageConsumer ic)
 | |
| 	{  }
 | |
| 
 | |
| 	/*
 | |
| 	 *	Go to next line in image, reallocating if necessary.
 | |
| 	 */
 | |
|     private void nextLine()
 | |
| 	{
 | |
| 	x = 0;
 | |
| 	++y;
 | |
| 	if (y >= height || image == null)
 | |
| 		{			// Got to reallocate.
 | |
| 		int oldSize = image == null ? 0 : width*height;
 | |
| 		height += STRIP_HEIGHT;	// Add more lines.
 | |
| 		int newSize = width*height;
 | |
| 		int[] newImage = new int[newSize];
 | |
| 		int i;
 | |
| 		if (oldSize != 0)	// Copy old data.
 | |
| 			for (i = 0; i < oldSize; i++)
 | |
| 				newImage[i] = image[i];
 | |
| 					// Fill new pixels with 0's, setting
 | |
| 					//   alpha channel.
 | |
| 		for (i = oldSize; i < newSize; i++)
 | |
| 			newImage[i] = (255 << 24);
 | |
| 		image = newImage;
 | |
| 		System.out.println("nextLine:  newSize="+newSize);
 | |
| 					// Tell clients.
 | |
| 		tellDimensions(width, height);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Process a buffer of data.
 | |
| 	 */
 | |
|     private boolean process(byte[] data, int readLen)
 | |
| 	{
 | |
| 	int prevY = y > 0 ? y : 0;	// Save current Y-coord.
 | |
| 	int i;
 | |
| 	switch (parms.format)
 | |
| 		{
 | |
| 	case SaneParameters.FRAME_RED:
 | |
| 	case SaneParameters.FRAME_GREEN:
 | |
| 	case SaneParameters.FRAME_BLUE:
 | |
| 		System.out.println("Process RED, GREEN or BLUE");
 | |
| 		int cindex = 2 - (parms.format - SaneParameters.FRAME_RED);
 | |
| 					// Single frame.
 | |
| 		for (i = 0; i < readLen; ++i)
 | |
| 			{		// Doing a single color frame.
 | |
| 			image[offset + i] |=
 | |
| 				(((int) data[i]) & 0xff) << (8*cindex);
 | |
| 			++x;
 | |
| 			if (x >= width)
 | |
| 				nextLine();
 | |
| 			}
 | |
| 		break;
 | |
| 	case SaneParameters.FRAME_RGB:
 | |
| 		for (i = 0; i < readLen; ++i)
 | |
| 			{
 | |
| 			int b = 2 - (offset + i)%3;
 | |
| 			image[(offset + i)/3] |=
 | |
| 				(((int) data[i]) & 0xff) << (8*b);
 | |
| 			if (b == 0)
 | |
| 				{
 | |
| 				++x;
 | |
| 				if (x >= width)
 | |
| 					nextLine();
 | |
| 				}
 | |
| 			}
 | |
| 		break;
 | |
| 	case SaneParameters.FRAME_GRAY:
 | |
| 		System.out.println("Process GREY");
 | |
| 					// Single frame.
 | |
| 		for (i = 0; i < readLen; ++i)
 | |
| 			{
 | |
| 			int v = ((int) data[i]) & 0xff;
 | |
| 			image[offset + i] |= (v<<16) | (v<<8) | (v);
 | |
| 			++x;
 | |
| 			if (x >= width)
 | |
| 				nextLine();
 | |
| 			}
 | |
| 		break;
 | |
| 		}
 | |
| 	offset += readLen;		// Update where we are.
 | |
| 					// Show it.
 | |
| 	System.out.println("PrevY = " + prevY + ", y = " + y);
 | |
| //	tellPixels(0, prevY, width, y - prevY);
 | |
| 	tellPixels(0, 0, width, height);
 | |
| 	return true;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Start scanning.
 | |
| 	 */
 | |
|     public void scan()
 | |
| 	{
 | |
| 	int dataLen = 32*1024;
 | |
| 	byte [] data = new byte[dataLen];
 | |
| 	int [] readLen = new int[1];
 | |
| 	int frameCnt = 0;
 | |
| 					// For now, use default RGB model.
 | |
| 	cm = ColorModel.getRGBdefault();
 | |
| 	int status;
 | |
| 	image = null;
 | |
| 	do				// Do each frame.
 | |
| 		{
 | |
| 		frameCnt++;
 | |
| 		x = 0;			// Init. position.
 | |
| 		y = -1;
 | |
| 		offset = 0;
 | |
| 		System.out.println("Reading frame #" + frameCnt);
 | |
| 		status = sane.start(handle);
 | |
| 		if (status != Sane.STATUS_GOOD)
 | |
| 			{
 | |
| 			System.out.println("start() failed.  Status= "
 | |
| 								+ status);
 | |
| 			tellStatus(ImageConsumer.IMAGEERROR);
 | |
| 			return;
 | |
| 			}
 | |
| 		status = sane.getParameters(handle, parms);
 | |
| 		if (status != Sane.STATUS_GOOD)
 | |
| 			{
 | |
| 			System.out.println("getParameters() failed.  Status= "
 | |
| 								+ status);
 | |
| 			tellStatus(ImageConsumer.IMAGEERROR);
 | |
| 			return;	//++++cleanup.
 | |
| 			}
 | |
| 		if (frameCnt == 1)	// First time?
 | |
| 			{
 | |
| 			width = parms.pixelsPerLine;
 | |
| 			if (parms.lines >= 0)
 | |
| 				height = parms.lines - STRIP_HEIGHT + 1;
 | |
| 			else		// Hand-scanner.
 | |
| 				height = 0;
 | |
| 			nextLine();	// Allocate image.
 | |
| 			}
 | |
| 		while ((status = sane.read(handle, data, dataLen, readLen))
 | |
| 							== Sane.STATUS_GOOD)
 | |
| 			{
 | |
| 			System.out.println("Read " + readLen[0] + " bytes.");
 | |
| 			if (!process(data, readLen[0]))
 | |
| 				{
 | |
| 				tellStatus(ImageConsumer.IMAGEERROR);
 | |
| 				return;
 | |
| 				}
 | |
| 			}
 | |
| 		if (status != Sane.STATUS_EOF)
 | |
| 			{
 | |
| 			System.out.println("read() failed.  Status= "
 | |
| 								+ status);
 | |
| 			tellStatus(ImageConsumer.IMAGEERROR);
 | |
| 			return;
 | |
| 			}
 | |
| 		}
 | |
| 	while (!parms.lastFrame);
 | |
| 	height = y;			// For now, send whole image here.
 | |
| 	tellDimensions(width, height);
 | |
| 	tellPixels(0, 0, width, height);
 | |
| 	if (outputFile != null)		// Write to file.
 | |
| 		{
 | |
| 		try
 | |
| 			{
 | |
| 			write(outputFile);
 | |
| 			}
 | |
| 		catch (IOException e)
 | |
| 			{	//+++++++++++++++
 | |
| 			System.out.println("I/O error writing file.");
 | |
| 			}
 | |
| 		outputFile = null;	// Clear for next time.
 | |
| 		}
 | |
| 	tellStatus(ImageConsumer.STATICIMAGEDONE);
 | |
| 	image = null;			// Allow buffer to be freed.
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 *	Write ppm/pnm output for last scan to a file.
 | |
| 	 */
 | |
|     private void write(OutputStream out) throws IOException
 | |
| 	{
 | |
| 	PrintWriter pout = new PrintWriter(out);
 | |
| 	BufferedWriter bout = new BufferedWriter(pout);
 | |
| 	int len = width*height;		// Get # of pixels.
 | |
| 	int i;
 | |
| 	switch (parms.format)
 | |
| 		{
 | |
| 	case SaneParameters.FRAME_RED:
 | |
| 	case SaneParameters.FRAME_GREEN:
 | |
| 	case SaneParameters.FRAME_BLUE:
 | |
| 	case SaneParameters.FRAME_RGB:
 | |
| 		pout.print("P6\n# SANE data follows\n" +
 | |
| 			width + ' ' + height + "\n255\n");
 | |
| 		for (i = 0; i < len; i++)
 | |
| 			{
 | |
| 			int pix = image[i];
 | |
| 			bout.write((pix >> 16) & 0xff);
 | |
| 			bout.write((pix >> 8) & 0xff);
 | |
| 			bout.write(pix & 0xff);
 | |
| 			}
 | |
| 		break;
 | |
| 	case SaneParameters.FRAME_GRAY:
 | |
| 		pout.print("P5\n# SANE data follows\n" +
 | |
| 			width + ' ' + height + "\n255\n");
 | |
| 		for (i = 0; i < len; i++)
 | |
| 			{
 | |
| 			int pix = image[i];
 | |
| 			bout.write(pix & 0xff);
 | |
| 			}
 | |
| 		break;
 | |
| 		}
 | |
| 
 | |
| 	bout.flush();			// Flush output.
 | |
| 	pout.flush();
 | |
| 	}
 | |
|     }
 |