一款非常简单酷炫的LoadingView动画效果

今天看到一个银行的APP上面的loadingview 挺好的,就尝试着自己实现,觉得很简单,但自己实现起来还是发现了一些问题。

LoadingView和下图类似:

实现的代码也不是很复杂,就是小球的运动轨迹需要计算,我自己手画了个计算的图,很简单的就是三角函数的使用。

然后代码就是代码实现了,主要的内容都有注释,代码如下:

public class LoadingView extends View {

  private final static String TAG = "LoadingView";

  private final static int LEFT_BALL_DOWN = 1;
  private final static int LEFT_BALL_UP = 2;
  private final static int RIGHT_BALL_DOWN = 3;
  private final static int RIGHT_BALL_UP = 4;

  private Paint paint1, paint2, paint3, paint4, paint5;
  private int mCurrentAnimatorValue;
  private int circleRadius = 10; //小球的半径
  private int distance = 60; //小球开始下落到最低点的距离
  private int mCurrentState = LEFT_BALL_DOWN;

  public LoadingView(Context context) {
    super(context);
    init(context);
  }

  public LoadingView(Context context, @Nullable AttributeSet attrs) {
    super(context, attrs);
    init(context);
  }

  public LoadingView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);

    init(context);
  }

  private void init(Context context) {

    paint1 = getPaint(Color.RED);
    paint2 = getPaint(Color.YELLOW);
    paint3 = getPaint(Color.GREEN);
    paint4 = getPaint(Color.BLUE);
    paint5 = getPaint(Color.CYAN);

    ValueAnimator animator = ValueAnimator.ofInt(0, 90);

    animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
      @Override
      public void onAnimationUpdate(ValueAnimator animation) {
        mCurrentAnimatorValue = (int) animation.getAnimatedValue();
        Log.e(TAG, "onAnimationUpdate : mCurrentAnimatorValue = " + mCurrentAnimatorValue);

        invalidate();
      }
    });

    animator.addListener(new Animator.AnimatorListener() {
      @Override
      public void onAnimationStart(Animator animation) {

      }

      @Override
      public void onAnimationEnd(Animator animation) {

      }

      @Override
      public void onAnimationCancel(Animator animation) {

      }

      @Override
      public void onAnimationRepeat(Animator animation) {
        Log.e(TAG, "onAnimationRepeat : mCurrentAnimatorValue = " + mCurrentAnimatorValue);
        switch (mCurrentState) {
          case LEFT_BALL_DOWN:
            mCurrentState = RIGHT_BALL_UP;
            break;
          case RIGHT_BALL_UP:
            mCurrentState = RIGHT_BALL_DOWN;
            break;
          case RIGHT_BALL_DOWN:
            mCurrentState = LEFT_BALL_UP;
            break;
          case LEFT_BALL_UP:
            mCurrentState = LEFT_BALL_DOWN;
            break;
        }
      }
    });
    animator.setStartDelay(500);
    animator.setDuration(600);
    animator.setRepeatCount(ValueAnimator.INFINITE);
    animator.setInterpolator(new DecelerateInterpolator());
    animator.start();

  }

  private Paint getPaint(int color) {
    Paint paint = new Paint();
    paint.setColor(color);
    paint.setAntiAlias(true);
    paint.setStyle(Paint.Style.FILL);
    return paint;
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  }

  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);

    int x, y;
    double cosValue = Math.cos(PI * mCurrentAnimatorValue / 180);
    double sinValue = Math.sin(PI * mCurrentAnimatorValue / 180);
    drawFourBall(canvas);
    switch (mCurrentState) {
      case LEFT_BALL_DOWN://最左边小球往下撞击
        x = circleRadius + (int) ((distance - circleRadius) * (1 - cosValue));
        y = getHeight() - distance + (int) ((distance - circleRadius) * sinValue);
        canvas.drawCircle(x, y, circleRadius, paint1);
        break;
      case RIGHT_BALL_UP://最右边小球往上撞击
        x = distance + 8 * circleRadius + (int) ((distance - circleRadius) * sinValue);
        y = getHeight() - distance + (int) (cosValue * (distance - circleRadius));
        canvas.drawCircle(x, y, circleRadius, paint5);
        break;
      case RIGHT_BALL_DOWN://最右边小球往下撞击
        x = distance + 8 * circleRadius + (int) ((distance - circleRadius) * (cosValue));
        y = (getHeight() - distance) + (int) ((distance - circleRadius) * (sinValue));
        canvas.drawCircle(x, y, circleRadius, paint5);
        break;
      case LEFT_BALL_UP://最左边小球往上撞击
        x = distance - (int) ((distance - circleRadius) * sinValue);
        y = getHeight() - distance + (int) ((distance - circleRadius) * cosValue);
        canvas.drawCircle(x, y, circleRadius, paint1);
        break;
    }

  }

  private void drawFourBall(Canvas canvas) {

    int y = getHeight() - circleRadius;
    canvas.drawCircle(distance + 2 * circleRadius, y, circleRadius, paint2);
    canvas.drawCircle(distance + 4 * circleRadius, y, circleRadius, paint3);
    canvas.drawCircle(distance + 6 * circleRadius, y, circleRadius, paint4);

    if (mCurrentState == LEFT_BALL_DOWN || mCurrentState == LEFT_BALL_UP) {//最左边球运动的时候,要绘制最右边的球
      canvas.drawCircle(distance + 8 * circleRadius, y, circleRadius, paint5);
    } else if (mCurrentState == RIGHT_BALL_UP || mCurrentState == RIGHT_BALL_DOWN) {//最右边球运动的时候,要绘制最左边的球
      canvas.drawCircle(distance, y, circleRadius, paint1);
    }

  }

}

