2

This is my first question here :D, first sorry about my english.

My question is basically how can i save a flash Movie Clip to FLV.

The Movie Clip is generated by users and it has various sounds and animations and i need to save an FLV to send it to Youtbue.

What have I tried: I found some question here about using an Alchemy Lib that I am using to grab the Movie Clip frame to frame and save it to Bitmap.

The Alchemy Lib converts those frames to FLV like a charm and supports saving chunks of sound using ByteArray.

In this situation my Problem is, how can i grab the sound of Movie Clip to send it to Alchemy Lib? I´ve tried using:

SoundMixer.computeSpectrum(sndData, false, 2);

Witch returns me a byteArray in sndData variable but is useless since it´s used for render Audio Wave forms on screen.

Thougth aboud using

Sound.extract();

but i believe the sound class is used only for one MP3 sound and I need to grab the mixed sounds generated by Movie Clip.

Is there another way to generate the FLV from a MovieClip?

Below some of my code:

I based my code under the tutorial that I found in this link: http://www.zeropointnine.com/blog/updated-flv-encoder-alchem/

private const OUTPUT_WIDTH:Number = 550;
private const OUTPUT_HEIGHT:Number = 400;
private const FLV_FRAMERATE:int = 24;
private var _baFlvEncoder:ByteArrayFlvEncoder;      
public var anime:MovieClip;


//Starts recording
public function startRecording()
{
    this.addEventListener(Event.ENTER_FRAME, enterFrame);

    //Initialize the Alchemy Lib
    _baFlvEncoder = new ByteArrayFlvEncoder(stage.frameRate);
    _baFlvEncoder.setVideoProperties(OUTPUT_WIDTH, OUTPUT_HEIGHT);
    _baFlvEncoder.setAudioProperties(FlvEncoder.SAMPLERATE_22KHZ);
    _baFlvEncoder.start();      

}
//Stops recording
public function stopRecording()
{
    this.removeEventListener(Event.ENTER_FRAME, enterFrame);
    _baFlvEncoder.updateDurationMetadata();

    // Save FLV file via FileReference
    var fileRef:FileReference = new FileReference();
    fileRef.save(_baFlvEncoder.byteArray, "test.flv");          

    _baFlvEncoder.kill();

}

//The Main Loop  activated by StartRecording
public function enterFrame(evt:Event)
{
    var bmpData:BitmapData = new  BitmapData(OUTPUT_WIDTH, OUTPUT_HEIGHT, false, 0xFFFFFFFF);
    bmpData.draw(anime);

    var sndData:ByteArray = new ByteArray();
    SoundMixer.computeSpectrum(sndData, false, 2);

    _baFlvEncoder.addFrame(bmpData, sndData);
    bmpData.dispose();


}
Lelis718
  • 627
  • 7
  • 16
  • Why do you need FLV specifically? Are you using the Flash IDE? If so, you can export the movie as an AVI (File | Export | "Export Movie..." ) and upload it to YouTube that way. – iND Jan 06 '12 at 19:09
  • At the moment there is no possibility to that because MovieClip has only soundTransform property but has no access to its' sound object. Maybe Adobe intentionaly didn't add that feature so no one would be able to create own swf2flv exporters in AS3. – Rytis Alekna Jan 07 '12 at 11:37
  • I have an idea how to do it hard way: it may be that sound data is stored withing MovieClip object, so you may try to serialize to byte array and then dig the byte code to find sound data and extract it :) – Rytis Alekna Jan 07 '12 at 17:20

1 Answers1

1

Done it! ( Well... almost :D )

It was complicated but since my problem was only the audio of MovieClip I've created a class that acts like a mixer.

The mixer is responsible to play all the sounds using a unique SampleDataEvent that mix the bytes of all my sounds. Also it generates a single sound data in ByteArray when i execute the startRecording funcion. Kind Complicated to explain but here is the code for the mixer:

/**** MySoundMixer initializations omitted ***/

