Android自定义View绘图实现渐隐动画

实现了一个有趣的小东西:使用自定义View绘图,一边画线,画出的线条渐渐变淡,直到消失。效果如下图所示:

用属性动画或者渐变填充(Shader)可以做到一笔一笔的变化,但要想一笔渐变(手指不抬起边画边渐隐),没在Android中找到现成的API可用。所以,自己做了一个。

基本的想法是这样的:

•在View的onTouchEvent中记录触摸点,生成一条一条的线LineElement,放在一个List中。给每个LineElement配置一个Paint实例。
•在onDraw中绘制线段。
•变换LineElement的Paint实例的Alpha值。
•根据Alpha值重组线段列表

别的不说了,上代码:

package com.example.disappearinglines;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.RectF;
import android.os.Handler;
import android.os.Message;
import android.os.SystemClock;
import android.support.annotation.NonNull;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;

public class DisappearingDoodleView extends View {
  final static String TAG = "DoodleView";
  class LineElement {
    static final public int ALPHA_STEP = 5;
    static final public int SUBPATH_DIMENSION = 8;
    public LineElement(){
      mPaint = new Paint();
      mPaint.setARGB(255, 255, 0, 0);
      mPaint.setAntiAlias(true);
      mPaint.setStrokeWidth(16);
      mPaint.setStrokeCap(Paint.Cap.BUTT);
      mPaint.setStyle(Paint.Style.STROKE);
    }
    public LineElement(Paint paint){
      mPaint = paint;
    }

    public void setPaint(Paint paint){
      mPaint = paint;
    }

    public void setAlpha(int alpha){
      mPaint.setAlpha(alpha);
    }

    public float mStartX = -1;
    public float mStartY = -1;
    public float mEndX = -1;
    public float mEndY = -1;
    public Paint mPaint;
  }

  private LineElement mCurrentLine = null;
  private List<LineElement> mLines = null;

  private long mElapsed = 0;
  private Handler mHandler = new Handler(){
    @Override
    public void handleMessage(Message msg){
      DisappearingDoodleView.this.invalidate();
    }
  };

  public DisappearingDoodleView(Context context){
    super(context);
  }

  public DisappearingDoodleView(Context context, AttributeSet attrs){
    super(context, attrs);
  }

  @Override
  protected void onDraw(Canvas canvas){
    mElapsed = SystemClock.elapsedRealtime();
    if(mLines != null) {
      for (LineElement e : mLines) {
        if(e.mStartX < 0 || e.mEndY < 0) continue;
        canvas.drawLine(e.mStartX, e.mStartY, e.mEndX, e.mEndY, e.mPaint);
      }
      compactPaths();
    }
  }

  @Override
  public boolean onTouchEvent(MotionEvent event){
    float x = event.getX();
    float y = event.getY();

    int action = event.getAction();
    if(action == MotionEvent.ACTION_UP){// end one line after finger release
      mCurrentLine.mEndX = x;
      mCurrentLine.mEndY = y;
      mCurrentLine = null;
      invalidate();
      return true;
    }

    if(action == MotionEvent.ACTION_DOWN){
      mCurrentLine = new LineElement();
      addToPaths(mCurrentLine);

      mCurrentLine.mStartX = x;
      mCurrentLine.mStartY = y;
      return true;
    }

    if(action == MotionEvent.ACTION_MOVE) {
      mCurrentLine.mEndX = x;
      mCurrentLine.mEndY = y;
      mCurrentLine = new LineElement();
      addToPaths(mCurrentLine);

      mCurrentLine.mStartX = x;
      mCurrentLine.mStartY = y;
    }

    if(mHandler.hasMessages(1)){
      mHandler.removeMessages(1);
    }
    Message msg = new Message();
    msg.what = 1;
    mHandler.sendMessageDelayed(msg, 0);

    return true;
  }

  private void addToPaths(LineElement element){
    if(mLines == null) {
      mLines = new ArrayList<LineElement>() ;
    }

    mLines.add(element);
  }

