Skip to content

Instantly share code, notes, and snippets.

@nsdevaraj
Created May 27, 2010 12:08
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save nsdevaraj/415729 to your computer and use it in GitHub Desktop.
Save nsdevaraj/415729 to your computer and use it in GitHub Desktop.
package
{
import flash.display.BitmapData;
import flash.filesystem.File;
import flash.filesystem.FileMode;
import flash.filesystem.FileStream;
import flash.utils.ByteArray;
public class SimpleFlvWriter
{
static private var _instance:SimpleFlvWriter;
private var frameWidth:int;
private var frameHeight:int;
private var frameRate:Number;
private var duration:Number;
private var fs:FileStream = new FileStream();
private var f:File;
public const blockWidth:int = 32;
public const blockHeight:int = 32;
private var previousTagSize:uint = 0;
private var iteration:int = 0;
private var bmp:BitmapData;
public static function getInstance():SimpleFlvWriter
{
if(SimpleFlvWriter._instance == null)
SimpleFlvWriter._instance = new SimpleFlvWriter(new SingletonEnforcer());
return SimpleFlvWriter._instance;
}
public function SimpleFlvWriter(singletonEnforcer:SingletonEnforcer)
{
}
public function createFile(pFile:File, pWidth:int, pHeight:int, pFramesPerSecond:Number, pDurationInSeconds:Number=0):void
{
/*
Parameters:
pFile: The file which will be created and written to
pWidth: Video height
pWidth: Video width
pFramesPerSecond: Determines framerate of created video
pDurationInSeconds: Duration of video file to be created. Used for metadata only. Optional.
*/
frameWidth = pWidth;
frameHeight = pHeight;
frameRate = pFramesPerSecond;
duration = pDurationInSeconds;
f = pFile;
fs.openAsync(f, FileMode.WRITE);
// create header
fs.writeBytes( header() );
// create metadata tag
fs.writeUnsignedInt( previousTagSize );
fs.writeBytes( flvTagOnMetaData() );
}
public function saveFrameByte(bmp:BitmapData):void
{
fs.writeUnsignedInt( previousTagSize );
fs.writeBytes( flvTagByteVideo(videoDataByte(bmp)) );
}
public function videoDataByte(bmp:BitmapData):ByteArray
{
var v:ByteArray = new ByteArray;
v.writeByte(0x13);
writeUI4_12(v, int(blockWidth/16) - 1, frameWidth);
writeUI4_12(v, int(blockHeight/16) - 1, frameHeight);
var yMax:int = int(frameHeight/blockHeight);
var yRemainder:int = frameHeight % blockHeight;
if (yRemainder > 0) yMax += 1;
var xMax:int = int(frameWidth/blockWidth);
var xRemainder:int = frameWidth % blockWidth;
if (xRemainder > 0) xMax += 1;
for (var y1:int = 0; y1 < yMax; y1++)
{
for (var x1:int = 0; x1 < xMax; x1++)
{
// create block
var block:ByteArray = new ByteArray();
var yLimit:int = blockHeight;
if (yRemainder > 0 && y1 + 1 == yMax) yLimit = yRemainder;
for (var y2:int = 0; y2 < yLimit; y2++)
{
var xLimit:int = blockWidth;
if (xRemainder > 0 && x1 + 1 == xMax) xLimit = xRemainder;
for (var x2:int = 0; x2 < xLimit; x2++)
{
var px:int = (x1 * blockWidth) + x2;
var py:int = frameHeight - ((y1 * blockHeight) + y2); // (flv's save from bottom to top)
var p:uint = bmp.getPixel(px, py);
block.writeByte( p & 0xff ); // blue
block.writeByte( p >> 8 & 0xff ); // green
block.writeByte( p >> 16 ); // red
}
}
block.compress();
writeUI16(v, block.length); // write block length (UI16)
v.writeBytes( block ); // write block
}
}
return v;
}
private function flvTagByteVideo(dat:ByteArray):ByteArray
{
var tag:ByteArray = new ByteArray();
var timeStamp:uint = uint(1000/frameRate * iteration++);
// tag 'header'
tag.writeByte( 0x09 ); // tagType = video
writeUI24(tag, dat.length); // data size
writeUI24(tag, timeStamp); // timestamp in ms
tag.writeByte(0); // timestamp extended, not using ***
writeUI24(tag, 0); // streamID always 0
// videodata
tag.writeBytes( dat );
previousTagSize = tag.length;
return tag;
}
public function closeFile():void
{
fs.close();
}
private function header():ByteArray
{
var ba:ByteArray = new ByteArray();
ba.writeByte(0x46) // 'F'
ba.writeByte(0x4C) // 'L'
ba.writeByte(0x56) // 'V'
ba.writeByte(0x01) // Version 1
ba.writeByte(0x01) // misc flags - video stream only
ba.writeUnsignedInt(0x09) // header length
return ba;
}
private function flvTagOnMetaData():ByteArray
{
var tag:ByteArray = new ByteArray();
var dat:ByteArray = metaData();
// tag 'header'
tag.writeByte( 18 ); // tagType = script data
writeUI24(tag, dat.length); // data size
writeUI24(tag, 0); // timestamp should be 0 for onMetaData tag
tag.writeByte(0); // timestamp extended
writeUI24(tag, 0); // streamID always 0
// data tag
tag.writeBytes( dat );
previousTagSize = tag.length;
return tag;
}
private function metaData():ByteArray
{
// onMetaData info goes in a ScriptDataObject of data type 'ECMA Array'
var b:ByteArray = new ByteArray();
// ObjectNameType (always 2)
b.writeByte(2);
// ObjectName (type SCRIPTDATASTRING):
writeUI16(b, "onMetaData".length); // StringLength
b.writeUTFBytes( "onMetaData" ); // StringData
// ObjectData (type SCRIPTDATAVALUE):
b.writeByte(8); // Type (ECMA array = 8)
b.writeUnsignedInt(7) // // Elements in array
// SCRIPTDATAVARIABLES...
if (duration > 0) {
writeUI16(b, "duration".length);
b.writeUTFBytes("duration");
b.writeByte(0);
b.writeDouble(duration);
}
writeUI16(b, "width".length);
b.writeUTFBytes("width");
b.writeByte(0);
b.writeDouble(frameWidth);
writeUI16(b, "height".length);
b.writeUTFBytes("height");
b.writeByte(0);
b.writeDouble(frameHeight);
writeUI16(b, "framerate".length);
b.writeUTFBytes("framerate");
b.writeByte(0);
b.writeDouble(frameRate);
writeUI16(b, "videocodecid".length);
b.writeUTFBytes("videocodecid");
b.writeByte(0);
b.writeDouble(3); // 'Screen Video' = 3
writeUI16(b, "canSeekToEnd".length);
b.writeUTFBytes("canSeekToEnd");
b.writeByte(1);
b.writeByte(int(true));
var mdc:String = "SimpleFLVWriter.as v0.8 zeropointnine.com";
writeUI16(b, "metadatacreator".length);
b.writeUTFBytes("metadatacreator");
b.writeByte(2);
writeUI16(b, mdc.length);
b.writeUTFBytes(mdc);
// VariableEndMarker1 (type UI24 - always 9)
writeUI24(b, 9);
return b;
}
private function writeUI24(stream:*, p:uint):void
{
var byte1:int = p >> 16;
var byte2:int = p >> 8 & 0xff;
var byte3:int = p & 0xff;
stream.writeByte(byte1);
stream.writeByte(byte2);
stream.writeByte(byte3);
}
public function writeUI16(stream:*, p:uint):void
{
stream.writeByte( p >> 8 )
stream.writeByte( p & 0xff );
}
public function writeUI4_12(stream:*, p1:uint, p2:uint):void
{
// writes a 4-bit value followed by a 12-bit value in two sequential bytes
var byte1a:int = p1 << 4;
var byte1b:int = p2 >> 8;
var byte1:int = byte1a + byte1b;
var byte2:int = p2 & 0xff;
stream.writeByte(byte1);
stream.writeByte(byte2);
}
}
}
class SingletonEnforcer {}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment