1主要思路

Image类的initWithImageFile()方法去初始化Image对象,在一开始时创建一次,用一个bool数组保存每个像素点是否透明度为0的信息。每次触发点击事件时,就根据这个数组的值来判断点击是否有效。IrregularButton类继承自Button类。

2详细设计

2.1成员变量

CC_SYNTHESIZE(int,m_iBtnID,iBtnID);
	int normalImageWidth_;
	int normalImageHeight_;
	bool* normalTransparent_;

m_iBtnID:按钮IDnormalImageWidth_:图片的宽度。normalImageHeight_:图片的高度。normalTransparent_:实际上是一个bool型的数组,用于存放每个像素点是否透明度为0,为0值为true

2.2Create函数

IrregularButton的静态创建函数create,覆盖父类create函数。

IrregularButton* IrregularButton::create(const std::string& normalImage,const std::string& selectedImage,const std::string& disableImage,TextureResType texType)
{
	IrregularButton* btn = new IrregularButton;
	if (btn && btn->init(normalImage,selectedImage,disableImage,texType)) {
		btn->autorelease();
		return btn;
	}
	CC_SAFE_DELETE(btn);
	return nullptr;
}

2.3init函数

bool IrregularButton::init(const std::string &normalImage,TextureResType texType)
{
	bool ret = true;
	do {
		if (!Button::init(normalImage,texType)) {
			ret = false;
			break;
		}
	} while (0);
	loadnormalTransparentInfo(normalImage);

	auto listener1 = EventListenerTouchOneByOne::create();	listener1->setSwallowtouches(true); 
	listener1->onTouchBegan = [=](Touch* touch,Event* event){
		Point locationInNode = convertToWorldspace(touch->getLocation());
		hitTest(locationInNode);
		return true;
	};
	listener1->onTouchMoved = [=](Touch* touch,Event* event){
	};
	listener1->onTouchEnded = [=](Touch* touch,Event* event){
	};
	_eventdispatcher->addEventListenerWithSceneGraPHPriority(listener1,this);
	return ret;
}

这里做以下几件事:1.先调用父类init函数,使得的_buttonnormalRenderer确定,即确定按钮的大小;2.调用loadnormalTransparentInfo函数,确定normalTransparent_的数组值,即得到了精灵的每一点的是否透明。3.绑定触摸响应,并进行判断点击的那一点是否点击在精灵的透明区域。

2.4loadnormalTransparentInfo函数

IrregularButtonloadnormalTransparentInfo函数,保存normalTransparent_的数组值。

void IrregularButton::loadnormalTransparentInfo(std::string sName)
{
	Image* normalImage = new Image();
	normalImage->initWithImageFile(sName);
	normalImageWidth_ = normalImage->getWidth();
	normalImageHeight_ = normalImage->getHeight();
	this->setContentSize(CCSizeMake(normalImageWidth_,normalImageHeight_));
	auto dataLen = normalImage->getDataLen();
	if (normalTransparent_ != nullptr) {
		delete[] normalTransparent_;
	}
	auto normalPixels = normalImage->getData();
	normalTransparent_ = new bool[dataLen / (sizeof(unsigned char)* 4)];
	for (auto i = 0; i < normalImageHeight_; i++) {
		for (auto j = 0; j < normalImageWidth_; j++) {
			normalTransparent_[i * normalImageWidth_ + j] = (normalPixels[(i * normalImageWidth_ + j) * 4] == 0);
		}
	}
	delete normalImage;
}

initWithImageFile得到精灵的RGBA信息。扫描精灵的每一点,当前点透明时为true放入normalTransparent中,否则为false放入normalTransparent中。

2.5hitTest函数

IrregularButtonhitTest函数,用来确定点击的是否透明区域。

bool IrregularButton::hitTest(const Vec2 &pt)
{
	Vec2 localLocation = _buttonnormalRenderer->convertToNodeSpace(pt);
	Rect validTouchedRect;
	validTouchedRect.size = _buttonnormalRenderer->getContentSize();
	if (validTouchedRect.containsPoint(localLocation) && getIsTransparentAtPoint(localLocation) == false)
	{
		cclOG("IN");
		NotificationCenter::getInstance()->postNotification("NotifyIrregularBtn",(Ref*)m_iBtnID);
		return true;
	}
	return false;
}

首先点击坐标转化为世界坐标,再转化为相对精灵坐标,当点击点包含在精灵的范围内并且点击的是非透明区域,说明点中了这个不规则按钮,向外发送通知,通知传递自身的m_iBtnID号。

2.6getIsTransparentAtPoint函数

