自定义控件PieView和六芒星图 - DatePicker

DatePicker

在这里插入图片描述

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <DatePicker

        android:id="@+id/date_picker"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:background="@color/list_coler"
        android:calendarTextColor="@color/item_select_d"
        android:calendarViewShown="false"
        android:datePickerMode="spinner"></DatePicker>

</RelativeLayout>

PieView

package com.example.zxtext.diyUi;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.view.View;

import com.example.zxtext.bean.PieData;

import java.util.ArrayList;
import java.util.List;

import androidx.annotation.Nullable;

public class PieView extends View {
    // 颜色表 (注意: 此处定义颜色使用的是ARGB,带Alpha通道的)
    private int[] mColors = {0xFFCCFF00, 0xFF6495ED, 0xFFE32636, 0xFF800000, 0xFF808000, 0xFFFF8C69, 0xFF808080,
            0xFFE6B800, 0xFF7CFC00};

    //初始角度
    private float startAngel = 0;
    //数据集合
    private List<PieData> mPieData;
    //宽高
    private int mWidth,mHeight;
    //画笔
    private Paint mPaint;

    public PieView(Context context) {
        super(context);
        initPaint();
    }

    public PieView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        initPaint();
    }

    private void initPaint(){
        mPaint = new Paint();
        mPaint.setStyle(Paint.Style.FILL);
        mPaint.setAntiAlias(true);
        mPaint.setTextSize(10);
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        mWidth = w;
        mHeight = h;
    }

    @Override
    protected void onDraw(Canvas canvas) {
        if(mPieData.size() == 0 || null  == mPieData){
            return;
        }
        float currentAngle = startAngel;
        canvas.translate(mWidth/2,mHeight/2);
        float r = (float) (Math.min(mWidth, mHeight) / 2 * 0.8);
        RectF rectF = new RectF(-r,-r,r,r);
        for(int i  = 0;i<mPieData.size();i++){
            PieData data = mPieData.get(i);
            mPaint.setColor(data.getColor());
            canvas.drawArc(rectF,currentAngle,data.getAngle(),true,mPaint);
            //绘制文字
            float textAngle = currentAngle + data.getAngle()/2;
            //圆周率π - PI = 180度
            float x = (float)(r /2 *Math.cos(textAngle * Math.PI / 180));
            //弧度=角度*Math.PI/180; Math.cos()中的参数的单位是弧度,所以要放弧度
            //toRadians()是将角度转换为弧度
            //toDegrees()是将弧度转换为角度
            float y = (float)(r /2 *Math.sin(textAngle * Math.PI / 180));
            mPaint.setColor(Color.BLACK);
            canvas.drawText(data.getName(),x,y,mPaint);

            currentAngle += data.getAngle();
        }
    }


    // 设置数据
    public void setData(ArrayList<PieData> mData) {
        this.mPieData = mData;
        initData(mData);
        invalidate();   // 刷新
    }

    private void initData(ArrayList<PieData> mData) {
        if(mData.size() == 0 || null  == mData){
            return;
        }

        float sumValue = 0;
        for(int i = 0;i<mPieData.size();i++){
            PieData data = mPieData.get(i);
            sumValue += data.getValue();

            int j = i%mColors.length;
            data.setColor(mColors[j]);
        }

        float sumAngle = 0;
        for(int i = 0;i<mPieData.size();i++){
            PieData data = mPieData.get(i);

            float percentage  = data.getValue()/sumValue;
            float angle = percentage *360;
            sumAngle += angle;
            data.setAngle(angle);
            data.setPercentage(percentage);
        }
    }

}

六芒星图

package com.example.zxtext.diyUi;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.Rect;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;

import androidx.annotation.Nullable;

/*
* 模仿LPL选手六芒星图
* 场均击杀、分均输出、生存能力、分均承伤、分均补刀、分均经济
* */
public class HexagramView extends View {
    private int count = 6; //数据个数
    private float radius; //网格最大半径
    private int centerX,centerY;//中心点
    private float angle = (float) (Math.PI*2/count);//角度
    private int layerCount;
    private String [] titles = {"场均击杀","分均输出","生存能力","分均承伤","分均补刀","分均经济"};
    private double [] data = {80,50,100,20,40,80};
    private float maxValue = 100;
    private Paint textPaint;//文本画笔
    private Paint viewPaint;//六边形画笔
    private Paint valuePaint; //数据区画笔

