480 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			480 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
////////////////////////////////////////////////////////////
 | 
						|
//
 | 
						|
// SFML - Simple and Fast Multimedia Library
 | 
						|
// Copyright (C) 2007-2018 Laurent Gomila (laurent@sfml-dev.org)
 | 
						|
//
 | 
						|
// This software is provided 'as-is', without any express or implied warranty.
 | 
						|
// In no event will the authors be held liable for any damages arising from the use of this software.
 | 
						|
//
 | 
						|
// Permission is granted to anyone to use this software for any purpose,
 | 
						|
// including commercial applications, and to alter it and redistribute it freely,
 | 
						|
// subject to the following restrictions:
 | 
						|
//
 | 
						|
// 1. The origin of this software must not be misrepresented;
 | 
						|
//    you must not claim that you wrote the original software.
 | 
						|
//    If you use this software in a product, an acknowledgment
 | 
						|
//    in the product documentation would be appreciated but is not required.
 | 
						|
//
 | 
						|
// 2. Altered source versions must be plainly marked as such,
 | 
						|
//    and must not be misrepresented as being the original software.
 | 
						|
//
 | 
						|
// 3. This notice may not be removed or altered from any source distribution.
 | 
						|
//
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
 | 
						|
#ifndef SFML_TRANSFORM_HPP
 | 
						|
#define SFML_TRANSFORM_HPP
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
// Headers
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
#include <SFML/Graphics/Export.hpp>
 | 
						|
#include <SFML/Graphics/Rect.hpp>
 | 
						|
#include <SFML/System/Vector2.hpp>
 | 
						|
 | 
						|
 | 
						|
