123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 |
- /* JPEGComponent.java --
- Copyright (C) 2005 Free Software Foundation, Inc.
- This file is part of GNU Classpath.
- GNU Classpath 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, or (at your option)
- any later version.
- GNU Classpath 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 GNU Classpath; see the file COPYING. If not, write to the
- Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- 02110-1301 USA.
- Linking this library statically or dynamically with other modules is
- making a combined work based on this library. Thus, the terms and
- conditions of the GNU General Public License cover the whole
- combination.
- As a special exception, the copyright holders of this library give you
- permission to link this library with independent modules to produce an
- executable, regardless of the license terms of these independent
- modules, and to copy and distribute the resulting executable under
- terms of your choice, provided that you also meet, for each linked
- independent module, the terms and conditions of the license of that
- module. An independent module is a module which is not derived from
- or based on this library. If you modify this library, you may extend
- this exception to your version of the library, but you are not
- obligated to do so. If you do not wish to do so, delete this
- exception statement from your version. */
- package gnu.javax.imageio.jpeg;
- import java.util.ArrayList;
- import java.io.IOException;
- import java.awt.image.WritableRaster;
- import javax.imageio.plugins.jpeg.JPEGHuffmanTable;
- /**
- * This class holds the methods to decode and write a component information to
- * a raster.
- */
- public class JPEGComponent
- {
- public byte factorH, factorV, component_id, quant_id;
- public int width = 0, height = 0, maxV = 0, maxH = 0;
- public HuffmanTable ACTable;
- public HuffmanTable DCTable;
- public int[] quantizationTable;
- public double previousDC = 0;
- ArrayList data = new ArrayList();
- /**
- * Initializes the component
- *
- * @param id
- * @param factorHorizontal
- * @param factorVertical
- * @param quantizationID
- */
- public JPEGComponent(byte id, byte factorHorizontal, byte factorVertical,
- byte quantizationID)
- {
- component_id = id;
- factorH = factorHorizontal;
- factorV = factorVertical;
- quant_id = quantizationID;
- }
- /**
- * If a restart marker is found with too little of an MCU count (i.e. our
- * Restart Interval is 63 and we have 61 we copy the last MCU until it's
- * full)
- *
- * @param index
- * @param length
- */
- public void padMCU(int index, int length)
- {
- double[] src = (double[]) data.get(index - 1);
- for (int i = 0; i < length; i++)
- data.add(index, src);
- }
- /**
- * Reset the interval by setting the previous DC value
- */
- public void resetInterval()
- {
- previousDC = 0;
- }
- /**
- * Run the Quantization backward method on all of the block data.
- */
- public void quantitizeData()
- {
- for (int i = 0; i < data.size(); i++)
- {
- double[] mydata = (double[]) data.get(i);
- for (int j = 0; j < mydata.length; j++)
- mydata[j] *= quantizationTable[j];
- }
- }
- public void setDCTable(JPEGHuffmanTable table)
- {
- DCTable = new HuffmanTable(table);
- }
- public void setACTable(JPEGHuffmanTable table)
- {
- ACTable = new HuffmanTable(table);
- }
- /**
- * Run the Inverse DCT method on all of the block data
- */
- public void idctData(DCT myDCT)
- {
- for (int i = 0; i < data.size(); i++)
- data.add(i,myDCT.fast_idct(ZigZag.decode8x8_map((double[]) data.remove(i))));
- }
- /**
- * This scales up the component size based on the factor size. This
- * calculates everyting up automatically so it's simply ran at the end of
- * the frame to normalize the size of all of the components.
- */
- public void scaleByFactors()
- {
- int factorUpVertical = maxV / factorV;
- int factorUpHorizontal = maxH / factorH;
- if (factorUpVertical > 1)
- {
- for (int i = 0; i < data.size(); i++)
- {
- double[][] src = (double[][]) data.remove(i);
- double[][] dest =
- new double[src.length * factorUpVertical][src[0].length];
- for (int j = 0; j < src.length; j++)
- {
- for (int u = 0; u < factorUpVertical; u++)
- {
- dest[j * factorUpVertical + u] = src[j];
- }
- }
- data.add(i, dest);
- }
- }
- if (factorUpHorizontal > 1)
- {
- for (int i = 0; i < data.size(); i++)
- {
- double[][] src = (double[][]) data.remove(i);
- double[][] dest =
- new double[src.length][src[0].length * factorUpHorizontal];
- for (int j = 0; j < src.length; j++)
- {
- for (int u = 0; u < src[0].length; u++)
- {
- for (int v = 0; v < factorUpHorizontal; v++)
- dest[j][u * factorUpHorizontal + v] = src[j][u];
- }
- }
- data.add(i, dest);
- }
- }
- }
- /**
- * This write the block of data to the raster throwing out anything that
- * spills over the raster width or height.
- *
- * @param raster
- * @param data
- * @param compIndex
- * @param x
- * @param y
- */
- public void writeBlock(WritableRaster raster, double[][] data,
- int compIndex, int x, int y)
- {
- for (int yIndex = 0; yIndex < data.length; yIndex++)
- {
- for (int xIndex = 0; xIndex < data[yIndex].length; xIndex++)
- {
- // The if statement is needed because blocks can spill over the
- // frame width because they are padded to make sure we keep the
- // height of the block the same as the width of the block
- if (x + xIndex < raster.getWidth()
- && y + yIndex < raster.getHeight())
- raster.setSample(x + xIndex, y + yIndex, compIndex,
- data[yIndex][xIndex]);
- }
- }
- }
- /**
- * This writes data to a raster block, so really it's reading not writing
- * but it writes the data to the raster block by factor size in a zig zag
- * fashion. This has the helper function writeBlock which does the actual
- * writing.
- *
- * @param raster
- * @param componentIndex
- */
- public void writeData(WritableRaster raster, int componentIndex)
- {
- int x = 0, y = 0, lastblockheight = 0, incrementblock = 0;
- // Keep looping through all of the blocks until there are no more.
- while(data.size() > 0)
- {
- int blockwidth = 0;
- int blockheight = 0;
- if (x >= raster.getWidth())
- {
- x = 0;
- y += incrementblock;
- }
- // Loop through the horizontal component blocks of the MCU first
- // then for each horizontal line write out all of the vertical
- // components
- for (int factorVIndex = 0; factorVIndex < factorV; factorVIndex++)
- {
- blockwidth = 0;
- for (int factorHIndex = 0; factorHIndex < factorH; factorHIndex++)
- {
- // Captures the width of this block so we can increment the
- // X coordinate
- double[][] blockdata = (double[][]) data.remove(0);
- // Writes the data at the specific X and Y coordinate of
- // this component
- writeBlock(raster, blockdata, componentIndex, x, y);
- blockwidth += blockdata[0].length;
- x += blockdata[0].length;
- blockheight = blockdata.length;
- }
- y += blockheight;
- x -= blockwidth;
- lastblockheight += blockheight;
- }
- y -= lastblockheight;
- incrementblock = lastblockheight;
- lastblockheight = 0;
- x += blockwidth;
- }
- }
- /**
- * Set the quantization table for this component.
- *
- * @param quanttable
- */
- public void setQuantizationTable(int[] quanttable)
- {
- quantizationTable = quanttable;
- }
- /**
- * Read in a partial MCU for this component
- *
- * @param stream TODO
- * @throws JPEGException TODO
- * @throws IOException TODO
- */
- public void readComponentMCU(JPEGImageInputStream stream)
- throws JPEGException, IOException
- {
- for (int i = 0; i < factorH * factorV; i++)
- {
- double dc = decode_dc_coefficient(stream);
- double[] datablock = decode_ac_coefficients(stream);
- datablock[0] = dc;
- data.add(datablock);
- }
- }
- /**
- * Generated from text on F-22, F.2.2.1 - Huffman decoding of DC
- * coefficients on ISO DIS 10918-1. Requirements and Guidelines.
- *
- * @param JPEGStream TODO
- *
- * @return TODO
- * @throws JPEGException TODO
- * @throws IOException TODO
- */
- public double decode_dc_coefficient(JPEGImageInputStream JPEGStream)
- throws JPEGException, IOException
- {
- int t = DCTable.decode(JPEGStream);
- double diff = JPEGStream.readBits(t);
- diff = HuffmanTable.extend((int) diff, t);
- diff = (previousDC + diff);
- previousDC = diff;
- return diff;
- }
- /**
- * Generated from text on F-23, F.13 - Huffman decoded of AC coefficients
- * on ISO DIS 10918-1. Requirements and Guidelines.
- *
- * @param JPEGStream TODO
- * @return TODO
- *
- * @throws JPEGException TODO
- * @throws IOException TODO
- */
- public double[] decode_ac_coefficients(JPEGImageInputStream JPEGStream)
- throws JPEGException, IOException
- {
- double[] zz = new double[64];
- for (int k = 1; k < 64; k++)
- {
- int s = ACTable.decode(JPEGStream);
- int r = s >> 4;
- s &= 15;
- if (s != 0)
- {
- k += r;
- r = (int) JPEGStream.readBits(s);
- s = HuffmanTable.extend(r, s);
- zz[k] = s;
- }
- else
- {
- if (r != 15)
- return (zz);
- k += 15;
- }
- }
- return zz;
- }
- }
|