/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.sanselan.formats.tiff;

import java.io.IOException;
import java.util.ArrayList;

import org.apache.sanselan.ImageReadException;
import org.apache.sanselan.common.byteSources.ByteSource;
import org.apache.sanselan.formats.tiff.constants.TagInfo;
import org.apache.sanselan.formats.tiff.constants.TiffConstants;

public class TiffDirectory extends TiffElement implements TiffConstants
//extends BinaryFileFunctions
{

	public String description()
	{
		return TiffDirectory.description(type);
	}

	public String getElementDescription(boolean verbose)
	{
		if (!verbose)
			return "TIFF Directory (" + description() + ")";

		int entryOffset = offset + TIFF_DIRECTORY_HEADER_LENGTH;

		StringBuffer result = new StringBuffer();
		for (int i = 0; i < entries.size(); i++)
		{
			TiffField entry = (TiffField) entries.get(i);

			result.append("\t");
			result.append("[" + entryOffset + "]: ");
			result.append(entry.tagInfo.name);
			result.append(" (" + entry.tag + ", 0x"
					+ Integer.toHexString(entry.tag) + ")");
			result.append(", " + entry.fieldType.name);
			result.append(", " + entry.fieldType.getRawBytes(entry).length);
			result.append(": " + entry.getValueDescription());

			result.append("\n");

			entryOffset += TIFF_ENTRY_LENGTH;
			//            entry.fillInValue(byteSource);
		}
		return result.toString();
	}

	public static final String description(int type)
	{
		switch (type)
		{
			case DIRECTORY_TYPE_UNKNOWN :
				return "Unknown";
			case DIRECTORY_TYPE_ROOT :
				return "Root";
			case DIRECTORY_TYPE_SUB :
				return "Sub";
			case DIRECTORY_TYPE_THUMBNAIL :
				return "Thumbnail";
			case DIRECTORY_TYPE_EXIF :
				return "Exif";
			case DIRECTORY_TYPE_GPS :
				return "Gps";
			case DIRECTORY_TYPE_INTEROPERABILITY :
				return "Interoperability";
			default :
				return "Bad Type";
		}
	}

	public final int type;
	public final ArrayList entries;
	//    public final int offset;
	public final int nextDirectoryOffset;

	public TiffDirectory(int type, ArrayList entries, final int offset,
			int nextDirectoryOffset)
	{
		super(offset, TIFF_DIRECTORY_HEADER_LENGTH + entries.size()
				* TIFF_ENTRY_LENGTH + TIFF_DIRECTORY_FOOTER_LENGTH);

		this.type = type;
		this.entries = entries;
		this.nextDirectoryOffset = nextDirectoryOffset;
	}

	public ArrayList getDirectoryEntrys()
	{
		return new ArrayList(entries);
	}

	protected void fillInValues(ByteSource byteSource)
			throws ImageReadException, IOException
	{
		for (int i = 0; i < entries.size(); i++)
		{
			TiffField entry = (TiffField) entries.get(i);

			entry.fillInValue(byteSource);
		}
	}

	public void dump()
	{
		for (int i = 0; i < entries.size(); i++)
		{
			TiffField entry = (TiffField) entries.get(i);
			entry.dump();
		}

	}

	public boolean hasJpegImageData() throws ImageReadException
	{
		if (null != findField(TIFF_TAG_JPEG_INTERCHANGE_FORMAT))
			return true;

		return false;
	}

	public boolean hasTiffImageData() throws ImageReadException
	{
		if (null != findField(TIFF_TAG_TILE_OFFSETS))
			return true;

		if (null != findField(TIFF_TAG_STRIP_OFFSETS))
			return true;

		return false;
	}

	public TiffField findField(TagInfo tag) throws ImageReadException
	{
		boolean failIfMissing = false;
		return findField(tag, failIfMissing);
	}

	public TiffField findField(TagInfo tag, boolean failIfMissing)
			throws ImageReadException
	{
		if (entries == null)
			return null;

		for (int i = 0; i < entries.size(); i++)
		{
			TiffField field = (TiffField) entries.get(i);
			if (field.tag == tag.tag)
				return field;
		}

		if (failIfMissing)
			throw new ImageReadException("Missing expected field: "
					+ tag.getDescription());

		return null;
	}

	public final class ImageDataElement extends TiffElement
	{
		public ImageDataElement(int offset, int length)
		{
			super(offset, length);
		}

		public String getElementDescription(boolean verbose)
		{
			if (verbose)
				return null;
			return "ImageDataElement";
		}
	}

	private ArrayList getRawImageDataElements(TiffField offsetsField,
			TiffField byteCountsField) throws ImageReadException
	{
		int offsets[] = offsetsField.getIntArrayValue();
		int byteCounts[] = byteCountsField.getIntArrayValue();

		if (offsets.length != byteCounts.length)
			throw new ImageReadException("offsets.length(" + offsets.length
					+ ") != byteCounts.length(" + byteCounts.length + ")");

		ArrayList result = new ArrayList();
		for (int i = 0; i < offsets.length; i++)
		{
			result.add(new ImageDataElement(offsets[i], byteCounts[i]));
		}
		return result;
	}

	public ArrayList getTiffRawImageDataElements() throws ImageReadException
	{
		TiffField tileOffsets = findField(TIFF_TAG_TILE_OFFSETS);
		TiffField tileByteCounts = findField(TIFF_TAG_TILE_BYTE_COUNTS);
		TiffField stripOffsets = findField(TIFF_TAG_STRIP_OFFSETS);
		TiffField stripByteCounts = findField(TIFF_TAG_STRIP_BYTE_COUNTS);

		if ((tileOffsets != null) && (tileByteCounts != null))
		{
			return getRawImageDataElements(tileOffsets, tileByteCounts);
		}
		else if ((stripOffsets != null) && (stripByteCounts != null))
		{
			return getRawImageDataElements(stripOffsets, stripByteCounts);
		}
		else
			throw new ImageReadException("Couldn't find image data.");
	}

	public boolean imageDataInStrips() throws ImageReadException
	{
		TiffField tileOffsets = findField(TIFF_TAG_TILE_OFFSETS);
		TiffField tileByteCounts = findField(TIFF_TAG_TILE_BYTE_COUNTS);
		TiffField stripOffsets = findField(TIFF_TAG_STRIP_OFFSETS);
		TiffField stripByteCounts = findField(TIFF_TAG_STRIP_BYTE_COUNTS);

		if ((tileOffsets != null) && (tileByteCounts != null))
			return false;
		else if ((stripOffsets != null) && (stripByteCounts != null))
			return true;
		else if ((stripOffsets != null) && (stripByteCounts != null))
			return true;
		else
			throw new ImageReadException("Couldn't find image data.");
	}

	public ImageDataElement getJpegRawImageDataElement()
			throws ImageReadException
	{
		TiffField jpegInterchangeFormat = findField(TIFF_TAG_JPEG_INTERCHANGE_FORMAT);
		TiffField jpegInterchangeFormatLength = findField(TIFF_TAG_JPEG_INTERCHANGE_FORMAT_LENGTH);

		if ((jpegInterchangeFormat != null)
				&& (jpegInterchangeFormatLength != null))
		{
			int offset = jpegInterchangeFormat.getIntArrayValue()[0];
			int byteCount = jpegInterchangeFormatLength.getIntArrayValue()[0];

			return new ImageDataElement(offset, byteCount);
		}
		else
			throw new ImageReadException("Couldn't find image data.");
	}

	private TiffImageData tiffImageData = null;

	public void setTiffImageData(TiffImageData rawImageData)
	{
		this.tiffImageData = rawImageData;
	}

	public TiffImageData getTiffImageData()
	{
		return tiffImageData;
	}

	private JpegImageData jpegImageData = null;

	public void setJpegImageData(JpegImageData value)
	{
		this.jpegImageData = value;
	}

	public JpegImageData getJpegImageData()
	{
		return jpegImageData;
	}

}