2015-04-23 00:51:47 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2008-2010 Ricardo Quesada
|
|
|
|
Copyright (c) 2010-2012 cocos2d-x.org
|
|
|
|
Copyright (c) 2011 Zynga Inc.
|
|
|
|
Copyright (c) 2013-2014 Chukong Technologies Inc.
|
|
|
|
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2015-05-29 15:03:36 +08:00
|
|
|
#ifndef COCOS_2D_CCAUTOPOLYGON_H__
|
|
|
|
#define COCOS_2D_CCAUTOPOLYGON_H__
|
2015-04-23 00:51:47 +08:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2015-06-02 10:10:09 +08:00
|
|
|
#include "platform/CCImage.h"
|
|
|
|
#include "renderer/CCTrianglesCommand.h"
|
2015-04-23 00:51:47 +08:00
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* @addtogroup _2d
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PolygonInfo is an object holding the required data to display Sprites。
|
|
|
|
* It can be a simple as a triangle, or as complex as a whole 3D mesh
|
|
|
|
*/
|
2015-06-02 10:10:09 +08:00
|
|
|
class CC_DLL PolygonInfo
|
|
|
|
{
|
|
|
|
public:
|
2015-06-08 13:59:44 +08:00
|
|
|
/// @name Creators
|
|
|
|
/// @{
|
|
|
|
/**
|
|
|
|
* Creates an empty Polygon info
|
|
|
|
* @memberof PolygonInfo
|
|
|
|
* @return PolygonInfo object
|
|
|
|
*/
|
2015-06-02 10:10:09 +08:00
|
|
|
PolygonInfo():
|
2015-06-04 15:43:31 +08:00
|
|
|
isVertsOwner(true),
|
2015-06-09 18:45:55 +08:00
|
|
|
rect(cocos2d::Rect::ZERO),
|
|
|
|
filename("")
|
2015-06-02 10:10:09 +08:00
|
|
|
{
|
2015-06-09 18:45:55 +08:00
|
|
|
triangles.verts = nullptr;
|
|
|
|
triangles.indices = nullptr;
|
|
|
|
triangles.vertCount = 0;
|
|
|
|
triangles.indexCount = 0;
|
2015-06-02 10:10:09 +08:00
|
|
|
};
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an polygoninfo from the data of another Polygoninfo
|
|
|
|
* @param other another PolygonInfo to be copied
|
|
|
|
* @return duplicate of the other PolygonInfo
|
|
|
|
*/
|
2015-06-02 10:10:09 +08:00
|
|
|
PolygonInfo(const PolygonInfo& other);
|
2015-06-08 13:59:44 +08:00
|
|
|
// end of creators group
|
|
|
|
/// @}
|
2015-06-04 15:43:31 +08:00
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* Copy the member of the other PolygonInfo
|
|
|
|
* @param other another PolygonInfo to be copied
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
PolygonInfo& operator= (const PolygonInfo &other);
|
2015-06-02 10:10:09 +08:00
|
|
|
~PolygonInfo();
|
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* set the data to be a pointer to a quad
|
|
|
|
* the member verts will not be released when this PolygonInfo destructs
|
|
|
|
* as the verts memory are managed by other objects
|
|
|
|
* @param quad a pointer to the V3F_C4B_T2F_Quad obje
|
|
|
|
*/
|
2015-06-04 18:27:58 +08:00
|
|
|
void setQuad(V3F_C4B_T2F_Quad *quad);
|
2015-06-09 18:45:55 +08:00
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* get vertex count
|
|
|
|
* @return number of vertices
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
const unsigned int getVertCount() const;
|
2015-06-08 13:59:44 +08:00
|
|
|
|
2015-06-11 14:38:38 +08:00
|
|
|
/**
|
|
|
|
* get triangles count
|
|
|
|
* @return number of triangles
|
|
|
|
*/
|
|
|
|
const unsigned int getTriaglesCount() const;
|
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* get sum of all triangle area size
|
|
|
|
* @return sum of all triangle area size
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
const float getArea() const;
|
|
|
|
|
2015-06-09 18:45:55 +08:00
|
|
|
Rect rect;
|
|
|
|
std::string filename;
|
|
|
|
TrianglesCommand::Triangles triangles;
|
|
|
|
|
2015-06-02 10:10:09 +08:00
|
|
|
protected:
|
|
|
|
bool isVertsOwner;
|
2015-06-09 18:45:55 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
void releaseVertsAndIndices();
|
2015-06-02 10:10:09 +08:00
|
|
|
};
|
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* AutoPolygon is a helper Object
|
|
|
|
* AutoPolygon's purpose is to process an image into 2d polygon mesh in runtime
|
|
|
|
* It has functions for each step in the process, from tracing all the points, to triangulation
|
|
|
|
* the result can be then passed to Sprite::create() to create a Polygon Sprite
|
|
|
|
*/
|
2015-05-20 06:07:50 +08:00
|
|
|
class CC_DLL AutoPolygon
|
2015-04-23 00:51:47 +08:00
|
|
|
{
|
|
|
|
public:
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* create an AutoPolygon and initialize it with an image file
|
|
|
|
* the image must be a 32bit PNG for current version 3.7
|
|
|
|
* @param filename a path to image file, e.g., "scene1/monster.png".
|
|
|
|
* @return an AutoPolygon object;
|
|
|
|
*/
|
2015-05-20 06:07:50 +08:00
|
|
|
AutoPolygon(const std::string &filename);
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
2015-06-11 14:38:38 +08:00
|
|
|
* Destructor of AutoPolygon.
|
2015-06-08 13:59:44 +08:00
|
|
|
*/
|
2015-05-20 06:07:50 +08:00
|
|
|
~AutoPolygon();
|
2015-05-18 15:10:41 +08:00
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* trace all the points along the outline of the image,
|
|
|
|
* @warning must create AutoPolygon with filename to use this function
|
|
|
|
* @param rect a texture rect for specify an area of the image, use Rect::Zero for the size of the image, default to Rect::Zero
|
|
|
|
* @param threshold the value when alpha is greater than this value will be counted as opaque, default to 0.0
|
|
|
|
* @return a vector of vec2 of all the points found in clockwise order
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon("grossini.png");
|
|
|
|
* std::vector<Vec2> points = ap.trace();//default to size of the image and threshold 0.0
|
|
|
|
* @endcode
|
|
|
|
*/
|
|
|
|
std::vector<Vec2> trace(const cocos2d::Rect& rect, const float& threshold = 0.0);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* reduce the ammount of points so its faster for GPU to process and draw
|
|
|
|
* based on Ramer-Douglas-Puecker algorithm
|
|
|
|
* @param points a vector of Vec2 points as input
|
|
|
|
* @param rect a texture rect for specify an area of the image to avoid over reduction
|
|
|
|
* @param epsilon the perpendicular distance where points smaller than this value will be discarded
|
|
|
|
* @return a vector of Vec2 of the remaining points in clockwise order
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon();
|
|
|
|
* std::vector<Vec2> reduced = ap.reduce(inputPoints, rect);//default epsilon is 2
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
std::vector<Vec2> reduce(const std::vector<Vec2>& points, const Rect& rect, const float& epsilon = 2.0);
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* expand the points along their edge, useful after you reduce the points that cuts into the sprite
|
|
|
|
* using ClipperLib
|
|
|
|
* @param points a vector of Vec2 points as input
|
|
|
|
* @param rect a texture rect for specify an area of the image, the expanded points will be clamped in this rect, ultimately resulting in a quad if the expansion is too great
|
|
|
|
* @param epsilon the distance which the edges will expand
|
|
|
|
* @return a vector of Vec2 as the result of the expansion
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon();
|
|
|
|
* std::vector<Vec2> expanded = ap.expand(inputPoints, rect, 2.0);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
std::vector<Vec2> expand(const std::vector<Vec2>& points, const Rect& rect, const float& epsilon);
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Triangulate the input points into triangles for rendering
|
|
|
|
* using poly2tri
|
|
|
|
* @warning points must be closed loop, cannot have 2 points sharing the same position and cannot intersect itself
|
|
|
|
* @param points a vector of vec2 points as input
|
|
|
|
* @return a Triangles object with points and indices
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon();
|
|
|
|
* TrianglesCommand::Triangles myPolygons = ap.triangulate(myPoints);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-02 10:10:09 +08:00
|
|
|
TrianglesCommand::Triangles triangulate(const std::vector<Vec2>& points);
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* calculate the UV coordinates for each points based on a texture rect
|
|
|
|
* @warning This method requires the AutoPolygon object to know the texture file dimension
|
|
|
|
* @param rect a texture rect to specify where to map the UV
|
|
|
|
* @param verts a pointer to the verts array, served both as input and output verts
|
|
|
|
* @param count the count for the verts arrac
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon("grossini.png");
|
|
|
|
* TrianglesCommand::Triangles myPolygons = ap.triangulate(myPoints);
|
|
|
|
* ap.calculateUV(rect, myPolygons.verts, 20);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
void calculateUV(const Rect& rect, V3F_C4B_T2F* verts, const ssize_t& count);
|
2015-05-20 06:07:50 +08:00
|
|
|
|
2015-06-08 13:59:44 +08:00
|
|
|
/**
|
|
|
|
* a helper function, packing trace, reduce, expand, triangulate and calculate uv in one function
|
|
|
|
* @param rect texture rect, use Rect::ZERO for the size of the texture, default is Rect::ZERO
|
|
|
|
* @param epsilon the value used to reduce and expand, default to 2.0
|
|
|
|
* @param threshold the value where bigger than the threshold will be counted as opaque, used in trace
|
|
|
|
* @return a PolygonInfo, to use with sprite
|
|
|
|
* @code
|
|
|
|
* auto ap = AutoPolygon("grossini.png");
|
|
|
|
* PolygonInfo myInfo = ap.generateTriangles();//use all default values
|
|
|
|
* auto sp1 = Sprite::create(myInfo);
|
|
|
|
* polygonInfo myInfo2 = ap.generateTriangles(Rect::ZERO, 5.0, 0.1);//ap can be reused to generate another set of PolygonInfo with different settings
|
|
|
|
* auto sp2 = Sprite::create(myInfo2);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
PolygonInfo generateTriangles(const Rect& rect = Rect::ZERO, const float& epsilon = 2.0, const float& threshold = 0.05);
|
2015-06-08 13:59:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* a helper function, packing autoPolygon creation, trace, reduce, expand, triangulate and calculate uv in one function
|
|
|
|
* @warning if you want to repetitively generate polygons, consider create an AutoPolygon object, and use generateTriangles function, as it only reads the file once
|
|
|
|
* @param filename A path to image file, e.g., "scene1/monster.png".
|
|
|
|
* @param rect texture rect, use Rect::ZERO for the size of the texture, default is Rect::ZERO
|
|
|
|
* @param epsilon the value used to reduce and expand, default to 2.0
|
|
|
|
* @param threshold the value where bigger than the threshold will be counted as opaque, used in trace
|
|
|
|
* @return a PolygonInfo, to use with sprite
|
|
|
|
* @code
|
|
|
|
* auto sp = Sprite::create(AutoPolygon::generatePolygon("grossini.png"));
|
|
|
|
* @endcode
|
|
|
|
*/
|
2015-06-04 15:43:31 +08:00
|
|
|
static PolygonInfo generatePolygon(const std::string& filename, const Rect& rect = Rect::ZERO, const float epsilon = 2.0, const float threshold = 0.05);
|
2015-04-23 00:51:47 +08:00
|
|
|
protected:
|
2015-06-04 15:43:31 +08:00
|
|
|
Vec2 findFirstNoneTransparentPixel(const Rect& rect, const float& threshold);
|
2015-06-02 10:10:09 +08:00
|
|
|
std::vector<cocos2d::Vec2> marchSquare(const Rect& rect, const Vec2& first, const float& threshold);
|
|
|
|
unsigned int getSquareValue(const unsigned int& x, const unsigned int& y, const Rect& rect, const float& threshold);
|
2015-05-15 17:18:19 +08:00
|
|
|
|
2015-05-18 15:10:41 +08:00
|
|
|
unsigned char getAlphaByIndex(const unsigned int& i);
|
2015-06-09 18:45:55 +08:00
|
|
|
unsigned char getAlphaByPos(const Vec2& pos);
|
2015-05-15 17:18:19 +08:00
|
|
|
|
2015-05-18 15:10:41 +08:00
|
|
|
int getIndexFromPos(const unsigned int& x, const unsigned int& y){return y*_width+x;};
|
|
|
|
cocos2d::Vec2 getPosFromIndex(const unsigned int& i){return cocos2d::Vec2(i%_width, i/_width);};
|
2015-05-15 17:18:19 +08:00
|
|
|
|
2015-05-20 06:07:50 +08:00
|
|
|
std::vector<cocos2d::Vec2> rdp(std::vector<cocos2d::Vec2> v, const float& optimization);
|
2015-05-18 15:10:41 +08:00
|
|
|
float perpendicularDistance(const cocos2d::Vec2& i, const cocos2d::Vec2& start, const cocos2d::Vec2& end);
|
2015-04-27 16:33:54 +08:00
|
|
|
|
2015-06-04 15:43:31 +08:00
|
|
|
//real rect is the size that is in scale with the texture file
|
|
|
|
Rect getRealRect(const Rect& rect);
|
2015-05-15 17:18:19 +08:00
|
|
|
|
2015-06-02 10:10:09 +08:00
|
|
|
Image* _image;
|
2015-05-15 17:18:19 +08:00
|
|
|
unsigned char * _data;
|
|
|
|
std::string _filename;
|
|
|
|
unsigned int _width;
|
|
|
|
unsigned int _height;
|
2015-05-20 06:07:50 +08:00
|
|
|
float _scaleFactor;
|
2015-05-15 17:18:19 +08:00
|
|
|
unsigned int _threshold;
|
2015-04-23 00:51:47 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
NS_CC_END
|
|
|
|
|
2015-05-29 15:03:36 +08:00
|
|
|
#endif // #ifndef COCOS_2D_CCAUTOPOLYGON_H__
|