服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Android - Android实现支付宝蚂蚁森林水滴浮动效果

Android实现支付宝蚂蚁森林水滴浮动效果

2022-03-06 20:34yangyong915 Android

这篇文章主要为大家详细介绍了Android实现支付宝蚂蚁森林水滴浮动效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

可以有多个水滴,可以控制位置,水滴上下浮动。点击水滴产生搜集动画,水滴向树移动并逐渐消失,如图:

Android实现支付宝蚂蚁森林水滴浮动效果

那么是如何实现的呢,下面我们一步步来分析:

1、定义一个继承Relativelayout 的子类作为容器放置多个水滴并在Onlayout()中设置子控件的位置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
@Override
  protected void onLayout(boolean changed, int l, int t, int r, int b) {
    final int count = getChildCount();
 
    for (int i = 0; i < count; i++) {
      View child = getChildAt(i);
      int childWidth = child.getMeasuredWidth();
      int childHeight = child.getMeasuredHeight();
      if (child.getVisibility() != GONE) {
        child.layout(listX.get(i), listY.get(i), childWidth + listX.get(i), childHeight + listY.get(i));
      }
    }
  }

上面代码最重要的就是child.layout()函数,前两个参数为子控件的位置,这我先去盗个图:

Android实现支付宝蚂蚁森林水滴浮动效果

如图,前两个参数分别为getLeft 和getTop,后两个参数分别为getRight和getBottom;前两个参数其实是我们重外界传进来的子坐标列表,代码如下:

?
1
2
3
4
5
6
7
List<Integer> listX = new ArrayList<>();
  List<Integer> listY = new ArrayList<>();
 
  public void setChildPosition(int posx, int posy) {
    listX.add(posx);
    listY.add(posy);
  }

对于后面两个参数我们需要先获取子控件的宽高;然后在叠加上前面两个参数就是我们需要的坐标,在上面代码中可以看到我们是通过child.getmeasure来获取的宽高,但在获取宽高之前我们还需要去测量子空间的宽高。这个测量需要在measure()中完成:

?
1
2
3
4
5
@Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    measureChildren(widthMeasureSpec, heightMeasureSpec);
  }

至此,我们的父容器已经设计完成,接下来我们需要自己定义子控件以及子控件的动画,首先是一个浮动的动画,因为我们这里后面需要监听点击动作,所以最好使用属性动画完成,如下:

?
1
2
3
4
5
6
7
private void doRepeatAnim() {
    ObjectAnimator animator = ObjectAnimator.ofFloat(this, "translationY", -padding, padding, -padding);
    animator.setRepeatMode(ObjectAnimator.REVERSE);
    animator.setRepeatCount(ObjectAnimator.INFINITE);
    animator.setDuration(1500);
    animator.start();
  }

就是让其沿Y轴上下移动,设置为INFINTE则为无限重复动画;第二个动画就是我们点击的时候,子控件会移动到某个特定的位置并逐渐消失:

?
1
2
3
4
5
6
this.setOnClickListener(new OnClickListener() {
     @Override
     public void onClick(View view) {
       doSetAnim();
     }
   });
?
1
2
3
4
5
6
7
8
9
10
11
12
private void doSetAnim() {
    if (isCollect) return;
    isCollect = true;
    ObjectAnimator move1 = ObjectAnimator.ofFloat(this, "translationX", startWidth, endWidth);
    ObjectAnimator move2 = ObjectAnimator.ofFloat(this, "translationY", startHeight, endHeight);
    ObjectAnimator move3 = ObjectAnimator.ofFloat(this, "alpha", 1, 0);
 
    AnimatorSet animatorSet = new AnimatorSet();
    animatorSet.playTogether(move1, move2, move3);
    animatorSet.setDuration(1500);
    animatorSet.start();
  }

里面我添加了isCollect 判断,用于处理点击事件重复生效的问题,这里是一个动画组合,重当前位置移动到特定位置同时透明度也不断的变淡。写动画的时候特别应该注意一个问题就是当前的所有位置都不是外面传进来的位置而是以当前控件初始位置为参考的相对位置,因为我们在父控件的时候就设定好了子控件的位置,不能再次进行重复设定不然会叠加,所以上面的startwidth 和startHeight其实都是0,endWidth 和endHeight也是结束位置减去控件移动的初始位置:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/**
   * @param context
   */
  public WaterView(Context context) {
    super(context);
    this.context = context;
    endWidth = (int) DeviceUtils.dpToPixel(context, 160);
    endHeight = (int) DeviceUtils.dpToPixel(context, 300);
    padding = (int) DeviceUtils.dpToPixel(context, 10);
    startWidth = 0;
    startHeight = 0;
  }
 
  /**
   * @param index
   * @param startWidth 开始坐标 X
   * @param startHeight 开始坐标 Y
   */
  public void setPosition(int index, int startWidth, int startHeight) {
    this.index = index;
    endWidth = endWidth - startWidth;
    endHeight = endHeight - startHeight;
  }