  public void compactPaths(){

    int size = mLines.size();
    int index = size - 1;
    if(size == 0) return;
    int baseAlpha = 255 - LineElement.ALPHA_STEP;
    int itselfAlpha;
    LineElement line;
    for(; index >=0 ; index--, baseAlpha -= LineElement.ALPHA_STEP){
      line = mLines.get(index);
      itselfAlpha = line.mPaint.getAlpha();
      if(itselfAlpha == 255){
        if(baseAlpha <= 0){
          ++index;
          break;
        }
        line.setAlpha(baseAlpha);
      }else{
        itselfAlpha -= LineElement.ALPHA_STEP;
        if(itselfAlpha <= 0){
          ++index;
          break;
        }
        line.setAlpha(itselfAlpha);
      }
    }

    if(index >= size){
      // all sub-path should disappear
      mLines = null;
    }
    else if(index >= 0){
      //Log.i(TAG, "compactPaths from " + index + " to " + (size - 1));
      mLines = mLines.subList(index, size);
    }else{
      // no sub-path should disappear
    }

    long interval = 40 - SystemClock.elapsedRealtime() + mElapsed;
    if(interval < 0) interval = 0;
    Message msg = new Message();
    msg.what = 1;
    mHandler.sendMessageDelayed(msg, interval);
  }
}

这个示例还可以添加一些效果,比如让线条一边变淡一边变细。

目前还有一些问题,线条粗的话,可以明显看到线段与线段之间有缝隙或裂口,哪位想到怎么优化?

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

(0)

