root/libcore/parser/filter_factory.cpp

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. read
  2. read
  3. read
  4. read
  5. read
  6. read
  7. read
  8. read
  9. read

// 
//   Copyright (C) 2007, 2008, 2009, 2010, 2011 Free Software Foundation, Inc.
// 
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 3 of the License, or
// (at your option) any later version.
// 
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
// 
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA


#include "filter_factory.h"

#include "log.h"
#include "SWFStream.h"
#include "Filters.h"

namespace gnash {

typedef enum
{
    DROP_SHADOW = 0,
    BLUR = 1,
    GLOW = 2,
    BEVEL = 3,
    GRADIENT_GLOW = 4,
    CONVOLUTION = 5,
    COLOR_MATRIX = 6,
    GRADIENT_BEVEL = 7
} filter_types;

int
filter_factory::read(SWFStream& in, bool read_multiple, Filters* store)
{
    //GNASH_REPORT_FUNCTION;

    int count = 1;

    if (read_multiple)
    {
        in.ensureBytes(1);
        count = static_cast<int> (in.read_u8());
    }

    IF_VERBOSE_PARSE(
    log_parse("   number of filters: %d", count);
    );

    for (int i = 0; i < count; ++i)
    {
        BitmapFilter *the_filter = NULL;

        in.ensureBytes(1);
        filter_types filter_type = static_cast<filter_types> (in.read_u8());

        switch (filter_type)
        {
            case DROP_SHADOW:
                the_filter = new DropShadowFilter;
                break;
            case BLUR:
                the_filter = new BlurFilter; 
                break;
            case GLOW:
                the_filter = new GlowFilter;
                break;
            case BEVEL:
                the_filter = new BevelFilter;
                break;
            case GRADIENT_GLOW:
                the_filter = new GradientGlowFilter;
                break;
            case CONVOLUTION:
                the_filter = new ConvolutionFilter;
                break;
            case COLOR_MATRIX:
                the_filter = new ColorMatrixFilter;
                break;
            case GRADIENT_BEVEL:
                the_filter = new GradientBevelFilter;
                break;
            default:
                IF_VERBOSE_MALFORMED_SWF(
                log_swferror(_("Invalid filter type %d."), filter_type);
                );
                return i; // We're already broken, no need to cause more pain.
        }

        // Protect against exceptions and such by storing before we read.
        boost::shared_ptr<BitmapFilter> p(the_filter);
        if (!p->read(in))
        {
            IF_VERBOSE_MALFORMED_SWF(
            log_swferror(_("Filter %d could not read."), filter_type);
            );
            return i; // We're already broken.
        }
        store->push_back(p);
    }

    return count;
}

bool DropShadowFilter::read(SWFStream& in)
{
    in.ensureBytes(4 + 8 + 8 + 2 + 1);

    m_color = in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8());
    m_alpha = in.read_u8();

    m_blurX = in.read_fixed();
    m_blurY = in.read_fixed();

    m_angle = in.read_fixed();
    m_distance = in.read_fixed();

    m_strength = in.read_short_sfixed();

    m_inner = in.read_bit(); 
    m_knockout = in.read_bit(); 
    m_hideObject = in.read_bit(); 

    static_cast<void> (in.read_uint(5)); // Throw these away on purpose.

    IF_VERBOSE_PARSE(
    log_parse("   DropShadowFilter: blurX=%f blurY=%f",
        m_blurX, m_blurY);
    );

    return true;
}

bool BlurFilter::read(SWFStream& in)
{
    in.ensureBytes(4 + 4 + 1);

    m_blurX = in.read_ufixed();
    m_blurY = in.read_ufixed();

    m_quality = static_cast<boost::uint8_t> (in.read_uint(5));

    static_cast<void> (in.read_uint(3)); // Throw these away.

    IF_VERBOSE_PARSE(
    log_parse("   BlurFilter: blurX=%f blurY=%f quality=%d",
        m_blurX, m_blurY, m_quality);
    );

    return true;
}

bool GlowFilter::read(SWFStream& in)
{
    //GNASH_REPORT_FUNCTION;

    in.ensureBytes(4 + 8 + 2 + 1);

    m_color = in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8());
    m_alpha = in.read_u8();

    m_blurX = in.read_fixed();
    m_blurY = in.read_fixed();

    m_strength = in.read_short_sfixed();

    m_inner = in.read_bit(); 
    m_knockout = in.read_bit(); 

    static_cast<void> (in.read_uint(6)); // Throw these away.

    IF_VERBOSE_PARSE(
    log_parse("   GlowFilter ");
    );

    return true;
}

bool BevelFilter::read(SWFStream& in)
{
    in.ensureBytes(4 + 4 + 8 + 8 + 2 + 1);

    // TODO: It is possible that the order of these two should be reversed.
    // highlight might come first. Find out for sure and then fix and remove
    // this comment.
    m_shadowColor = in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8());
    m_shadowAlpha = in.read_u8();

    m_highlightColor = in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8());
    m_highlightAlpha = in.read_u8();

    m_blurX = in.read_fixed();
    m_blurY = in.read_fixed();

    m_angle = in.read_fixed();
    m_distance = in.read_fixed();
    
    m_strength = in.read_short_sfixed();

    bool inner_shadow = in.read_bit(); 
    m_knockout = in.read_bit(); 
    in.read_bit();  // reserved ?
    bool on_top = in.read_bit(); 

    // Set the bevel type. top and inner is full, top is outer, inner is inner
    m_type = on_top ? (inner_shadow ? FULL_BEVEL : OUTER_BEVEL) : INNER_BEVEL;
    
    static_cast<void> (in.read_uint(4)); // Throw these away.

    IF_VERBOSE_PARSE(
    log_parse("   BevelFilter ");
    );

    return true;
}

bool GradientGlowFilter::read(SWFStream& in)
{
    in.ensureBytes(1);

    boost::uint8_t count = in.read_u8(); // How many colorings.

    m_colors.reserve(count);
    m_alphas.reserve(count);
    m_ratios.reserve(count);

    in.ensureBytes(count*5 + 8 + 8 + 2 + 1); 

    for (int i = 0; i < count; ++i)
    {
        m_colors.push_back(in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8()));
        m_alphas.push_back(in.read_u8());
    }

    for (int i = 0; i < count; ++i)
    {
        m_ratios.push_back(in.read_u8());
    }

    m_blurX = in.read_fixed();
    m_blurY = in.read_fixed();

    m_angle = in.read_fixed();
    m_distance = in.read_fixed();

    m_strength = in.read_short_sfixed();

    bool inner = in.read_bit();
    m_knockout = in.read_bit();
    in.read_bit(); // reserved ?
    bool outer = in.read_bit(); 

    m_type = outer ? (inner ? FULL_GLOW : OUTER_GLOW) : INNER_GLOW;

    m_quality = static_cast<boost::uint8_t> (in.read_uint(4));

    IF_VERBOSE_PARSE(
    log_parse("   GradientGlowFilter ");
    );

    return true;
}

bool
ConvolutionFilter::read(SWFStream& in)
{
    in.ensureBytes(2 + 8);

    _matrixX = in.read_u8(); // 1 byte
    _matrixY = in.read_u8(); // 1 byte

    _divisor = in.read_long_float(); // 4 bytes
    _bias = in.read_long_float(); // 4 bytes

    size_t SWFMatrixCount = _matrixX * _matrixY;

    in.ensureBytes(SWFMatrixCount*4 + 4 + 1);

    _matrix.reserve(SWFMatrixCount);
    for (size_t i = 0; i < SWFMatrixCount; ++i)
    {
        _matrix.push_back(in.read_long_float());
    }

    _color = in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8());
    _alpha = in.read_u8();

    static_cast<void> (in.read_uint(6)); // Throw away.

    _clamp = in.read_bit(); 
    _preserveAlpha = in.read_bit(); 

    IF_VERBOSE_PARSE(
    log_parse("   ConvolutionFilter ");
    );

    return true;
}

bool ColorMatrixFilter::read(SWFStream& in)
{
    in.ensureBytes(20 * 4); 

    m_matrix.reserve(20);
    for (int i = 0; i < 20; ++i)
    {
        m_matrix.push_back(in.read_long_float());
    }

    IF_VERBOSE_PARSE(
    log_parse("   ColorMatrixFilter: ");
    log_parse("     %g, %g, %g, %g, %g",
        m_matrix[0], m_matrix[1], m_matrix[2], m_matrix[3], m_matrix[4]);
    log_parse("     %g, %g, %g, %g, %g",
        m_matrix[5], m_matrix[6], m_matrix[7], m_matrix[8], m_matrix[9]);
    log_parse("     %g, %g, %g, %g, %g",
        m_matrix[10], m_matrix[11], m_matrix[12], m_matrix[13], m_matrix[14]);
    log_parse("     %g, %g, %g, %g, %g",
        m_matrix[15], m_matrix[16], m_matrix[17], m_matrix[18], m_matrix[19]);
    );

    return true;
}

bool GradientBevelFilter::read(SWFStream& in)
{
    in.ensureBytes(1);
    boost::uint8_t count = in.read_u8(); // How many colorings.

    in.ensureBytes(count*5 + 8 + 8 + 2 + 1);

    m_colors.reserve(count);
    m_alphas.reserve(count);
    m_ratios.reserve(count);
    for (int i = 0; i < count; ++i)
    {
        m_colors.push_back(in.read_u8() << (16 + in.read_u8()) << (8 + in.read_u8()));
        m_alphas.push_back(in.read_u8());
    }

    for (int i = 0; i < count; ++i)
    {
        m_ratios.push_back(in.read_u8());
    }

    m_blurX = in.read_fixed();
    m_blurY = in.read_fixed();

    m_angle = in.read_fixed();
    m_distance = in.read_fixed();

    m_strength = in.read_short_sfixed();

    bool inner = in.read_bit();
    m_knockout = in.read_bit();
    in.read_bit(); // reserved ?
    bool outer = in.read_bit();

    m_type = outer ? (inner ? FULL_BEVEL : OUTER_BEVEL) : INNER_BEVEL;

    m_quality = static_cast<boost::uint8_t> (in.read_uint(4));

    IF_VERBOSE_PARSE(
    log_parse("   GradientBevelFilter ");
    );

    return true;
}

} // gnash namespace

/* [<][>][^][v][top][bottom][index][help] */