bool IrregularButton::getIsTransparentAtPoint(cocos2d::Vec2 point)
{
	point.y = _buttonnormalRenderer->getContentSize().height - point.y;
	int x = (int)point.x - 1;
	if (x < 0) {
		x = 0;
	}
	else if (x >= normalImageWidth_) {
		x = normalImageWidth_ - 1;
	}
	int y = (int)point.y - 1;
	if (y < 0) {
		y = 0;
	}
	else if (y >= normalImageHeight_) {
		y = normalImageHeight_ - 1;
	}
	return normalTransparent_[normalImageWidth_ * y + x];
}

这里值得注意的是相对精灵坐标的是从精灵的左下角为原点,这里要转化为以左上角为原点的坐标。

2.7调用

const string sNameN = "smile.png";
	const string sNameP = "angry.png";
	IrregularButton* alphaBtn = IrregularButton::create(sNameN,sNameP,sNameP);
	alphaBtn->setiBtnID(0);
	alphaBtn->setPosition(ccp(400,400));
	this->addChild(alphaBtn);
NotificationCenter::getInstance()->addobserver(this,callfuncO_selector(HelloWorld::onNotifyIrregularBtn),"NotifyIrregularBtn",NULL);

在调用的类中绑定通知,把转换的传入参数转为BtnID即可知道是哪个不规则按钮的通知。

void HelloWorld::onNotifyIrregularBtn(Ref* ref)
{
	int iBtnID = (int)ref;
}

实现效果

没点击时:

当点击到笑脸的圆形区域(即非透明区域)时:

点击精灵其他区域(透明区域)是不会响应的。IrregularButton类继承自Button类,这样按钮就可以通过传入三张精灵,实现正常,按压,禁止的按钮三种状态

工程源码下载:
http://download.csdn.net/detail/luoyikun/9099903

cocos3.x创建不规则按钮的更多相关文章

  1. HTML5 input新增type属性color颜色拾取器的实例代码

    type 属性规定 input 元素的类型。本文较详细的给大家介绍了HTML5 input新增type属性color颜色拾取器的实例代码,感兴趣的朋友跟随脚本之家小编一起看看吧

  2. amazeui模态框弹出后立马消失并刷新页面

    这篇文章主要介绍了amazeui模态框弹出后立马消失并刷新页面,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  3. 移动HTML5前端框架—MUI的使用

    这篇文章主要介绍了移动HTML5前端框架—MUI的使用的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  4. AmazeUI 模态窗口的实现代码

    这篇文章主要介绍了AmazeUI 模态窗口的实现代码,代码简单易懂,非常不错,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  5. ios – UIPopoverController出现在错误的位置

    所以我花了一些时间寻找答案,但到目前为止还没有找到任何答案.我正在尝试从UIInputAccessoryView上的按钮呈现弹出窗口.UIBarButtonItem我想显示popover来自定制视图,所以我可以使用图像.我创建这样的按钮:当需要显示popover时,我这样做:但我得到的是:弹出窗口看起来很好,但它应该出现在第一个按钮上时出现在第二个按钮上.然后我发现了这个问题:UIBarButto

  6. ios – 关闭UIBarButtonItem上的突出显示

    我正在尝试使用UIBarButtonItem在我的UIToolbar上添加标题.我使用简单的风格,看起来很好,但我似乎无法让它停止突出显示触摸.“突出显示时触摸”选项不适用于条形按钮项目.有没有快速简便的方法来做到这一点?

  7. 以编程方式调整iOS中的按钮大小

    我正在使用XCode4.6.1并开发iOS6.我在故事板上添加了一个按钮.我在我的实现文件ViewController.m中创建了一个插座:我尝试按如下所示更改按钮b1的属性(在同一个文件中:ViewController.m):当我在模拟器中运行应用程序时,按钮的alpha成功设置为0.5.但是,按钮的位置和大小不会改变.我尝试了各种方法来实现它.然而似乎没有任何作用.我想知道我做错了什么.我对O

  8. 如何在iOS / Swift的顶部导航栏中添加“继续”按钮

    我想在导航栏的右侧添加一个“继续”按钮.如何实现这一目标?我一直在尝试使用UIBarButtonItem上的一些方法,但无法使其正常工作.我迄今为止的最大努力是:但我在第一行遇到错误.它不喜欢“style”参数.我也试过了但没有运气.仍然停留在样式参数上.有任何想法吗?

  9. ios – 将图像添加到界面构建器中的按钮

    我想在我的按钮而不是文本中添加图像.我可以在界面构建器中这样做吗?我可以看一下这个例子吗?

  10. ios – 自定义BackBarButtonItem

    主要原因是您丢失了标准后退按钮滑动动画以更改视图.此外,这意味着我不需要使用自定义按钮或编写自定义方法返回.它只是起作用.希望这也能解决别人的问题,我知道我已经坚持了3个小时!

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部