All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.googlecode.mp4parser.BasicContainer Maven / Gradle / Ivy

Go to download

A generic parser and writer for all ISO 14496 based files (MP4, Quicktime, DCF, PDCF, ...)

The newest version!
package com.googlecode.mp4parser;

import com.coremedia.iso.BoxParser;
import com.coremedia.iso.boxes.Box;
import com.coremedia.iso.boxes.Container;
import com.googlecode.mp4parser.util.LazyList;
import com.googlecode.mp4parser.util.Logger;

import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.EOFException;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.Channels;
import java.nio.channels.WritableByteChannel;
import java.util.*;

import static com.googlecode.mp4parser.util.CastUtils.l2i;

/**
 * Created by sannies on 18.05.13.
 */
public class BasicContainer implements Container, Iterator, Closeable {
    private static final Box EOF = new AbstractBox("eof ") {

        @Override
        protected long getContentSize() {
            return 0;
        }

        @Override
        protected void getContent(ByteBuffer byteBuffer) {
        }

        @Override
        protected void _parseDetails(ByteBuffer content) {
        }
    };
    private static Logger LOG = Logger.getLogger(BasicContainer.class);
    protected BoxParser boxParser;
    protected DataSource dataSource;
    Box lookahead = null;
    long parsePosition = 0;
    long startPosition = 0;
    long endPosition = 0;
    private List boxes = new ArrayList();

    public BasicContainer() {
    }

    public List getBoxes() {
        if (dataSource != null && lookahead != EOF) {
            return new LazyList(boxes, this);
        } else {
            return boxes;
        }
    }

    public void setBoxes(List boxes) {
        this.boxes = new ArrayList(boxes);
        this.lookahead = EOF;
        this.dataSource = null;
    }

    protected long getContainerSize() {
        long contentSize = 0;
        for (int i = 0; i < getBoxes().size(); i++) {
            // it's quicker to iterate an array list like that since no iterator
            // needs to be instantiated
            contentSize += boxes.get(i).getSize();
        }
        return contentSize;
    }

    @SuppressWarnings("unchecked")
    public  List getBoxes(Class clazz) {
        List boxesToBeReturned = null;
        T oneBox = null;
        List boxes = getBoxes();
        for (int i = 0; i < boxes.size(); i++) {
            Box boxe = boxes.get(i);
            //clazz.isInstance(boxe) / clazz == boxe.getClass()?
            // I hereby finally decide to use isInstance

            if (clazz.isInstance(boxe)) {
                if (oneBox == null) {
                    oneBox = (T) boxe;
                } else {
                    if (boxesToBeReturned == null) {
                        boxesToBeReturned = new ArrayList(2);
                        boxesToBeReturned.add(oneBox);
                    }
                    boxesToBeReturned.add((T) boxe);
                }
            }
        }
        if (boxesToBeReturned != null) {
            return boxesToBeReturned;
        } else if (oneBox != null) {
            return Collections.singletonList(oneBox);
        } else {
            return Collections.emptyList();
        }
    }

    @SuppressWarnings("unchecked")
    public  List getBoxes(Class clazz, boolean recursive) {
        List boxesToBeReturned = new ArrayList(2);
        List boxes = getBoxes();
        for (int i = 0; i < boxes.size(); i++) {
            Box boxe = boxes.get(i);
            //clazz.isInstance(boxe) / clazz == boxe.getClass()?
            // I hereby finally decide to use isInstance

            if (clazz.isInstance(boxe)) {
                boxesToBeReturned.add((T) boxe);
            }

            if (recursive && boxe instanceof Container) {
                boxesToBeReturned.addAll(((Container) boxe).getBoxes(clazz, recursive));
            }
        }
        return boxesToBeReturned;
    }

    /**
     * Add box to the container and sets the parent correctly. If box is null
     * nochange will be performed and no error thrown.
     *
     * @param box will be added to the container
     */
    public void addBox(Box box) {
        if (box != null) {
            boxes = new ArrayList(getBoxes());
            box.setParent(this);
            boxes.add(box);
        }
    }

