cocos2d-x多分辨率和随后的自适应CCListView的bug修复
cocos2d-x是一款众所周知的跨平台的游戏开发引擎。因为其跨平台的特性。多分辨率支持也自然就有其需求。
因此。在某一次更新中(抱歉,笔者已经忘了是哪次更新了),cocos2d-x加入了一个新的方法。能够很简便地让cocos2d依据屏幕尺寸的大小做自适配。
方法名:
virtual void setDesignResolutionSize(float width, float height, ResolutionPolicy resolutionPolicy);
该方法所传的參数中,前两个參数width、height,指的是开发人员在设计界面时默认设计的尺寸。比如:开发人员在设计界面时以iphone5为准,则width和height就是568*320;若是以iphone4为准。则width和height就是480*320;以此类推。
第三个參数resolutionPolicy指的是cocos2d在缩放时须要遵守的规则。ResolutionPolicy这个自己定义结构例如以下:

enum ResolutionPolicy
{
// The entire application is visible in the specified area without trying to preserve the original aspect ratio.
// Distortion can occur, and the application may appear stretched or compressed.
kResolutionExactFit,
// The entire application fills the specified area, without distortion but possibly with some cropping,
// while maintaining the original aspect ratio of the application.
kResolutionNoBorder,
// The entire application is visible in the specified area without distortion while maintaining the original
// aspect ratio of the application. Borders can appear on two sides of the application.
kResolutionShowAll, kResolutionUnKnown,
};

通过凝视能够看出:
kResolutionExactFit是指cocos2d缩放的时候不考虑画面失真,直接用画面填充满整个屏幕
kResolutionNoBorder是指cocos2d缩放的时候尽量顶满画面,假设因此有超出画面的部分,就会把超出的部分切除掉(举例来说:假设设计大小是568*320,在480*320大小的屏幕上自适配的时候,就会把宽度超出的部分切除)
kResolutionShowAll是指cocos2d缩放的时候显示全部的画面,假设有填不满屏幕的地方就会显示黑条。
这三种规则各有优缺点,详细使用哪种规则须要考量项目的须要进行合理的选择。
然而。须要特别注意的一点是:setDesignResolutionSize和enableRetina这两个方法是不能够同一时候使用的。
这即意味着使用自适配的时候是不能够做高清版的。
假设查看cocos2d-x源码。能够看到在setDesignResolutionSize方法中有一句代码:
CCAssert(m_bIsRetinaEnabled == false, "can not enable retina while set design resolution size!");
在下觉得这是由于高清模式下图片资源都是进行了缩小的,全部的坐标都是定义的point,而不是pixel。因此高清模式和自适配可能有某些冲突存在。
(以上仅仅是笔者的推測,不靠谱勿怪)
总之,假设想要使用自适配。就必须在项目一開始就放弃高清。假设要做高清版,眼下看来就仅仅能自己依据屏幕尺寸的比例变化设置setContentScaleFactor,这一部分的设置方法这里就不展开了。
另外,笔者在刚開始使用自适配功能时,误以为开发时要把UI元素的位置坐标和大小都写成相对于屏幕大小的一定比例。比如:
CCSprite* avatarSprite = CCSprite::create("avatar.png");
avatarSprite->setPosition(ccp(winSize.width * 0.000878, winSize.height * 0.653125));
addChild(avatarSprite);
但实际上这是没有必要的。
自适配功能是直接调整整个CCDirector的画布大小,因此不论什么加入到CCDirector中的元素都会被缩放对应的比例。没有必要再特意将坐标写成相对屏幕比例的模式。
以下笔者将分享一下在使用自适配功能的时候遇到的一个问题,即CCListView在多分辨率模式下可能出现的被过多地切除一部分内容的问题。
首先先介绍下笔者发现问题时的背景:当时笔者是在实现一款跨平台的游戏的当中一个界面,该游戏的画面是依照iphone5的低清版设计。DesignResolutionSize设置的是568*320。在安装到iphone5上时一切正常。可是随后,在安装到iphone4机器上进行调试时,发现画面上全部的CCListView都被切除了左边的一部分内容。
因为在iphone5上一切正常。在iphone4上画面被切除了一部分。因此笔者马上怀疑是CCListView与setDesignResolutionSize之间有某些冲突。
通过debug和log。最后将问题锁定到了CCListView类的visit方法。
方法的源码例如以下:

