[cocos2dx]让CCScrollView支持分页

做过IOS开发的朋友, 肯定知道UIScrollView有一个isPaged属性. 当设置其为true的时候, 滑动会自动分页. 即, 每次滑动之后, 会停止在整页的位置. 当开始介入cocos2dx开发的时候, 却发现跟UIScrollView接口十分相似的CCScrollView却没有这个分页属性. 于是手动实现了一个.



关键字

cocos2dx, CCScrollView, page, UIScrollView

基础知识

在常见的图形引擎中, 滑动组件的定义里有两个重要的概念

  • viewSize: 这个大小值得是组件占用屏幕的大小. 即实际大小.
  • contentSize: 这个大小是一个虚拟的大小. 我们之所以要滚动, 必然是因为需要展示的内容比现实的屏幕空间大. 我们需要滚动屏幕, 才能浏览到所有的显示内容. 这个contentSize即是我们虚拟出来的, 需要展示的所有内容加起来的大小.
  • 分页: 最典型的例子就是iPhone的主界面. 我们不能任意指定一个位置, 让滑动固定在那里. 它要么停留在第一页, 要么停留在第二页, 不会是第0.5页. 每一页的大小是viewSize决定的. 那么总页数就是total_page_count = ceil(viewSize / contentSize )

CCScrollView源码查看

我们知道, cocos2dx的触摸都是通过CCTouchDelegate来实现的. 如果对cocos2dx的touch机制不熟悉的, 可以参考博客.

简单介绍ccTouchBegan方法的功能:

ccTouchBegancocos2dxtouch机制的第一个方法. 这个方法的接口如下:

bool CCScrollView::ccTouchBegan(CCTouch* touch, CCEvent* event)

返回的bool值, 告诉cocos2dx中touch事件的管理者CCTouchDispatcher, 当前组件是否处理这一次触摸:

  • 如果返回true, 则CCTouchDispatcher会根据用户的touch动作, 在后续调用本组件的ccTouchMoved, ccTouchEnded, ccTouchCancelled方法.
  • 如果返回false, 则表示当前组件不处理此次touch事件. 后续的三个方法不会被调用.

CCScrollView::ccTouchBegan()解析

无码无真相, 先贴一张代码图:

bool CCScrollView::ccTouchBegan(CCTouch* touch, CCEvent* event)
{
//tag-1
if (!this->isVisible())
{
return false;
} //tag-2
CCRect frame = getViewRect();
//dispatcher does not know about clipping. reject touches outside visible bounds.
if (m_pTouches->count() > 2 ||
m_bTouchMoved ||
!frame.containsPoint(m_pContainer->convertToWorldSpace(m_pContainer->convertTouchToNodeSpace(touch))))
{
return false;
}
... //tag-3
if (m_pTouches->count() == 1)
{ // scrolling
m_tTouchPoint = this->convertTouchToNodeSpace(touch);
m_bTouchMoved = false;
m_bDragging = true; //dragging started
m_tScrollDistance = ccp(0.0f, 0.0f);
m_fTouchLength = 0.0f;
}
//tag-4
else if (m_pTouches->count() == 2)
{
...
m_bDragging = false;
}
return true;
}

我把跟本博文主题关系不大的代码用省略号(…)代替了. 下面是相关代码解释:

  • tag-1 当此组件隐藏的时候, 不予处理.
  • tag-2 当此触摸点数大于2或者触摸点不在当前组件的显示范围内的时候, 不予处理.
  • tag-3 这是我们的重点. 当触摸点数等于1的时候, 处理滑动操作.
  • tag-4 触摸点等于2的时候, 响应缩放处理.

源代码的修改

变量申明及方法增加

CCScrollView.h文件中增加以下成员变量和方法:

//CCScrollView.h
public:
bool isPaged(){ return m_bPaged; };
void setPaged( bool value ){ m_bPaged = value; };
protected:
clock_t m_touchBeganTime;
int m_touchBeganOffset;
int m_targetPage;
int m_currPage;
float m_pageAccSpeed;
float m_distanceRatioOfTurn;
bool m_bPaged; void __pageTouchBegan(); //在CCScrollView的滑动被触发的时候调用
bool __pageTouchEnd(); //在ScrollView的滑动停止的时候调用
void __pageTouchCancel(); //在滑动被取消的时候调用
void __pageClearTouch(); //在一次滑动结束的时候调用

方法的实现:

四个方法的代码可能要占用一些篇幅, 所以在这里, 先简要讲一下原理:

  1. 在touch开始的时候, 记录一下当时时间m_touchBeganTime和开始滑动的位置m_touchBeganOffset.
  2. 在touch结束的时候, 获取结束时刻的时间和位置. 我们有两个标准来判断应该翻页还是停留在上一页:
    • 如果滑动距离超过一页距离的一半(或者是其他阈值),那么判断为用户希望翻到下一页(或上一页)
    • 如果滑动速度超过一个阈值, 那么判断为用户希望翻到下一页(或上一页)
  3. 当做了判断之后, 即可滑动到对应的位置.

