bl双性强迫侵犯h_国产在线观看人成激情视频_蜜芽188_被诱拐的少孩全彩啪啪漫画

Android自定義View——扇形統(tǒng)計圖的實現(xiàn)代碼

Android 扇形統(tǒng)計圖

成都創(chuàng)新互聯(lián)主營開魯網站建設的網絡公司,主營網站建設方案,成都APP應用開發(fā),開魯h5成都小程序開發(fā)搭建,開魯網站營銷推廣歡迎開魯?shù)鹊貐^(qū)企業(yè)咨詢

先看看效果:

看上去如果覺得還行就繼續(xù)往下看吧!

Android自定義View——扇形統(tǒng)計圖的實現(xiàn)代碼

Android自定義View——扇形統(tǒng)計圖的實現(xiàn)代碼

自定義View

定義成員變量

  private int mHeight, mWidth;//寬高
  private Paint mPaint;//扇形的畫筆
  private Paint mTextPaint;//畫文字的畫筆

  private int centerX, centerY;//中心坐標

  //"其他"的value
  //扇形圖分成太多快 所以要合并一部分為其他 即圖中灰色部分
  private double rest;

  private int maxNum = 5;//扇形圖的最大塊數(shù) 超過的item就合并到其他

  String others = "其他";//“其他”塊要顯示的文字
  double total;//數(shù)據(jù)的總和 
  double[] datas;//數(shù)據(jù)集
  String[] texts;//每個數(shù)據(jù)對應的文字集

  //顏色 默認的顏色
  private int[] mColors = {
      Color.parseColor("#FF4081"), Color.parseColor("#ffc0cb"),
      Color.parseColor("#00ff00"), Color.parseColor("#0066ff"), Color.parseColor("#ffee00")
  };

  private int mTextSize;//文字大小 單位:像素

  private int radius = 1000;//半徑 在畫圖時初始化

測量寬高

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

    //獲取寬高 不要設置wrap_content
    mHeight = MeasureSpec.getSize(heightMeasureSpec);
    mWidth = MeasureSpec.getSize(widthMeasureSpec);

  }

畫圖

@Override
protected void onDraw(Canvas canvas) {
  super.onDraw(canvas);
  //無數(shù)據(jù) 直接返回
  if (datas == null || datas.length == 0) return;

  centerX = (getRight() - getLeft()) / 2;
  centerY = (getBottom() - getTop()) / 2;
  int min = mHeight > mWidth ? mWidth : mHeight;
  if (radius > min / 2) {
    radius = (int) ((min - getPaddingTop() - getPaddingBottom()) / 3.5);
  }

  //畫各個扇形
  drawCircle(canvas);

  //畫線與文字
  drawLineAndText(canvas);

}

畫扇形

一個圓形統(tǒng)計圖是由許多個扇形組成的,我們根據(jù)數(shù)據(jù)計算出每個扇形的角度即可。注意,畫弧度的時候,角度是順時針變大的!即與我們平時的坐標是反過來的

  //畫扇形
  private void drawCircle(Canvas canvas) {
    int centerX =( getRight() - getLeft() )/2;//中點
    int centerY = ( getBottom() - getTop()) /2;
    RectF rect = new RectF((float) (centerX - radius), centerY-radius,
        centerX+radius,centerY+radius);//圓形區(qū)域

    int start = 0;//扇形開始的角度
    for (int i = 0; i < (maxNum<datas.length?maxNum:datas.length); i++) {
      float angles = (float) ((datas[i] * 1.0f /total) * 360);//計算扇形的角度
      mPaint.setColor(mColors[i%mColors.length]);//顏色
      canvas.drawArc(rect,start,angles,true,mPaint);//畫扇形
      start += angles;//下一個扇形開始的角度
    }

    //畫"其他"部分 即圖中灰色的部分
    rest =0;//保存其他部分的value
    for(int i=maxNum;i<datas.length;i++){
      rest+=datas[i];
    }
    float angles = (float) 360 - start;//角度
    mPaint.setColor(Color.GRAY);
    canvas.drawArc(rect,start,angles,true,mPaint);

  }

畫線條和文字

主要是計算各個點的坐標很煩

這里我畫出一個圖 大家把代碼和圖對照理解一下

Android自定義View——扇形統(tǒng)計圖的實現(xiàn)代碼

 

