Android万能的指示器MagicIndicator

说到 ViewPager 指示器,想必大家都不陌生,绝大部分应用中都有这个。使用频率非常之高。但系统对它的支持并不好,自带的 PagerTabStrip 和 PagerTitleStrip 太弱,很难满足需求。因此往往程序中需要用到指示器的时候,大家基本上都是选择自己写。当然也有第三方框架诸如 Jake Wharton 大神的ViewPagerIndicator 等,我曾经尝试着用它们,但还是被它们的可定制能力给吓退了。


magicindicator.gif

背景


近期交互改版,需要在指示器上增加吸附效果(类似上图第7个),刚开始我有点懵逼,因为之前的指示器只是简单的使用了 HorizontalScrollView + 横向 LinearLayout ,向 LinearLayout 里面添加一些 TextView 当做标题,选中的时候只是简单的改变 TextView 的颜色,没有任何动画,因此实现起来相对简单(项目前期时间紧迫)。这估计也是大部分应用的做法吧。

考虑到后面如果交互再改版,那我又会懵逼了,所以干脆自己来打造一个可扩展、可定制能力 很强 真TM强 的 ViewPager 指示器框架 ——MagicIndicator

关于命名


之所以叫 MagicIndicator,是因为 鸿神 之前搞了一个 MagicViewPager, 我觉得这两个可以很好的搭配使用,并且正如大家看到的,它确实比较 Magic。

如何使用


这期就不打算给大家讲原理性文章了,只讲如何集成(主要是这两天都在写这个,被媳妇骂惨了,没时间写了),后面我会有一个系列的文章来讲这个,涉及到的内容大概有:

  • MagicIndicator 原理

  • 如何扩展 MagicIndicator 打造任意的切换效果

  • 如何使用 MagicIndicator 来打造主流应用(诸如微信主页的切换)的指示器效果

好,我们开始。

首先,你需要从我的 Github 上将工程代码 check 下来,直接导入即可运行,里面包含源码和 demo。工程中有个 Module 叫做 magicindicator,直接拷到你的项目中即可。包结构如下:


Paste_Image.png

不要忘了添加依赖哦:


  
  
  1. dependencies {
  2. compile project( ':magicindicator')
  3. }

导入成功后,就可以使用啦。你需要先在布局文件里引入 MagicIndicator:


  
  
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:tools = "http://schemas.android.com/tools"
  4. android:layout_width = "match_parent"
  5. android:layout_height = "match_parent"
  6. android:orientation = "vertical"
  7. tools:context = "net.lucode.hackware.magicindicatordemo.MainActivity" >
  8. <net.lucode.hackware.magicindicator.MagicIndicator
  9. android:id = "@+id/magic_indicator"
  10. android:layout_width = "match_parent"
  11. android:layout_height = "@dimen/navigator_common_height"
  12. android:background = "#d43d3d" />
  13. <android.support.v4.view.ViewPager
  14. android:id = "@+id/view_pager"
  15. android:layout_width = "match_parent"
  16. android:layout_height = "0dp"
  17. android:layout_weight = "1"
  18. android:background = "@android:color/white" />
  19. </LinearLayout>

再在代码里面找到它,并进行简单设置:


  
  
  1. final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
  2. final CommonNavigator commonNavigator = new CommonNavigator(this);
  3. commonNavigator.setAdapter(new CommonNavigatorAdapter() {
  4. @Override
  5. public int getCount() {
  6. return mDataList == null ? 0 : mDataList.size();
  7. }
  8. @Override
  9. public IPagerTitleView getItemView(Context context, final int index) {
  10. ClipPagerTitleView clipPagerTitleView = new ClipPagerTitleView(context);
  11. clipPagerTitleView.setText(mDataList.get(index));
  12. clipPagerTitleView.setTextColor(Color.parseColor("#f2c4c4"));
  13. clipPagerTitleView.setClipColor(Color.WHITE);
  14. clipPagerTitleView.setOnClickListener(new View.OnClickListener() {
  15. @Override
  16. public void onClick(View v) {
  17. mPager.setCurrentItem(index);
  18. }
  19. });
  20. return clipPagerTitleView;
  21. }
  22. @Override
  23. public IPagerIndicator getIndicator(Context context) {
  24. return null; // 没有指示器,因为title的指示作用已经很明显了
  25. }
  26. });
  27. magicIndicator.setNavigator(commonNavigator);

