整理+源码实现
我们都知道,在iOS里面有一种控件------滚筒控件(Wheel View),这通常用于设置时间/日期,非常方便,但Android SDK并没有提供类似的控件。这里介绍一下如何Android实现WheelView。
源码下载:http://download.csdn.net/detail/heng615975867/6521189

先来看一看iOS中的WheelView的效果图:

这个效果不错吧,我们应该如何实现呢?

那在Android如果也要实现这样一个效果,应该怎么做呢?

1.Android WheelView效果图

上图是我实现的DEMO的运行效果图。

2.网上的开源代码

我们从网上找到了一个开源的代码,它也实现了这样的效果,而且效果也不错,大家可以用SVN来checkout:

http://android-wheel.googlecode.com/svn/trunk

它这个Demo最本质是自己写布局,好像是利用一个LinearLayout来布局child,然后调用LinearLayout.draw(canvas)方法,把child绘制在指定的canvas上面。它同时还提供了类似AdapterView的访问方式,用户可以设置Adapter来提供数据。我在这里主要不是讲解这个Demo的结构,如果大家感兴趣,可以自己下载代码研究。

3.实现思路

由于我之前修改过Gallery的源代码,可以使其循环滚动,并且第一个child可以排列在最左端,所以,我在想,如果我能把Gallery修改成竖的(垂直排列),那这个不就是OK了吗?基于这样的想法,我就准备修改代码了。

我们这里需要把Gallery的源码复制到我们的工程中,然后修改,保证能编译通过。

与Gallery相关的的几个文件如下所示,它们都是放在widget文件夹和res/value文件夹下面。

  • AbsSpinner.java
  • AdapterView.java
  • Gallery.java
  • attr.xml

修改的过程比较麻烦,我这里不详细说明(要细说的话,内容太多了),在修改之后,我们的Gallery提供了一个方法:setOrientation(int),你可以让这个Gallery水平滑动,也可以垂直滑动。

我们还应该提供以下几个核心方法:

  • setOnEndFlingListener ------ 当Gallery停止滑动时的回调用,这样调用者可以在停止滑动时来得到当前选中的项。
  • setOrientation(int) ------ 支持布局方向:HORIZONTAL和VERTICAL。
  • setScrollCycle(boolean) ------ 是否支持循环滑动。
  • setSlotInCenter(boolean) ------ 是否让Gallery选中的项居中。

4. 扩展Gallery

[java] view plaincopy
  1. package com.nj1s.lib.widget;
  2. import android.content.Context;
  3. import android.graphics.Canvas;
  4. import android.graphics.Rect;
  5. import android.graphics.drawable.Drawable;
  6. import android.graphics.drawable.GradientDrawable;
  7. import android.graphics.drawable.GradientDrawable.Orientation;
  8. import android.util.AttributeSet;
  9. import android.view.Gravity;
  10. import android.view.View;
  11. import com.nj1s.lib.R;
  12. public class WheelView extends TosGallery
  13. {
  14. private Drawable mSelectorDrawable      = null;
  15. private Rect mSelectorBound             = new Rect();
  16. private GradientDrawable mTopShadow     = null;
  17. private GradientDrawable mBottomShadow  = null;
  18. private static final int[] SHADOWS_COLORS =
  19. {
  20. 0xFF111111,
  21. 0x00AAAAAA,
  22. 0x00AAAAAA
  23. };
  24. public WheelView(Context context)
  25. {
  26. super(context);
  27. initialize(context);
  28. }
  29. public WheelView(Context context, AttributeSet attrs)
  30. {
  31. super(context, attrs);
  32. initialize(context);
  33. }
  34. public WheelView(Context context, AttributeSet attrs, int defStyle)
  35. {
  36. super(context, attrs, defStyle);
  37. initialize(context);
  38. }
  39. private void initialize(Context context)
  40. {
  41. this.setVerticalScrollBarEnabled(false);
  42. this.setSlotInCenter(true);
  43. this.setOrientation(TosGallery.VERTICAL);
  44. this.setGravity(Gravity.CENTER_HORIZONTAL);
  45. this.setUnselectedAlpha(1.0f);
  46. // This lead the onDraw() will be called.
  47. this.setWillNotDraw(false);
  48. // The selector rectangle drawable.
  49. this.mSelectorDrawable =
  50. getContext().getResources().getDrawable(R.drawable.wheel_val);
  51. this.mTopShadow    =
  52. new GradientDrawable(Orientation.TOP_BOTTOM, SHADOWS_COLORS);
  53. this.mBottomShadow =
  54. new GradientDrawable(Orientation.BOTTOM_TOP, SHADOWS_COLORS);
  55. // The default background.
  56. this.setBackgroundResource(R.drawable.wheel_bg);
  57. }
  58. @Override
  59. protected void dispatchDraw(Canvas canvas)
  60. {
  61. super.dispatchDraw(canvas);
  62. // After draw child, we do the following things:
  63. // +1, Draw the center rectangle.
  64. // +2, Draw the shadows on the top and bottom.
  65. drawCenterRect(canvas);
  66. drawShadows(canvas);
  67. }
  68. /**
  69. * setOrientation
  70. */
  71. @Override
  72. public void setOrientation(int orientation)
  73. {
  74. if (TosGallery.HORIZONTAL == orientation)
  75. {
  76. throw new IllegalArgumentException("The orientation must be VERTICAL");
  77. }
  78. super.setOrientation(orientation);
  79. }
  80. @Override
  81. protected void onLayout(boolean changed, int l, int t, int r, int b)
  82. {
  83. super.onLayout(changed, l, t, r, b);
  84. int galleryCenter = getCenterOfGallery();
  85. View v = this.getChildAt(0);
  86. int height = (null != v) ? v.getMeasuredHeight() : 50;
  87. int top = galleryCenter - height / 2;
  88. int bottom = top + height;
  89. mSelectorBound.set(
  90. getPaddingLeft(),
  91. top,
  92. getWidth() - getPaddingRight(),
  93. bottom);
  94. }
  95. private void drawCenterRect(Canvas canvas)
  96. {
  97. if (null != mSelectorDrawable)
  98. {
  99. mSelectorDrawable.setBounds(mSelectorBound);
  100. mSelectorDrawable.draw(canvas);
  101. }
  102. }
  103. private void drawShadows(Canvas canvas)
  104. {
  105. int height = (int)(2.0 * mSelectorBound.height());
  106. mTopShadow.setBounds(0, 0, getWidth(), height);
  107. mTopShadow.draw(canvas);
  108. mBottomShadow.setBounds(0, getHeight() - height, getWidth(), getHeight());
  109. mBottomShadow.draw(canvas);
  110. }
  111. }