    public HexagramView(Context context) {
        this(context,null);
        init();
    }

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

    private void init(){
        textPaint = new Paint();
        textPaint.setTextSize(10);
        textPaint.setColor(Color.BLACK);
        textPaint.setAntiAlias(true);

        viewPaint = new Paint();
        viewPaint.setAntiAlias(true);
        viewPaint.setColor(Color.WHITE);
        viewPaint.setStrokeWidth(2);
        viewPaint.setStyle(Paint.Style.STROKE);

        valuePaint = new Paint();
        valuePaint.setColor(Color.RED);
        valuePaint.setAlpha(125);
        valuePaint.setStrokeWidth(1);
        valuePaint.setStyle(Paint.Style.FILL_AND_STROKE);
        valuePaint.setAntiAlias(true);
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        radius = Math.min(h,w)/2 * 0.9f;//半径
        //中心坐标
        centerX = w/2;
        centerY = h/2;
        postInvalidate();
        super.onSizeChanged(w, h, oldw, oldh);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int width = MeasureSpec.getSize(widthMeasureSpec);
        int height = MeasureSpec.getSize(heightMeasureSpec);
        int th = Math.max(width,height);
        setMeasuredDimension(th,th);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        //canvas.drawColor(Color.RED);
        drawPolygon(canvas);
        //drawLinkLine(canvas);
        drawValueShape(canvas);
        drawTextTitles(canvas);
    }

    /*画正多边形*/
    private void drawPolygon(Canvas canvas){
        //PI  = π=180度  (180度 * 2 /6) 每个角的角度
       // float angle = (float) (Math.PI*2/count);//角度
        Path path = new Path();
        float r = radius /(count-1);//每个六边形的半径 我们需要画5个

        //开始绘制六边形
        for(int i = 1;i<count;i++){//中心点不要绘制
            //确定当前六边形的半径
            float curR = r * i;
            //连线  连6条边
            for(int j = 0 ;j<=count;j++){
                if(j == 0){
                    //起点 == 顶点 顺时针   屏幕的 x轴 y轴 是坐标系的第二象限  x向右 y向下
                    path.moveTo(centerX,centerY-curR);
                }else {
                    //顺时针记录其余顶角的点坐标
                    float x = (float) (centerX + Math.sin(angle * j) * curR);
                    float y = (float) (centerY - Math.cos(angle * j) * curR);
                    path.lineTo(x, y);
                }
            }

            //最外层的顶角外面的五个小圆点(图中红色部分)
           /* if (i == count-1) {
                for (int j = 0; j < count; j++) {
                    float x = (float) (centerX + Math.sin(angle * j) * (curR + 12));
                    float y = (float) (centerY - Math.cos(angle * j) * (curR + 12));
                    canvas.drawCircle(x, y, 4, textPaint);
                }
            }
            path.close();
            canvas.drawPath(path, textPaint);*/

        }
        path.close();//闭合路径
        canvas.drawPath(path,viewPaint);
    }

    /*对角连线*/
    private void drawLinkLine(Canvas canvas){
        float r = radius / (count);
        for (int i = 0; i <count; i++) {
            //起始坐标 从中心开始的话
            float startX=centerX ;
            float startY=centerY;
           /* float startX = (float) (centerX + Math.sin(angle * i) * r);
            float startY = (float) (centerY - Math.cos(angle * i) * r);*/
            //末端坐标
            float endX = (float) (centerX + Math.sin(angle * i) * radius);
            float endY = (float) (centerY - Math.cos(angle * i) * radius);
            canvas.drawLine(startX, startY, endX, endY, viewPaint);
        }
    }