这样,你就可以轻松的实现效果图中 今日头条 式(效果图中第一个)切换效果。

额,上面这代码明显没有和 ViewPager 相关联,因此滑动 ViewPager 时是看不到切换效果的,我们给 ViewPager 添加一个监听器:


  
  
  1. mPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
  2. @Override
  3. public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
  4. magicIndicator.onPageScrolled(position, positionOffset, positionOffsetPixels);
  5. }
  6. @Override
  7. public void onPageSelected(int position) {
  8. magicIndicator.onPageSelected(position);
  9. }
  10. @Override
  11. public void onPageScrollStateChanged(int state) {
  12. magicIndicator.onPageScrollStateChanged(state);
  13. }
  14. });
  15. mPager.setCurrentItem(1);

这样, MagicIndicator 就算成功引入项目啦。

  • 注意,以上代码明显没有和 ViewPager 强关联,因此在不使用 ViewPager 的情况下,仍然可以使用 MagicIndicator。只是需要你手动调用 onPageXXX 系列方法回调即可。当然,后续我会提供帮助类来简化这个过程。
  • 等 MagicIndicator 基本稳定、成型后,我会把它提交到 MavenCenter 和 JCenter 中,方便大家使用。

内建的指示器


MagicIndicator 目前内建了好几种指示器,基本可以满足绝大部分需求,并且每一种指示器都支持通过 插值器(Interpolator) 来微调效果(如果你还不会巧用插值器,可以参考我的前一篇博文Android水波纹特效的简单实现 ),后面我还会不定期的往里面添加更多炫酷的效果,敬请期待。下面演示一下使用内建的指示器实现效果图中的小尖角 式切换效果:


  
  
  1. final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
  2. final CommonNavigator commonNavigator = new CommonNavigator(this);
  3. commonNavigator.setAlwaysScrollToCenter(true);
  4. commonNavigator.setAdapter(new CommonNavigatorAdapter() {
  5. @Override
  6. public int getCount() {
  7. return mDataList == null ? 0 : mDataList.size();
  8. }
  9. @Override
  10. public IPagerTitleView getItemView(Context context, final int index) {
  11. SimplePagerTitleView simplePagerTitleView = new SimplePagerTitleView(context);
  12. simplePagerTitleView.setText(mDataList.get(index));
  13. simplePagerTitleView.setNormalColor(Color.parseColor("#333333"));
  14. simplePagerTitleView.setSelectedColor(Color.parseColor("#e94220"));
  15. simplePagerTitleView.setOnClickListener(new View.OnClickListener() {
  16. @Override
  17. public void onClick(View v) {
  18. mPager.setCurrentItem(index);
  19. }
  20. });
  21. return simplePagerTitleView;
  22. }
  23. @Override
  24. public IPagerIndicator getIndicator(Context context) {
  25. TriangularPagerIndicator indicator = new TriangularPagerIndicator(context);
  26. indicator.setLineColor(Color.parseColor("#e94220"));
  27. return indicator;
  28. }
  29. });
  30. magicIndicator.setNavigator(commonNavigator);

看到没有,如果你想换一个效果,只需在 getItemView 里返回不同的指示器标题(IPagerTitleView),在getIndicator 里返回不同的指示器(IPagerIndicator)就可以啦。

如何扩展