相关推荐

  • Android编程实现canvas绘制柱状统计图功能【自动计算宽高及分度值、可左右滑动】

    本文实例讲述了Android编程实现canvas绘制柱状统计图功能.分享给大家供大家参考,具体如下: 这里实现了一个简单的柱状统计图,如下:   特点: 1.根据数据源自动计算每个条目的高度.宽度.间距,自动计算分度值. 2.当条目数较多时,可左右滑动查看全部内容,图形.文字同步滑动,并且松手后会渐渐的停下来(而不是立刻停下来). 代码: (1)核心代码:BarChartView.Java package com.sina.appbarchart; import android.app.Acti

  • Android编程之canvas绘制各种图形(点,直线,弧,圆,椭圆,文字,矩形,多边形,曲线,圆角矩形)

    本文实例讲述了Android编程之canvas绘制各种图形的方法.分享给大家供大家参考,具体如下: 1.首先说一下canvas类: Class Overview The Canvas class holds the "draw" calls. To draw something, you need 4 basic components: A Bitmap to hold the pixels, a Canvas to host the draw calls (writing into

  • Android 游戏开发之Canvas画布的介绍及方法

    Canvas,在英语中,这个单词的意思是帆布.在Android中,则把Canvas当做画布,只要我们借助设置好的画笔(Paint类)就可以在画布上绘制我们想要的任何东西:另外它也是显示位图(Bitmap类)的核心类.随用户的喜好,Canvas还可设置一些关于画布的属性,比如,画布的颜色.尺寸等.Canvas提供了如下一些方法:    Canvas(): 创建一个空的画布,可以使用setBitmap()方法来设置绘制具体的画布.    Canvas(Bitmap bitmap): 以bitmap对

  • Android画图之抗锯齿paint和Canvas两种方式实例

    在画图的时候,图片如果旋转或缩放之后,总是会出现那些华丽的锯齿.其实Android自带了解决方式. 方法一:给Paint加上抗锯齿标志.然后将Paint对象作为参数传给canvas的绘制方法. paint.setAntiAlias(true); 方法二:给Canvas加上抗锯齿标志. 有些地方不能用paint的,就直接给canvas加抗锯齿,更方便. 复制代码 代码如下: canvas.setDrawFilter(new PaintFlagsDrawFilter(0, Paint.ANTI_AL

  • Android编程之OpenGL绘图技巧总结

    本文实例讲述了Android编程之OpenGL绘图技巧.分享给大家供大家参考,具体如下: 很久不用OpenGL ES绘图,怕自己忘记了,于是重新复习一遍,顺便原理性的东西总结如下: 1. Android 3D坐标系统 如图: Android的三维坐标系统中: 坐标原点位于中央, X轴从左向右延伸,原点左边的值为负数,右边为正数: Y轴从下向上延伸,原点下边的值为负数,上边为正数: Z轴屏幕里面向外面延伸,屏幕里面为负数,外面为正数. 2. 开发工具(OpenGL和OpenGL ES)介绍 Ope

  • Android 通过onDraw实现在View中绘图操作的示例

    Android绘图操作,通过继承View实现,在onDraw函数中实现绘图.下面是一个简单的例子: 复制代码 代码如下: public class AndroidTest extends Activity {    /** Called when the activity is first created. */    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(save

  • Android中使用Canvas绘制南丁格尔玫瑰图(Nightingale rose diagram)

    南丁格尔玫瑰图 在常规图表中实在很惊艳,但我初看没看懂,一查原来南丁格尔这么伟大,确实值得尊敬. 再仔细研究了下这种图的构成,发现原来就是把柱形图的柱形换成了扇形图的半径来表示,当然,变种有好多,我这只是说我理解的这种. 知道了其构成方式后就好实现了,依传入参数个数决定其扇形角度,依百分比决定其扇形的半径长度,然后就一切都水到渠成了. 漂亮的美图献上: 附上实现代码: package com.xcl.chart; /** * Canvas练习 * 自已画南丁格尔玫瑰图(Nightingale r

  • Android编程之绘图canvas基本用法示例

    本文实例讲述了Android编程之绘图canvas基本用法.分享给大家供大家参考,具体如下: MainActivity的代码如下: package example.com.myapplication; import android.os.Bundle; import android.app.Activity; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedIns

  • Android编程开发之在Canvas中利用Path绘制基本图形(圆形,矩形,椭圆,三角形等)

    本文实例讲述了Android编程开发之在Canvas中利用Path绘制基本图形的方法.分享给大家供大家参考,具体如下: 在Android中绘制基本的集合图形,本程序就是自定义一个View组件,程序重写该View组件的onDraw(Canvase)方法,然后在该Canvas上绘制大量的基本的集合图形. 直接上代码: 1.自定义的View组件代码: package com.infy.configuration; import android.content.Context; import andro

  • Android使用Canvas绘制圆形进度条效果

    前言 Android自定义控件经常会用到Canvas绘制2D图形,在优化自己自定义控件技能之前,必须熟练掌握Canvas绘图机制.本文从以下三个方面对Canvas绘图机制进行讲解: 画布Canvas 画笔Paint 示例圆形进度条 画布Canvas 首先,来看一下Android官网对Canvas类的定义: The Canvas class holds the "draw" calls.To draw something, you need 4 basic components: A B

  • Android编程实现canvas绘制饼状统计图功能示例【自动适应条目数量与大小】

    本文实例讲述了Android编程实现canvas绘制饼状统计图功能.分享给大家供大家参考,具体如下: 本例的目的是实现一个简单的饼状统计图,效果如下:    特点: 1.使用非常方便,可放在xml布局文件中,然后在代码中设置内容,即: PieChartView pieChartView = (PieChartView) findViewById(R.id.pie_chart); PieChartView.PieItemBean[] items = new PieChartView.PieItem

  • Android中Canvas的常用方法总结

    一.对Canvas进行操作 对Canvas的一系列操作,是指对Canvas进行旋转.平移.缩放等操作. 这些操作可以让Canvas对象使用起来更加便捷. 二.Canvas平移 /** * 画布向(100,50)方向平移 * * 参数1: 向X轴方向移动100距离 * 参数2: 向Y轴方向移动50距离 */ canvas.translate(100, 50); 三.Canvas缩放 /** * 在X轴方向放大为原来2倍,Y轴方向方大为原来的4倍 * 参数1: X轴的放大倍数 * 参数2: Y轴的放

随机推荐