Android自定义控件实现支付宝记账饼图

 更新时间:2018年04月09日 11:39:43   作者:匡效国  
这篇文章主要为大家详细介绍了Android自定义控件实现支付宝记账饼图,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android实现支付宝记账饼图,点击旋转到最下面,供大家参考,具体内容如下

代码:

package com.example.a_102.myapplication7.ui; 
 
import java.util.ArrayList; 
import java.util.List; 
 
 
import android.animation.Animator; 
import android.animation.ValueAnimator; 
import android.annotation.TargetApi; 
import android.content.Context; 
import android.graphics.BitmapFactory; 
import android.graphics.Canvas; 
import android.graphics.Color; 
import android.graphics.Paint; 
import android.graphics.Rect; 
import android.graphics.RectF; 
import android.os.Build; 
import android.text.TextUtils; 
import android.util.AttributeSet; 
import android.util.Log; 
import android.view.MotionEvent; 
import android.view.View; 
import android.view.animation.AccelerateInterpolator; 
 
import com.example.a_102.myapplication7.util.Util; 
 
/** 
 * 
 */ 
public class SelectPieView extends View { 
 
 private static final String TAG = "CustomPie_tag"; 
 private int width; 
 private SelectPieCallBack mCallBack; 
 private boolean initPostion = false; 
 /** 
  * 当前选中的区域 
  */ 
 private int currentDownPostion; 
 
 public SelectPieView(Context context) { 
  super(context); 
 } 
 
 public SelectPieView(Context context, AttributeSet attrs) { 
  super(context, attrs); 
 } 
 
 public SelectPieView(Context context, AttributeSet attrs, int defStyleAttr) { 
  super(context, attrs, defStyleAttr); 
 } 
 
 @Override 
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
  super.onMeasure(widthMeasureSpec, heightMeasureSpec); 
  int width; 
  int hight; 
 
  int widthmode = MeasureSpec.getMode(widthMeasureSpec); 
  int widthsize = MeasureSpec.getSize(widthMeasureSpec); 
 
  int hightmode = MeasureSpec.getMode(heightMeasureSpec); 
  int hightsize = MeasureSpec.getSize(heightMeasureSpec); 
 
  if (MeasureSpec.EXACTLY == widthmode) { 
   width = widthsize; 
  } else { 
   width = 200; 
   if (MeasureSpec.AT_MOST == widthmode) { 
    width = Math.min(widthsize, 200); 
   } 
  } 
 
  if (MeasureSpec.EXACTLY == hightmode) { 
   hight = hightsize; 
  } else { 
   hight = 200; 
   if (MeasureSpec.AT_MOST == hightmode) { 
    hight = Math.min(hightsize, 200); 
   } 
  } 
 