实现的效果如图一,有问题的话互相讨论。最后贴上想xml文件,后续会完善设置loadingview的大小和颜色之类的参数。
xml如下:

<com.define_view.LoadingView
    android:layout_marginTop="20px"
    android:background="#999999"
    android:layout_width="200px"
    android:layout_height="200px" />

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

(0)

相关推荐

  • Android通用LoadingView加载框架详解

    手写一个通用加载中.显示数据.加载失败.空数据的LoadingView框架. 定义3个布局:加载中,加载失败,空数据 加载中: <?xml version="1.0" encoding="utf-8"?> <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent&q

  • Android 自定义通用的loadingview实现代码

    功能 1.显示加载视图,加载失败的时候显示加载失败视图,数据为空时显示数据为空视图,支持为失败视图设置点击事件重新加载数据. 2.支持个性化设置,自定义设置 加载.失败.空数据视图. 先放一张效果图压压惊 实现 实现思路其实就是一个FrameLayout里添加三个布局做处理显示隐藏,自定义视图其实就是替换里面的view ,代码比较简单,如果直接看过我的自定义view系列文章,或者对自定义view有所了解,都很容易看懂,所有直接上代码了. 具体代码 Java 代码 public class Com

  • Android自定义环形LoadingView效果

    最近项目有要用到环形的进度条,Github上有一个类似的DashedCircularProgress控件,但是他画的进度是通过设置画笔的虚线效果来实现间隔的:progressPaint.setPathEffect(new DashPathEffect(new float[]{dashWith, dashSpace}, dashSpace));如果内层还有一层圆环,在动态设置时,内层和外层有细微的偏差.于是我在原有基础上改了一个,实现了我要的效果(设置进度时可以选择加动画或者不加动画): 控件实现

  • Android项目实战手把手教你画圆形水波纹loadingview

    本文实例讲解的是如何画一个满满圆形水波纹loadingview,这类效果应用场景很多,比如内存占用百分比之类的,分享给大家供大家参考,具体内容如下 效果图如下: 预备的知识: 1.贝塞尔曲线    如果你不了解,可以来这里进行基础知识储备:神奇的贝塞尔曲线 2.Paint.setXfermode()  以及PorterDuffXfermode 千万不要被这个b的名字吓到,不熟悉看到可能会认为很难记,其实 只要站在巨人的丁丁上 还是很简单的. 好了 废话不多说 ,跟我一步步来做一个炫酷的view吧

  • Android实现创意LoadingView动画效果

    Android上的热火锅煮萝卜蔬菜的Loading动画效果. 这是一个锅煮萝卜的Loading动画,效果仿照自之前IOS上看到的一个效果,觉得挺有意思,就移植过来了,在此完成了Dialog的样式,方便使用者作为LoadingView去使用. 关键性代码: package yellow5a5.demo.boilingloadingview.View; import android.animation.Animator; import android.animation.AnimatorListen

  • 一款非常简单酷炫的LoadingView动画效果

    今天看到一个银行的APP上面的loadingview 挺好的,就尝试着自己实现,觉得很简单,但自己实现起来还是发现了一些问题. LoadingView和下图类似: 实现的代码也不是很复杂,就是小球的运动轨迹需要计算,我自己手画了个计算的图,很简单的就是三角函数的使用. 然后代码就是代码实现了,主要的内容都有注释,代码如下: public class LoadingView extends View { private final static String TAG = "LoadingView&

  • Flutter实战教程之酷炫的开关动画效果

    前言 此动画效果是我在浏览文章时发现的一个非常酷炫的效果,于是就使用 Flutter 实现了. 更多动画效果及Flutter资源: https://github.com/781238222/flutter-do 添加依赖 在项目的 pubspec.yaml 文件中添加依赖: dependencies: wheel_switch: ^0.0.1 执行命令: flutter pub get 使用 WheelSwitch( value: false, ) 组件默认的宽高分别是80.30,也可以指定宽高

  • 通过FancyView提供 Android 酷炫的开屏动画实例代码

    效果 使用 compile 'site.gemus:openingstartanimation:1.0.0' //在gradle中导入项目 OpeningStartAnimation openingStartAnimation = new OpeningStartAnimation.Builder(this) .setDrawStategy(new NormalDrawStrategy()) //设置动画效果 .create(); openingStartAnimation.show(this)

  • 通过JQuery实现win8一样酷炫的动态磁贴效果(示例代码)

    我个人表示非常喜欢微软新一代的产品,先不管它产品的成熟与否,但是它带来的是全新的产品.所谓全新,是指在用户体验上,苹果这些年的成功使得所有产品都在模仿它的界面,包括安卓在内,不知道大家的感觉如何,反正我是对这些圆角矩形产生了审美疲劳(苹果以及安卓的粉丝勿喷,这里仅仅是从界面上评价,事实上从整体上来说,微软还是有差距的),当年wp的推出让我眼前一亮,马上喜欢上了Metro风格的产品,直至今天wp8以及win8开始越来越成熟. 写的不好,欢迎各位看官指正批评,不欢迎无故猛喷.大神请绕道. 废话少说,

  • JavaScript+Canvas实现酷炫的粒子和流星效果

    目录 一:粒子效果 二:流星效果 一:粒子效果 <html> <head> <meta charset="utf-8"> <title>www.husonghe.com</title> <style> html { height: 100%; background-image: -webkit-radial-gradient(ellipse farthest-corner at center center, #1b

  • JS实现超炫网页烟花动画效果的方法

    本文实例讲述了JS实现超炫网页烟花动画效果的方法.分享给大家供大家参考.具体分析如下: 非常炫的使用JS实现的一个网页烟花燃放动画效果,能适应JS做出这样的动画来 复制代码 代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns

  • Flutter 实现酷炫的3D效果示例代码

    此文讲解3个酷炫的3D动画效果. 下面是要实现的效果: Flutter 中3D效果是通过 Transform 组件实现的,没有变换效果的实现: class TransformDemo extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('3D 变换Demo'), ), body: Container( alignm

  • 5分钟快速实现Android爆炸破碎酷炫动画特效的示例

    这个破碎动画,是一种类似小米系统删除应用时的爆炸破碎效果的动画. 效果图展示 先来看下是怎样的动效,要是感觉不是理想的学习目标,就跳过,避免浪费大家的时间.�� 源码在这里:point_right: https://github.com/ReadyShowShow/explosion 一行代码即可调用该动画 new ExplosionField(this).explode(view, null)) 下面开始我们酷炫的Android动画特效正式讲解:point_down: 先来个整体结构的把握 整

  • 简单JS打造酷炫代码雨(黑客高逼格)

    电影黑客帝国有个代码雨效果,满满的既视感,身为程序猿的你羡慕吗?只要很简单的HTML+JavaScript就能实现,甚至不需要懂任何技术.这篇文章主要介绍了简单JS打造酷炫代码雨(黑客高逼格),需要的朋友可以参考下 <!doctype html> <html> <head> <meta charset="utf-8" /> <title>流星雨</title> <meta name="keyword

随机推荐