void CCListView::visit(void)
{
if (!m_pListViewParent)
{
CCRect rectSelf;
float factor = CC_CONTENT_SCALE_FACTOR();
rectSelf.origin = convertToWorldSpace(CCPoint(0,0));
rectSelf.origin.x *= factor;
rectSelf.origin.y *= factor;
rectSelf.size = this->getContentSize();
rectSelf.size.width *= factor;
rectSelf.size.height *= factor;
glScissor((GLsizei)rectSelf.origin.x, (GLsizei)rectSelf.origin.y, (GLsizei)rectSelf.size.width , (GLsizei)rectSelf.size.height);
glEnable(GL_SCISSOR_TEST);
}
CCLayerColor::visit();
if (!m_pListViewParent)
{
glDisable(GL_SCISSOR_TEST);
}
}

这段代码中调用了一个方法:glScissor。这种方法是框定了CCListView将会显示在屏幕上的范围。
而因为在使用自适配模式。对CC_CONTENT_SCALE_FACTOR没有做不论什么改动,因此这个框出来的范围是相应于DesignResolutionSize设定的原始设计画面大小的。而在屏幕尺寸发生了变化。整个CCDirector被缩放的情况下。这个rectSelf却没有做不论什么改变,所以就导致了上述的被切除一部分的问题。
随后,笔者通过搜索。发现已经有人遇到了同一个问题。并且还提供了修复的方法:http://www.cocos2d-x.org/boards/18/topics/14464
简单讲就是,用CCEGLView::sharedOpenGLView()->setScissorInPoints方法替换glScissor方法。其他不论什么地方都不用变。
改动好以后在iphone4和5上调试都没问题,因此这个bug本身能够算是修复了。
PS:因为对这种方法可以奏效感到好奇,笔者查看了setScissorInPoints方法的源文件。代码例如以下:

void CCEGLViewProtocol::setScissorInPoints(float x , float y , float w , float h)
{
glScissor((GLint)(x * m_fScaleX + m_obViewPortRect.origin.x),
(GLint)(y * m_fScaleY + m_obViewPortRect.origin.y),
(GLsizei)(w * m_fScaleX),
(GLsizei)(h * m_fScaleY));
}

