/**************************************************************************** Copyright (c) 2013-2016 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. ****************************************************************************/ #include "ui/UIScale9Sprite.h" #include "2d/CCSprite.h" #include "2d/CCSpriteFrameCache.h" #include "base/CCVector.h" #include "base/CCDirector.h" #include "base/ccUTF8.h" #include "renderer/CCGLProgram.h" #include "renderer/ccShaders.h" #include "platform/CCImage.h" #include "base/CCNinePatchImageParser.h" #include "2d/CCDrawNode.h" #include "2d/CCCamera.h" #include "renderer/CCRenderer.h" NS_CC_BEGIN namespace ui { Scale9Sprite::Scale9Sprite() : _spriteFrameRotated(false) , _scale9Image(nullptr) , _scale9Enabled(true) , _insetLeft(0) , _insetTop(0) , _insetRight(0) , _insetBottom(0) ,_flippedX(false) ,_flippedY(false) ,_isPatch9(false) ,_brightState(State::NORMAL) ,_nonSliceSpriteAnchor(Vec2::ANCHOR_MIDDLE) ,_sliceVertices(nullptr) ,_sliceIndices(nullptr) ,_sliceSpriteDirty(false) ,_renderingType(RenderingType::SLICE) ,_insideBounds(true) { this->setAnchorPoint(Vec2(0.5,0.5)); #if CC_SPRITE_DEBUG_DRAW _debugDrawNode = DrawNode::create(); addChild(_debugDrawNode); #endif //CC_SPRITE_DEBUG_DRAW } Scale9Sprite::~Scale9Sprite() { this->cleanupSlicedSprites(); CC_SAFE_RELEASE(_scale9Image); } bool Scale9Sprite::initWithFile(const Rect& capInsets, const std::string& file) { bool pReturn = this->initWithFile(file, Rect::ZERO, capInsets); return pReturn; } bool Scale9Sprite::initWithFile(const std::string& file) { bool pReturn = this->initWithFile(file, Rect::ZERO); return pReturn; } bool Scale9Sprite::initWithSpriteFrame(SpriteFrame* spriteFrame, const Rect& capInsets) { bool ret = false; do { Texture2D* texture = spriteFrame->getTexture(); CCASSERT(texture != nullptr, "Texture2D must be not null"); if(texture == nullptr) break; Sprite *sprite = Sprite::createWithSpriteFrame(spriteFrame); CCASSERT(sprite != nullptr, "Sprite must be not null"); if(sprite == nullptr) break; ret = this->init(sprite, spriteFrame->getRect(), spriteFrame->isRotated(), spriteFrame->getOffset(), spriteFrame->getOriginalSize(), capInsets); } while (false); return ret; } bool Scale9Sprite::initWithSpriteFrame(SpriteFrame* spriteFrame) { CCASSERT(spriteFrame != nullptr, "Invalid spriteFrame for sprite"); bool pReturn = this->initWithSpriteFrame(spriteFrame, Rect::ZERO); return pReturn; } bool Scale9Sprite::initWithSpriteFrameName(const std::string& spriteFrameName, const Rect& capInsets) { bool ret = false; do { auto spriteFrameCache = SpriteFrameCache::getInstance(); CCASSERT(spriteFrameCache != nullptr, "SpriteFrameCache::getInstance() must be non-NULL"); if(spriteFrameCache == nullptr) break; SpriteFrame *frame = spriteFrameCache->getSpriteFrameByName(spriteFrameName); CCASSERT(frame != nullptr, StringUtils::format("CCSpriteFrame: %s must be non-NULL ", spriteFrameName.c_str()).c_str()); if (frame == nullptr) break; ret = initWithSpriteFrame(frame, capInsets); } while (false); return ret; } bool Scale9Sprite::initWithSpriteFrameName(const std::string& spriteFrameName) { bool pReturn = this->initWithSpriteFrameName(spriteFrameName, Rect::ZERO); return pReturn; } bool Scale9Sprite::init() { return this->init(nullptr, Rect::ZERO, Rect::ZERO); } bool Scale9Sprite::init(Sprite* sprite, const Rect& rect, const Rect& capInsets) { return this->init(sprite, rect, false, capInsets); } bool Scale9Sprite::init(Sprite* sprite, const Rect& rect, bool rotated, const Rect& capInsets) { return init(sprite, rect, rotated, Vec2::ZERO, rect.size, capInsets); } bool Scale9Sprite::init(Sprite* sprite, const Rect& rect, bool rotated, const Vec2 &offset, const Size &originalSize, const Rect& capInsets) { bool ret = true; if(sprite) { auto texture = sprite->getTexture(); auto spriteFrame = sprite->getSpriteFrame(); Rect actualCapInsets = capInsets; if (texture->isContain9PatchInfo()) { auto& parsedCapInset = texture->getSpriteFrameCapInset(spriteFrame); if(!parsedCapInset.equals(Rect::ZERO)) { this->_isPatch9 = true; if(capInsets.equals(Rect::ZERO)) { actualCapInsets = parsedCapInset; } } } ret = this->updateWithSprite(sprite, rect, rotated, offset, originalSize, actualCapInsets); } return ret; } bool Scale9Sprite::initWithBatchNode(cocos2d::SpriteBatchNode *batchnode, const cocos2d::Rect &rect, bool rotated, const cocos2d::Rect &capInsets) { Sprite *sprite = Sprite::createWithTexture(batchnode->getTexture()); return init(sprite, rect, rotated, capInsets); } bool Scale9Sprite::initWithBatchNode(cocos2d::SpriteBatchNode *batchnode, const cocos2d::Rect &rect, const cocos2d::Rect &capInsets) { auto sprite = Sprite::createWithTexture(batchnode->getTexture()); return init(sprite, rect, false, capInsets); } bool Scale9Sprite::initWithFile(const std::string& file, const Rect& rect, const Rect& capInsets) { CCASSERT(!file.empty(), "file must not be empty string!"); if(file.empty()) { return false; } auto sprite = Sprite::create(file); return init(sprite, rect, capInsets); } bool Scale9Sprite::initWithFile(const std::string& file, const Rect& rect) { return initWithFile(file, rect, Rect::ZERO); } Scale9Sprite* Scale9Sprite::create() { Scale9Sprite *pReturn = new (std::nothrow) Scale9Sprite(); if (pReturn && pReturn->init()) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::create(const std::string& file, const Rect& rect, const Rect& capInsets) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithFile(file, rect, capInsets) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::create(const std::string& file, const Rect& rect) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithFile(file, rect) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::create(const Rect& capInsets, const std::string& file) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithFile(capInsets, file) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::create(const std::string& file) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithFile(file) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::createWithSpriteFrame(SpriteFrame* spriteFrame, const Rect& capInsets) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithSpriteFrame(spriteFrame, capInsets) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::createWithSpriteFrame(SpriteFrame* spriteFrame) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithSpriteFrame(spriteFrame) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::createWithSpriteFrameName(const std::string& spriteFrameName, const Rect& capInsets) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithSpriteFrameName(spriteFrameName, capInsets) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite* Scale9Sprite::createWithSpriteFrameName(const std::string& spriteFrameName) { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->initWithSpriteFrameName(spriteFrameName) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); log("Could not allocate Scale9Sprite()"); return nullptr; } void Scale9Sprite::cleanupSlicedSprites() { CC_SAFE_DELETE_ARRAY(_sliceIndices); CC_SAFE_DELETE_ARRAY(_sliceVertices); } void Scale9Sprite::setBlendFunc(const BlendFunc &blendFunc) { _blendFunc = blendFunc; applyBlendFunc(); } const BlendFunc &Scale9Sprite::getBlendFunc() const { return _blendFunc; } void Scale9Sprite::updateBlendFunc(Texture2D *texture) { // it is possible to have an untextured sprite if (! texture || ! texture->hasPremultipliedAlpha()) { _blendFunc = BlendFunc::ALPHA_NON_PREMULTIPLIED; setOpacityModifyRGB(false); } else { _blendFunc = BlendFunc::ALPHA_PREMULTIPLIED; setOpacityModifyRGB(true); } } void Scale9Sprite::applyBlendFunc() { if(_scale9Image) _scale9Image->setBlendFunc(_blendFunc); } bool Scale9Sprite::updateWithBatchNode(cocos2d::SpriteBatchNode *batchnode, const cocos2d::Rect &originalRect, bool rotated, const cocos2d::Rect &capInsets) { Sprite *sprite = Sprite::createWithTexture(batchnode->getTexture()); return this->updateWithSprite(sprite, originalRect, rotated, Vec2::ZERO, originalRect.size, capInsets); } bool Scale9Sprite::updateWithSprite(Sprite* sprite, const Rect& rect, bool rotated, const Rect& capInsets) { return updateWithSprite(sprite, rect, rotated, Vec2::ZERO, rect.size, capInsets); } bool Scale9Sprite::updateWithSprite(Sprite* sprite, const Rect& textureRect, bool rotated, const Vec2 &offset, const Size &originalSize, const Rect& capInsets) { // Release old sprites this->cleanupSlicedSprites(); updateBlendFunc(sprite?sprite->getTexture():nullptr); if(nullptr != sprite) { if (nullptr == sprite->getSpriteFrame()) { return false; } if (nullptr == _scale9Image) { _scale9Image = sprite; _scale9Image->retain(); } else { _scale9Image->setSpriteFrame(sprite->getSpriteFrame()); } } else { CC_SAFE_RELEASE_NULL(_scale9Image); } if (!_scale9Image) { return false; } Rect rect(textureRect); Size size(originalSize); // If there is no given rect if ( rect.equals(Rect::ZERO) ) { // Get the texture size as original Size textureSize = _scale9Image->getTexture()->getContentSize(); rect = Rect(0, 0, textureSize.width, textureSize.height); } if( size.equals(Size::ZERO) ) { size = rect.size; } // Set the given rect's size as original size _spriteRect = rect; _spriteFrameRotated = rotated; _originalSize = size; _preferredSize = size; _offset = offset; _capInsetsInternal = capInsets; if (_scale9Enabled) { _scale9Image->setAnchorPoint(Vec2::ZERO); _scale9Image->setPosition(Vec2::ZERO); _sliceSpriteDirty = true; } applyBlendFunc(); this->setState(_brightState); if(this->_isPatch9) { size.width = size.width - 2; size.height = size.height - 2; } this->setContentSize(size); return true; } void Scale9Sprite::configureSimpleModeRendering() { this->_insetLeft = 0; this->_insetTop = 0; this->_insetRight = 0; this->_insetBottom = 0; this->updateCapInset(); } void Scale9Sprite::createSlicedSprites() { //todo create sliced sprite if (_scale9Enabled) { Texture2D *tex = _scale9Image ? _scale9Image->getTexture() : nullptr; if (tex == nullptr) { return; } if (_renderingType == RenderingType::SIMPLE) { this->configureSimpleModeRendering(); } auto capInsets = CC_RECT_POINTS_TO_PIXELS(_capInsetsInternal); auto textureRect = CC_RECT_POINTS_TO_PIXELS(_spriteRect); auto spriteRectSize = _spriteRect.size; auto originalSize = CC_SIZE_POINTS_TO_PIXELS(_originalSize); auto offset = CC_POINT_POINTS_TO_PIXELS(_offset); Vec4 offsets; offsets.x = offset.x + (originalSize.width - textureRect.size.width) / 2; offsets.w = offset.y + (originalSize.height - textureRect.size.height) / 2; offsets.z = originalSize.width - textureRect.size.width - offsets.x; offsets.y = originalSize.height - textureRect.size.height - offsets.w; //handle .9.png if (_isPatch9) { originalSize = Size(originalSize.width - 2, originalSize.height-2); } if(capInsets.equals(Rect::ZERO)) { capInsets = Rect(originalSize.width/3, originalSize.height/3, originalSize.width/3, originalSize.height/3); } auto uv = this->calculateUV(tex, capInsets, originalSize, offsets); auto vertices = this->calculateVertices(capInsets, originalSize, offsets); auto triangles = this->calculateTriangles(uv, vertices); auto polyInfo = _scale9Image->getPolygonInfo(); polyInfo.setTriangles(triangles); _scale9Image->setPolygonInfo(polyInfo); } } void Scale9Sprite::setContentSize(const Size &size) { if (_contentSize.equals(size)) { return; } Node::setContentSize(size); _preferredSize = size; _sliceSpriteDirty = true; this->adjustNoneScale9ImagePosition(); } Scale9Sprite* Scale9Sprite::resizableSpriteWithCapInsets(const Rect& capInsets) const { Scale9Sprite* pReturn = new (std::nothrow) Scale9Sprite(); if ( pReturn && pReturn->init(_scale9Image, _spriteRect, _spriteFrameRotated, Vec2::ZERO, _originalSize, capInsets) ) { pReturn->autorelease(); return pReturn; } CC_SAFE_DELETE(pReturn); return nullptr; } Scale9Sprite::State Scale9Sprite::getState()const { return _brightState; } void Scale9Sprite::setState(cocos2d::ui::Scale9Sprite::State state) { auto getScale9ImageTexture = [this] { return _scale9Image != nullptr ? _scale9Image->getTexture() : nullptr; }; GLProgramState *glState = nullptr; switch (state) { case State::NORMAL: { glState = GLProgramState::getOrCreateWithGLProgramName(GLProgram::SHADER_NAME_POSITION_TEXTURE_COLOR_NO_MVP, getScale9ImageTexture()); } break; case State::GRAY: { glState = GLProgramState::getOrCreateWithGLProgramName(GLProgram::SHADER_NAME_POSITION_GRAYSCALE, getScale9ImageTexture()); } default: break; } setGLProgramState(glState); _brightState = state; } /** sets the opacity. @warning If the texture has premultiplied alpha then, the R, G and B channels will be modified. Values goes from 0 to 255, where 255 means fully opaque. */ void Scale9Sprite::updateCapInset() { Rect insets; insets = Rect(_insetLeft, _insetTop, _originalSize.width-_insetLeft-_insetRight, _originalSize.height-_insetTop-_insetBottom); this->setCapInsets(insets); } void Scale9Sprite::setSpriteFrame(SpriteFrame * spriteFrame, const Rect& capInsets) { Sprite * sprite = Sprite::createWithTexture(spriteFrame->getTexture()); this->updateWithSprite(sprite, spriteFrame->getRect(), spriteFrame->isRotated(), spriteFrame->getOffset(), spriteFrame->getOriginalSize(), capInsets); // Reset insets this->_insetLeft = capInsets.origin.x; this->_insetTop = capInsets.origin.y; this->_insetRight = _originalSize.width - _insetLeft - capInsets.size.width; this->_insetBottom = _originalSize.height - _insetTop - capInsets.size.height; } void Scale9Sprite::setPreferredSize(const Size& preferredSize) { this->setContentSize(preferredSize); } void Scale9Sprite::setCapInsets(const Rect& capInsets) { Size contentSize = this->_contentSize; this->updateWithSprite(this->_scale9Image, _spriteRect, _spriteFrameRotated, _offset, _originalSize, capInsets); this->_insetLeft = capInsets.origin.x; this->_insetTop = capInsets.origin.y; this->_insetRight = _originalSize.width - _insetLeft - capInsets.size.width; this->_insetBottom = _originalSize.height - _insetTop - capInsets.size.height; this->setContentSize(contentSize); } void Scale9Sprite::setInsetLeft(float insetLeft) { this->_insetLeft = insetLeft; this->updateCapInset(); } void Scale9Sprite::setInsetTop(float insetTop) { this->_insetTop = insetTop; this->updateCapInset(); } void Scale9Sprite::setInsetRight(float insetRight) { this->_insetRight = insetRight; this->updateCapInset(); } void Scale9Sprite::setInsetBottom(float insetBottom) { this->_insetBottom = insetBottom; this->updateCapInset(); } void Scale9Sprite::draw(Renderer *renderer, const Mat4 &transform, uint32_t flags) { if (_scale9Image && _scale9Enabled) { #if CC_USE_CULLING // Don't do calculate the culling if the transform was not updated auto visitingCamera = Camera::getVisitingCamera(); auto defaultCamera = Camera::getDefaultCamera(); if (visitingCamera == defaultCamera) { _insideBounds = ((flags & FLAGS_TRANSFORM_DIRTY)|| visitingCamera->isViewProjectionUpdated()) ? renderer->checkVisibility(transform, _contentSize) : _insideBounds; } else { _insideBounds = renderer->checkVisibility(transform, _contentSize); } if(_insideBounds) #endif { auto texture = _scale9Image->getTexture(); auto programState = _scale9Image->getGLProgramState(); auto blendFunc = _scale9Image->getBlendFunc(); auto& polyInfo = _scale9Image->getPolygonInfo(); auto globalZOrder = _scale9Image->getGlobalZOrder(); // ETC1 ALPHA support _trianglesCommand.init(globalZOrder,texture, programState, blendFunc, polyInfo.triangles, transform, flags); renderer->addCommand(&_trianglesCommand); #if CC_SPRITE_DEBUG_DRAW _debugDrawNode->clear(); auto count = polyInfo.triangles.indexCount/3; auto indices = polyInfo.triangles.indices; auto verts = polyInfo.triangles.verts; for(ssize_t i = 0; i < count; i++) { //draw 3 lines Vec3 from =verts[indices[i*3]].vertices; Vec3 to = verts[indices[i*3+1]].vertices; _debugDrawNode->drawLine(Vec2(from.x, from.y), Vec2(to.x,to.y), Color4F::WHITE); from =verts[indices[i*3+1]].vertices; to = verts[indices[i*3+2]].vertices; _debugDrawNode->drawLine(Vec2(from.x, from.y), Vec2(to.x,to.y), Color4F::WHITE); from =verts[indices[i*3+2]].vertices; to = verts[indices[i*3]].vertices; _debugDrawNode->drawLine(Vec2(from.x, from.y), Vec2(to.x,to.y), Color4F::WHITE); } #endif //CC_SPRITE_DEBUG_DRAW } } } void Scale9Sprite::visit(Renderer *renderer, const Mat4 &parentTransform, uint32_t parentFlags) { // quick return if not visible. children won't be drawn. if (!_visible) { return; } if (_scale9Enabled && _sliceSpriteDirty) { this->createSlicedSprites(); _sliceSpriteDirty = false; } uint32_t flags = processParentFlags(parentTransform, parentFlags); // IMPORTANT: // To ease the migration to v3.0, we still support the Mat4 stack, // but it is deprecated and your code should not rely on it Director* director = Director::getInstance(); CCASSERT(nullptr != director, "Director is null when setting matrix stack"); director->pushMatrix(MATRIX_STACK_TYPE::MATRIX_STACK_MODELVIEW); director->loadMatrix(MATRIX_STACK_TYPE::MATRIX_STACK_MODELVIEW, _modelViewTransform); int i = 0; // used by _children sortAllChildren(); // // draw children and protectedChildren zOrder < 0 // for(auto size = _children.size(); i < size; i++) { auto node = _children.at(i); if ( node && node->getLocalZOrder() < 0 ) node->visit(renderer, _modelViewTransform, flags); else break; } if (!_scale9Enabled && _scale9Image && _scale9Image->getLocalZOrder() < 0 ) { _scale9Image->visit(renderer, _modelViewTransform, flags); } // draw self // if (isVisitableByVisitingCamera()) this->draw(renderer, _modelViewTransform, flags); if (!_scale9Enabled && _scale9Image && _scale9Image->getLocalZOrder() >= 0 ) { _scale9Image->visit(renderer, _modelViewTransform, flags); } for(auto it=_children.cbegin()+i, itCend = _children.cend(); it != itCend; ++it) (*it)->visit(renderer, _modelViewTransform, flags); // FIX ME: Why need to set _orderOfArrival to 0?? // Please refer to https://github.com/cocos2d/cocos2d-x/pull/6920 // setOrderOfArrival(0); director->popMatrix(MATRIX_STACK_TYPE::MATRIX_STACK_MODELVIEW); } Size Scale9Sprite::getOriginalSize()const { return _originalSize; } Size Scale9Sprite::getPreferredSize() const { return _preferredSize; } Rect Scale9Sprite::getCapInsets()const { return _capInsetsInternal; } float Scale9Sprite::getInsetLeft()const { return this->_insetLeft; } float Scale9Sprite::getInsetTop()const { return this->_insetTop; } float Scale9Sprite::getInsetRight()const { return this->_insetRight; } float Scale9Sprite::getInsetBottom()const { return this->_insetBottom; } void Scale9Sprite::setScale9Enabled(bool enabled) { if (_scale9Enabled == enabled) { return; } _scale9Enabled = enabled; this->cleanupSlicedSprites(); //we must invalid the transform when toggling scale9enabled _transformUpdated = _transformDirty = _inverseDirty = true; if (_scale9Enabled) { if (_scale9Image) { this->updateWithSprite(this->_scale9Image, _spriteRect, _spriteFrameRotated, Vec2::ZERO, _originalSize, _capInsetsInternal); } } else { if (_scale9Image) { auto quad = _scale9Image->getQuad(); PolygonInfo polyInfo; polyInfo.setQuad(&quad); _scale9Image->setPolygonInfo(polyInfo); } } this->adjustNoneScale9ImagePosition(); } bool Scale9Sprite::isScale9Enabled() const { return _scale9Enabled; } void Scale9Sprite::setAnchorPoint(const cocos2d::Vec2 &position) { Node::setAnchorPoint(position); if (!_scale9Enabled) { if (_scale9Image) { _nonSliceSpriteAnchor = position; _scale9Image->setAnchorPoint(position); this->adjustNoneScale9ImagePosition(); } } } void Scale9Sprite::adjustNoneScale9ImagePosition() { if (_scale9Image) { if (!_scale9Enabled) { _scale9Image->setAnchorPoint(_nonSliceSpriteAnchor); _scale9Image->setPosition(_contentSize.width * _scale9Image->getAnchorPoint().x, _contentSize.height * _scale9Image->getAnchorPoint().y); } } } void Scale9Sprite::updateDisplayedColor(const cocos2d::Color3B &parentColor) { _displayedColor.r = _realColor.r * parentColor.r/255.0; _displayedColor.g = _realColor.g * parentColor.g/255.0; _displayedColor.b = _realColor.b * parentColor.b/255.0; updateColor(); if (_scale9Image) { _scale9Image->updateDisplayedColor(_displayedColor); } if (_cascadeColorEnabled) { for(const auto &child : _children) { child->updateDisplayedColor(_displayedColor); } } } void Scale9Sprite::updateDisplayedOpacity(GLubyte parentOpacity) { _displayedOpacity = _realOpacity * parentOpacity/255.0; updateColor(); if (_scale9Image) { _scale9Image->updateDisplayedOpacity(_displayedOpacity); } if (_cascadeOpacityEnabled) { for(auto child : _children) { child->updateDisplayedOpacity(_displayedOpacity); } } } void Scale9Sprite::disableCascadeColor() { for(auto child : _children) { child->updateDisplayedColor(Color3B::WHITE); } if (_scale9Image) { _scale9Image->updateDisplayedColor(Color3B::WHITE); } } void Scale9Sprite::disableCascadeOpacity() { _displayedOpacity = _realOpacity; for(auto child : _children){ child->updateDisplayedOpacity(255); } } void Scale9Sprite::setGLProgram(GLProgram *glprogram) { Node::setGLProgram(glprogram); if (_scale9Image) { _scale9Image->setGLProgram(glprogram); } } void Scale9Sprite::setGLProgramState(GLProgramState *glProgramState) { Node::setGLProgramState(glProgramState); if (_scale9Image) { _scale9Image->setGLProgramState(glProgramState); } } Sprite* Scale9Sprite::getSprite()const { return _scale9Image; } void Scale9Sprite::setFlippedX(bool flippedX) { float realScale = this->getScaleX(); _flippedX = flippedX; this->setScaleX(realScale); } void Scale9Sprite::setFlippedY(bool flippedY) { float realScale = this->getScaleY(); _flippedY = flippedY; this->setScaleY(realScale); } bool Scale9Sprite::isFlippedX()const { return _flippedX; } bool Scale9Sprite::isFlippedY()const { return _flippedY; } void Scale9Sprite::setScaleX(float scaleX) { if (_flippedX) { scaleX = scaleX * -1; } Node::setScaleX(scaleX); } void Scale9Sprite::setScaleY(float scaleY) { if (_flippedY) { scaleY = scaleY * -1; } Node::setScaleY(scaleY); } void Scale9Sprite::setScale(float scale) { this->setScaleX(scale); this->setScaleY(scale); this->setScaleZ(scale); } void Scale9Sprite::setScale(float scaleX, float scaleY) { this->setScaleX(scaleX); this->setScaleY(scaleY); } float Scale9Sprite::getScaleX()const { float originalScale = Node::getScaleX(); if (_flippedX) { originalScale = originalScale * -1.0f; } return originalScale; } float Scale9Sprite::getScaleY()const { float originalScale = Node::getScaleY(); if (_flippedY) { originalScale = originalScale * -1.0f; } return originalScale; } float Scale9Sprite::getScale()const { CCASSERT(this->getScaleX() == this->getScaleY(), "Scale9Sprite#scale. ScaleX != ScaleY. Don't know which one to return"); return this->getScaleX(); } void Scale9Sprite::setCameraMask(unsigned short mask, bool applyChildren) { Node::setCameraMask(mask, applyChildren); if(_scale9Image) _scale9Image->setCameraMask(mask,applyChildren); } // (0,0) O = capInsets.origin // v0---------------------- // | | | | // | | | | // v1-------O------+------| // | | | | // | | | | // v2-------+------+------| // | | | | // | | | | // v3-------------------- (1,1) (texture coordinate is flipped) // u0 u1 u2 u3 std::vector Scale9Sprite::calculateUV(Texture2D *tex, const Rect& capInsets, const Size& originalSize, const Vec4& offsets) { auto atlasWidth = tex->getPixelsWide(); auto atlasHeight = tex->getPixelsHigh(); //calculate texture coordinate float leftWidth = 0, centerWidth = 0, rightWidth = 0; float topHeight = 0, centerHeight = 0, bottomHeight = 0; if (_spriteFrameRotated) { rightWidth = capInsets.origin.y - offsets.y; centerWidth = capInsets.size.height; leftWidth = originalSize.height - centerWidth - capInsets.origin.y - offsets.w; topHeight = capInsets.origin.x - offsets.x; centerHeight = capInsets.size.width; bottomHeight = originalSize.width - (capInsets.origin.x + centerHeight) - offsets.z; } else { leftWidth = capInsets.origin.x - offsets.x; centerWidth = capInsets.size.width; rightWidth = originalSize.width - (capInsets.origin.x + centerWidth) - offsets.z; topHeight = capInsets.origin.y - offsets.y; centerHeight = capInsets.size.height; bottomHeight = originalSize.height - (capInsets.origin.y + centerHeight) - offsets.w; } if(leftWidth<0) { centerWidth += leftWidth; leftWidth = 0; } if(rightWidth<0) { centerWidth += rightWidth; rightWidth = 0; } if(topHeight<0) { centerHeight += topHeight; topHeight = 0; } if(bottomHeight<0) { centerHeight += bottomHeight; bottomHeight = 0; } auto textureRect = CC_RECT_POINTS_TO_PIXELS(_spriteRect); //handle .9.png if (_isPatch9) { //This magic number is used to avoiding artifact with .9.png format. float offset = 1.3f; textureRect = Rect(textureRect.origin.x + offset, textureRect.origin.y + offset, textureRect.size.width - 2, textureRect.size.height - 2); } //uv computation should take spritesheet into account. float u0, u1, u2, u3; float v0, v1, v2, v3; if (_spriteFrameRotated) { u0 = textureRect.origin.x / atlasWidth; u1 = (leftWidth + textureRect.origin.x) / atlasWidth; u2 = (leftWidth + centerWidth + textureRect.origin.x) / atlasWidth; u3 = (textureRect.origin.x + textureRect.size.height) / atlasWidth; v3 = textureRect.origin.y / atlasHeight; v2 = (topHeight + textureRect.origin.y) / atlasHeight; v1 = (topHeight + centerHeight + textureRect.origin.y) / atlasHeight; v0 = (textureRect.origin.y + textureRect.size.width) / atlasHeight; } else { u0 = textureRect.origin.x / atlasWidth; u1 = (leftWidth + textureRect.origin.x) / atlasWidth; u2 = (leftWidth + centerWidth + textureRect.origin.x) / atlasWidth; u3 = (textureRect.origin.x + textureRect.size.width) / atlasWidth; v0 = textureRect.origin.y / atlasHeight; v1 = (topHeight + textureRect.origin.y) / atlasHeight; v2 = (topHeight + centerHeight + textureRect.origin.y) / atlasHeight; v3 = (textureRect.origin.y + textureRect.size.height) / atlasHeight; } std::vector uvCoordinates; if (_renderingType == RenderingType::SIMPLE) { uvCoordinates = {Vec2(u0,v3), Vec2(u3,v0)}; } else { uvCoordinates = {Vec2(u0,v3), Vec2(u1,v2), Vec2(u2,v1), Vec2(u3,v0)}; } return uvCoordinates; } // // y3----------------------(preferedSize.width, preferedSize.height) // | | | | // | | | | // y2-------O------+------| // | | | | // | | | | // y1-------+------+------| // | | | | // | | | | //x0,y0-------------------- // x1 x2 x3 std::vector Scale9Sprite::calculateVertices(const Rect& capInsets, const Size& originalSize, const Vec4& offsets) { float offsetLeft = offsets.x / CC_CONTENT_SCALE_FACTOR(); float offsetTop = offsets.y / CC_CONTENT_SCALE_FACTOR(); float offsetRight = offsets.z / CC_CONTENT_SCALE_FACTOR(); float offsetBottom = offsets.w / CC_CONTENT_SCALE_FACTOR(); std::vector vertices; if (_renderingType == RenderingType::SIMPLE) { float hScale = _preferredSize.width / (originalSize.width / CC_CONTENT_SCALE_FACTOR()); float vScale = _preferredSize.height / (originalSize.height / CC_CONTENT_SCALE_FACTOR()); vertices = {Vec2(offsetLeft * hScale, offsetBottom * vScale), Vec2(_preferredSize.width - offsetRight * hScale, _preferredSize.height - offsetTop * vScale)}; } else { float leftWidth = 0, centerWidth = 0, rightWidth = 0; float topHeight = 0, centerHeight = 0, bottomHeight = 0; leftWidth = capInsets.origin.x; centerWidth = capInsets.size.width; rightWidth = originalSize.width - (leftWidth + centerWidth); topHeight = capInsets.origin.y; centerHeight = capInsets.size.height; bottomHeight = originalSize.height - (topHeight + centerHeight); leftWidth = leftWidth / CC_CONTENT_SCALE_FACTOR(); rightWidth = rightWidth / CC_CONTENT_SCALE_FACTOR(); centerWidth = centerWidth / CC_CONTENT_SCALE_FACTOR(); topHeight = topHeight / CC_CONTENT_SCALE_FACTOR(); bottomHeight = bottomHeight / CC_CONTENT_SCALE_FACTOR(); centerHeight = centerHeight / CC_CONTENT_SCALE_FACTOR(); float sizableWidth = _preferredSize.width - leftWidth - rightWidth; float sizableHeight = _preferredSize.height - topHeight - bottomHeight; leftWidth -= offsetLeft; rightWidth -= offsetRight; topHeight -= offsetTop; bottomHeight -= offsetBottom; float hScale = sizableWidth / centerWidth; float vScale = sizableHeight / centerHeight; if(leftWidth<0) { offsetLeft -= leftWidth * (hScale - 1.0f); sizableWidth += leftWidth * hScale; leftWidth = 0; } if(rightWidth<0) { sizableWidth += rightWidth * hScale; rightWidth = 0; } if(topHeight<0) { sizableHeight += topHeight * vScale; topHeight = 0; } if(bottomHeight<0) { offsetBottom -= bottomHeight * (vScale - 1.0f); sizableHeight += bottomHeight * vScale; bottomHeight = 0; } float x0,x1,x2,x3; float y0,y1,y2,y3; if(sizableWidth >= 0) { x0 = offsetLeft; x1 = x0 + leftWidth; x2 = x1 + sizableWidth; x3 = x2 + rightWidth; } else { float xScale = _preferredSize.width / (leftWidth + rightWidth); x0 = offsetLeft; x1 = x2 = offsetLeft + leftWidth * xScale; x3 = x2 + rightWidth * xScale; } if(sizableHeight >= 0) { y0 = offsetBottom; y1 = y0 + bottomHeight; y2 = y1 + sizableHeight; y3 = y2 + topHeight; } else { float yScale = _preferredSize.height / (topHeight + bottomHeight); y0 = offsetBottom; y1 = y2 = y0 + bottomHeight * yScale; y3 = y2 + topHeight * yScale; } vertices = {Vec2(x0,y0), Vec2(x1,y1), Vec2(x2,y2), Vec2(x3,y3)}; } return vertices; } TrianglesCommand::Triangles Scale9Sprite::calculateTriangles(const std::vector& uv, const std::vector& vertices) { const unsigned short slicedTotalVertexCount = powf(uv.size(),2); const unsigned short slicedTotalIndices = 6 * powf(uv.size() -1, 2); CC_SAFE_DELETE_ARRAY(_sliceVertices); CC_SAFE_DELETE_ARRAY(_sliceIndices); _sliceVertices = new (std::nothrow) V3F_C4B_T2F[slicedTotalVertexCount]; _sliceIndices = new (std::nothrow) unsigned short[slicedTotalIndices]; unsigned short indicesStart = 0; const unsigned short indicesOffset = 6; const unsigned short sliceQuadIndices[] = {4,0,5, 1,5,0}; const unsigned short simpleQuadIndices[] = {0,1,2, 3,2,1}; auto displayedColor = _scale9Image->getDisplayedColor(); auto displayedOpacity = _scale9Image->getDisplayedOpacity(); Color4B color4( displayedColor.r, displayedColor.g, displayedColor.b, displayedOpacity ); // special opacity for premultiplied textures if (_scale9Image->isOpacityModifyRGB()) { color4.r *= displayedOpacity/255.0f; color4.g *= displayedOpacity/255.0f; color4.b *= displayedOpacity/255.0f; } int vertexCount = (int)(vertices.size() - 1); for (int j = 0; j <= vertexCount; ++j) { for (int i = 0; i <= vertexCount; ++i) { V3F_C4B_T2F vertextData; vertextData.vertices.x = vertices[i].x; vertextData.vertices.y = vertices[j].y; if (_spriteFrameRotated) { vertextData.texCoords.u = uv[j].x; vertextData.texCoords.v = uv[i].y; } else { vertextData.texCoords.u = uv[i].x; vertextData.texCoords.v = uv[j].y; } vertextData.colors = color4; //if slice mode if (_renderingType == RenderingType::SLICE) { memcpy(_sliceVertices + i + j * 4, &vertextData, sizeof(V3F_C4B_T2F)); } else { memcpy(_sliceVertices + i + j * 2, &vertextData, sizeof(V3F_C4B_T2F)); } } } if (_renderingType == RenderingType::SLICE) { for (int j = 0; j <= vertexCount; ++j) { for (int i = 0; i <= vertexCount; ++i) { if (i < 3 && j < 3) { memcpy(_sliceIndices + indicesStart, sliceQuadIndices, indicesOffset * sizeof(unsigned short)); for (int k = 0; k < indicesOffset; ++k) { unsigned short actualIndex = (i + j * 3) * indicesOffset; _sliceIndices[k + actualIndex] = _sliceIndices[k + actualIndex] + j * 4 + i; } indicesStart = indicesStart + indicesOffset; } } } } if (_renderingType == RenderingType::SIMPLE) { memcpy(_sliceIndices, simpleQuadIndices, indicesOffset * sizeof(unsigned short)); } TrianglesCommand::Triangles triangles; triangles.vertCount = slicedTotalVertexCount; triangles.indexCount = slicedTotalIndices; triangles.verts = _sliceVertices; triangles.indices = _sliceIndices; return triangles; } void Scale9Sprite::setRenderingType(cocos2d::ui::Scale9Sprite::RenderingType type) { if (_renderingType == type) { return; } _renderingType = type; _sliceSpriteDirty = true; } Scale9Sprite::RenderingType Scale9Sprite::getRenderingType()const { return _renderingType; } void Scale9Sprite::resetRender() { // Release old sprites this->cleanupSlicedSprites(); CC_SAFE_RELEASE_NULL(this->_scale9Image); } void Scale9Sprite::setGlobalZOrder(float globalZOrder) { Node::setGlobalZOrder(globalZOrder); if (_scale9Image) { _scale9Image->setGlobalZOrder(globalZOrder); } } }}