mirror of https://github.com/axmolengine/axmol.git
179 lines
4.7 KiB
C++
179 lines
4.7 KiB
C++
//
|
|
// CCNewSprite.cpp
|
|
// cocos2d_libs
|
|
//
|
|
// Created by NiTe Luo on 10/31/13.
|
|
//
|
|
//
|
|
|
|
#include "CCNewSprite.h"
|
|
#include "RenderCommand.h"
|
|
#include "Renderer.h"
|
|
#include "QuadCommand.h"
|
|
#include "CCMenuItem.h"
|
|
#include "Frustum.h"
|
|
#include "CCDirector.h"
|
|
|
|
NS_CC_BEGIN
|
|
|
|
#if CC_SPRITEBATCHNODE_RENDER_SUBPIXEL
|
|
#define RENDER_IN_SUBPIXEL
|
|
#else
|
|
#define RENDER_IN_SUBPIXEL(__ARGS__) (ceil(__ARGS__))
|
|
#endif
|
|
|
|
NewSprite* NewSprite::create()
|
|
{
|
|
NewSprite* sprite = new NewSprite();
|
|
if(sprite && sprite->init())
|
|
{
|
|
sprite->autorelease();
|
|
return sprite;
|
|
}
|
|
CC_SAFE_DELETE(sprite);
|
|
return NULL;
|
|
}
|
|
|
|
NewSprite* NewSprite::create(const char *filename)
|
|
{
|
|
NewSprite* sprite = new NewSprite();
|
|
if(sprite && sprite->initWithFile(filename))
|
|
{
|
|
sprite->autorelease();
|
|
return sprite;
|
|
}
|
|
CC_SAFE_DELETE(sprite);
|
|
return NULL;
|
|
}
|
|
|
|
NewSprite::NewSprite()
|
|
:Sprite()
|
|
{
|
|
|
|
}
|
|
|
|
NewSprite::~NewSprite(void)
|
|
{
|
|
}
|
|
|
|
bool NewSprite::initWithTexture(Texture2D *texture, const Rect &rect, bool rotated)
|
|
{
|
|
bool result = Sprite::initWithTexture(texture, rect, rotated);
|
|
_recursiveDirty = true;
|
|
setDirty(true);
|
|
return result;
|
|
}
|
|
|
|
void NewSprite::updateQuadVertices()
|
|
{
|
|
|
|
#ifdef CC_USE_PHYSICS
|
|
updatePhysicsTransform();
|
|
setDirty(true);
|
|
#endif
|
|
|
|
//TODO optimize the performance cache affineTransformation
|
|
|
|
// recalculate matrix only if it is dirty
|
|
if(isDirty())
|
|
{
|
|
|
|
// if( ! _parent || _parent == (Node*)_batchNode )
|
|
// {
|
|
// _transformToBatch = getNodeToParentTransform();
|
|
// }
|
|
// else
|
|
// {
|
|
// CCASSERT( dynamic_cast<NewSprite*>(_parent), "Logic error in Sprite. Parent must be a Sprite");
|
|
// _transformToBatch = AffineTransformConcat( getNodeToParentTransform() , static_cast<NewSprite*>(_parent)->_transformToBatch );
|
|
// }
|
|
|
|
//TODO optimize this transformation, should use parent's transformation instead
|
|
_transformToBatch = getNodeToWorldTransform();
|
|
|
|
//
|
|
// calculate the Quad based on the Affine Matrix
|
|
//
|
|
|
|
Size size = _rect.size;
|
|
|
|
float x1 = _offsetPosition.x;
|
|
float y1 = _offsetPosition.y;
|
|
|
|
float x2 = x1 + size.width;
|
|
float y2 = y1 + size.height;
|
|
float x = _transformToBatch.tx;
|
|
float y = _transformToBatch.ty;
|
|
|
|
float cr = _transformToBatch.a;
|
|
float sr = _transformToBatch.b;
|
|
float cr2 = _transformToBatch.d;
|
|
float sr2 = -_transformToBatch.c;
|
|
float ax = x1 * cr - y1 * sr2 + x;
|
|
float ay = x1 * sr + y1 * cr2 + y;
|
|
|
|
float bx = x2 * cr - y1 * sr2 + x;
|
|
float by = x2 * sr + y1 * cr2 + y;
|
|
|
|
float cx = x2 * cr - y2 * sr2 + x;
|
|
float cy = x2 * sr + y2 * cr2 + y;
|
|
|
|
float dx = x1 * cr - y2 * sr2 + x;
|
|
float dy = x1 * sr + y2 * cr2 + y;
|
|
|
|
_quad.bl.vertices = Vertex3F( RENDER_IN_SUBPIXEL(ax), RENDER_IN_SUBPIXEL(ay), _vertexZ );
|
|
_quad.br.vertices = Vertex3F( RENDER_IN_SUBPIXEL(bx), RENDER_IN_SUBPIXEL(by), _vertexZ );
|
|
_quad.tl.vertices = Vertex3F( RENDER_IN_SUBPIXEL(dx), RENDER_IN_SUBPIXEL(dy), _vertexZ );
|
|
_quad.tr.vertices = Vertex3F( RENDER_IN_SUBPIXEL(cx), RENDER_IN_SUBPIXEL(cy), _vertexZ );
|
|
|
|
_recursiveDirty = false;
|
|
setDirty(false);
|
|
}
|
|
}
|
|
|
|
void NewSprite::draw(void)
|
|
{
|
|
updateQuadVertices();
|
|
if(false == culling())
|
|
{
|
|
//static int count =0;
|
|
//CCLOG("culling Sprite New to not visible %d ",++count);
|
|
return;
|
|
}
|
|
|
|
//TODO implement z order
|
|
QuadCommand* renderCommand = new QuadCommand(0, _vertexZ, _texture->getName(), _shaderProgram, _blendFunc, &_quad, 1);
|
|
|
|
Renderer::getInstance()->addCommand(renderCommand);
|
|
}
|
|
|
|
bool NewSprite::culling() const
|
|
{
|
|
Frustum* frustum = Director::getInstance()->getFrustum();
|
|
AffineTransform worldTM = getNodeToWorldTransform();
|
|
//generate aabb
|
|
Point lowLeft(0,0);
|
|
Point topRight = lowLeft + Point(getContentSize());
|
|
Point lowRight(topRight.x,0);
|
|
Point topLeft(0, topRight.y);
|
|
|
|
lowLeft = PointApplyAffineTransform(lowLeft,worldTM);
|
|
lowRight = PointApplyAffineTransform(lowRight,worldTM);
|
|
topRight = PointApplyAffineTransform(topRight,worldTM);
|
|
topLeft = PointApplyAffineTransform(topLeft,worldTM);
|
|
|
|
kmVec3 point = {lowLeft.x, lowLeft.y, _vertexZ};
|
|
|
|
AABB aabb(point,point);
|
|
point = {lowRight.x, lowRight.y, _vertexZ};
|
|
aabb.expand(point);
|
|
point = {topLeft.x, topLeft.y, _vertexZ};
|
|
aabb.expand(point);
|
|
point = {topRight.x, topRight.y, _vertexZ};
|
|
aabb.expand(point);
|
|
|
|
return Frustum::IntersectResult::OUTSIDE !=frustum->intersectAABB(aabb);
|
|
}
|
|
|
|
|
|
NS_CC_END |