namespace sf
 | 
						|
{
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \brief Define a 3x3 transform matrix
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
class SFML_GRAPHICS_API Transform
 | 
						|
{
 | 
						|
public:
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Default constructor
 | 
						|
    ///
 | 
						|
    /// Creates an identity transform (a transform that does nothing).
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform();
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Construct a transform from a 3x3 matrix
 | 
						|
    ///
 | 
						|
    /// \param a00 Element (0, 0) of the matrix
 | 
						|
    /// \param a01 Element (0, 1) of the matrix
 | 
						|
    /// \param a02 Element (0, 2) of the matrix
 | 
						|
    /// \param a10 Element (1, 0) of the matrix
 | 
						|
    /// \param a11 Element (1, 1) of the matrix
 | 
						|
    /// \param a12 Element (1, 2) of the matrix
 | 
						|
    /// \param a20 Element (2, 0) of the matrix
 | 
						|
    /// \param a21 Element (2, 1) of the matrix
 | 
						|
    /// \param a22 Element (2, 2) of the matrix
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform(float a00, float a01, float a02,
 | 
						|
              float a10, float a11, float a12,
 | 
						|
              float a20, float a21, float a22);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Return the transform as a 4x4 matrix
 | 
						|
    ///
 | 
						|
    /// This function returns a pointer to an array of 16 floats
 | 
						|
    /// containing the transform elements as a 4x4 matrix, which
 | 
						|
    /// is directly compatible with OpenGL functions.
 | 
						|
    ///
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform = ...;
 | 
						|
    /// glLoadMatrixf(transform.getMatrix());
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \return Pointer to a 4x4 matrix
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    const float* getMatrix() const;
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Return the inverse of the transform
 | 
						|
    ///
 | 
						|
    /// If the inverse cannot be computed, an identity transform
 | 
						|
    /// is returned.
 | 
						|
    ///
 | 
						|
    /// \return A new transform which is the inverse of self
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform getInverse() const;
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Transform a 2D point
 | 
						|
    ///
 | 
						|
    /// \param x X coordinate of the point to transform
 | 
						|
    /// \param y Y coordinate of the point to transform
 | 
						|
    ///
 | 
						|
    /// \return Transformed point
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Vector2f transformPoint(float x, float y) const;
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Transform a 2D point
 | 
						|
    ///
 | 
						|
    /// \param point Point to transform
 | 
						|
    ///
 | 
						|
    /// \return Transformed point
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Vector2f transformPoint(const Vector2f& point) const;
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Transform a rectangle
 | 
						|
    ///
 | 
						|
    /// Since SFML doesn't provide support for oriented rectangles,
 | 
						|
    /// the result of this function is always an axis-aligned
 | 
						|
    /// rectangle. Which means that if the transform contains a
 | 
						|
    /// rotation, the bounding rectangle of the transformed rectangle
 | 
						|
    /// is returned.
 | 
						|
    ///
 | 
						|
    /// \param rectangle Rectangle to transform
 | 
						|
    ///
 | 
						|
    /// \return Transformed rectangle
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    FloatRect transformRect(const FloatRect& rectangle) const;
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with another one
 | 
						|
    ///
 | 
						|
    /// The result is a transform that is equivalent to applying
 | 
						|
    /// *this followed by \a transform. Mathematically, it is
 | 
						|
    /// equivalent to a matrix multiplication.
 | 
						|
    ///
 | 
						|
    /// \param transform Transform to combine with this transform
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& combine(const Transform& transform);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a translation
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.translate(100, 200).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param x Offset to apply on X axis
 | 
						|
    /// \param y Offset to apply on Y axis
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see rotate, scale
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& translate(float x, float y);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a translation
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.translate(sf::Vector2f(100, 200)).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param offset Translation offset to apply
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see rotate, scale
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& translate(const Vector2f& offset);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a rotation
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.rotate(90).translate(50, 20);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param angle Rotation angle, in degrees
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, scale
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& rotate(float angle);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a rotation
 | 
						|
    ///
 | 
						|
    /// The center of rotation is provided for convenience as a second
 | 
						|
    /// argument, so that you can build rotations around arbitrary points
 | 
						|
    /// more easily (and efficiently) than the usual
 | 
						|
    /// translate(-center).rotate(angle).translate(center).
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.rotate(90, 8, 3).translate(50, 20);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param angle Rotation angle, in degrees
 | 
						|
    /// \param centerX X coordinate of the center of rotation
 | 
						|
    /// \param centerY Y coordinate of the center of rotation
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, scale
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& rotate(float angle, float centerX, float centerY);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a rotation
 | 
						|
    ///
 | 
						|
    /// The center of rotation is provided for convenience as a second
 | 
						|
    /// argument, so that you can build rotations around arbitrary points
 | 
						|
    /// more easily (and efficiently) than the usual
 | 
						|
    /// translate(-center).rotate(angle).translate(center).
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.rotate(90, sf::Vector2f(8, 3)).translate(sf::Vector2f(50, 20));
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param angle Rotation angle, in degrees
 | 
						|
    /// \param center Center of rotation
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, scale
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& rotate(float angle, const Vector2f& center);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a scaling
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.scale(2, 1).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param scaleX Scaling factor on the X axis
 | 
						|
    /// \param scaleY Scaling factor on the Y axis
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, rotate
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& scale(float scaleX, float scaleY);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a scaling
 | 
						|
    ///
 | 
						|
    /// The center of scaling is provided for convenience as a second
 | 
						|
    /// argument, so that you can build scaling around arbitrary points
 | 
						|
    /// more easily (and efficiently) than the usual
 | 
						|
    /// translate(-center).scale(factors).translate(center).
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.scale(2, 1, 8, 3).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param scaleX Scaling factor on X axis
 | 
						|
    /// \param scaleY Scaling factor on Y axis
 | 
						|
    /// \param centerX X coordinate of the center of scaling
 | 
						|
    /// \param centerY Y coordinate of the center of scaling
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, rotate
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& scale(float scaleX, float scaleY, float centerX, float centerY);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a scaling
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.scale(sf::Vector2f(2, 1)).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param factors Scaling factors
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, rotate
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& scale(const Vector2f& factors);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    /// \brief Combine the current transform with a scaling
 | 
						|
    ///
 | 
						|
    /// The center of scaling is provided for convenience as a second
 | 
						|
    /// argument, so that you can build scaling around arbitrary points
 | 
						|
    /// more easily (and efficiently) than the usual
 | 
						|
    /// translate(-center).scale(factors).translate(center).
 | 
						|
    ///
 | 
						|
    /// This function returns a reference to *this, so that calls
 | 
						|
    /// can be chained.
 | 
						|
    /// \code
 | 
						|
    /// sf::Transform transform;
 | 
						|
    /// transform.scale(sf::Vector2f(2, 1), sf::Vector2f(8, 3)).rotate(45);
 | 
						|
    /// \endcode
 | 
						|
    ///
 | 
						|
    /// \param factors Scaling factors
 | 
						|
    /// \param center Center of scaling
 | 
						|
    ///
 | 
						|
    /// \return Reference to *this
 | 
						|
    ///
 | 
						|
    /// \see translate, rotate
 | 
						|
    ///
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    Transform& scale(const Vector2f& factors, const Vector2f& center);
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    // Static member data
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    static const Transform Identity; ///< The identity transform (does nothing)
 | 
						|
 | 
						|
private:
 | 
						|
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    // Member data
 | 
						|
    ////////////////////////////////////////////////////////////
 | 
						|
    float m_matrix[16]; ///< 4x4 matrix defining the transformation
 | 
						|
};
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \relates sf::Transform
 | 
						|
/// \brief Overload of binary operator * to combine two transforms
 | 
						|
///
 | 
						|
/// This call is equivalent to calling Transform(left).combine(right).
 | 
						|
///
 | 
						|
/// \param left Left operand (the first transform)
 | 
						|
/// \param right Right operand (the second transform)
 | 
						|
///
 | 
						|
/// \return New combined transform
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
SFML_GRAPHICS_API Transform operator *(const Transform& left, const Transform& right);
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \relates sf::Transform
 | 
						|
/// \brief Overload of binary operator *= to combine two transforms
 | 
						|
///
 | 
						|
/// This call is equivalent to calling left.combine(right).
 | 
						|
///
 | 
						|
/// \param left Left operand (the first transform)
 | 
						|
/// \param right Right operand (the second transform)
 | 
						|
///
 | 
						|
/// \return The combined transform
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
SFML_GRAPHICS_API Transform& operator *=(Transform& left, const Transform& right);
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \relates sf::Transform
 | 
						|
/// \brief Overload of binary operator * to transform a point
 | 
						|
///
 | 
						|
/// This call is equivalent to calling left.transformPoint(right).
 | 
						|
///
 | 
						|
/// \param left Left operand (the transform)
 | 
						|
/// \param right Right operand (the point to transform)
 | 
						|
///
 | 
						|
/// \return New transformed point
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
SFML_GRAPHICS_API Vector2f operator *(const Transform& left, const Vector2f& right);
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \relates sf::Transform
 | 
						|
/// \brief Overload of binary operator == to compare two transforms
 | 
						|
///
 | 
						|
/// Performs an element-wise comparison of the elements of the
 | 
						|
/// left transform with the elements of the right transform.
 | 
						|
///
 | 
						|
/// \param left Left operand (the first transform)
 | 
						|
/// \param right Right operand (the second transform)
 | 
						|
///
 | 
						|
/// \return true if the transforms are equal, false otherwise
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
SFML_GRAPHICS_API bool operator ==(const Transform& left, const Transform& right);
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \relates sf::Transform
 | 
						|
/// \brief Overload of binary operator != to compare two transforms
 | 
						|
///
 | 
						|
/// This call is equivalent to !(left == right).
 | 
						|
///
 | 
						|
/// \param left Left operand (the first transform)
 | 
						|
/// \param right Right operand (the second transform)
 | 
						|
///
 | 
						|
/// \return true if the transforms are not equal, false otherwise
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
SFML_GRAPHICS_API bool operator !=(const Transform& left, const Transform& right);
 | 
						|
 | 
						|
} // namespace sf
 | 
						|
 | 
						|
 | 
						|