    //绘制内部  绘制覆盖区域,百分比取连线长度的百分比(如果从中心点开始的连线,则是半径的百分比),
    // 此处用半径radius减去间隔r即连线长度
    public void drawValueShape(Canvas canvas){
        Path path = new Path();
        float r = radius / (count-1);//每层的间距
        for (int i = 0; i <count; i++) {
            if (i == 0) {
                path.moveTo(centerX, (float) (centerY - r - (radius - r) * (data[i]/100)));
            } else {
                float x = (float) (centerX + Math.sin(angle * i) * ((data[i]/100) * (radius - r) + r-3));
                float y = (float) (centerY - Math.cos(angle * i) * ((data[i]/100) * (radius - r) + r-3));
                path.lineTo(x, y);
                //canvas.drawCircle(x,y,1,valuePaint);
            }
        }
        path.close();
        canvas.drawPath(path, valuePaint);
    }

//绘制文字还存在漏洞  
    public void drawTextTitles(Canvas canvas){
        Paint.FontMetrics fontMetrics = textPaint.getFontMetrics();
        float fontHeight = fontMetrics.descent - fontMetrics.ascent;
        for(int i=0;i<count;i++){
            float x = (float) (centerX+(radius+fontHeight/2)*Math.cos(angle*i));
            float y = (float) (centerY+(radius+fontHeight/2)*Math.sin(angle*i));
            float dis = textPaint.measureText(titles[i]);//文本长度
            if (angle * i == 0) {
                //平角 场均击杀 0
                textPaint.setTextAlign(Paint.Align.CENTER);
                canvas.drawText(titles[i], x,y-85, textPaint);
                textPaint.setTextAlign(Paint.Align.LEFT);
            } else if (angle * i > 0 && angle * i < Math.PI / 2) {
                //微调  分均输出  60
                canvas.drawText(titles[i], x+70 ,y-65 , textPaint);
            } else if (angle * i >= Math.PI / 2 && angle * i < Math.PI) {
                canvas.drawText(titles[i], x+75, y+35, textPaint);//生存能力 120
            } else if (angle * i >= Math.PI && angle * i <= Math.PI *1.2) {
                String txt = titles[i];
                Rect bounds = new Rect();
                textPaint.getTextBounds(txt, 0, txt.length(), bounds);
                float width = textPaint.measureText(txt);
                float height = bounds.bottom - bounds.top; //180
                canvas.drawText(txt, x - width * 0.4f, y+95, textPaint);
            } else if (angle * i >= Math.PI && angle * i <= Math.PI *1.5 ) {
                String txt = titles[i];
                float width = textPaint.measureText(txt);//240
                canvas.drawText(txt, x-110, y+75, textPaint);
            }else if(angle * i >= Math.PI && angle * i <= Math.PI *2 ){
                //300
                String txt = titles[i];
                canvas.drawText(txt, x-110, y-25, textPaint);
            }

        }
    }



    //重新设置数据
    public void  setData(double [] data){
        this.data = data;
        invalidate();
    }

    //需要绘制几边形
    public void setCount(int count){
        this.count =count;
        angle = (float) ((Math.PI *2)/count);
        invalidate();
    }
    //设置层数
    public void setLayerCount(int layerCount){
        this.layerCount  = layerCount;
        invalidate();
    }
}

CircleImageView

package com.naman14.timber.widgets;

import android.animation.ObjectAnimator;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Bitmap;
import android.graphics.BitmapShader;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.ColorFilter;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.RectF;
import android.graphics.Shader;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.ColorDrawable;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import androidx.annotation.ColorRes;
import androidx.annotation.DrawableRes;
import androidx.annotation.RequiresApi;
import androidx.core.content.ContextCompat;

import android.os.Build;
import android.util.AttributeSet;
import android.view.animation.LinearInterpolator;
import android.widget.ImageView;

import com.naman14.timber.R;

public class CircleImageView extends ImageView {

    private static final ScaleType SCALE_TYPE = ScaleType.CENTER_CROP;

    private static final Bitmap.Config BITMAP_CONFIG = Bitmap.Config.ARGB_8888;
    private static final int COLORDRAWABLE_DIMENSION = 2;

    private static final int DEFAULT_BORDER_WIDTH = 0;
    private static final int DEFAULT_BORDER_COLOR = Color.BLACK;
    private static final boolean DEFAULT_BORDER_OVERLAY = false;

    private final RectF mDrawableRect = new RectF();
    private final RectF mBorderRect = new RectF();