上面代码没有什么特别的东西,只是有几点需要注意:

[1] 不要重写onDraw(),为什么呢?因为onDraw()是绘制自己,如果你在onDraw()中来绘制阴影的话,那么最后的效果可能是Child在上面,阴影在下面。因此,我们应该是在绘制完Child之后,再绘制阴影,怎么做呢?请看第二步。
[2] 重写dispatchDraw(),如果对这个方法不明白的话,请自己看文档,这里不解释,总之,这个方法是用来绘制Child的,因此,重写这个方法,先调用super.dispatchDraw()方法,然后再绘制阴影,OK,万事大吉。
[3] 你可以调用#setScrollCycle(boolean)来指定这个WheelView是否可以循环滑动。

5. 如何使用

[java] view plaincopy
  1. // 设置listener
  2. mDateWheel.setOnEndFlingListener(mListener);
  3. // 设置滑动时的声音
  4. mDateWheel.setSoundEffectsEnabled(true);
  5. // 设置adapter
  6. mDateWheel.setAdapter(new WheelTextAdapter(this));
  7. // Adapter的实现
  8. protected class WheelTextAdapter extends BaseAdapter
  9. {
  10. ArrayList<TextInfo> mData = null;
  11. int mWidth  = ViewGroup.LayoutParams.MATCH_PARENT;
  12. int mHeight = 50;
  13. Context mContext = null;
  14. public WheelTextAdapter(Context context)
  15. {
  16. mContext = context;
  17. }
  18. public void setData(ArrayList<TextInfo> data)
  19. {
  20. mData = data;
  21. this.notifyDataSetChanged();
  22. }
  23. public void setItemSize(int width, int height)
  24. {
  25. mWidth  = width;
  26. mHeight = height;
  27. }
  28. @Override
  29. public int getCount()
  30. {
  31. return (null != mData) ? mData.size() : 0;
  32. }
  33. @Override
  34. public Object getItem(int position)
  35. {
  36. return null;
  37. }
  38. @Override
  39. public long getItemId(int position)
  40. {
  41. return 0;
  42. }
  43. @Override
  44. public View getView(int position, View convertView, ViewGroup parent)
  45. {
  46. TextView textView = null;
  47. if (null == convertView)
  48. {
  49. convertView = new TextView(mContext);
  50. convertView.setLayoutParams(new TosGallery.LayoutParams(mWidth, mHeight));
  51. textView = (TextView)convertView;
  52. textView.setGravity(Gravity.CENTER);
  53. textView.setTextSize(26);
  54. textView.setTextColor(Color.BLACK);
  55. }
  56. if (null == textView)
  57. {
  58. textView = (TextView)convertView;
  59. }
  60. TextInfo info = mData.get(position);
  61. textView.setText(info.mText);
  62. textView.setTextColor(info.mColor);
  63. return convertView;
  64. }
  65. }