#endif // SFML_TRANSFORM_HPP
 | 
						|
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////
 | 
						|
/// \class sf::Transform
 | 
						|
/// \ingroup graphics
 | 
						|
///
 | 
						|
/// A sf::Transform specifies how to translate, rotate, scale,
 | 
						|
/// shear, project, whatever things. In mathematical terms, it defines
 | 
						|
/// how to transform a coordinate system into another.
 | 
						|
///
 | 
						|
/// For example, if you apply a rotation transform to a sprite, the
 | 
						|
/// result will be a rotated sprite. And anything that is transformed
 | 
						|
/// by this rotation transform will be rotated the same way, according
 | 
						|
/// to its initial position.
 | 
						|
///
 | 
						|
/// Transforms are typically used for drawing. But they can also be
 | 
						|
/// used for any computation that requires to transform points between
 | 
						|
/// the local and global coordinate systems of an entity (like collision
 | 
						|
/// detection).
 | 
						|
///
 | 
						|
/// Example:
 | 
						|
/// \code
 | 
						|
/// // define a translation transform
 | 
						|
/// sf::Transform translation;
 | 
						|
/// translation.translate(20, 50);
 | 
						|
///
 | 
						|
/// // define a rotation transform
 | 
						|
/// sf::Transform rotation;
 | 
						|
/// rotation.rotate(45);
 | 
						|
///
 | 
						|
/// // combine them
 | 
						|
/// sf::Transform transform = translation * rotation;
 | 
						|
///
 | 
						|
/// // use the result to transform stuff...
 | 
						|
/// sf::Vector2f point = transform.transformPoint(10, 20);
 | 
						|
/// sf::FloatRect rect = transform.transformRect(sf::FloatRect(0, 0, 10, 100));
 | 
						|
/// \endcode
 | 
						|
///
 | 
						|
/// \see sf::Transformable, sf::RenderStates
 | 
						|
///
 | 
						|
////////////////////////////////////////////////////////////
 |