org.jcodec.movtool.streaming.VirtualMovie Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jcodec Show documentation
Show all versions of jcodec Show documentation
Pure Java implementation of video/audio codecs and formats
package org.jcodec.movtool.streaming;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
/**
* This class is part of JCodec ( www.jcodec.org ) This software is distributed
* under FreeBSD License
*
* Virtual movie. A movie constructed on-the-fly from virtual track data.
*
* Generic muxing
*
* @author The JCodec project
*
*/
public abstract class VirtualMovie {
public MovieSegment[] chunks;
public MovieSegment headerChunk;
protected long size;
protected VirtualTrack[] tracks;
public VirtualMovie(VirtualTrack... tracks) throws IOException {
this.tracks = tracks;
}
protected void muxTracks() throws IOException {
List chch = new ArrayList();
VirtualPacket[] heads = new VirtualPacket[tracks.length], tails = new VirtualPacket[tracks.length];
for (int curChunk = 1;; curChunk++) {
int min = -1;
for (int i = 0; i < heads.length; i++) {
if (heads[i] == null) {
heads[i] = tracks[i].nextPacket();
if (heads[i] == null)
continue;
}
min = min == -1 || heads[i].getPts() < heads[min].getPts() ? i : min;
}
if (min == -1)
break;
chch.add(packetChunk(tracks[min], heads[min], curChunk, min, size));
if (heads[min].getDataLen() >= 0)
size += heads[min].getDataLen();
else
System.err.println("WARN: Negative frame data len!!!");
tails[min] = heads[min];
heads[min] = tracks[min].nextPacket();
}
headerChunk = headerChunk(chch, tracks, size);
size += headerChunk.getDataLen();
chunks = chch.toArray(new MovieSegment[0]);
}
protected abstract MovieSegment packetChunk(VirtualTrack track, VirtualPacket pkt, int chunkNo, int trackNo,
long pos);
protected abstract MovieSegment headerChunk(List chunks, VirtualTrack[] tracks, long dataSize)
throws IOException;
public void close() throws IOException {
for (VirtualTrack virtualTrack : tracks) {
virtualTrack.close();
}
}
public MovieSegment getPacketAt(long position) throws IOException {
if (position >= 0 && position < headerChunk.getDataLen())
return headerChunk;
for (int i = 0; i < chunks.length - 1; i++) {
if (chunks[i + 1].getPos() > position)
return chunks[i];
}
if (position < size)
return chunks[chunks.length - 1];
return null;
}
public MovieSegment getPacketByNo(int no) {
if (no > chunks.length)
return null;
if (no == 0)
return headerChunk;
return chunks[no - 1];
}
public long size() {
return size;
}
}