带进度条显示的按钮, 其效果如下所示:


其由三部分动画组成: 初始状态->圆环状态->完成状态.
0. 实现从初始到圆环的简单实现:

继承自button 类, 设置其背景
  1. public class CircleButton extends Button implements View.OnClickListener {
  2. private StateListDrawable mIdleStateDrawable;
  3. private StrokeGradientDrawable background;
  4. public CircleButton(Context context, AttributeSet attrs) {
  5. super(context, attrs);
  6. background = new StrokeGradientDrawable();
  7. // init();
  8. // setBackground(mIdleStateDrawable);
  9. setBackground(background.getGradientDrawable());
  10. setOnClickListener(this);
  11. }
  12. //.......
  13. }
其中的StrokeGradientDrawable(照搬源码) 如下:
  1. public class StrokeGradientDrawable {
  2. private int mStrokeColor; //描边颜色
  3. private int mStrokeWidth; //描边宽度
  4. public int getStrokeColor() {
  5. return mStrokeColor;
  6. }
  7. public void setStrokeColor(int strokeColor) {
  8. mStrokeColor = strokeColor;
  9. mGradientDrawable.setStroke(getStrokeWidth(), strokeColor);
  10. }
  11. public int getStrokeWidth() {
  12. return mStrokeWidth;
  13. }
  14. public void setStrokeWidth(int strokeWidth) {
  15. mStrokeWidth = strokeWidth;
  16. mGradientDrawable.setStroke(strokeWidth, getStrokeColor());
  17. }
  18. public StrokeGradientDrawable() {
  19. mGradientDrawable = new GradientDrawable();
  20. mGradientDrawable.setColor(0xff99cc00);
  21. mStrokeWidth = 5;
  22. }
  23. public GradientDrawable getGradientDrawable() {
  24. return mGradientDrawable;
  25. }
  26. public void setmGradientDrawable(GradientDrawable mGradientDrawable) {
  27. this.mGradientDrawable = mGradientDrawable;
  28. }
  29. private GradientDrawable mGradientDrawable;
  30. }
其中的GradientDrawable 为动画的关键所在
在该类中,可以设置其圆角, 填充颜色, 描边, 尺寸等. 因此,当点击按钮时,可通过动画的方式渐变到目的状态, 其实现如下(源码见MorphingAnimation 类):
  1. @Override
  2. public void onClick(View v) {
  3. final GradientDrawable gradientDrawable = background.getGradientDrawable();
  4. final int mFromWidth = getWidth();
  5. final int mToWidth = getHeight();
  6. //宽度动画
  7. ValueAnimator widthAnimation = ValueAnimator.ofInt(mFromWidth, mToWidth);
  8. widthAnimation.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
  9. @Override
  10. public void onAnimationUpdate(ValueAnimator animation) {
  11. Integer value = (Integer) animation.getAnimatedValue();
  12. int leftOffset;
  13. int rightOffset;
  14. int padding;
  15. if (mFromWidth > mToWidth) {
  16. leftOffset = (mFromWidth - value) / 2;
  17. rightOffset = mFromWidth - leftOffset;
  18. } else {
  19. leftOffset = (mToWidth - value) / 2;
  20. rightOffset = mToWidth - leftOffset;
  21. }
  22. gradientDrawable.setBounds(leftOffset, 0, rightOffset, getHeight());
  23. }
  24. });
  25. //填充颜色
  26. ObjectAnimator bgColorAnimation = ObjectAnimator.ofInt(gradientDrawable, "color", 0xff99cc00, Color.WHITE);
  27. bgColorAnimation.setEvaluator(new ArgbEvaluator());
  28. //描边
  29. ObjectAnimator strokeColorAnimation =
  30. ObjectAnimator.ofInt(background, "strokeColor", 0xff99cc00, Color.GRAY);
  31. strokeColorAnimation.setEvaluator(new ArgbEvaluator());
  32. //圆角
  33. ObjectAnimator cornerAnimation =
  34. ObjectAnimator.ofFloat(gradientDrawable, "cornerRadius", 0, 30);
  35. AnimatorSet animatorSet = new AnimatorSet();
  36. animatorSet.setDuration(2000);
  37. animatorSet.playTogether(bgColorAnimation, cornerAnimation, widthAnimation,strokeColorAnimation);
  38. animatorSet.start();
  39. }
1. 源码分析(此处以Sample2Activity 为实例):
按钮的几种状态, [, IDLE, ,]


初始为IDLE 状态.
在Sample2Activity 中, 仅仅只是调用了setProgress()来实现其整个过程.其value的值从0-100不断的递增

在setProgress()方法中,其中的从初始化进入加载的圆环状态如下

 在morphToProgress() 方法,实现了从初始化到圆环状态的过度.在setListener(mProgressStateListener)中当动画完成的时候将mState 置为Progress

在createProgressMorphing() 中为创建一个MorphingAnimation实例, 其主要设置动画的圆角,宽度,颜色等等 ,在0中的简单实现也用到了这个类中的一些代码(MorphingAnimation.start() 方法中的代码片段).

在动画完成后,在setProgress 中将进行不断的界面刷新invalidate() 在调用此方法,则系统将进行重绘调用onDraw()方法


