mirror of https://github.com/axmolengine/axmol.git
445 lines
14 KiB
C++
445 lines
14 KiB
C++
/*
|
|
*
|
|
* Copyright 2012 Yannick Loriot. All rights reserved.
|
|
* http://yannickloriot.com
|
|
*
|
|
* 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 "CCControlSwitch.h"
|
|
#include "cocos2d.h"
|
|
|
|
NS_CC_EXT_BEGIN
|
|
// CCControlSwitchSprite
|
|
|
|
class CCControlSwitchSprite : public CCSprite, public CCActionTweenDelegate
|
|
{
|
|
public:
|
|
CCControlSwitchSprite();
|
|
virtual ~CCControlSwitchSprite();
|
|
bool initWithMaskSprite(
|
|
CCSprite *maskSprite,
|
|
CCSprite *onSprite,
|
|
CCSprite *offSprite,
|
|
CCSprite *thumbSprite,
|
|
CCLabelTTF* onLabel,
|
|
CCLabelTTF* offLabel);
|
|
void draw();
|
|
void needsLayout();
|
|
void setSliderXPosition(CCFloat sliderXPosition);
|
|
CCFloat getSliderXPosition() {return m_fSliderXPosition;}
|
|
CCFloat onSideWidth();
|
|
CCFloat offSideWidth();
|
|
virtual void updateTweenAction(float value, const char* key);
|
|
/** Contains the position (in x-axis) of the slider inside the receiver. */
|
|
CCFloat m_fSliderXPosition;
|
|
CC_SYNTHESIZE(CCFloat, m_fOnPosition, OnPosition)
|
|
CC_SYNTHESIZE(CCFloat, m_fOffPosition, OffPosition)
|
|
|
|
CC_SYNTHESIZE_RETAIN(CCTexture2D*, m_pMaskTexture, MaskTexture)
|
|
CC_SYNTHESIZE(GLuint, m_uTextureLocation, TextureLocation)
|
|
CC_SYNTHESIZE(GLuint, m_uMaskLocation, MaskLocation)
|
|
|
|
CC_SYNTHESIZE_RETAIN(CCSprite*, m_pOnSprite, OnSprite)
|
|
CC_SYNTHESIZE_RETAIN(CCSprite*, m_pOffSprite, OffSprite)
|
|
CC_SYNTHESIZE_RETAIN(CCSprite*, m_ThumbSprite, ThumbSprite)
|
|
CC_SYNTHESIZE_RETAIN(CCLabelTTF*, m_pOnLabel, OnLabel)
|
|
CC_SYNTHESIZE_RETAIN(CCLabelTTF*, m_pOffLabel, OffLabel)
|
|
};
|
|
|
|
CCControlSwitchSprite::CCControlSwitchSprite()
|
|
: m_fSliderXPosition(0.0f)
|
|
, m_fOnPosition(0.0f)
|
|
, m_fOffPosition(0.0f)
|
|
, m_pMaskTexture(NULL)
|
|
, m_uTextureLocation(0)
|
|
, m_uMaskLocation(0)
|
|
, m_pOnSprite(NULL)
|
|
, m_pOffSprite(NULL)
|
|
, m_ThumbSprite(NULL)
|
|
, m_pOnLabel(NULL)
|
|
, m_pOffLabel(NULL)
|
|
{
|
|
|
|
}
|
|
|
|
CCControlSwitchSprite::~CCControlSwitchSprite()
|
|
{
|
|
CC_SAFE_RELEASE(m_pOnSprite);
|
|
CC_SAFE_RELEASE(m_pOffSprite);
|
|
CC_SAFE_RELEASE(m_ThumbSprite);
|
|
CC_SAFE_RELEASE(m_pOnLabel);
|
|
CC_SAFE_RELEASE(m_pOffLabel);
|
|
CC_SAFE_RELEASE(m_pMaskTexture);
|
|
}
|
|
|
|
bool CCControlSwitchSprite::initWithMaskSprite(
|
|
CCSprite *maskSprite,
|
|
CCSprite *onSprite,
|
|
CCSprite *offSprite,
|
|
CCSprite *thumbSprite,
|
|
CCLabelTTF* onLabel,
|
|
CCLabelTTF* offLabel)
|
|
{
|
|
if (CCSprite::initWithTexture(maskSprite->getTexture()))
|
|
{
|
|
// Sets the default values
|
|
m_fOnPosition = 0;
|
|
m_fOffPosition = -onSprite->getContentSize().width + thumbSprite->getContentSize().width / 2;
|
|
m_fSliderXPosition = m_fOnPosition;
|
|
|
|
setOnSprite(onSprite);
|
|
setOffSprite(offSprite);
|
|
setThumbSprite(thumbSprite);
|
|
setOnLabel(onLabel);
|
|
setOffLabel(offLabel);
|
|
|
|
addChild(m_ThumbSprite);
|
|
|
|
// Set up the mask with the Mask shader
|
|
setMaskTexture(maskSprite->getTexture());
|
|
CCGLProgram* pProgram = new CCGLProgram();
|
|
pProgram->initWithVertexShaderByteArray(ccPositionTextureColor_vert, ccExSwitchMask_frag);
|
|
setShaderProgram(pProgram);
|
|
pProgram->release();
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
getShaderProgram()->addAttribute(kCCAttributeNamePosition, kCCVertexAttrib_Position);
|
|
getShaderProgram()->addAttribute(kCCAttributeNameColor, kCCVertexAttrib_Color);
|
|
getShaderProgram()->addAttribute(kCCAttributeNameTexCoord, kCCVertexAttrib_TexCoords);
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
getShaderProgram()->link();
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
getShaderProgram()->updateUniforms();
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
m_uTextureLocation = glGetUniformLocation( getShaderProgram()->getProgram(), "u_texture");
|
|
m_uMaskLocation = glGetUniformLocation( getShaderProgram()->getProgram(), "u_mask");
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
setContentSize(m_pMaskTexture->getContentSize());
|
|
|
|
needsLayout();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CCControlSwitchSprite::updateTweenAction(float value, const char* key)
|
|
{
|
|
CCLOG("key = %s, value = %f", key, value);
|
|
setSliderXPosition(value);
|
|
}
|
|
|
|
void CCControlSwitchSprite::draw()
|
|
{
|
|
CC_NODE_DRAW_SETUP();
|
|
|
|
ccGLEnableVertexAttribs(kCCVertexAttribFlag_PosColorTex);
|
|
ccGLBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
getShaderProgram()->setUniformForModelViewProjectionMatrix();
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
glBindTexture( GL_TEXTURE_2D, getTexture()->getName());
|
|
glUniform1i(m_uTextureLocation, 0);
|
|
|
|
glActiveTexture(GL_TEXTURE1);
|
|
glBindTexture( GL_TEXTURE_2D, m_pMaskTexture->getName() );
|
|
glUniform1i(m_uMaskLocation, 1);
|
|
|
|
#define kQuadSize sizeof(m_sQuad.bl)
|
|
long offset = (long)&m_sQuad;
|
|
|
|
// vertex
|
|
int diff = offsetof( ccV3F_C4B_T2F, vertices);
|
|
glVertexAttribPointer(kCCVertexAttrib_Position, 3, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff));
|
|
|
|
// texCoods
|
|
diff = offsetof( ccV3F_C4B_T2F, texCoords);
|
|
glVertexAttribPointer(kCCVertexAttrib_TexCoords, 2, GL_FLOAT, GL_FALSE, kQuadSize, (void*)(offset + diff));
|
|
|
|
// color
|
|
diff = offsetof( ccV3F_C4B_T2F, colors);
|
|
glVertexAttribPointer(kCCVertexAttrib_Color, 4, GL_UNSIGNED_BYTE, GL_TRUE, kQuadSize, (void*)(offset + diff));
|
|
|
|
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
|
|
glActiveTexture(GL_TEXTURE0);
|
|
}
|
|
|
|
void CCControlSwitchSprite::needsLayout()
|
|
{
|
|
m_pOnSprite->setPosition(ccp(m_pOnSprite->getContentSize().width / 2 + m_fSliderXPosition,
|
|
m_pOnSprite->getContentSize().height / 2));
|
|
m_pOffSprite->setPosition(ccp(m_pOnSprite->getContentSize().width + m_pOffSprite->getContentSize().width / 2 + m_fSliderXPosition,
|
|
m_pOffSprite->getContentSize().height / 2));
|
|
m_ThumbSprite->setPosition(ccp(m_pOnSprite->getContentSize().width + m_fSliderXPosition,
|
|
m_pMaskTexture->getContentSize().height / 2));
|
|
|
|
if (m_pOnLabel)
|
|
{
|
|
m_pOnLabel->setPosition(ccp(m_pOnSprite->getPosition().x - m_ThumbSprite->getContentSize().width / 6,
|
|
m_pOnSprite->getContentSize().height / 2));
|
|
}
|
|
if (m_pOffLabel)
|
|
{
|
|
m_pOffLabel->setPosition(ccp(m_pOffSprite->getPosition().x + m_ThumbSprite->getContentSize().width / 6,
|
|
m_pOffSprite->getContentSize().height / 2));
|
|
}
|
|
|
|
CCRenderTexture *rt = CCRenderTexture::create((int)m_pMaskTexture->getContentSize().width, (int)m_pMaskTexture->getContentSize().height);
|
|
|
|
rt->begin();
|
|
m_pOnSprite->visit();
|
|
m_pOffSprite->visit();
|
|
|
|
if (m_pOnLabel)
|
|
{
|
|
m_pOnLabel->visit();
|
|
}
|
|
if (m_pOffLabel)
|
|
{
|
|
m_pOffLabel->visit();
|
|
}
|
|
|
|
rt->end();
|
|
|
|
setTexture(rt->getSprite()->getTexture());
|
|
setFlipY(true);
|
|
}
|
|
|
|
void CCControlSwitchSprite::setSliderXPosition(CCFloat sliderXPosition)
|
|
{
|
|
if (sliderXPosition <= m_fOffPosition)
|
|
{
|
|
// Off
|
|
sliderXPosition = m_fOffPosition;
|
|
} else if (sliderXPosition >= m_fOnPosition)
|
|
{
|
|
// On
|
|
sliderXPosition = m_fOnPosition;
|
|
}
|
|
|
|
m_fSliderXPosition = sliderXPosition;
|
|
|
|
needsLayout();
|
|
}
|
|
|
|
|
|
CCFloat CCControlSwitchSprite::onSideWidth()
|
|
{
|
|
return m_pOnSprite->getContentSize().width;
|
|
}
|
|
|
|
CCFloat CCControlSwitchSprite::offSideWidth()
|
|
{
|
|
return m_pOffSprite->getContentSize().height;
|
|
}
|
|
|
|
|
|
// CCControlSwitch
|
|
|
|
CCControlSwitch::CCControlSwitch()
|
|
{
|
|
|
|
}
|
|
|
|
CCControlSwitch::~CCControlSwitch()
|
|
{
|
|
CC_SAFE_RELEASE(m_pSwitchSprite);
|
|
}
|
|
|
|
bool CCControlSwitch::initWithMaskSprite(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite)
|
|
{
|
|
return initWithMaskSprite(maskSprite, onSprite, offSprite, thumbSprite, NULL, NULL);
|
|
}
|
|
|
|
CCControlSwitch* CCControlSwitch::switchWithMaskSprite(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite)
|
|
{
|
|
return CCControlSwitch::create(maskSprite, onSprite, offSprite, thumbSprite);
|
|
}
|
|
|
|
CCControlSwitch* CCControlSwitch::create(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite)
|
|
{
|
|
CCControlSwitch* pRet = new CCControlSwitch();
|
|
if (pRet && pRet->initWithMaskSprite(maskSprite, onSprite, offSprite, thumbSprite, NULL, NULL))
|
|
{
|
|
pRet->autorelease();
|
|
}
|
|
else
|
|
{
|
|
CC_SAFE_DELETE(pRet);
|
|
}
|
|
return pRet;
|
|
}
|
|
|
|
bool CCControlSwitch::initWithMaskSprite(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite, CCLabelTTF* onLabel, CCLabelTTF* offLabel)
|
|
{
|
|
if (CCControl::init())
|
|
{
|
|
CCAssert(maskSprite, "Mask must not be nil.");
|
|
CCAssert(onSprite, "onSprite must not be nil.");
|
|
CCAssert(offSprite, "offSprite must not be nil.");
|
|
CCAssert(thumbSprite, "thumbSprite must not be nil.");
|
|
|
|
setIsTouchEnabled(true);
|
|
m_bOn = true;
|
|
|
|
m_pSwitchSprite = new CCControlSwitchSprite();
|
|
m_pSwitchSprite->initWithMaskSprite(maskSprite,
|
|
onSprite,
|
|
offSprite,
|
|
thumbSprite,
|
|
onLabel,
|
|
offLabel);
|
|
m_pSwitchSprite->setPosition(ccp (m_pSwitchSprite->getContentSize().width / 2, m_pSwitchSprite->getContentSize().height / 2));
|
|
addChild(m_pSwitchSprite);
|
|
|
|
setIgnoreAnchorPointForPosition(false);
|
|
setAnchorPoint(ccp (0.5f, 0.5f));
|
|
setContentSize(m_pSwitchSprite->getContentSize());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
CCControlSwitch* CCControlSwitch::switchWithMaskSprite(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite, CCLabelTTF* onLabel, CCLabelTTF* offLabel)
|
|
{
|
|
return CCControlSwitch::create(maskSprite, onSprite, offSprite, thumbSprite, onLabel, offLabel);
|
|
}
|
|
|
|
CCControlSwitch* CCControlSwitch::create(CCSprite *maskSprite, CCSprite * onSprite, CCSprite * offSprite, CCSprite * thumbSprite, CCLabelTTF* onLabel, CCLabelTTF* offLabel)
|
|
{
|
|
CCControlSwitch* pRet = new CCControlSwitch();
|
|
if (pRet && pRet->initWithMaskSprite(maskSprite, onSprite, offSprite, thumbSprite, onLabel, offLabel))
|
|
{
|
|
pRet->autorelease();
|
|
}
|
|
else
|
|
{
|
|
CC_SAFE_DELETE(pRet);
|
|
}
|
|
return pRet;
|
|
}
|
|
|
|
void CCControlSwitch::setOn(bool isOn)
|
|
{
|
|
setOn(isOn, false);
|
|
}
|
|
|
|
void CCControlSwitch::setOn(bool isOn, bool animated)
|
|
{
|
|
m_bOn = isOn;
|
|
|
|
m_pSwitchSprite->runAction
|
|
(
|
|
CCActionTween::create
|
|
(
|
|
0.2f,
|
|
"sliderXPosition",
|
|
m_pSwitchSprite->getSliderXPosition(),
|
|
(m_bOn) ? m_pSwitchSprite->getOnPosition() : m_pSwitchSprite->getOffPosition()
|
|
)
|
|
);
|
|
|
|
sendActionsForControlEvents(CCControlEventValueChanged);
|
|
}
|
|
|
|
void CCControlSwitch::setIsEnabled(bool enabled)
|
|
{
|
|
m_bEnabled = enabled;
|
|
|
|
m_pSwitchSprite->setOpacity((enabled) ? 255 : 128);
|
|
}
|
|
|
|
CCPoint CCControlSwitch::locationFromTouch(CCTouch* pTouch)
|
|
{
|
|
CCPoint touchLocation = pTouch->locationInView(); // Get the touch position
|
|
touchLocation = CCDirector::sharedDirector()->convertToGL(touchLocation); // Convert the position to GL space
|
|
touchLocation = this->convertToNodeSpace(touchLocation); // Convert to the node space of this class
|
|
|
|
return touchLocation;
|
|
}
|
|
|
|
bool CCControlSwitch::ccTouchBegan(CCTouch *pTouch, CCEvent *pEvent)
|
|
{
|
|
if (!this->isTouchInside(pTouch)
|
|
|| !this->getIsEnabled())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
m_bMoved = false;
|
|
|
|
CCPoint location = this->locationFromTouch(pTouch);
|
|
|
|
m_fInitialTouchXPosition = location.x - m_pSwitchSprite->getSliderXPosition();
|
|
|
|
m_pSwitchSprite->getThumbSprite()->setColor(ccGRAY);
|
|
m_pSwitchSprite->needsLayout();
|
|
|
|
return true;
|
|
}
|
|
|
|
void CCControlSwitch::ccTouchMoved(CCTouch *pTouch, CCEvent *pEvent)
|
|
{
|
|
CCPoint location = this->locationFromTouch(pTouch);
|
|
location = ccp (location.x - m_fInitialTouchXPosition, 0);
|
|
|
|
m_bMoved = true;
|
|
|
|
m_pSwitchSprite->setSliderXPosition(location.x);
|
|
}
|
|
|
|
void CCControlSwitch::ccTouchEnded(CCTouch *pTouch, CCEvent *pEvent)
|
|
{
|
|
CCPoint location = this->locationFromTouch(pTouch);
|
|
|
|
m_pSwitchSprite->getThumbSprite()->setColor(ccWHITE);
|
|
|
|
if (hasMoved())
|
|
{
|
|
setOn(!(location.x < m_pSwitchSprite->getContentSize().width / 2), true);
|
|
}
|
|
else
|
|
{
|
|
setOn(!m_bOn, true);
|
|
}
|
|
}
|
|
|
|
void CCControlSwitch::ccTouchCancelled(CCTouch *pTouch, CCEvent *pEvent)
|
|
{
|
|
CCPoint location = this->locationFromTouch(pTouch);
|
|
|
|
m_pSwitchSprite->getThumbSprite()->setColor(ccWHITE);
|
|
|
|
if (hasMoved())
|
|
{
|
|
setOn(!(location.x < m_pSwitchSprite->getContentSize().width / 2), true);
|
|
} else
|
|
{
|
|
setOn(!m_bOn, true);
|
|
}
|
|
}
|
|
|
|
NS_CC_EXT_END
|