//畫線與文字
  private void drawLineAndText(Canvas canvas) {
    int start = 0;
    //平移畫布到中心 所以下面的坐標是從中點開始算起的
    canvas.translate(centerX, centerY);
    mPaint.setStrokeWidth(4);//線條寬度

    //如果數(shù)據(jù)集過大 那么要合并到其他
    for (int i = 0; i < (maxNum < datas.length ? maxNum : datas.length); i++) {
      float angles = (float) ((datas[i] * 1.0f / total) * 360);
      //畫線條和文字
      drawLine(canvas, start, angles, texts[i], mColors[i % mColors.length]);
      start += angles;
    }
    //畫其他部分的線條和文字
    if (start < 360)//如果start小于360 說明有其他部分
      drawLine(canvas, start, 360 - start, others, Color.GRAY);

  }

  private void drawLine(Canvas canvas, int start, float angles, String text, int color) {
    mPaint.setColor(color);
    float stopX, stopY;
    stopX = (float) ((radius + 40) * Math.cos((2 * start + angles) / 2 * Math.PI / 180));
    stopY = (float) ((radius + 40) * Math.sin((2 * start + angles) / 2 * Math.PI / 180));

    canvas.drawLine((float) ((radius - 20) * Math.cos((2 * start + angles) / 2 * Math.PI / 180)),
        (float) ((radius - 20) * Math.sin((2 * start + angles) / 2 * Math.PI / 180)),
        stopX, stopY, mPaint
    );
    //畫橫線
    int dx;//判斷橫線是畫在左邊還是右邊
    int endX;
    if (stopX > 0) {
      endX = (centerX - getPaddingRight() - 20);
    } else {
      endX = (-centerX + getPaddingLeft() + 20);
    }
    //畫橫線
    canvas.drawLine(stopX, stopY,
        endX, stopY, mPaint
    );
    dx = (int) (endX - stopX);

    //測量文字大小
    Rect rect = new Rect();
    mTextPaint.getTextBounds(text, 0, text.length(), rect);
    int w = rect.width();
    int h = rect.height();
    int offset = 20;//文字在橫線的偏移量
    //畫文字 文字的Y坐標值的是文字底部的Y坐標
    canvas.drawText(text, 0, text.length(), dx > 0 ? stopX + offset : stopX - w - offset, stopY + h, mTextPaint);

    //測量百分比大小
    String percentage = angles / 3.60 + "";
    percentage = percentage.substring(0, percentage.length() > 4 ? 4 : percentage.length()) + "%";
    mTextPaint.getTextBounds(percentage, 0, percentage.length(), rect);
    w = rect.width() - 10;
    //畫百分比
    canvas.drawText(percentage, 0, percentage.length(), dx > 0 ? stopX + offset : stopX - w - offset, stopY - 5, mTextPaint);

  }

這樣我們就已經完成了繪制的工作了,接下來就是綁定數(shù)據(jù)啦!

大家只要把datas和texts填充好數(shù)據(jù)就可以啦,最好調用一下invalidate()方法請求重繪(如果已經繪制了)。

我使用了一個內部抽象類來實現(xiàn)數(shù)據(jù)綁定的:

  public abstract class ArcViewAdapter<T> {

    public void setData(List<T> list) {
      datas = new double[list.size()];
      texts = new String[list.size()];
      for (int i = 0; i < list.size(); i++) {
        total += getValue(list.get(i));
        datas[i] = getValue(list.get(i));
        texts[i] = getText(list.get(i));
      }
      invalidate();//請求重繪
    }

    //通過傳來的數(shù)據(jù)集的某個元素 得到具體的數(shù)字
    public abstract double getValue(T t);

    //通過傳來的數(shù)據(jù)集的某個元素 得到具體的描述
    public abstract String getText(T t);
  }

在布局文件里面引用這個ArcView,然后在MainActivity中綁定數(shù)據(jù):

ArcView arcView = (ArcView) findViewById(R.id.arc);
    List<Times> times = new ArrayList<>();
    for (int i = 6; i > 0; i--) {
      Times t = new Times();//這個類就只有兩個變量
      t.hour = i;
      t.text = "Number"+i;
      times.add(t);
    }

    //初始化適配器
    ArcView.ArcViewAdapter myAdapter = arcView.new ArcViewAdapter<Times>(){
      @Override
      public double getValue(Times times) {
        return times.hour;
      }

      @Override
      public String getText(Times times) {
        return times.text;
      }
    };
    myAdapter.setData(times);//綁定數(shù)據(jù)

大家可以設置各種setter以便在Activity中調用,來提高ArcView的靈活性,比如設置顏色、半徑、最大塊數(shù)等等。

demo下載地址:SectorDiagram_jb51.rar

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

分享標題:Android自定義View——扇形統(tǒng)計圖的實現(xiàn)代碼
文章源于:http://vcdvsql.cn/article44/gjsjhe.html

成都網站建設公司_創(chuàng)新互聯(lián),為您提供定制開發(fā)企業(yè)建站網站排名網站導航Google靜態(tài)網站

廣告

聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)

h5響應式網站建設