  setMeasuredDimension(Math.min(width, hight), Math.min(width, hight)); 
 } 
 
 /** 
  * 笔宽 
  */ 
 private int mPaintWid; 
 /** 
  * 外边圆半径 
  */ 
 private int mOutRoot; 
 /** 
  * 内边圆半径 
  */ 
 private int mIntRoot; 
 /** 
  * 空白处宽度 
  */ 
 private int emptysize = -1; 
 /** 
  * 点击前的圆和点击后的圆半径差距 
  */ 
 private float betweenSize = 10; 
 /** 
  * 向限 
  */ 
 private int XIANGXAIN; 
 /** 
  * 开始的角度 
  */ 
 private float start = 360; 
 /** 
  * 旋转过的角度 
  */ 
 private List<startAndRoatData> haveRoats = new ArrayList<>(); 
 /** 
  * 
  */ 
 private String mTitle = "总消费"; 
 /** 
  * 
  */ 
 private String mSubTitle = "00"; 
 /** 
  * 
  */ 
 private String mSubTitleDot = "00"; 
 /** 
  * 是否在运行 
  */ 
 private boolean isRun = true; 
 
 private Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); 
 
 private Paint textPaint = new Paint(Paint.ANTI_ALIAS_FLAG); 
 /** 
  * 整数部分 
  */ 
 private Paint textPaintSubTitle= new Paint(Paint.ANTI_ALIAS_FLAG); 
 /** 
  * 小数部分 
  */ 
 private Paint textPaintSubTitleDot= new Paint(Paint.ANTI_ALIAS_FLAG); 
 
 @Override 
 protected void onDraw(Canvas canvas) { 
  // reSetData(); 
  if (null == datas || datas.size() == 0) 
   return; 
  width = getWidth(); 
  mOutRoot = width / 2; 
  mPaintWid = Util.dip2px(getContext(), 40); 
  mIntRoot = mOutRoot - mPaintWid; 
 
  paint.setStyle(Paint.Style.STROKE); 
  paint.setColor(Color.RED); 
  paint.setStrokeWidth(mPaintWid); 
  RectF rt = new RectF(mPaintWid / 2 + betweenSize, mPaintWid / 2 + betweenSize, 
    width - mPaintWid / 2 - betweenSize, width - mPaintWid / 2 - betweenSize); 
  RectF rt2 = new RectF(mPaintWid / 2, mPaintWid / 2, width - mPaintWid / 2, width - mPaintWid / 2); 
 
  int size = datas.size(); 
  float allValues = 0; 
  for (int i = 0; i < datas.size(); i++) { 
   allValues += datas.get(i).getValuse(); 
  } 
  // allValues = allValues + emptysize * size; 
  float sigleSize = (360 - emptysize * datas.size()) / (allValues * 1f); 
 
  float end = 0; 
  haveRoats.clear(); 
  for (int i = 0; i < size; i++) { 
   paint.setColor(getResources().getColor(datas.get(i).getColor())); 
   end = datas.get(i).getValuse() * sigleSize; 
 
   if (!isRun && datas.get(i).getPostion() == currentDownPostion && datas.size()>1) { 
    canvas.drawArc(rt2, start + 3, end - 6, false, paint); 
    canvas.drawArc(rt, start + 3, end - 6, false, paint); 
   } else { 
    canvas.drawArc(rt, start, end, false, paint); 
   } 
   Log.i(TAG, "first=" + start % 360 + "==" + end + "postion=" + datas.get(i).getPostion()); 
 
   haveRoats.add(new startAndRoatData(datas.get(i).getPostion(), start % 360, end)); 
 
   start = start + end + emptysize; 
 
  } 
 
  textPaint.setStrokeWidth(Util.dip2px(getContext(), 1)); 
  /** 画图片 */ 
  for (int i = 0; i < haveRoats.size(); i++) { 
   startAndRoatData startAndRoatData = haveRoats.get(i); 
 
   float x = 0; 
   float y = 0; 
 
   if (!isRun && currentDownPostion == haveRoats.get(i).getPostion() && datas.size()>1) { 
 
    x = (float) (Math 
      .cos(Math.PI / 180 * (startAndRoatData.getStartAng() + startAndRoatData.getRoatAng() / 2)) 
      * (mIntRoot + mPaintWid / 2) + mOutRoot); 
    y = (float) (Math 
      .sin(Math.PI / 180 * (startAndRoatData.getStartAng() + startAndRoatData.getRoatAng() / 2)) 
      * (mIntRoot + mPaintWid / 2) + mOutRoot); 
   } else { 
    x = (float) (Math 
      .cos(Math.PI / 180 * (startAndRoatData.getStartAng() + startAndRoatData.getRoatAng() / 2)) 
      * (mIntRoot + mPaintWid / 2 - betweenSize) + mOutRoot); 
    y = (float) (Math 
      .sin(Math.PI / 180 * (startAndRoatData.getStartAng() + startAndRoatData.getRoatAng() / 2)) 
      * (mIntRoot + mPaintWid / 2 - betweenSize) + mOutRoot); 
   } 
 
   Rect rect = new Rect((int) (x - mPaintWid / 3), (int) (y - mPaintWid / 3), (int) (x + mPaintWid / 3), 
     (int) (y + mPaintWid / 3)); 
 
   int width = BitmapFactory.decodeResource(getResources(), datas.get(i).getIcon()).getWidth(); 
   // L=n(圆心角度数)× π(圆周率)× r(半径)/180(角度制) 
   if (startAndRoatData.getRoatAng() * Math.PI * (mIntRoot + mPaintWid / 2) / 180 > width) { 
    canvas.drawBitmap(BitmapFactory.decodeResource(getResources(), datas.get(i).getIcon()), null, rect, 
      null); 
   } 
  } 
 
  textPaint.setStyle(Paint.Style.FILL); 
  textPaint.setTextSize(Util.dip2px(getContext(), 14)); 
  /** 写文字 */ 
  canvas.drawText(mTitle, width / 2 - (textPaint.measureText(mTitle)) / 2, 
    width / 2 - Util.dip2px(getContext(), 8), textPaint); 
 
  textPaintSubTitle.setTextSize(Util.dip2px(getContext(), 20)); 
  canvas.drawText(mSubTitle, 
    width / 2 - (textPaintSubTitle.measureText(mSubTitle)) / 2 - (textPaintSubTitleDot.measureText("."+mSubTitleDot) / 2), 
    width / 2 + Util.dip2px(getContext(), 15), textPaintSubTitle); 
 
  textPaintSubTitleDot.setTextSize(Util.dip2px(getContext(), 15)); 
  canvas.drawText("." + mSubTitleDot, 
    width / 2 + textPaintSubTitle.measureText(mSubTitle) /2 - (textPaintSubTitleDot.measureText("." + mSubTitleDot))/2, 
    width / 2 + Util.dip2px(getContext(), 15), textPaintSubTitleDot); 
  //Toast.makeText(getContext(), "=="+textPaint.measureText(mSubTitle), Toast.LENGTH_SHORT).show(); 
  /** 测试基线 */ 
  /* 
   * paint.setColor(Color.BLACK); 
   * paint.setStrokeWidth(Util.dip2px(getContext(), 1)); 
   * 
   * canvas.drawLine(0, width / 2, width, width / 2, paint); 
   * canvas.drawLine(width / 2, 0, width / 2, width, paint); 
   */ 
  /** 
   * 初始化位置 
   * */ 
  if (!initPostion) { 
   initPostion = true; 
 
   startAndRoatData roatData = haveRoats.get(0); 
   float currentCenterAng = roatData.getStartAng() + roatData.getRoatAng() / 2; 
   if (currentCenterAng < 90) { 
    starRoat(start, start + 90 - (roatData.getStartAng() + roatData.getRoatAng() / 2), false); 
   } else if (currentCenterAng > 90 && currentCenterAng < 270) { 
    starRoat(start, start - (currentCenterAng - 90), false); 
   } else { 
    starRoat(start, start + 360 + 90 - (roatData.getStartAng() + roatData.getRoatAng() / 2), false); 
   } 
   currentDownPostion = roatData.getPostion(); 
   isRun = false; 
   invalidate(); 
  } 
 
 } 
 
 /** 
  * 设置显示金额 
  * 
  * @param number 
  */ 
 public void setNumber(String number) { 
  if (TextUtils.isEmpty(number)) { 
   number = "0.00"; 
  } 
  if (number.contains(".")) { 
   String[] split = number.split("\\."); 
   mSubTitle = split[0]; 
   mSubTitleDot = split[1]; 
  } else { 
   mSubTitle = number; 
   mSubTitleDot = "00"; 
  } 
  if (mSubTitleDot.length() > 2) { 
   mSubTitleDot = mSubTitleDot.substring(0, 2); 
  } 
 } 
 
 private int Lxy2; 
 
 @Override 
 public boolean onTouchEvent(MotionEvent event) { 
 
  switch (event.getAction()) { 
  case MotionEvent.ACTION_DOWN: 
   float x = event.getX(); 
   float y = event.getY(); 
   double atan = Math.atan((x - mOutRoot) / (mOutRoot - y)); 
   double currntAngle = (atan / Math.PI * 180); 
   double clickAngle = 0; 
   Lxy2 = (int) Math.pow(x - mOutRoot, 2) + (int) Math.pow(mOutRoot - y, 2); 
   if (Math.pow(mIntRoot, 2) < Lxy2 && Lxy2 < Math.pow(mOutRoot, 2)) { 
 
    if (x > width / 2 && y > width / 2) { 
     /** currntAngle第四象限是负数 */ 
     // starRoat(start, (float) (start - currntAngle), true); 
     clickAngle = currntAngle +90; 
 
    } else if (x > width / 2 && y < width / 2) { 
     /** currntAngle第一象限是负数 */ 
     //starRoat(start, (float) (start + 180 - currntAngle), true); 
     clickAngle = currntAngle +270; 
    } else if (x < width / 2 && y < width / 2) { 
     /** currntAngle第二象限是正数 */ 
     //starRoat(start, (float) (start - (180 - Math.abs(currntAngle))), true); 
     clickAngle = currntAngle +270; 
    } else { 
     /** currntAngle第三象限是正数 */ 
     //starRoat(start, (float) (start - Math.abs(currntAngle)), true); 
     clickAngle = currntAngle +90; 
    } 
 
    int i = clickDownPostion(clickAngle); 
    startAndRoatData roatData = haveRoats.get(i); 
    currentDownPostion = roatData.getPostion(); 
    float currentCenterAng = roatData.getStartAng() + roatData.getRoatAng() / 2; 
    if (currentCenterAng < 90) { 
     starRoat(start, start + 90 - (roatData.getStartAng() + roatData.getRoatAng() / 2), true); 
    } else if (currentCenterAng > 90 && currentCenterAng < 270) { 
     starRoat(start, start - (currentCenterAng - 90), true); 
    } else { 
     starRoat(start, start + 360 + 90 - (roatData.getStartAng() + roatData.getRoatAng() / 2), true); 
    } 
 
   } 
 
   return true; 
  } 
 
  return super.onTouchEvent(event); 
 } 
 
 private int clickDownPostion(double clickAngle) { 
  for (int i = 0; i < haveRoats.size(); i++) { 
   startAndRoatData data = haveRoats.get(i); 
   if ((data.getStartAng() < clickAngle && data.getStartAng() + data.getRoatAng() > clickAngle) 
     || (data.getStartAng() + data.getRoatAng() > 360 
       && ((data.getStartAng() + data.getRoatAng()) % 360) > clickAngle)) { 
    return i; 
   } 
  } 
  return 0; 
 } 
 
 @TargetApi(Build.VERSION_CODES.HONEYCOMB) 
 public void starRoat(final float star, final float end, boolean isSmooth) { 
  ValueAnimator valueAnimator = ValueAnimator.ofFloat(star, end); 
  valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { 
   @Override 
   public void onAnimationUpdate(ValueAnimator animation) { 
    float animatedValue = (float) animation.getAnimatedValue(); 
    start = animatedValue; 
    isRun = true; 
    invalidate(); 
   } 
  }); 
  valueAnimator.setDuration(isSmooth ? (long) (700) : 10); 
  valueAnimator.setInterpolator(new AccelerateInterpolator()); 
  valueAnimator.start(); 
  valueAnimator.addListener(new Animator.AnimatorListener() { 
   @Override 
   public void onAnimationStart(Animator animation) { 
 
   } 
 
   @Override 
   public void onAnimationEnd(Animator animation) { 
 
    if (currentDownPostion == -1) { 
     start++; 
     starRoat(start, start++, false); 
    } else { 
     isRun = false; 
     invalidate();//画突出部分 
     mCallBack.currentPostion(currentDownPostion); 
    } 
   } 
 
   @Override 
   public void onAnimationCancel(Animator animation) { 
 
   } 
 
   @Override 
   public void onAnimationRepeat(Animator animation) { 
 
   } 
  }); 
 } 
 
 private List<PieData> datas = new ArrayList<>(); 
 
 public void reSetData(List<PieData> data) { 
  datas.clear(); 
  datas.addAll(data); 
  float all =0; 
  for (PieData da : datas) { 
   all += da.getValuse(); 
  } 
  if (all < 360) { 
   for (PieData da : datas) { 
    da.setValuse(da.getValuse() * 200); 
   } 
  } 
  for (PieData da : datas) { 
   all += da.getValuse(); 
  } 
  /**强制设置最低值*/ 
  for (PieData da : datas) { 
   if (da.getValuse() / all < 0.03) { 
    da.setValuse((float) (all * 0.03)); 
   } 
  } 
 
  invalidate(); 
 
 } 
 
 /** 
  * 判断当前选择的所在区间 
  * 
  * @return 
  */ 
 private int findCurrentDownPostion() { 
  if (haveRoats == null || haveRoats.size() <= 0) { 
   return 1; 
  } 
 
  for (int i = 0; i < haveRoats.size(); i++) { 
 
   float startAng = haveRoats.get(i).getStartAng(); 
   float roatAng = haveRoats.get(i).getRoatAng(); 
   //Utility.Logi(TAG, "currentpostion=sstar=" + startAng + "===rroat=" + roatAng); 
   if (startAng < 90 && (startAng <= 90 && startAng + roatAng > 90)) { 
    // Utility.Logi(TAG, "currentpostion=" + haveRoats.get(i).getPostion()); 
    return haveRoats.get(i).getPostion(); 
   } else if (startAng > 90 && startAng - 360 + roatAng > 90) { 
    //Utility.Logi(TAG, "currentpostion=" + haveRoats.get(i).getPostion()); 
    return haveRoats.get(i).getPostion(); 
   } 
  } 
  return -1; 
 } 
 
 public void setCallBack(SelectPieCallBack callBack) { 
  this.mCallBack = callBack; 
 } 
 
 public interface SelectPieCallBack { 
  void currentPostion(int postion); 
 } 
 
 public static class PieData { 
 
  public PieData(int postion, float valuse, int color, int icon) { 
   this.postion = postion; 
   this.valuse = valuse; 
   this.color = color; 
   this.icon = icon; 
  } 
 
  private int postion; 
 
  private float valuse; 
 
  private int color; 
 
  private int icon; 
 
  public int getPostion() { 
   return postion; 
  } 
 
  public void setPostion(int postion) { 
   this.postion = postion; 
  } 
 
  public float getValuse() { 
   return valuse; 
  } 
 
  public void setValuse(float valuse) { 
   this.valuse = valuse; 
  } 
 
  public int getColor() { 
   return color; 
  } 
 
  public void setColor(int color) { 
   this.color = color; 
  } 
 
  public int getIcon() { 
   return icon; 
  } 
 
  public void setIcon(int icon) { 
   this.icon = icon; 
  } 
 } 
 
 class startAndRoatData { 
 
  private int postion; 
 
  private float startAng; 
 
  private float roatAng; 
 
  public startAndRoatData(int postion, float startAng, float roatAng) { 
   this.postion = postion; 
   this.startAng = startAng; 
   this.roatAng = roatAng; 
  } 
 
  public int getPostion() { 
   return postion; 
  } 
 
  public void setPostion(int postion) { 
   this.postion = postion; 
  } 
 
  public float getStartAng() { 
   return startAng; 
  } 
 
  public void setStartAng(float startAng) { 
   this.startAng = startAng; 
  } 
 
  public float getRoatAng() { 
   return roatAng; 
  } 
 
  public void setRoatAng(float roatAng) { 
   this.roatAng = roatAng; 
  } 
 
  @Override 
  public String toString() { 
   return "startAndRoatData{" + "postion=" + postion + ", startAng=" + startAng + ", roatAng=" + roatAng + '}'; 
  } 
 } 
} 