下面是四个方法的实现, 实现了上述原理.

void CCScrollView::__pageTouchBegan()
{
//仅在设置了分页属性, 并且只有一个滑动方向的时候, 才支持分页.
if( !m_bPaged || ( m_eDirection != kCCScrollViewDirectionHorizontal && m_eDirection != kCCScrollViewDirectionVertical )) return ; //记录初试时间和位置
m_touchBeganTime = clock();
m_touchBeganOffset = m_eDirection == kCCScrollViewDirectionHorizontal ? getContentOffset().x : getContentOffset().y; }
bool CCScrollView::__pageTouchEnd()
{
if( !m_bPaged || ( m_eDirection != kCCScrollViewDirectionHorizontal && m_eDirection != kCCScrollViewDirectionVertical )) return false ; //constant
const float PAGE_DISTENCE = m_eDirection == kCCScrollViewDirectionHorizontal ? getViewSize().width : getViewSize().height ;
if( PAGE_DISTENCE <= 0 ) return false; const float MAX_PAGE = ( m_eDirection == kCCScrollViewDirectionHorizontal ? getContentSize().width : getContentSize().height ) / PAGE_DISTENCE;
const float MIN_PAGE = 0; float currOffset = m_eDirection == kCCScrollViewDirectionHorizontal ? getContentOffset().x : getContentOffset().y;
float deltaOffset = -(currOffset - m_touchBeganOffset);
clock_t currTime = clock();
float speed = currTime != m_touchBeganTime ? deltaOffset / ( currTime - m_touchBeganTime ) : 0; m_targetPage = m_currPage;
if( abs(deltaOffset) >= TURN_PAGE_MIN_OFFSET_RATIO*PAGE_DISTENCE )
{//滑动距离大于某一阈值. if( deltaOffset > 0 )
{
m_targetPage = m_currPage + 1;
}
else if( deltaOffset < 0 )
{
m_targetPage = m_currPage - 1;
}
}
else if( abs(speed) >= TURN_PAGE_SPEED )
{//速度大于某一阈值.
if( speed > 0 )
{
m_targetPage = m_currPage + 1;
}
else if( speed < 0 )
{
m_targetPage = m_currPage - 1;
}
} if( m_targetPage > MAX_PAGE ) m_targetPage = MAX_PAGE;
else if( m_targetPage < MIN_PAGE ) m_targetPage = MIN_PAGE; float targetOffset = -m_targetPage*( m_eDirection == kCCScrollViewDirectionHorizontal ? getViewSize().width : getViewSize().height );
float pageDurateion = 0.5;
CCPoint targetPointOffset = m_eDirection == kCCScrollViewDirectionHorizontal ? ccp( targetOffset, getContentOffset().y ) : ccp(getContentOffset().x, targetOffset );
setContentOffsetInDuration(targetPointOffset, pageDurateion); m_currPage = m_targetPage; return true;
}
void CCScrollView::__pageTouchCancel()
{
if( !m_bPaged || ( m_eDirection != kCCScrollViewDirectionHorizontal && m_eDirection != kCCScrollViewDirectionVertical )) return ; __pageClearTouch(); }
void CCScrollView::__pageClearTouch()
{
//clear所有状态
m_touchBeganOffset = 0;
m_touchBeganTime = 0;
m_targetPage = m_currPage;
}

修改后的CCScrollView.h, CCScrollView.cpp

这里是修改后的文件, 可以直接下载覆盖.

CCScrollView.zip

注意:上述代码仅在cocos2dx-2.2.2cocos2dx-2.2.1版本上验证通过. 其他版本请根据上述原理做适当的修改~~~

Written with StackEdit.

