使用MagicIndicator打造千变万化的ViewPager指示器

本文主要是介绍使用MagicIndicator打造千变万化的ViewPager指示器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


转载地址:http://www.jianshu.com/p/2865812fed41#rd


额,又到周末了,该更新博客了,这周我们来谈一谈 ViewPager 指示器。

说到 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

不要忘了添加依赖哦:

dependencies {compile project(':magicindicator')
}

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

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"tools:context="net.lucode.hackware.magicindicatordemo.MainActivity"><net.lucode.hackware.magicindicator.MagicIndicatorandroid:id="@+id/magic_indicator"android:layout_width="match_parent"android:layout_height="@dimen/navigator_common_height"android:background="#d43d3d" /><android.support.v4.view.ViewPagerandroid:id="@+id/view_pager"android:layout_width="match_parent"android:layout_height="0dp"android:layout_weight="1"android:background="@android:color/white" /></LinearLayout>

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

final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
final CommonNavigator commonNavigator = new CommonNavigator(this);
commonNavigator.setAdapter(new CommonNavigatorAdapter() {@Overridepublic int getCount() {return mDataList == null ? 0 : mDataList.size();}@Overridepublic IPagerTitleView getItemView(Context context, final int index) {ClipPagerTitleView clipPagerTitleView = new ClipPagerTitleView(context);clipPagerTitleView.setText(mDataList.get(index));clipPagerTitleView.setTextColor(Color.parseColor("#f2c4c4"));clipPagerTitleView.setClipColor(Color.WHITE);clipPagerTitleView.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {mPager.setCurrentItem(index);}});return clipPagerTitleView;}@Overridepublic IPagerIndicator getIndicator(Context context) {return null;    // 没有指示器,因为title的指示作用已经很明显了}
});
magicIndicator.setNavigator(commonNavigator);

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

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

mPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {@Overridepublic void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {magicIndicator.onPageScrolled(position, positionOffset, positionOffsetPixels);}@Overridepublic void onPageSelected(int position) {magicIndicator.onPageSelected(position);}@Overridepublic void onPageScrollStateChanged(int state) {magicIndicator.onPageScrollStateChanged(state);}
});mPager.setCurrentItem(1);

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

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

内建的指示器


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

final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
final CommonNavigator commonNavigator = new CommonNavigator(this);
commonNavigator.setAlwaysScrollToCenter(true);
commonNavigator.setAdapter(new CommonNavigatorAdapter() {@Overridepublic int getCount() {return mDataList == null ? 0 : mDataList.size();}@Overridepublic IPagerTitleView getItemView(Context context, final int index) {SimplePagerTitleView simplePagerTitleView = new SimplePagerTitleView(context);simplePagerTitleView.setText(mDataList.get(index));simplePagerTitleView.setNormalColor(Color.parseColor("#333333"));simplePagerTitleView.setSelectedColor(Color.parseColor("#e94220"));simplePagerTitleView.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {mPager.setCurrentItem(index);}});return simplePagerTitleView;}@Overridepublic IPagerIndicator getIndicator(Context context) {TriangularPagerIndicator indicator = new TriangularPagerIndicator(context);indicator.setLineColor(Color.parseColor("#e94220"));return indicator;}
});
magicIndicator.setNavigator(commonNavigator);

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

如何扩展


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

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

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

final MagicIndicator magicIndicator = (MagicIndicator) findViewById(R.id.magic_indicator);
CommonNavigator commonNavigator = new CommonNavigator(this);
commonNavigator.setAdapter(new CommonNavigatorAdapter() {@Overridepublic int getCount() {return mDataList == null ? 0 : mDataList.size();}@Overridepublic IPagerTitleView getItemView(Context context, final int index) {CommonPagerTitleView commonPagerTitleView = new CommonPagerTitleView(MainActivity.this);commonPagerTitleView.setContentView(R.layout.simple_pager_title_layout);// 初始化final ImageView titleImg = (ImageView) commonPagerTitleView.findViewById(R.id.title_img);titleImg.setImageResource(R.mipmap.ic_launcher);final TextView titleText = (TextView) commonPagerTitleView.findViewById(R.id.title_text);titleText.setText(mDataList.get(index));commonPagerTitleView.setOnPagerTitleChangeListener(new CommonPagerTitleView.OnPagerTitleChangeListener() {@Overridepublic void onSelected(int index) {titleText.setTextColor(Color.RED);}@Overridepublic void onDeselected(int index) {titleText.setTextColor(Color.BLACK);}@Overridepublic void onLeave(int index, float leavePercent, boolean leftToRight) {titleImg.setScaleX(1.3f + (0.8f - 1.3f) * leavePercent);titleImg.setScaleY(1.3f + (0.8f - 1.3f) * leavePercent);}@Overridepublic void onEnter(int index, float enterPercent, boolean leftToRight) {titleImg.setScaleX(0.8f + (1.3f - 0.8f) * enterPercent);titleImg.setScaleY(0.8f + (1.3f - 0.8f) * enterPercent);}});commonPagerTitleView.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {mPager.setCurrentItem(index);}});return commonPagerTitleView;}@Overridepublic IPagerIndicator getIndicator(Context context) {return null;}
});
magicIndicator.setNavigator(commonNavigator);

效果图:


custom.gif

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

更多


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

mDataList.clear();
mDataList.add("欢迎关注");
mDataList.add("我的博客");
mDataList.add("hackware.lucode.net");
commonNavigator.notifyDataSetChanged();
mAdapter.notifyDataSetChanged();

结语


如果大家觉得 MagicIndicator 很好,对你有帮助,欢迎多多 star + fork,关注!关注!,也请帮忙推广一下哈。感激不尽。

MagicIndicator 是我计划长期维护的项目,由于才刚开始,现在的指示器效果还不是很多,后续会不断加入更多炫酷的效果,如果你想加入我,打造更好的用户体验,私信我吧。

MagicIndicator 疑难解答,交流请加QQ群:373360748。

今天就到此为止吧。周末愉快!



文/hackware(简书作者)
原文链接:http://www.jianshu.com/p/2865812fed41#rd
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

这篇关于使用MagicIndicator打造千变万化的ViewPager指示器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/173932

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

git使用的说明总结

Git使用说明 下载安装(下载地址) macOS: Git - Downloading macOS Windows: Git - Downloading Windows Linux/Unix: Git (git-scm.com) 创建新仓库 本地创建新仓库:创建新文件夹,进入文件夹目录,执行指令 git init ,用以创建新的git 克隆仓库 执行指令用以创建一个本地仓库的

【北交大信息所AI-Max2】使用方法

BJTU信息所集群AI_MAX2使用方法 使用的前提是预约到相应的算力卡,拥有登录权限的账号密码,一般为导师组共用一个。 有浏览器、ssh工具就可以。 1.新建集群Terminal 浏览器登陆10.126.62.75 (如果是1集群把75改成66) 交互式开发 执行器选Terminal 密码随便设一个(需记住) 工作空间:私有数据、全部文件 加速器选GeForce_RTX_2080_Ti

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念