Android Gallary扩展实现 WheelView(Gallary竖着滑实现时间现实)相关推荐

  1. android 横屏竖屏设置,Android中强制设置横屏与竖屏,与横竖屏生命周期的设置

    许多小的知识点,是面试中经常会被提问到的,面试题也方式也会不一样,比如,让你在配置文件中设置Activity 为竖屏模式,答题方式是把相关的配置节点代码也给写出来,多少有点懵逼... 全屏 在Acti ...

  2. android kotlin扩展添加域名

    android kotlin扩展添加域名? 你是否常为和系统方法变量起名冲突而烦恼? 你是否常为过多的方法名查找起来而而困惑? 你是否常为不知道谁写出的垃圾代码而头痛? 今天,给大家介绍一种添加域名的 ...

  3. android仿微信发布动态功能,Android GridView扩展仿微信微博发图动态添加删除图片功能.pdf...

    Android GridView扩扩展展仿仿微微信信微微博博发发图图动动态态添添加加删删除除图图片片功功能能 这篇文章主要为大家详细介绍了Android GridView扩展仿微信微博发图动态添加删除 ...

  4. android PopupWindow实现从底部弹出或滑出选择菜单或窗口

    android PopupWindow实现从底部弹出或滑出选择菜单或窗口 http://www.open-open.com/lib/view/open1379383271818.html http:/ ...

  5. Android 应用《离线全唐诗》。利用碎片化时间,无障碍欣赏唐诗。

    QuanTangshi 项目地址:animalize/QuanTangshi 简介:Android 应用<离线全唐诗>.利用碎片化时间,无障碍欣赏唐诗. 更多:作者   提 Bug 标签: ...

  6. android 动画x轴旋转,Android Roate3dAnimation实现围绕y轴竖直方向或者绕x轴方向旋转的3d动画效果...

    概要: ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Roate3dAnim ...

  7. android百分比扩展枯,Android 增强版百分比布局库 为了适配而扩展

    一 概述 上周一我们发布了Android 百分比布局库(percent-support-lib) 解析与扩展中对percent-support这个库进行了解析和添加了PercentLinearLayo ...

  8. android可扩展列表,android-可扩展列表视图未显示

    代码可以正常运行,没有任何错误或警告,但可扩展列表未显示,这是代码 public class MainActivity extends Activity implements ExpandableLi ...

  9. 027 Android 可扩展的listview:ExpandableListView的使用案例

    1.ExpandableListView简介 ExpandableListView是一种用于垂直滚动展示两级列表的视图,和 ListView 的不同之处就是它可以展示两级列表,分组可以单独展开显示子选 ...

最新文章

  1. Visual Basic 2005 - 如何将色彩字符串转换成 Color 结构
  2. 图像纹理合成及纹理传输算法学习(附源码)。
  3. 11G中自动收集统计信息
  4. 【实验手册】使用Visual Studio Code 开发.NET Core应用程序
  5. 面试准备系列01----面试中的链表题目汇总
  6. 牛客网_PAT乙级_1018人口普查(20)
  7. opencv mat初始化_【OpenCV入门之十二】看起来一样的图像竟然存在这么大的差别!...
  8. Python数据结构与算法(三)--数据结构的概念
  9. 万圣节头像小程序源码
  10. 全网最全的git命令大全
  11. flash请求来源Refer测试
  12. 如何进行 JVM 调优
  13. 关于使用FileZilla软件出现中文乱码解决方法
  14. R语言【分解时间序列并检验时间序列的自相关性】
  15. 基于Java医院网上预约挂号系统设计与实现(含源代码)
  16. 梦参老和尚:糊涂人念〈大悲咒〉往生的故事
  17. 最危害程序员职业生涯的三大观念
  18. CLIP改进工作串讲(上)
  19. python爬取拉钩python数据分析职位招聘信息
  20. android最新主流机型,就要Android 市售主流机型随你挑

热门文章

  1. web前端工程师做什么
  2. 【华为机试】HJ2 计算某字符出现次数
  3. 【k哥爬虫普法】程序员183并发爬取官方网站,直接获刑3年?
  4. Snagit_日文输入法:促音,长音,小写的输入方法
  5. 计算机无法屏幕亮度,电脑更新windows 10 四月更新1803后,笔记本电脑亮度无法调节。...
  6. 【教程】批量删除B站抽奖动态
  7. python人力成本数据测算_历史数据法进行人力成本分析
  8. Appweb+ESP学习笔记
  9. IMPERVA-WAF 设备替换方案
  10. 云服务器ecs是虚拟主机,云服务器ecs是虚拟主机吗