快过年了,提前祝大家新年快乐哦。马上就可以闲下来休息几天了,好幸福啊。。。不多说了,最近项目中有个地方要用柱状图显示客户的消费情况,本着一名程序员的优秀品质,上gitbub上连续搜罗了有关画图的,不得不说,真心蛮多的,不仅仅功能多,而且很多都已经高度封装,很好用了。但是唯一的缺点就是它们有些功能我们产品用不上,就为了显示一个小柱状图就compile一个别人的项目,有点得不偿失啊。所以我就自己撸了一个,呵呵,撸的蛮简单的。。

如图:

MPAndroidChart 柱状图绘制多条柱子_柱状图

MPAndroidChart 柱状图绘制多条柱子_柱状图_02

MPAndroidChart 柱状图绘制多条柱子_控件_03

本着一个菜鸟的气质,也非常羡慕那些会自定义View的人,先不说封装的非常优秀,让人用起来非常顺手;看自定义代码时,各种自定义的属性让人忧郁,我就是这种人,每次下载别人看上去很好的代码,唉,自定义的属性真实太多了,本想顺着他们的思路学习下去,但是很多时候就是看这自定义的高度,宽度,长度,颜色,type值,字体大小值,padding值,margin值…….我就害怕得崩溃了。所以我就想,能不能有些简单的方法让自己能理清思路呢。作为一只有想法的菜鸟,我想就这个柱状图说说自己的看法。

我想做一个怎么样的控件

在android中写个自定义控件并不是为了好玩[我是这么理解的],大多数自定义控件的诞生,都是与我们的业务逻辑相关的,然后通过代码抽象,提取了一个符合多种场合的东西。所以,我们的控件,不管是找的,还是自己写的,到底能不能实现自己业务中的功能。说到这里,简单列举我自己项目中柱状图要的功能:
1.显示每个季度/一年中 用户的消费情况
2.科技点击每个季度/一年中任意一个月,可以看到备份比/数值
3.可以自定义颜色

由于不是电商,这个柱状图的很多功能是弱化的,所以我去下载guahub上千star的项目的确有些浪费啊。明确的需求,那就开始写吧。

开始

这就是说自己的想法了,我先特么不管onMeasure(),onSizeChanged(),我先画个草图再说,这对我onDraw觉得是有帮助的:

MPAndroidChart 柱状图绘制多条柱子_柱状图_04

恩,那么我就先把width,height钉死吧,恩,我就定死你,我先把最重要的写完再写你。

那好啊,我就先画坐标吧。是的,画坐标,画之前,先写死一下:

MPAndroidChart 柱状图绘制多条柱子_控件_05

画坐标的代码我也就不说了,也就一个Paint.drawLine(),你传入一个起始坐标与结束坐标就行,应该来说是简单的了啊。

坐标画完了,那么我就来画我们的柱状图了,这个我想应该也是简单的,就是一个Paint.drawRoundRect(),我想难的还是那个传入的坐标了,

MPAndroidChart 柱状图绘制多条柱子_自定义_06

这里我们假设所有的柱状图的宽度是一致的,为mItemWidth , 另外还有一个就是两个柱状图之间的距离值,我们也假设他们之间的距离也是一样的,那么也可以设置mItemBetWidth;可以看出结论如下:
n * mItemWidth + (n+1) * mItemBetWidth = View的宽度 - View.paddingLeft - View.paddingRight ;这里的n就是我们设置的要显示的柱状图的个数,唉,先不管了,设置n=12吧。这个mItemWidth与mItemBetWdith怎么办,还能咋办,先让他们相等嘛,如果测试妹妹说不行,我们改还不行吗?
那么我们得到的结论就是 mItemWidth = (View.宽度 - View.paddingLeft - View.paddingRight ) / (2 * n + 1) ;
好了,宽度确定了,高度呢?? 先设置假的啊,用random,呵呵,恩。

说到这里,相信大家也已经画个大致的样子了吧:

MPAndroidChart 柱状图绘制多条柱子_控件_07

细节

写到这里,自定义View已经完成了很多一部分,为了能让我们的View拿得出手,需要定义一些细节,哪些细节呢?举个例子啊:

1.坐标系的Paint,怎么定义?颜色/字体大小/strokeWidth
2.具状图的Paint, 怎么定义? 颜色[普通颜色/渐变颜色]/字体大小/strokeWidth/显示方式[百分比还是数值]
3.坐标系X轴该显示什么?有没有Paint,显示文字,设置前缀/后缀? 颜色/大小/strokeWidth/位置摆放?

别急,你一个一个来,就像这样:

mTextPaint = new Paint();
        mTextPaint.setStyle(Paint.Style.FILL);
        mTextPaint.setAntiAlias(true);
        mTextPaint.setColor(mTextColor);
        mTextPaint.setTextSize(mTextSize);
        Paint.FontMetrics metrics = mTextPaint.getFontMetrics();
        mTextBottomHeight = metrics.ascent + metrics.descent;

        mChartPaint = new Paint();
        mChartPaint.setStyle(Paint.Style.FILL);
        mChartPaint.setAntiAlias(true);
        mChartPaint.setColor(mStartChartViewColor);

        mTextTopPaint = new Paint();
        mTextTopPaint.setStyle(Paint.Style.FILL);
        mTextTopPaint.setAntiAlias(true);
        mTextTopPaint.setTextSize(mTopTextSize);
        mTextTopPaint.setColor(mTopTextColor);

        mCoordinatePaint = new Paint();
        mCoordinatePaint.setColor(Color.RED);
        mCoordinatePaint.setAntiAlias(true);
        mCoordinatePaint.setStrokeWidth(4);

先标上自己喜欢的颜色,然后再去细致的改吧。当然这是刚刚开始,view的onDraw方法我感觉就像是画家在画画,慢慢去调节。当然啊前提是我们得学习一个常用的技巧,想Paint.draw到底可以draw什么,这个我们得自己知道啊。

微调了坐标系,柱状图,现在还剩下点击颜色发生变化了吧,好吧,那就是OnTouch事件了:

MPAndroidChart 柱状图绘制多条柱子_柱状图_08

那主要是判断我们的MotionEvent是否落在了这个柱状图里面了,如果是那么就修改这个柱状图的Paint中的颜色了,就是这样:

private boolean contains(float x, float y) {
        for (int i = 1; i <= mChartItemCount; i++) {
            int tempValue = mValues.get(i - 1);

            RectF rectF = new RectF();
            rectF.top = mTextLocationHeight + mChartPaddingTop - tempValue * mHeight * 1.0f / (mMaxValue - mMinValue);
            rectF.left = (i - 1) * mBetweenWidth + (i - 1) * mChartWidth;
            rectF.right = i * (mBetweenWidth + mChartWidth) + mChartWidth;
            rectF.bottom = mTextLocationHeight + mChartPaddingTop;

            if (rectF.contains(x, y)) {
                mCurrentIndex = i - 1;
                return true;
            }
        }
        return false;
    }

该变画笔的颜色:

mTextPaint.setColor(mCurrentIndex == i - 1 ? mChooseColor : mTopTextColor);
 canvas.drawText(result, rectF.left + mChartWidth / 2 - width / 2, rectF.top - 10, mTextTopPaint);

好了,基本上就是这样了,还是蛮简单的啊,下面是我定义的属性,也是一点一点加上去的:

<declare-styleable name="CustomChartView">

        <!--底部文字颜色 -->
        <attr name="text_color" format="color"/>
        <!--底部文字大小 -->
        <attr name="text_size" format="dimension"/>
        <!--底部文字后缀 当然你可以定义前缀-->
        <attr name="text_suffix" format="string"/>

        <!--chartView开始显示的颜色-->
        <attr name="chart_start_color" format="color"/>
        <!--chartView结束显示的颜色-->
        <attr name="chart_end_color" format="color"/>
        <!--chartView选中的颜色-->
        <attr name="chart_choose_color" format="color"/>
        <!--chartView的圆角大小-->
        <attr name="chart_view_circle_radius" format="integer"/>

        <!--顶部文字的颜色-->
        <attr name="text_top_color" format="color"/>
        <!--顶部文字大小-->
        <attr name="text_top_size" format="dimension"/>
        <!--顶部文字显示方式 百分比还是数值-->
        <attr name="text_top_type" format="enum">
            <enum name="percent" value="1"/>
            <enum name="value" value="2"/>
        </attr>

        <!--显示chartView的个数-->
        <attr name="chart_item_count" format="integer"/>
        <!--最大值-->
        <attr name="chart_max_value" format="integer"/>
        <!--最小值-->
        <attr name="chart_min_value" format="integer"/>

        <!--chartView显示图与View顶部的距离-->
        <attr name="chart_padding_top" format="dimension"/>
        <!--chartView显示图与View底部的距离-->
        <attr name="chart_padding_bottom" format="dimension"/>

    </declare-styleable>

基本上都有注释,我想也还算简单的,希望大家喜欢。
代码

好了,上海的下午,太阳落山了还是很冷的,我要去买菜做饭了。。。