当内建的指示器不能满足你的需求时,你可以轻易的扩展,如果你的需求貌似可以使用 HorizontalScrollView + 横向 LinearLayout 方式实现,建议继承IPagerTitleViewIPagerIndicator 即可,如果不行,那就完全自定义吧,继承IPagerNavigator (导航器 —— 我觉得上面的那些效果本质是一个导航器,指示器只是包含在导航器中的一个元素而已) 吧。效果图中的最后一个效果就是如此:


  
  
  1. final MagicIndicator magicIndicator = ( MagicIndicator) findViewById( R.id.magic_indicator);
  2. final CircleNavigator circleNavigator = new CircleNavigator( this);
  3. circleNavigator.setCount(mDataList.size());
  4. circleNavigator.setCircleColor( Color. RED);
  5. magicIndicator.setNavigator(circleNavigator);

当然,我可不希望每当内置指示器达不到你的需求时,你总是去继承这些个类,为了简化,我在最新的代码里增加了 CommonPagerTitleView,使用方法如下:


  
  
  1. final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
  2. CommonNavigator commonNavigator = new CommonNavigator(this);
  3. commonNavigator.setAdapter(new CommonNavigatorAdapter() {
  4. @Override
  5. public int getCount() {
  6. return mDataList == null ? 0 : mDataList.size();
  7. }
  8. @Override
  9. public IPagerTitleView getItemView(Context context, final int index) {
  10. CommonPagerTitleView commonPagerTitleView = new CommonPagerTitleView(MainActivity.this);
  11. commonPagerTitleView.setContentView(R.layout.simple_pager_title_layout);
  12. // 初始化
  13. final ImageView titleImg = (ImageView) commonPagerTitleView.findViewById(R.id.title_img);
  14. titleImg.setImageResource(R.mipmap.ic_launcher);
  15. final TextView titleText = (TextView) commonPagerTitleView.findViewById(R.id.title_text);
  16. titleText.setText(mDataList.get(index));
  17. commonPagerTitleView.setOnPagerTitleChangeListener(new CommonPagerTitleView.OnPagerTitleChangeListener() {
  18. @Override
  19. public void onSelected(int index) {
  20. titleText.setTextColor(Color.RED);
  21. }
  22. @Override
  23. public void onDeselected(int index) {
  24. titleText.setTextColor(Color.BLACK);
  25. }
  26. @Override
  27. public void onLeave(int index, float leavePercent, boolean leftToRight) {
  28. titleImg.setScaleX(1.3f + (0.8f - 1.3f) * leavePercent);
  29. titleImg.setScaleY(1.3f + (0.8f - 1.3f) * leavePercent);
  30. }
  31. @Override
  32. public void onEnter(int index, float enterPercent, boolean leftToRight) {
  33. titleImg.setScaleX(0.8f + (1.3f - 0.8f) * enterPercent);
  34. titleImg.setScaleY(0.8f + (1.3f - 0.8f) * enterPercent);
  35. }
  36. });
  37. commonPagerTitleView.setOnClickListener(new View.OnClickListener() {
  38. @Override
  39. public void onClick(View v) {
  40. mPager.setCurrentItem(index);
  41. }
  42. });
  43. return commonPagerTitleView;
  44. }
  45. @Override
  46. public IPagerIndicator getIndicator(Context context) {
  47. return null;
  48. }
  49. });
  50. magicIndicator.setNavigator(commonNavigator);

效果图:


custom.gif

关于扩展 MagicIndicator,后面会有详细的博文来讲解,敬请留意。如果你扩展 MagicIndicator 实现了很炫酷的指示器效果,欢迎共享出来。

更多


MagicIndicator 同样考虑到了 ViewPager 内容变化的情况,当你的 ViewPager 内容发生变化时,除了调用 PagerAdapter.notifyDataSetChanged ,还记得先调用 IPagerNavigator.notifyDataSetChanged 哦:


  
  
  1. mDataList.clear();
  2. mDataList.add("欢迎关注");
  3. mDataList.add("我的博客");
  4. mDataList.add("hackware.lucode.net");
  5. commonNavigator.notifyDataSetChanged();
  6. mAdapter.notifyDataSetChanged();

源码下载:http://download.csdn.net/detail/xiangzhihong8/9571280

著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

猜你喜欢

转载自blog.csdn.net/weixin_41805792/article/details/88814245