用法如下:

SelectPieView mPie = (SelectPieView) findViewById(R.id.pie); 
mPie.setCallBack(new SelectPieView.SelectPieCallBack() { 
 @Override 
 public void currentPostion(int postion) { 
  Toast.makeText(CustomViewActivity.this, "postion="+postion, Toast.LENGTH_SHORT).show(); 
 } 
}); 
ArrayList<SelectPieView.PieData> datas = new ArrayList<>(); 
datas.add(new SelectPieView.PieData(1, 200.3f, R.color.read_color, R.drawable.arrow)); 
datas.add(new SelectPieView.PieData(2, 200.3f, R.color.hx_red, R.drawable.arrow)); 
datas.add(new SelectPieView.PieData(3, 200.3f, R.color.text_blue, R.drawable.arrow)); 
mPie.reSetData(datas); 
mPie.setNumber("333.33"); 

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

相关文章

  • 了解Android OpenGLES2.0(一)

    了解Android OpenGLES2.0(一)

    OpenGLES2.0是一个功能强大,调用方便的底层图形库,这篇文章主要为大家详细介绍了Android OpenGLES2.0的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • Android实现跑马灯效果的两种简单方式

    Android实现跑马灯效果的两种简单方式

    这篇文章主要给大家介绍了关于Android实现跑马灯的两种简单方式,文中介绍了两种方法,分别说了每个方法的优缺点,需要的朋友可以选择性使用,下面来一起看看吧
    2021-07-07
  • Android中关于屏幕的三个小众知识(宽屏适配、禁止截屏和保持屏幕常亮)

    Android中关于屏幕的三个小众知识(宽屏适配、禁止截屏和保持屏幕常亮)

    这篇文章主要给大家介绍了Android中关于屏幕的三个小众知识,分别是宽屏适配、禁止截屏和保持屏幕常亮的相关资料,文中通过示例代码介绍的非常详细,需要的朋友们可以参考学习,下面随着小编来一起学习学习吧。
    2017-12-12
  • Android仿QQ微信实时监测网络状态

    Android仿QQ微信实时监测网络状态

    这篇文章主要为大家详细介绍了Android仿QQ微信实时监测网络状态,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • Android模仿美团顶部的滑动菜单实例代码

    Android模仿美团顶部的滑动菜单实例代码

    最近在工作遇到一个需要,要做一个滑动菜单,实现的效果类似美团顶部的滑动菜单,所以下面这篇文章主要给大家介绍了关于Android如何模仿美团顶部滑动菜单的相关资料,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2017-08-08
  • 详解Android四种存储方式

    详解Android四种存储方式

    在Android程序开发中我们经常遇到四种数据存储方式,每种存储方式都各不相同,下面通过本篇文章给大家介绍android四种存储方式,对此感兴趣的朋友一起学习吧
    2015-12-12
  • Android实现平铺图片效果

    Android实现平铺图片效果

    这篇文章主要为大家介绍了Android实现平铺图片效果的相关方法,感兴趣的小伙伴们可以参考一下
    2016-02-02
  • Android指纹识别API讲解,一种更快更好的用户体验

    Android指纹识别API讲解,一种更快更好的用户体验

    今天小编就为大家分享一篇关于Android指纹识别API讲解,一种更快更好的用户体验,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-10-10
  • Android清空编辑框内容功能的实现实例代码

    Android清空编辑框内容功能的实现实例代码

    本篇文章主要介绍了Android清空编辑框数据功能的实现实例代码,非常具有实用价值,需要的朋友可以参考下。
    2017-03-03
  • Android实现pdf在线预览或本地预览的方法

    Android实现pdf在线预览或本地预览的方法

    下面小编就为大家分享一篇Android实现pdf在线预览或本地预览的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01

最新评论