首页 > Unity3D频道 > 【NGUI研究院之Unity插件】 > NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四)
2012
07-10

NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四)

         MOMO与图灵合作的《Unity 3D游戏开发》没想到开售不到1个月就售罄了,心里实在是太高兴了。不过高兴归高兴,学习生活工作我是不能耽误的,加油哇咔咔。 NGUI系列的文章在上一章提到的活灵活现ListView相信大家已经熟练掌握,但是这些是满足不了程序猿的欲望的,哇咔咔~~NGUI中提供了两种Scroll View 一种是通过手指或鼠标滑动视图时移动平面物体,另一种则是直接移动摄像机,他们各有各的好处。但是NGUI提供的Scroll View很难实现类似Android 与 IOS 中的Scroll View 滚动相册的那种效果,至少MOMO没有很快的找到方法,不过程序猿的力量是伟大无穷的。虽然不能用它提供的API做出来,但是我们可以通过另外的手打巧妙的实现。不用担心MOMO会在这篇文章仔细向大家介绍如何实现自制Scroll View实现滚动相册。

        如下图所示,这是我们的工程页面,程序的实现原理是将相册在Unity3D世界中呈横向队列,摄像机固定的照射在第一个Item相册,当手指发生滑动事件时,计算向左滑动还是向右滑动,此时整体移动相册队列,而摄像机不动。为了让滑动效果更加好看我们需要使用插值计算滑动的时间,使滑动队列不是直接移动过去,而是以一定惯性移动过去。相册下方我们制作一个小白点用来记录当前滑动的位置,在做几个灰色的点表示队列一共的长度,滑动时下方的小白点也会跟随移动,这样就更想高级控件啦。当然小白点与小灰点是要根据item的数量居中显示的喔。

注解1:滚动相册一般可分为两种,第一种为数量已知的情况,第二种为数量未知的情况。因为第一种比较简单所以我们主要探讨第二种。

Script historyInit.cs: 该脚本用于相册队列的初始化工作。在这里初始化相册队列的数量,计算完毕让队列以横向线性的排列方式在Unity3D中。

Prefab item:每个相册的预设,我这里每个相册上还会有一些文字的描述,我需要动态的修改它们的内容。大家也可根据自己的情况制作自己的相册item。

Prefabhui:相册滚动时下方用来记录相册队列总数的灰色小点。

Prefabbai :相册滚动时下方用来记录当前滚动页面ID的白色小点。

Point :因为灰色、白色的点不能和相册队列在一个面板之上,否则会跟着相册队列移动而移动,所以这里将灰色白色的点放在两外一个面板之上。

注解2:这个面板上的4个item就是我们通过historyinit脚本初始化时动态生成赋值的、当界面发生触摸事件时,会整体移动该面板让自对象的相册队列跟随移动。

注解3:button0 – button3 是下方的Tabar。bai(Clone)表示白色的小点,hui(Clone)表示灰色的小点,它们的位置是需要根据滑动事件而改变的。

NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四) - 雨松MOMO程序研究院 - 1

 

因为我们需要监听每一个Item的滑动事件,所以肯定要在每一个item预设之上绑定监听事件的脚本,如下图所示。

注解1:因为需要监听触摸滑动事件,所以肯定要绑定Box Collider组件,这个是NGUI的标准用法。

注解2:Move脚本用来监听向左滑动 向右滑动 点击事件。

注解3:这个就是每一个相册的item,在上图中挂在historyInit脚本之上。

NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四) - 雨松MOMO程序研究院 - 2

 

historyInit.cs

 

如下图所示,我们可以看出Tabbar 、 下方记录界面的灰色、白色小点、摄像机  它们是不会发生改变的。唯一改变的就是面板之上的相册队列。为了让滑动界面的效果更加连贯,我们需要以插值的形式来计算真个相册面板的坐标。

 

NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四) - 雨松MOMO程序研究院 - 3

 

触摸的事件全都写在Move.cs脚本中。

 

还有两个辅助的类,我也贴出来。

UserData.cs

 

Globe.cs 这个静态类用来共享记录多个脚本甚至多个场景所需的数据。

 