    private final Matrix mShaderMatrix = new Matrix();
    private final Paint mBitmapPaint = new Paint();
    private final Paint mBorderPaint = new Paint();

    private int mBorderColor = DEFAULT_BORDER_COLOR;
    private int mBorderWidth = DEFAULT_BORDER_WIDTH;

    private Bitmap mBitmap;
    private BitmapShader mBitmapShader;
    private int mBitmapWidth;
    private int mBitmapHeight;

    private float mDrawableRadius;
    private float mBorderRadius;

    private ColorFilter mColorFilter;

    private boolean mReady;
    private boolean mSetupPending;
    private boolean mBorderOverlay;


    private ObjectAnimator mObjectAnimator;
    public static final int STATE_PLAYING =1;//正在播放
    public static final int STATE_PAUSE =2;//暂停
    public static final int STATE_STOP =3;//停止
    public int state;

    public CircleImageView(Context context) {
        super(context);

        init();
    }

    public CircleImageView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public CircleImageView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);

        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleImageView, defStyle, 0);

        mBorderWidth = a.getDimensionPixelSize(R.styleable.CircleImageView_border_width, DEFAULT_BORDER_WIDTH);
        mBorderColor = a.getColor(R.styleable.CircleImageView_border_color, DEFAULT_BORDER_COLOR);
        mBorderOverlay = a.getBoolean(R.styleable.CircleImageView_border_overlay, DEFAULT_BORDER_OVERLAY);

        a.recycle();

        init();
    }

    private void init() {
        super.setScaleType(SCALE_TYPE);
        mReady = true;

        if (mSetupPending) {
            setup();
            mSetupPending = false;
        }

        state = STATE_STOP;
        mObjectAnimator = ObjectAnimator.ofFloat(this, "rotation", 0f, 359f);//添加旋转动画,旋转中心默认为控件中点
        mObjectAnimator.setDuration(80000);//设置动画时间
        mObjectAnimator.setInterpolator(new LinearInterpolator());//动画时间线性渐变
        mObjectAnimator.setRepeatCount(ObjectAnimator.INFINITE);
        mObjectAnimator.setRepeatMode(ObjectAnimator.RESTART);
    }

    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    public void playMusic(){
        if(state == STATE_STOP){
            mObjectAnimator.start();//动画开始
            state = STATE_PLAYING;
        }else if(state == STATE_PAUSE){
            mObjectAnimator.resume();//动画重新开始
            state = STATE_PLAYING;
        }else if(state == STATE_PLAYING){
            mObjectAnimator.pause();//动画暂停
            state = STATE_PAUSE;
        }
    }

    public void stopMusic(){
        mObjectAnimator.end();//动画结束
        state = STATE_STOP;
    }

    @Override
    public ScaleType getScaleType() {
        return SCALE_TYPE;
    }

    @Override
    public void setScaleType(ScaleType scaleType) {
        if (scaleType != SCALE_TYPE) {
            throw new IllegalArgumentException(String.format("ScaleType %s not supported.", scaleType));
        }
    }

    @Override
    public void setAdjustViewBounds(boolean adjustViewBounds) {
        if (adjustViewBounds) {
            throw new IllegalArgumentException("adjustViewBounds not supported.");
        }
    }

    @Override
    protected void onDraw(Canvas canvas) {
        if (getDrawable() == null) {
            return;
        }

        canvas.drawCircle(getWidth() / 2, getHeight() / 2, mDrawableRadius, mBitmapPaint);
        if (mBorderWidth != 0) {
            canvas.drawCircle(getWidth() / 2, getHeight() / 2, mBorderRadius, mBorderPaint);
        }
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        setup();
    }

    public int getBorderColor() {
        return mBorderColor;
    }

    public void setBorderColor(int borderColor) {
        if (borderColor == mBorderColor) {
            return;
        }

        mBorderColor = borderColor;
        mBorderPaint.setColor(mBorderColor);
        invalidate();
    }

    public void setBorderColorResource(@ColorRes int borderColorRes) {
        setBorderColor(ContextCompat.getColor(getContext(), borderColorRes));
    }

    public int getBorderWidth() {
        return mBorderWidth;
    }

    public void setBorderWidth(int borderWidth) {
        if (borderWidth == mBorderWidth) {
            return;
        }

        mBorderWidth = borderWidth;
        setup();
    }

    public boolean isBorderOverlay() {
        return mBorderOverlay;
    }

    public void setBorderOverlay(boolean borderOverlay) {
        if (borderOverlay == mBorderOverlay) {
            return;
        }

        mBorderOverlay = borderOverlay;
        setup();
    }

    @Override
    public void setImageBitmap(Bitmap bm) {
        super.setImageBitmap(bm);
        mBitmap = bm;
        setup();
    }

    @Override
    public void setImageDrawable(Drawable drawable) {
        super.setImageDrawable(drawable);
        mBitmap = getBitmapFromDrawable(drawable);
        setup();
    }

    @Override
    public void setImageResource(@DrawableRes int resId) {
        super.setImageResource(resId);
        mBitmap = getBitmapFromDrawable(getDrawable());
        setup();
    }

    @Override
    public void setImageURI(Uri uri) {
        super.setImageURI(uri);
        mBitmap = getBitmapFromDrawable(getDrawable());
        setup();
    }

    @Override
    public void setColorFilter(ColorFilter cf) {
        if (cf == mColorFilter) {
            return;
        }

        mColorFilter = cf;
        mBitmapPaint.setColorFilter(mColorFilter);
        invalidate();
    }

    private Bitmap getBitmapFromDrawable(Drawable drawable) {
        if (drawable == null) {
            return null;
        }

        if (drawable instanceof BitmapDrawable) {
            return ((BitmapDrawable) drawable).getBitmap();
        }

        try {
            Bitmap bitmap;

            if (drawable instanceof ColorDrawable) {
                bitmap = Bitmap.createBitmap(COLORDRAWABLE_DIMENSION, COLORDRAWABLE_DIMENSION, BITMAP_CONFIG);
            } else {
                bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), BITMAP_CONFIG);
            }

            Canvas canvas = new Canvas(bitmap);
            drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
            drawable.draw(canvas);
            return bitmap;
        } catch (OutOfMemoryError e) {
            return null;
        }
    }

    private void setup() {
        if (!mReady) {
            mSetupPending = true;
            return;
        }

        if (mBitmap == null) {
            return;
        }

        mBitmapShader = new BitmapShader(mBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);

        mBitmapPaint.setAntiAlias(true);
        mBitmapPaint.setShader(mBitmapShader);

        mBorderPaint.setStyle(Paint.Style.STROKE);
        mBorderPaint.setAntiAlias(true);
        mBorderPaint.setColor(mBorderColor);
        mBorderPaint.setStrokeWidth(mBorderWidth);

        mBitmapHeight = mBitmap.getHeight();
        mBitmapWidth = mBitmap.getWidth();

        mBorderRect.set(0, 0, getWidth(), getHeight());
        mBorderRadius = Math.min((mBorderRect.height() - mBorderWidth) / 2, (mBorderRect.width() - mBorderWidth) / 2);

        mDrawableRect.set(mBorderRect);
        if (!mBorderOverlay) {
            mDrawableRect.inset(mBorderWidth, mBorderWidth);
        }
        mDrawableRadius = Math.min(mDrawableRect.height() / 2, mDrawableRect.width() / 2);

        updateShaderMatrix();
        invalidate();
    }

    private void updateShaderMatrix() {
        float scale;
        float dx = 0;
        float dy = 0;

        mShaderMatrix.set(null);

        if (mBitmapWidth * mDrawableRect.height() > mDrawableRect.width() * mBitmapHeight) {
            scale = mDrawableRect.height() / (float) mBitmapHeight;
            dx = (mDrawableRect.width() - mBitmapWidth * scale) * 0.5f;
        } else {
            scale = mDrawableRect.width() / (float) mBitmapWidth;
            dy = (mDrawableRect.height() - mBitmapHeight * scale) * 0.5f;
        }

        mShaderMatrix.setScale(scale, scale);
        mShaderMatrix.postTranslate((int) (dx + 0.5f) + mDrawableRect.left, (int) (dy + 0.5f) + mDrawableRect.top);

        mBitmapShader.setLocalMatrix(mShaderMatrix);
    }

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值