当setProgress() 到达100的时候则会调用到morphProgressToComplete() 来转换到完成的状态.

---------------------------------------------------END---------------------------------------------------------------------------------------
StateListDrawable 类, 在xml 中,通过设置<selector/> 来的background来进行按钮点击时候的背景的切换,此类则是该xml 的实现.
ColorStateList  类似.

开源项目: circular-progress-button的更多相关文章

  1. 用开源项目circular progress button实现有进度条的Button

    circular progress button可以让button实现进度条,效果和动画都做的很赞,只是有点小bug.需要注意的是按钮上的文字不能太大,否则会出现错位. 项目的地址:https://g ...

  2. Android开源项目发现---TextView,Button篇(持续更新)

    android-flowtextview 文字自动环绕其他View的Layout 项目地址:https://code.google.com/p/android-flowtextview/ 效果图:ht ...

  3. Github上关于iOS的各种开源项目集合(强烈建议大家收藏,查看,总有一款你需要)

    下拉刷新 EGOTableViewPullRefresh - 最早的下拉刷新控件. SVPullToRefresh - 下拉刷新控件. MJRefresh - 仅需一行代码就可以为UITableVie ...

  4. iOS及Mac开源项目和学习资料【超级全面】

    UI 下拉刷新 EGOTableViewPullRefresh – 最早的下拉刷新控件. SVPullToRefresh – 下拉刷新控件. MJRefresh – 仅需一行代码就可以为UITable ...

  5. iOS开发--iOS及Mac开源项目和学习资料

    文/零距离仰望星空(简书作者)原文链接:http://www.jianshu.com/p/f6cdbc8192ba著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”. 原文出处:codecl ...

  6. iOS、mac开源项目及库汇总

    原文地址:http://blog.csdn.net/qq_26359763/article/details/51076499    iOS每日一记------------之 中级完美大整理 iOS.m ...

  7. github上关于iOS的各种开源项目集合(转)

    UI 下拉刷新 EGOTableViewPullRefresh - 最早的下拉刷新控件. SVPullToRefresh - 下拉刷新控件. MJRefresh - 仅需一行代码就可以为UITable ...

  8. 分享海量 iOS 及 Mac 开源项目和学习资料

    UI 下拉刷新 EGOTableViewPullRefresh - 最早的下拉刷新控件. SVPullToRefresh - 下拉刷新控件. MJRefresh - 仅需一行代码就可以为UITable ...

  9. iOS -- 开源项目和库

    TimLiu-iOS 目录 UI 下拉刷新 模糊效果 AutoLayout 富文本 图表 表相关与Tabbar 隐藏与显示 HUD与Toast 对话框 其他UI 动画 侧滑与右滑返回手势 gif动画 ...

  10. 最全面的iOS和Mac开源项目和第三方库汇总

    标签: UI 下拉刷新 EGOTableViewPullRefresh – 最早的下拉刷新控件. SVPullToRefresh – 下拉刷新控件. MJRefresh – 仅需一行代码就可以为UIT ...

随机推荐

  1. Flutter实战视频-移动电商-38.路由_Fluro中Handler编写方法

    38.路由_Fluro中Handler编写方法 在main.dart中初始化Fluro 编写handler 在lib下新建routers文件夹,表示里面要很多路由相关的文件 我们声明一个Handler ...

  2. 09.客户端集成IdentityServer

    09.客户端集成IdentityServer 新建API的项目 dotnet new webapi --name ClientCredentialApi 在我们上一节课的代码IdentityServe ...

  3. JS鼠标响应事件经过、移动、点击示例介绍

    原文: http://www.jb51.net/article/41124.htm onMouseDown 按下鼠标时触发 onMouseOver 鼠标经过时触发 onMouseUp 按下鼠标松开鼠标 ...

  4. C#获得当前执行的函数名、当前代码行、源代码文件名

    http://blog.csdn.net/newegg2009/article/details/6220385 C#获得当前执行的函数名.当前代码行.源代码文件名 [日期:2010-10-18 11: ...

  5. sql server 2012 导出sql文件

    导出表数据和表结构sql文件 在工作中,经常需要导出某个数据库中,某些表数据:或者,需要对某个表的结构,数据进行修改的时候,就需要在数据库中导出表的sql结构,包括该表的建表语句和数据存储语句!在这个 ...

  6. get 和 post 请求的区别(转)

    转自 http://www.cnblogs.com/hyddd/archive/2009/03/31/1426026.html http://www.nowamagic.net/librarys/ve ...

  7. unicode官网 unicode码表和标准下载

  8. pip 参数

    pip 自带参数 pip --help pip install 自带参数 pip install --help

  9. PAT甲级——1118 Birds in Forest (并查集)

    此文章 同步发布在CSDN:https://blog.csdn.net/weixin_44385565/article/details/89819984   1118 Birds in Forest  ...

  10. GYM 101673E(暴搜预处理)

    1.不会超过500个不同的串-- 2.样例没给has到has是怎么样的,实测是true. 3.记忆化别剪错枝就好,嘤嘤嘤-- const int maxn = 505 + 5; int n, m, t ...