能够看出这种方法内部仍然是调用了glScissor方法,可是传递的參数则考考虑到在屏幕尺寸的变化,因此,为了确保这种方法不切割过多CCListView内容。
其他,在project搜索glScissor发现CCScrollView这个方法也被称为,假定的使用CCScrollView建议同时更换的源代码glScissor方法。
cocos2d-x多分辨率和随后的自适应CCListView的bug修复的更多相关文章
- 常用样式制作思路 自定义按钮~自适应布局~常见bug seajs简记 初学者必知的HTML规范 不容忽略的——CSS规范
常用样式制作思路 学习常用样式总结参考来自这里 带点文字链接列表利用:before实现 1 <!DOCTYPE html> 2 <html lang="en" ...
- 【原】文本图片自适应高度小bug以及解决办法
自定义cell的文本图片自适应高度代码,如果存在自定义的cell赋值封装,就必须将自适应高度代码写在这个方法中
- 文本图片自适应高度小bug以及解决办法
自定义cell的文本图片自适应高度代码,如果存在自定义的cell赋值封装,就必须将自适应高度代码写在这个方法中 点击效果: 注:- (void)layoutSubviews 方法不能同时操作,否则会出 ...
- 自定义按钮~自适应布局~常见bug
一.元件 自定义按钮可用button或a display为 inline-block 方便设置格式,通过 padding,height,line-height,font-size设置按钮的大小 & ...
- Cocos2d-x项目移植到WP8小记
Cocos2d-x项目移植到WP8小记 作者: K.C. 日期: 10/24/2013 Date: 2013-10-24 00:33 Title: Cocos2d-x项目移植到WP8小记 Tags: ...
- margin设置为负数
1.为负margin“平反” 我们在CSS中都会使用margin,但将margin设置成负数,那可能就不大好处理了.在网页设计中,人们对负margin用法的态度大相径庭,有的人非常喜欢,而有的人则认为 ...
- 我的Android进阶之旅------>经典的大牛博客推荐(排名不分先后)!!
本文来自:http://blog.csdn.net/ouyang_peng/article/details/11358405 今天看到一篇文章,收藏了很多大牛的博客,在这里分享一下 谦虚的天下 柳志超 ...
- [PHP] 2018年终总结
去掉敏感信息后的不完整版 ==========================================================================2018年12月29日 记 ...
- 集显也能硬件编码:Intel SDK && 各种音视频编解码学习详解
http://blog.sina.com.cn/s/blog_4155bb1d0100soq9.html INTEL MEDIA SDK是INTEL推出的基于其内建显示核心的编解码技术,我们在播放高清 ...
随机推荐
- 【iOS发展-61】更换plist经过资源,执行iOS一旦数据仍显示在模拟器的外观,如何解决?
(1)案例介绍 --我们首先导入plist文件做项目,模拟的观看效果. --删除plist,更换一个新的plist,CMD+R模拟执行,或者找到该程序界面上显示最后一个数据. (2)原因 是由于第一次 ...
- eclipse设备SVN插入
原文:http://www.cnblogs.com/ruiati/p/3584120.html 1.下载最新的Eclipse.我的版本号是3.7.2 indigo(Eclipse IDE for Ja ...
- telnet发电子邮件
无聊今天的工作,想想一个学生被提到最后一次telnet发电子邮件,所以我想试试.最后,成功的实践,这里做个总结. 首先,cmd进telnet打开回话: 下面红色字体为命令. 1.open smtp.1 ...
- 最小二乘法拟合非线性函数及其Matlab/Excel 实现(转)
1.最小二乘原理 Matlab直接实现最小二乘法的示例: close x = 1:1:100; a = -1.5; b = -10; y = a*log(x)+b; yrand = y + 0.5*r ...
- 谷歌上不去,长期的解决方案。在稳定高速Google和Gmail
对稳定Google神器 国内Google很不稳定,缓慢并经常上不去,由"我想去Google",安全和稳定的使用Google.Gmail.Google+所以通常需要特殊的手段岗位胜任 ...
- SWFUpload多文件上传 文件数限制 setStats()
使用swfupload仿公平图片上传 SWFUpload它是基于flash与javascript的client文件上传组件. handlers.js文件 完毕文件入列队(fileQueued) → 完 ...
- Unity3d C# Socket 下载文件 (同步到)
续篇 Unity3d C# HttpWebRequest 异步下载文件 ,由于project编译为IL2CPP的情况下仍然无效.提示HttpWebrequest 在当前版本号不支持.所以还是寻求其他的 ...
- celery最佳实践
作为一个Celery使用重度用户.看到Celery Best Practices这篇文章.不由得菊花一紧. 干脆翻译出来,同一时候也会添加我们项目中celery的实战经验. 至于Celery为何物,看 ...
- 探索static——不需要能够使用该类实例?
在一般情况下.需要使用一个上课时间.你必须先实例化类,它调用的能力.在编程过程中发现.有些类不能直接实例来使用,利用其场.法等等. 这时候.靠的就是static作用.static英文意思为" ...
- 7.oracle学习门户系列七---网络管理和配置
oracle学习门户系列七 网络管理和配置 们学习了模式和用户.包含模式定义以及模式的作用. 这篇我么来看下ORACLE数据库中的网络管理和配置.只是这篇好像和上篇没有继承啊.这怎么看? Ok,事实上 ...