,在设置动画之前,我们还缺少初始化控件的步骤,这个步骤就是绘制背景控件,这个过程在ondraw()方法中进行:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
@Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    mMWidth = getMeasuredWidth();
    mHeight = getMeasuredHeight();
  }
 
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    mPaint.setColor(getResources().getColor(R.color.color_87d1ab));
    mPaint.setStyle(Paint.Style.FILL);
    canvas.drawCircle(mMWidth / 2, (float) mHeight / 2, DeviceUtils.dpToPixel(context, 30), mPaint);
 
    mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    mTextPaint.setColor(Color.WHITE);
    mTextPaint.setTextSize(DeviceUtils.dpToPixel(context, 30));
    mTextPaint.setColor(getResources().getColor(R.color.text_color_fc));
    mTextPaint.setStyle(Paint.Style.FILL);
    float width = mTextPaint.measureText(text);
    canvas.drawText(text, mMWidth / 2 - width / 2, (float) mHeight * 0.65f, mTextPaint);
 
    doRepeatAnim();
    this.setOnClickListener(new OnClickListener() {
      @Override
      public void onClick(View view) {
        doSetAnim();
      }
    });
  }

如上,我绘制了一个纯色的园和特定的文字。当子控件绘制完成后才进行的动画。

最后就是如何使用我们刚才做好了轮子啦,请看代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
@Override
  protected void onStart() {
    super.onStart();
    int posx = (int) DeviceUtils.dpToPixel(this, 70);
    int posy = (int) DeviceUtils.dpToPixel(this, 70);
    addChildView(this, relative, 1, posx, posy);
    addChildView(this, relative, 2, 2 * posx, 2 * posy);
    addChildView(this, relative, 3, 3 * posx, posy);
  }
 
  /**
   * 添加子水滴
   *
   * @param relative
   * @param index  第几个
   * @param posx   子控件初始位置x
   * @param posy   子控件初始位置y
   */
  private void addChildView(final Context context, final WaterContainer relative, final int index, final int posx, final int posy) {
    relative.postDelayed(new Runnable() {
      @Override
      public void run() {
        int width = (int) DeviceUtils.dpToPixel(context, 60);
        int height = (int) DeviceUtils.dpToPixel(context, 60);
        ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(width, height);
        WaterView waterView = new WaterView(context);
        waterView.setPosition(index, posx, posy);
        waterView.setLayoutParams(layoutParams);
        relative.setChildPosition(posx, posy);
        relative.addView(waterView);
      }
    }, (index - 1) * 300);
  }

在添加代码里面,我添加了一个延时,这样每个添加的子水滴就会不同步的上下跳动,看起来更为真实,如果你有更好的办法请一定记得告诉我,上面的代码就是通过LayoutParams先设定子控件的布局,再把子控件添加到父容器中去。可以实现重复调用,就是这么简单。

最后给出项目的github地址:链接地址

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/yangyong915/article/details/79012025

延伸 · 阅读

精彩推荐
  • AndroidAndroid实现Service获取当前位置(GPS+基站)的方法

    Android实现Service获取当前位置(GPS+基站)的方法

    这篇文章主要介绍了Android实现Service获取当前位置(GPS+基站)的方法,较为详细的分析了Service基于GPS位置的技巧,具有一定参考借鉴价值,需要的朋友可以参考下...

    Ruthless8342021-03-31
  • AndroidAndroid编程解析XML方法详解(SAX,DOM与PULL)

    Android编程解析XML方法详解(SAX,DOM与PULL)

    这篇文章主要介绍了Android编程解析XML方法,结合实例形式详细分析了Android解析XML文件的常用方法与相关实现技巧,需要的朋友可以参考下...

    liuhe68810052021-05-03
  • AndroidAndroid中AsyncTask详细介绍

    Android中AsyncTask详细介绍

    这篇文章主要介绍了Android中AsyncTask详细介绍,AsyncTask是一个很常用的API,尤其异步处理数据并将数据应用到视图的操作场合,需要的朋友可以参考下...

    Android开发网7452021-03-11
  • AndroidAndroid界面效果UI开发资料汇总(附资料包)

    Android界面效果UI开发资料汇总(附资料包)

    android ui界面设计,友好的界面会提高用户体验度;同时也增强了android ui界面设计的难度,本文提供了一些常用开发资料(有下载哦)感兴趣的朋友可以了解下...

    Android开发网4672021-01-03
  • AndroidAndroid实现固定屏幕显示的方法

    Android实现固定屏幕显示的方法

    这篇文章主要介绍了Android实现固定屏幕显示的方法,实例分析了Android屏幕固定显示所涉及的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下...

    鉴客6192021-03-27
  • AndroidAndroid CardView+ViewPager实现ViewPager翻页动画的方法

    Android CardView+ViewPager实现ViewPager翻页动画的方法

    本篇文章主要介绍了Android CardView+ViewPager实现ViewPager翻页动画的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    Abby代黎明9602022-03-02
  • Android汇总Android视频录制中常见问题

    汇总Android视频录制中常见问题

    这篇文章主要汇总了Android视频录制中常见问题,帮助大家更好地解决Android视频录制中常见的问题,需要的朋友可以参考下...

    yh_thu5192021-04-28
  • AndroidAndroid程序设计之AIDL实例详解

    Android程序设计之AIDL实例详解

    这篇文章主要介绍了Android程序设计的AIDL,以一个完整实例的形式较为详细的讲述了AIDL的原理及实现方法,需要的朋友可以参考下...

    Android开发网4642021-03-09