[cocos2dx]让CCScrollView支持分页的更多相关文章

  1. ASP.NET 为GridView添加序号列,且支持分页连续累计显示

    为GridView添加序号列,且支持分页连续累计显示,废话不多说,直接上代码: <%@ Page Language="C#" AutoEventWireup="tr ...

  2. 移动开发之浅析cocos2d-x的中文支持问题

    题记:这阵子一直在学习cocos2d-x,其跨平台的特性确实让人舒爽,引擎的框架概念也很成熟,虽然相应的第三方工具略显单薄,但也无愧是一件移动开发的利器啊,有兴趣的朋友有时间就多了解吧. 使用引擎的过 ...

  3. 【jsPDF】jsPDF插件实现将html页面转换成PDF,并下载,支持分页

    1.目的:在前段是 jQuery库 或者 VUE库 或者两者混合库,将html 页面和数据 转换成PDF格式并下载,支持分页 1.项目背景: 对客户报修记录进行分类统计,并生成各种饼图.柱状图.线性图 ...

  4. 页面直接导出为PDF文件,支持分页与页边距

    将WEB页面直接导出为pdf文件是经常会用到的一个功能,尤其是各种报表系统.总结了一下目前几种主流的做法: 在后端用代码生成pdf文件,比如iText一类: 在后端抓取页面并生成pdf文件,比如pha ...

  5. Cocos2d-x多语言支持解决方式

    很多其它相关内容请查看本人博客:http://www.bokeyi.com/ll/category/cocos2d-x/ 利用.plist文件让Cocos2d-x轻松支持多语言. .plist文件类似 ...

  6. cocos2dx旧版本支持arm64修改

    修改的版本是cocos2dx.2.2 1.在neon_matrix_impl.c中修改 #if defined(__ARM_NEON__)为 #if defined(_ARM_ARCH_7) 2.在m ...

  7. [cocos2d-x]OPENGL ES支持的像素格式

    OPENGL ES最多支持32位颜色值. 支持的像素格式有以下几种: 客户端格式 GL格式 GL数据类型 字节数 RGBA8888 GL_RGBA GL_UNSIGNED_BYTE 4 RGB888 ...

  8. [cocos2dx] 让UIButton支持disable状态

    摘要: 主要解决cocos2dx-2.2.2版本中, UIButton显示不了disable状态图的问题. 顺便, 理解了一下cocos2dx中UIWidget的渲染原理. 博客: http://ww ...

  9. 【转】让 cocos2d-x 的 CCHttpRequest 支持https

    肖锐(Cooki)个人原创,欢迎转载,转载请注明地址,肖锐(Cooki)的技术博客 http://blog.csdn.net/xiao0026  由于游戏用到了网络头像, 今天发现换成facebook ...

随机推荐

  1. Spring管理 hibernate 事务配置的五种方式

    Spring配置文件中关于事务配置总是由三个组成部分,DataSource.TransactionManager和代理机制这三部分,无论是那种配置方法,一般变化的只是代理机制这块! 首先我创建了两个类 ...

  2. 【JavaEE】SSH+Spring Security自定义Security的部分处理策略

    本文建立在 SSH与Spring Security整合 一文的基础上,从这篇文章的example上做修改,或者从 配置了AOP 的example上做修改皆可.这里主要补充我在实际使用Spring Se ...

  3. SQLSERVER数据库表各种同步技术

    1 --SQLSERVER数据库表各种同步技术 减少SQLServer中每次的同步数据量 2 3 --说到数据库,我就不由地想到同步数据,如何尽可能地减少每次的同步数据量,以此来提高同步效率,降低对网 ...

  4. 参加:白帽子活动-赠三星(SAMSUNG) PRO....

    参加:白帽子活动-—赠三星(SAMSUNG) PRO.... Everybody~小i在这里提前祝大家国庆假期愉快,咱们期待已久的国庆活动终于开始拉,下面进入正题,恩,很正的题! 活动地址:http: ...

  5. 【读书笔记】iOS-GCD-使用方法

    代码: -(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event { dispatch_async(dispatch_get_gl ...

  6. 【转】IOS设备旋转的内部处理流程以及一些【优化建议】

    加速计是整个IOS屏幕旋转的基础,依赖加速计,设备才可以判断出当前的设备方向,IOS系统共定义了以下七种设备方向: typedef NS_ENUM(NSInteger, UIDeviceOrienta ...

  7. NSURLSession/NSURLConnection的上传文件方法(已做了更新)

    最好的学习方法就是 领悟 + 证悟. 此篇文章的理论基础主要是与HTTP网络通信协议相关.为集中精力,可以先把TCP/IP协议这些置之不理,也就是先只关注HTTP的请求和响应的结构.HTTP完整的原理 ...

  8. ios 删除系统从相册压缩的视频

    iOS的沙盒机制,应用只能访问自己应用目录下的文件.iOS不像android,没有SD卡概念,不能直接访问图像.视频等内容.iOS应用产生的内容,如图像.文件.缓存内容等都必须存储在自己的沙盒内.默认 ...

  9. C++中static用法总结

    1用于局部变量 C++中局部变量有三种: (1)auto:此关键词常常省略.auto type a 常常简写为type a. 如: int a=auto int a 存储在内存的栈中,只在此局部区域有 ...

  10. 记录JVM垃圾回收算法

    垃圾回收算法可以分为三类,都基于标记-清除(复制)算法: Serial算法(单线程) 并行算法 并发算法 JVM会根据机器的硬件配置对每个内存代选择适合的回收算法,比如,如果机器多于1个核,会对年轻代 ...