如何在Android中对非摄像机视频进行编码

时间:2011-09-13 22:35:04

标签: android ffmpeg video-encoding

我正在开发一个Android应用程序,其中通过合成一系列动画帧动态生成视频。我尝试使用Android Media Recorder API,但还没有找到一种方法让它接受非相机源作为输入。我一直在尝试使用FFMPEG端口(基于Rockplayer构建),但由于我将其用作编码器而不是解码器,因此遇到了缺少功能的困难。

此应用程序的iPhone版本使用AVFoundation框架中的AVAssetWriter。

是否有更简单的方法可以做到这一点,还是我坚持用FFMPEG阻止它?

3 个答案:

答案 0 :(得分:0)

这可能会有所帮助(请参阅有关决议的说明): -

How to encode using the FFMpeg in Android (using H263)

我不确定他们是否进行了ffmpeg的自定义构建,如果是这样的话,他们可能会提供有关移植更多功能完整版本的建议。

- 安东尼

答案 1 :(得分:0)

Opencv有ViewBase类,它将摄像机的输入作为一个帧并将帧表示为一个位图,你可以直接使用View类库并让它供你自己使用,即使在android上安装opencv不是很容易。

当您扩展SampleCvViewBase时,您将拥有以下功能,您可以使用相当多的功能,但我能想到的最好。

 @Override
protected Bitmap processFrame(VideoCapture capture) {


    capture.retrieve(picture, Highgui.CV_CAP_ANDROID_COLOR_FRAME_RGBA);



    if (Utils.matToBitmap(picture, bmp))
        return bmp;

    bmp.recycle();
    return null;


}

答案 2 :(得分:-1)

您可以使用名为JCodec(http://jcodec.org)的纯Java开源库。

它包含一个简单但可用的H.264编码器和MP4复用器。下面的类使用JCodec低级API,应该是您需要的(已更正):

    public class SequenceEncoder {
    private SeekableByteChannel ch;
    private Picture toEncode;
    private RgbToYuv420 transform;
    private H264Encoder encoder;
    private ArrayList<ByteBuffer> spsList;
    private ArrayList<ByteBuffer> ppsList;
    private CompressedTrack outTrack;
    private ByteBuffer _out;
    private int frameNo;
    private MP4Muxer muxer;

    public SequenceEncoder(File out) throws IOException {
        this.ch = NIOUtils.writableFileChannel(out);

        // Transform to convert between RGB and YUV
        transform = new RgbToYuv420(0, 0);

        // Muxer that will store the encoded frames
        muxer = new MP4Muxer(ch, Brand.MP4);

        // Add video track to muxer
        outTrack = muxer.addTrackForCompressed(TrackType.VIDEO, 25);

        // Allocate a buffer big enough to hold output frames
        _out = ByteBuffer.allocate(1920 * 1080 * 6);

        // Create an instance of encoder
        encoder = new H264Encoder();

        // Encoder extra data ( SPS, PPS ) to be stored in a special place of
        // MP4
        spsList = new ArrayList<ByteBuffer>();
        ppsList = new ArrayList<ByteBuffer>();

    }

    public void encodeImage(BufferedImage bi) throws IOException {
        if (toEncode == null) {
            toEncode = Picture.create(bi.getWidth(), bi.getHeight(), ColorSpace.YUV420);
        }

        // Perform conversion
        for (int i = 0; i < 3; i++)
            Arrays.fill(toEncode.getData()[i], 0);
        transform.transform(AWTUtil.fromBufferedImage(bi), toEncode);

        // Encode image into H.264 frame, the result is stored in '_out' buffer
        _out.clear();
        ByteBuffer result = encoder.encodeFrame(_out, toEncode);

        // Based on the frame above form correct MP4 packet
        spsList.clear();
        ppsList.clear();
        H264Utils.encodeMOVPacket(result, spsList, ppsList);

        // Add packet to video track
        outTrack.addFrame(new MP4Packet(result, frameNo, 25, 1, frameNo, true, null, frameNo, 0));

        frameNo++;
    }

    public void finish() throws IOException {
        // Push saved SPS/PPS to a special storage in MP4
        outTrack.addSampleEntry(H264Utils.createMOVSampleEntry(spsList, ppsList));

        // Write MP4 header and finalize recording
        muxer.writeHeader();
        NIOUtils.closeQuietly(ch);
    }

    public static void main(String[] args) throws IOException {
        SequenceEncoder encoder = new SequenceEncoder(new File("video.mp4"));
        for (int i = 1; i < 100; i++) {
            BufferedImage bi = ImageIO.read(new File(String.format("folder/img%08d.png", i)));
            encoder.encodeImage(bi);
        }
        encoder.finish();
    }
}

您可以从project web-site获取JCodec jar。