写到这里,本篇博文就写的差不多了。这篇文章我是用NGUI来实现的触摸滚动效果,仔细想想,其实不用NGUI完全也能实现这样的效果。在脚本中完全可以通过射线 以及 触摸的时间去计算当前用户操作的事件,这篇文章里的工程最后我是打包在Android上面的,效果挺不错,滑动的效果图不好截取我也不截取了,主要还是文章的书写内容,希望大家学习愉快,雨松MOMO祝大家晚安,哇咔咔,啦啦啦。

最后编辑:
作者:雨松MOMO
专注移动互联网,Unity3D游戏开发
捐 赠写博客不易,如果您想请我喝一杯星巴克的话?就进来看吧!

NGUI研究院之自制 Scroll View实现触摸滚动相册效果(四)》有 52 条评论

  1. 哟哟哟 说:

    雨松大神,插值方法的第二个参数的y坐标为啥是0呢

  2. 可以看看你的源码吗306891402@qq.com,谢谢

  3. 杨静轩 说:

    在Move.cs文件中的OnPress函数结尾应该加句isOnDrag = false;要不在一次滑动事件之后得点击第二下才能响应按键事件了

  4. 橘南小巷゛ 说:

    雨松大大 这个项目有源文件么?希望带上源文件

  5. MTT 说:

    雨松老师 有没有 2dtookit 的呢?

  6. 求学 说:

    请问item(clone)和hui(clone)是什么啊?他们是什么类型的GameObject?而且希望能求到工程源文件,麻烦发到214137022@qq.com,谢谢了。。。

  7. 雨松老师讲的真好,有些地方有点不懂。求工程源文件呀。我的邮箱是365206763@qq.com

  8. zm 说:

    为什么 我的OnDrag 和 OnPress 没有反应

  9. MENG 说:

    我也购买了博主大人的《unity3d 游戏开发》,真是入门的好教材,谢谢博主大人!另外博主大人这篇ScrollView的完整工程能发给我看一下吗?我的邮箱是m_iSnow@hotmail.com再次感谢!

  10. peng 说:

    请问如果scrollview的范围设置得比较大,但是里面的内容比较少,就会造成内容显示在区域中央,有没有方法当内容不够的时候,让内容显示在顶部?

    • twntwn 说:

      修改UIPanelpublic bool isTop=false;public Vector3 CalculateConstrainOffset (Vector2 min, Vector2 max) { float offsetX = clipRange.z * 0.5f; float offsetY = clipRange.w * 0.5f; Vector2 minRect = new Vector2(min.x, min.y); Vector2 maxRect = new Vector2(max.x, max.y); Vector2 minArea = new Vector2(clipRange.x – offsetX, clipRange.y – offsetY); Vector2 maxArea = new Vector2(clipRange.x + offsetX, clipRange.y + offsetY); if (clipping == UIDrawCall.Clipping.SoftClip) { minArea.x += clipSoftness.x; minArea.y += clipSoftness.y; maxArea.x -= clipSoftness.x; maxArea.y -= clipSoftness.y; } Vector2 v2 =NGUIMath.ConstrainRect(minRect, maxRect, minArea, maxArea); if(isTop&&(max.y-min.y)

    • twntwn 说:

      修改UIPanelpublic bool isTop=false;public Vector3 CalculateConstrainOffset (Vector2 min, Vector2 max){float offsetX = clipRange.z * 0.5f;float offsetY = clipRange.w * 0.5f;Vector2 minRect = new Vector2(min.x, min.y);Vector2 maxRect = new Vector2(max.x, max.y);Vector2 minArea = new Vector2(clipRange.x – offsetX, clipRange.y – offsetY);Vector2 maxArea = new Vector2(clipRange.x offsetX, clipRange.y offsetY);if (clipping == UIDrawCall.Clipping.SoftClip){minArea.x = clipSoftness.x;minArea.y = clipSoftness.y;maxArea.x -= clipSoftness.x;maxArea.y -= clipSoftness.y;}Vector2 v2 =NGUIMath.ConstrainRect(minRect, maxRect, minArea, maxArea);if(isTop&&(max.y-min.y)

  11. 程宝 说:

    老师,能给个源代码么,谢谢啦~我的邮箱是244157539@qq.com

  12. VIP 说:

    看了你的书,但觉得自己还处于一个小小有基础的人,这个能给个源代码吗,谢谢谢。我的邮箱1975209805@qq.com

  13. 史金亮 说:

    雨松可以给下这个的源码吗?看你做的挺好看的。资源也是难求的。呵呵。谢谢。sjl_leaf@163.com

留下一个回复

你的email不会被公开。