/*
* Copyright 2002-2017 Drew Noakes
*
* Licensed 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.
*
* More information about this project is available at:
*
* https://drewnoakes.com/code/exif/
* https://github.com/drewnoakes/metadata-extractor
*/
package com.drew.imaging.jpeg;
import com.drew.lang.annotations.NotNull;
import com.drew.lang.annotations.Nullable;
import java.util.*;
/**
* Holds a collection of JPEG data segments. This need not necessarily be all segments
* within the JPEG. For example, it may be convenient to store only the non-image
* segments when analysing metadata.
*
* Segments are keyed via their {@link JpegSegmentType}. Where multiple segments use the
* same segment type, they will all be stored and available.
*
* Each segment type may contain multiple entries. Conceptually the model is:
* Map<JpegSegmentType, Collection<byte[]>>
. This class provides
* convenience methods around that structure.
*
* @author Drew Noakes https://drewnoakes.com
*/
public class JpegSegmentData
{
// TODO key this on JpegSegmentType rather than Byte, and hopefully lose much of the use of 'byte' with this class
@NotNull
private final HashMap> _segmentDataMap = new HashMap>(10);
/**
* Adds segment bytes to the collection.
*
* @param segmentType the type of the segment being added
* @param segmentBytes the byte array holding data for the segment being added
*/
@SuppressWarnings({"MismatchedQueryAndUpdateOfCollection"})
public void addSegment(byte segmentType, @NotNull byte[] segmentBytes)
{
getOrCreateSegmentList(segmentType).add(segmentBytes);
}
/**
* Gets the set of JPEG segment type identifiers.
*/
public Iterable getSegmentTypes()
{
Set segmentTypes = new HashSet();
for (Byte segmentTypeByte : _segmentDataMap.keySet())
{
JpegSegmentType segmentType = JpegSegmentType.fromByte(segmentTypeByte);
if (segmentType == null) {
throw new IllegalStateException("Should not have a segmentTypeByte that is not in the enum: " + Integer.toHexString(segmentTypeByte));
}
segmentTypes.add(segmentType);
}
return segmentTypes;
}
/**
* Gets the first JPEG segment data for the specified type.
*
* @param segmentType the JpegSegmentType for the desired segment
* @return a byte[] containing segment data or null if no data exists for that segment
*/
@Nullable
public byte[] getSegment(byte segmentType)
{
return getSegment(segmentType, 0);
}
/**
* Gets the first JPEG segment data for the specified type.
*
* @param segmentType the JpegSegmentType for the desired segment
* @return a byte[] containing segment data or null if no data exists for that segment
*/
@Nullable
public byte[] getSegment(@NotNull JpegSegmentType segmentType)
{
return getSegment(segmentType.byteValue, 0);
}
/**
* Gets segment data for a specific occurrence and type. Use this method when more than one occurrence
* of segment data for a given type exists.
*
* @param segmentType identifies the required segment
* @param occurrence the zero-based index of the occurrence
* @return the segment data as a byte[], or null if no segment exists for the type & occurrence
*/
@Nullable
public byte[] getSegment(@NotNull JpegSegmentType segmentType, int occurrence)
{
return getSegment(segmentType.byteValue, occurrence);
}
/**
* Gets segment data for a specific occurrence and type. Use this method when more than one occurrence
* of segment data for a given type exists.
*
* @param segmentType identifies the required segment
* @param occurrence the zero-based index of the occurrence
* @return the segment data as a byte[], or null if no segment exists for the type & occurrence
*/
@Nullable
public byte[] getSegment(byte segmentType, int occurrence)
{
final List segmentList = getSegmentList(segmentType);
return segmentList != null && segmentList.size() > occurrence
? segmentList.get(occurrence)
: null;
}
/**
* Returns all instances of a given JPEG segment. If no instances exist, an empty sequence is returned.
*
* @param segmentType a number which identifies the type of JPEG segment being queried
* @return zero or more byte arrays, each holding the data of a JPEG segment
*/
@NotNull
public Iterable getSegments(@NotNull JpegSegmentType segmentType)
{
return getSegments(segmentType.byteValue);
}
/**
* Returns all instances of a given JPEG segment. If no instances exist, an empty sequence is returned.
*
* @param segmentType a number which identifies the type of JPEG segment being queried
* @return zero or more byte arrays, each holding the data of a JPEG segment
*/
@NotNull
public Iterable getSegments(byte segmentType)
{
final List segmentList = getSegmentList(segmentType);
return segmentList == null ? new ArrayList() : segmentList;
}
@Nullable
private List getSegmentList(byte segmentType)
{
return _segmentDataMap.get(segmentType);
}
@NotNull
private List getOrCreateSegmentList(byte segmentType)
{
List segmentList;
if (_segmentDataMap.containsKey(segmentType)) {
segmentList = _segmentDataMap.get(segmentType);
} else {
segmentList = new ArrayList();
_segmentDataMap.put(segmentType, segmentList);
}
return segmentList;
}
/**
* Returns the count of segment data byte arrays stored for a given segment type.
*
* @param segmentType identifies the required segment
* @return the segment count (zero if no segments exist).
*/
public int getSegmentCount(@NotNull JpegSegmentType segmentType)
{
return getSegmentCount(segmentType.byteValue);
}
/**
* Returns the count of segment data byte arrays stored for a given segment type.
*
* @param segmentType identifies the required segment
* @return the segment count (zero if no segments exist).
*/
public int getSegmentCount(byte segmentType)
{
final List segmentList = getSegmentList(segmentType);
return segmentList == null ? 0 : segmentList.size();
}
/**
* Removes a specified instance of a segment's data from the collection. Use this method when more than one
* occurrence of segment data exists for a given type exists.
*
* @param segmentType identifies the required segment
* @param occurrence the zero-based index of the segment occurrence to remove.
*/
@SuppressWarnings({"MismatchedQueryAndUpdateOfCollection"})
public void removeSegmentOccurrence(@NotNull JpegSegmentType segmentType, int occurrence)
{
removeSegmentOccurrence(segmentType.byteValue, occurrence);
}
/**
* Removes a specified instance of a segment's data from the collection. Use this method when more than one
* occurrence of segment data exists for a given type exists.
*
* @param segmentType identifies the required segment
* @param occurrence the zero-based index of the segment occurrence to remove.
*/
@SuppressWarnings({"MismatchedQueryAndUpdateOfCollection"})
public void removeSegmentOccurrence(byte segmentType, int occurrence)
{
final List segmentList = _segmentDataMap.get(segmentType);
segmentList.remove(occurrence);
}
/**
* Removes all segments from the collection having the specified type.
*
* @param segmentType identifies the required segment
*/
public void removeSegment(@NotNull JpegSegmentType segmentType)
{
removeSegment(segmentType.byteValue);
}
/**
* Removes all segments from the collection having the specified type.
*
* @param segmentType identifies the required segment
*/
public void removeSegment(byte segmentType)
{
_segmentDataMap.remove(segmentType);
}
/**
* Determines whether data is present for a given segment type.
*
* @param segmentType identifies the required segment
* @return true if data exists, otherwise false
*/
public boolean containsSegment(@NotNull JpegSegmentType segmentType)
{
return containsSegment(segmentType.byteValue);
}
/**
* Determines whether data is present for a given segment type.
*
* @param segmentType identifies the required segment
* @return true if data exists, otherwise false
*/
public boolean containsSegment(byte segmentType)
{
return _segmentDataMap.containsKey(segmentType);
}
}