    public void initContainer(DataSource dataSource, long containerSize, BoxParser boxParser) throws IOException {

        this.dataSource = dataSource;
        this.parsePosition = this.startPosition = dataSource.position();
        dataSource.position(dataSource.position() + containerSize);
        this.endPosition = dataSource.position();
        this.boxParser = boxParser;
    }

    public void remove() {
        throw new UnsupportedOperationException();
    }

    public boolean hasNext() {
        if (lookahead == EOF) {
            return false;
        }
        if (lookahead != null) {
            return true;
        } else {
            try {
                lookahead = next();
                return true;
            } catch (NoSuchElementException e) {
                lookahead = EOF;
                return false;
            }
        }
    }

    public Box next() {
        if (lookahead != null && lookahead != EOF) {
            Box b = lookahead;
            lookahead = null;
            return b;
        } else {
           // LOG.logDebug("Parsing next() box");
            if (dataSource == null || parsePosition >= endPosition) {
                lookahead = EOF;
                throw new NoSuchElementException();
            }

            try {
                synchronized (dataSource) {
                    dataSource.position(parsePosition);
                    Box b = boxParser.parseBox(dataSource, this);
                    //System.err.println(b.getType());
                    parsePosition = dataSource.position();
                    return b;
                }
            } catch (EOFException e) {
                throw new NoSuchElementException();
            } catch (IOException e) {
                throw new NoSuchElementException();
            }
        }

    }

    public String toString() {
        StringBuilder buffer = new StringBuilder();

        buffer.append(this.getClass().getSimpleName()).append("[");
        for (int i = 0; i < boxes.size(); i++) {
            if (i > 0) {
                buffer.append(";");
            }
            buffer.append(boxes.get(i).toString());
        }
        buffer.append("]");
        return buffer.toString();
    }


    public final void writeContainer(WritableByteChannel bb) throws IOException {
        for (Box box : getBoxes()) {
            box.getBox(bb);
        }
    }

    public ByteBuffer getByteBuffer(long rangeStart, long size) throws IOException {
        if (this.dataSource != null) {
            synchronized (this.dataSource) {
                return this.dataSource.map(this.startPosition + rangeStart, size);
            }
        } else {
            ByteBuffer out = ByteBuffer.allocate(l2i(size));
            long rangeEnd = rangeStart + size;
            long boxStart;
            long boxEnd = 0;
            for (Box box : boxes) {
                boxStart = boxEnd;
                boxEnd = boxStart + box.getSize();
                if (!(boxEnd <= rangeStart || boxStart >= rangeEnd)) {
                    ByteArrayOutputStream baos = new ByteArrayOutputStream();
                    WritableByteChannel wbc = Channels.newChannel(baos);
                    box.getBox(wbc);
                    wbc.close();

                    if (boxStart >= rangeStart && boxEnd <= rangeEnd) {
                        out.put(baos.toByteArray());
                        // within -> use full box
                    } else if (boxStart < rangeStart && boxEnd > rangeEnd) {
                        // around -> use 'middle' of box
                        int length = l2i(box.getSize() - (rangeStart - boxStart) - (boxEnd - rangeEnd));
                        out.put(baos.toByteArray(), l2i(rangeStart - boxStart), length);
                    } else if (boxStart < rangeStart && boxEnd <= rangeEnd) {
                        // endwith
                        int length = l2i(box.getSize() - (rangeStart - boxStart));
                        out.put(baos.toByteArray(), l2i(rangeStart - boxStart), length);
                    } else if (boxStart >= rangeStart && boxEnd > rangeEnd) {
                        int length = l2i(box.getSize() - (boxEnd - rangeEnd));
                        out.put(baos.toByteArray(), 0, length);
                    }
                }
            }
            return (ByteBuffer) out.rewind();
        }
    }

    public void close() throws IOException {
        dataSource.close();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy