root/libcore/SWFMatrix.h

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

INCLUDED FROM


// 
//   Copyright (C) 2005, 2006, 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

// 
//
// Original author: Thatcher Ulrich <tu@tulrich.com> 2003
//
//

#ifndef GNASH_MATRIX_H
#define GNASH_MATRIX_H

#include "dsodefs.h" // for DSOEXPORT

#include <ostream> 
#include <boost/cstdint.hpp>

// Forward declarations
namespace gnash {
    class SWFRect;
    namespace geometry {
        class Point2d;
        template<typename T> class Range2d;
    }
}


namespace gnash {

/// The SWF SWFMatrix record.
/// 
/// Conceptually, it represents a 3*3 linear transformation SWFMatrix like this:
/// 
///   | scale_x       rotateSkew_y  translate_x |
///   | rotateSkey_x  scale_y       traslate_y  |
///   | 0             0             1           |
/// 
class DSOEXPORT SWFMatrix
{
public:

    /// Construct an identity SWFMatrix
    SWFMatrix()
        :
        _a(65536),
        _b(0),
        _c(0),
        _d(65536),
        _tx(0),
        _ty(0)
    {}

    /// Construct a SWFMatrix with all values.
    SWFMatrix(int a, int b, int c, int d, int x, int y)
        :
        _a(a),
        _b(b),
        _c(c),
        _d(d),
        _tx(x),
        _ty(y)
    {}

    boost::int32_t a() const {
        return _a;
    }

    boost::int32_t b() const {
        return _b;
    }

    boost::int32_t c() const {
        return _c;
    }

    boost::int32_t d() const {
        return _d;
    }

    boost::int32_t tx() const {
        return _tx;
    }

    boost::int32_t ty() const {
        return _ty;
    }

    /// Set the SWFMatrix to identity.
    void set_identity();

    /// Concatenate m's transform onto ours. 
    //
    /// When transforming points, m happens first,
    /// then our original xform.
    void concatenate(const SWFMatrix& m);

    /// Concatenate a translation onto the front of our SWFMatrix.
    //
    /// When transforming points, the translation
    /// happens first, then our original xform.
    void concatenate_translation(int _tx, int _ty);

    /// Concatenate scale x and y to the front of our SWFMatrix 
    //
    /// When transforming points, these scales happen first, then
    /// our original SWFMatrix. 
    void concatenate_scale(double x, double y);

    /// Set this SWFMatrix to a blend of m1 and m2, parameterized by t.
    void set_lerp(const SWFMatrix& m1, const SWFMatrix& m2, float t);

    /// Set the scale & rotation part of the SWFMatrix. angle in radians.
    void set_scale_rotation(double x_scale, double y_scale, double rotation);

    /// Set x and y scales, rotation is unchanged.
    void set_scale(double x_scale, double y_scale);

    /// Set x scale, rotation any y scale are unchanged.
    void set_x_scale(double scale);

    /// Set y scale, rotation and x scale are unchanged.
    void set_y_scale(double scale);

    /// Set rotation in radians, scales component are unchanged.
    void set_rotation(double rotation);

    /// Set x translation in TWIPS
    void set_x_translation(int x) {
        _tx = x;
    }

    /// Set y translation in TWIPS.
    void set_y_translation(int y) {
        _ty = y;
    }

    /// Set x and y translation in TWIPS.
    void set_translation(int x, int y) {
        _tx = x;
        _ty = y;
    }

    /// Transform a given point by our SWFMatrix
    void transform(geometry::Point2d& p) const;

    /// Transform the given point by our SWFMatrix.
    void transform(boost::int32_t& x, boost::int32_t& y) const;
    
    /// Transform point 'p' by our SWFMatrix. 
    //
    /// Put the result in *result.
    ///
    void transform(geometry::Point2d* result, const geometry::Point2d& p) const;

    /// Transform Range2d<float> 'r' by our SWFMatrix. 
    //
    /// NULL and WORLD ranges are untouched.
    ///
    void transform(geometry::Range2d<boost::int32_t>& r) const;

    void transform(SWFRect& r) const;
    
    /// Invert this SWFMatrix and return the result.
    SWFMatrix& invert();
    
    /// return the magnitude scale of our x coord output
    double get_x_scale() const;

    /// return the magnitude scale of our y coord output
    double get_y_scale() const;

    /// return rotation component in radians.
    double get_rotation() const;

    /// return x translation n TWIPS unit.
    int get_x_translation() const {
        return _tx;
    }

    /// return y translation in TWIPS unit.
    int get_y_translation() const {
        return _ty;
    }

    /// Allow direct access to values for equality
    friend bool operator==(const SWFMatrix& a, const SWFMatrix& b);

private: 

    /// Return the determinant of this SWFMatrix in 32.32 fixed point format.
    boost::int64_t  determinant() const;

    /// Xscale, 16.16 fixed point. xx in swfdec. 'a' in AS Matrix.
    boost::int32_t _a; 

    /// Xshear, 16.16 fixed point. yx in swfdec. 'b' in AS Matrix.
    boost::int32_t _b;

    /// Yshear, 16.16 fixed point. xy in swfdec. 'c' in AS Matrix.
    boost::int32_t _c;

    /// Yscale, 16.16 fixed point. yy in swfdec. 'd' in AS Matrix.
    boost::int32_t _d; 

    /// Xtranslation, TWIPS. x0 in swfdec. '_tx' in AS Matrix.
    boost::int32_t _tx; 

    /// Ytranslation, TWIPS. y0 in swfdec. '_ty' in AS Matrix.
    boost::int32_t _ty; 
             

}; //end of SWFMatrix

inline bool
operator==(const SWFMatrix& a, const SWFMatrix& b)
{
    return  
        a.a()  == b._a  &&
        a._b == b._b &&
        a._tx  == b._tx  &&
        a._d  == b._d  &&
        a._c == b._c &&
        a._ty  == b._ty;
}

std::ostream& operator<<(std::ostream& o, const SWFMatrix& m);

} // namespace gnash

#endif 


// Local Variables:
// mode: C++
// indent-tabs-mode: t
// End:
// 

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