/*
* MusicTag Copyright (C)2003,2004
*
* This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser
* General Public License as published by the Free Software Foundation; either version 2.1 of the License,
* or (at your option) any later version.
*
* This library 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License along with this library; if not,
* you can get a copy from http://www.opensource.org/licenses/lgpl-license.php or write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package org.jaudiotagger.tag.id3.framebody;
import org.jaudiotagger.tag.InvalidTagException;
import org.jaudiotagger.tag.datatype.NumberFixedLength;
import org.jaudiotagger.tag.datatype.NumberVariableLength;
import org.jaudiotagger.tag.id3.ID3v24Frames;
import java.nio.ByteBuffer;
/**
* Audio files with variable bit rates are intrinsically difficult to
* deal with in the case of seeking within the file. The ASPI frame
* makes seeking easier by providing a list a seek points within the
* audio file. The seek points are a fractional offset within the audio
* data, providing a starting point from which to find an appropriate
* point to start decoding. The presence of an ASPI frame requires the
* existence of a TLEN frame, indicating the duration of the file in
* milliseconds. There may only be one 'audio seek point index' frame in
* a tag.
* <p/>
* <Header for 'Seek Point Index', ID: "ASPI">
* Indexed data start (S) $xx xx xx xx
* Indexed data length (L) $xx xx xx xx
* Number of index points (N) $xx xx
* Bits per index point (b) $xx
* <p/>
* Then for every index point the following data is included;
* <p/>
* Fraction at index (Fi) $xx (xx)
* <p/>
* 'Indexed data start' is a byte offset from the beginning of the file.
* 'Indexed data length' is the byte length of the audio data being
* indexed. 'Number of index points' is the number of index points, as
* the name implies. The recommended number is 100. 'Bits per index
* point' is 8 or 16, depending on the chosen precision. 8 bits works
* well for short files (less than 5 minutes of audio), while 16 bits is
* advantageous for long files. 'Fraction at index' is the numerator of
* the fraction representing a relative position in the data. The
* denominator is 2 to the power of b.
* <p/>
* Here are the algorithms to be used in the calculation. The known data
* must be the offset of the start of the indexed data (S), the offset
* of the end of the indexed data (E), the number of index points (N),
* the offset at index i (Oi). We calculate the fraction at index i
* (Fi).
* <p/>
* Oi is the offset of the frame whose start is soonest after the point
* for which the time offset is (i/N * duration).
* <p/>
* The frame data should be calculated as follows:
* <p/>
* Fi = Oi/L * 2^b (rounded down to the nearest integer)
* <p/>
* Offset calculation should be calculated as follows from data in the
* frame:
* <p/>
* Oi = (Fi/2^b)*L (rounded up to the nearest integer)
*
* @author : Paul Taylor
* @author : Eric Farng
* @version $Id: FrameBodyASPI.java 832 2009-11-12 13:25:38Z paultaylor $
*/
public class FrameBodyASPI extends AbstractID3v2FrameBody implements ID3v24FrameBody
{
private static final int DATA_START_FIELD_SIZE = 4;
private static final int DATA_LENGTH_FIELD_SIZE = 4;
private static final int NO_OF_INDEX_POINTS_FIELD_SIZE = 2;
private static final int BITS_PER_INDEX_POINTS_FIELD_SIZE = 1;
private static final int FRACTION_AT_INDEX_MINIMUM_FIELD_SIZE = 1;
private static final String INDEXED_DATA_START = "IndexedDataStart";
private static final String INDEXED_DATA_LENGTH = "IndexedDataLength";
private static final String NUMBER_OF_INDEX_POINTS = "NumberOfIndexPoints";
private static final String BITS_PER_INDEX_POINT = "BitsPerIndexPoint";
private static final String FRACTION_AT_INDEX = "FractionAtIndex";
/**
* Creates a new FrameBodyASPI datatype.
*/
public FrameBodyASPI()
{
}
/**
* Creates a new FrameBodyASPI from another FrameBodyASPI
*
* @param copyObject
*/
public FrameBodyASPI(FrameBodyASPI copyObject)
{
super(copyObject);
}
/**
* Creates a new FrameBodyASPI datatype.
*
* @param byteBuffer
* @param frameSize
* @throws InvalidTagException
*/
public FrameBodyASPI(ByteBuffer byteBuffer, int frameSize) throws InvalidTagException
{
super(byteBuffer, frameSize);
}
/**
* The ID3v2 frame identifier
*
* @return the ID3v2 frame identifier for this frame type
*/
public String getIdentifier()
{
return ID3v24Frames.FRAME_ID_AUDIO_SEEK_POINT_INDEX;
}
protected void setupObjectList()
{
objectList.add(new NumberFixedLength(INDEXED_DATA_START, this, DATA_START_FIELD_SIZE));
objectList.add(new NumberFixedLength(INDEXED_DATA_LENGTH, this, DATA_LENGTH_FIELD_SIZE));
objectList.add(new NumberFixedLength(NUMBER_OF_INDEX_POINTS, this, NO_OF_INDEX_POINTS_FIELD_SIZE));
objectList.add(new NumberFixedLength(BITS_PER_INDEX_POINT, this, BITS_PER_INDEX_POINTS_FIELD_SIZE));
objectList.add(new NumberVariableLength(FRACTION_AT_INDEX, this, FRACTION_AT_INDEX_MINIMUM_FIELD_SIZE));
}
}
|