//Generates the sound object to start the stream 
//for reproduce the mixed sounds. 
public function startStream()
{
    fullStreamSound= new Sound();
    fullStreamSoundData = new ByteArray();            
    this.fullStreamSoundChannel = this.fullStreamSound.play();
}

//Adds a sound in the soundlib 
//(See: MySound object for more details)
public function addSound(sound:MySound, key:String)
{
    sound.initialize();
    sounds.push({sound:sound, key:key});
}

//Play a sound in the sound lib
public function play(key)
{
    var founded:MySound = null;
    for (var i = 0; i < sounds.length; i++)
    {
        if (key == sounds[i].key)
        {
            founded = sounds[i].sound;
            break;
        }
    }
    if (founded != null)
    {
        founded.play();
    }
}

// The SampleDataEvent function to Play the sound and
// if recording is activated record the sound to fullStreamSoundData
public function processSampleData(event:SampleDataEvent)
{
    var pos = 0;

    var normValue:Number = 1 / this.sounds.length;
    while (pos < BUFFER)
    {

        var leftChannel:Number = 0;
        var rightChannel:Number = 0;

        for (var i = 0; i < this.sounds.length; i++)
        {
            var currentSound:MySound = this.sounds[i].sound;
            var result = currentSound.getSampleData();
            leftChannel += result.leftChannel * normValue;
            rightChannel += result.rightChannel * normValue;
        }

        event.data.writeFloat(leftChannel);
        event.data.writeFloat(rightChannel);

        if (isRecording)
        {
            fullStreamSoundData.writeFloat(leftChannel);
            fullStreamSoundData.writeFloat(rightChannel);
        }

        pos++;
    }
}

//Starts recording
public function startRecording()
{
    this.isRecording = true;
}

//Stops recording
public function stopRecording()
{
    this.isRecording = false;
}

The SampleDataEvent is used for play and extract the mixed sounds at the same time.

I also had to create a MySound class that extends the Sound object in order to extract the sampleData of each sound (method getSampleData()) at current buffer used by processSampleData method. The MySound class also starts playing when Mixer starts (sending 0 bytes for both channels) and will stop also when the mixer stops, it will only start to send the byte information of the music when play() function is called.

The class that i have created is something like this:

/**** MySound initializations omitted ***/
public function initialize()
{
    this.extractInformation(null);
}

//Override the play function to avoid playing.
//(The play act will be triggered by SoundMixer class)
override public function play(startTime:Number = 0, loops:int = 0, sndTransform:SoundTransform = null):SoundChannel 
{
    this.isPlaying = true;
    this.currentPhase = 0;
    return null;
}

// On each buffer in sampledata i read the chunk of sound bytes
public function getSampleData()
{
    var leftChannel:Number = 0;
    var rightChannel:Number = 0;
    if (this.isPlaying) {
        if (currentPhase < totalPhases)
        {
            this.soundData.position = currentPhase * 8;
            leftChannel = this.soundData.readFloat();
            rightChannel = this.soundData.readFloat();
            this.currentPhase ++;
        } else 
        {
            stopPlaying();
        }
    }
    return { leftChannel:leftChannel, rightChannel:rightChannel };
}

//Extracts information of the sound object in order to 
//split it in several chunks of sounds.
public function extractInformation(evt:Event)
{
    trace("Inicializando o som " + this.id3);
    this.soundData = new ByteArray();
    this.extract(soundData, int(this.length * SAMPLE_44HZ + 10000));
    this.totalPhases = this.soundData.length / 8;
    this.currentPhase = 0;
}

///Stop playing means stop extracting bytes
public function stopPlaying()
{
    this.isPlaying = false;
}

With that, i have generated a unique ByteArray object that contains the hole sound information of the mixer. all i had to do is start the mixer when the movieclip starts and stop it also when the movieclip stops. The ByteArray information with the sound object is passed addFrame(bitmapData, sndData) of Alchemy Lib that records it successfully.

It is working well in my project but i probably will need to optmize the code.

Thanks all guys that helped me!

Lelis718
  • 627
  • 7
  • 16