4.坐标轴的绘制(MPAndroidChart源码解析)

1.绘制坐标轴的思路:

1.计算x,y的起始值和结束值
2.根据显示的值范围将坐标轴显示的数据进行分组,获取间隔值,并确定每个间隔要显示的value
3.计算坐标轴label的需要的绘制空间和位置,位置就是将上面的value转换为px值

2.涉及的主要类

包含配置类和render类,如下图所示:


相关类.png

2.1 配置类:

//只看主要的成员变量和方法
public abstract class AxisBase extends ComponentBase {

    /**
     * label显示值的格式化类
     */
    protected IAxisValueFormatter mAxisValueFormatter;
    /**
     * 是否自定义mAxisMin,比如y轴的最小值大于0,我们可以将y轴的最小值设置为0
     */
    protected boolean mCustomAxisMin = false;
    /**
     * 所要绘制的元素集合
     */
    public float[] mEntries = new float[]{};
    /**
     * 坐标轴绘制的元素数量
     */
    public int mEntryCount;
    /**
     * 这条轴上绘制label的数量,比如y轴显示了20条数据,如果每一条都显示label的话可能会造成重叠,
     * 所以这个参数可以设置可显示的最大的label绘制个数
     */
    private int mLabelCount = 6;

    /**
     * 绘制的线是以每个item的中心点为起点的
     */
    protected boolean mDrawGridLines = true;

    /**
     * 是否绘制坐标轴线
     */
    protected boolean mDrawAxisLine = true;
    /**
     * 是否要绘制label
     */
    protected boolean mDrawLabels = true;

 /**
     * 获取所有label中最长的,使用场景在于,当label设置了旋转角度,绘制chart的区域就要根据角度和这个长度来计算,感觉这种设定并不好,会造成chart的区域大小不一,不如添加设置label显示的最大最小长度,否则名称过长了会挤压content的绘制区域
     *注意:这个方法只是返回了最大字符数的label,中英文的话就不准确了
     */
    public String getLongestLabel() {
        String longest = "";
        for (int i = 0; i < mEntries.length; i++) {
            String text = getFormattedLabel(i);
            if (text != null && longest.length() < text.length())
                longest = text;
        }
        return longest;
    }

    public String getFormattedLabel(int index) {

        if (index < 0 || index >= mEntries.length)
            return "";
        else
            return getValueFormatter().getFormattedValue(mEntries[index], this);
    }
}

2.2 绘制类

public abstract class AxisRenderer extends Renderer {

    protected AxisBase mAxis;
    protected Transformer mTrans;
    ...
    public AxisRenderer(ViewPortHandler viewPortHandler, AxisBase xAxis, Transformer transformer) {
        super(viewPortHandler);
        this.mAxis = xAxis;
        this.mTrans = transformer;
    }
    
    //绘制文本
    public abstract void renderAxisLabels(Canvas c);
    //绘制分组线
    public abstract void renderGridLines(Canvas c);
    //绘制坐标轴线
    public abstract void renderAxisLine(Canvas c);
    //绘制限制线
    public abstract void renderLimitLines(Canvas c);
   
    /**
     * 计算坐标轴的值
     * @param min
     * @param max
     * @param invert
     */
    public void computeAxis(float min, float max, boolean invert) {
        //这里只判断y的缩放是因为XAxisRenderer会对这个方法进行覆写
        //如果y轴的缩放比例不是最小,就重新计算最大最小值
        if (mViewPortHandler != null && mViewPortHandler.contentWidth() > 10 && !mViewPortHandler.isFullyZoomedOutY()) {
            //y轴上获得当前显示top点和bottom点,然后获取这两个坐标点(px值)所代表的实际的value值
            MPPointD p1 = mTrans.getValuesByTouchPoint(mViewPortHandler.contentLeft(), mViewPortHandler.contentTop());
            MPPointD p2 = mTrans.getValuesByTouchPoint(mViewPortHandler.contentLeft(), mViewPortHandler.contentBottom());

            if(!invert){
                min = (float)p2.y;
                max = (float)p1.y;
            }else{
                min = (float)p1.y;
                max = (float)p2.y;
            }

            MPPointD.recycleInstance(p1);
            MPPointD.recycleInstance(p2);
        }
        //计算索要绘制的标签
        computeAxisValues(min, max);
    }

/**
 * 计算最大值和最小值之间所需的标签数量
 * */
    protected void computeAxisValues(float min,float max){

         float yMin = min;
         float yMax = max;

         int labelCount = mAxis.getLabelCount();
        //1.计算所要绘制的数据区间
         double range = Math.abs(yMax - yMin);
        ...
        //2.根据设置的显示label的个数和数值区间初步获得 间隔值
         double rawInterval = range/labelCount;
         //向下取整,间隔当然得是整数
         double interval = Utils.roundToNextSignificant(rawInterval);
        if (mAxis.isGranularityEnabled())
            interval = interval < mAxis.getGranularity() ? mAxis.getGranularity() : interval;
        //一个间隔值中内部的间隔值,比如groupbar这种情况
        double intervalMagnitude = Utils.roundToNextSignificant(Math.pow(10,(int)Math.log10(interval)));
    ...
        //3.根据间隔值,起始值和终点值计算出具体需要好绘制的value,比如x轴的mEntries 包含的是 2,3,4 表示要绘制的是2,3,4的值
        int n = mAxis.isCenterAxisLabelsEnabled() ? 1 : 0;
    ...

        double first = interval == 0.0 ? 0.0 : Math.ceil(yMin / interval) * interval;
        if(mAxis.isCenterAxisLabelsEnabled()) {
            first -= interval;
        }

        double last = interval == 0.0 ? 0.0 : Utils.nextUp(Math.floor(yMax / interval) * interval);

        double f;
        int i;

        if (interval != 0.0) {
            for (f = first; f <= last; f += interval) {
                ++n;
            }
        }

        mAxis.mEntryCount = n;
        if (mAxis.mEntries.length < n) {
            mAxis.mEntries = new float[n];
        }

        for(f = first, i = 0; i < n; f += interval, ++i) {
            if (f == 0.0) 
                f = 0.0;
            mAxis.mEntries[i] = (float) f;
        }
        ...
    }
}

从上面的代码我们获得的mEntries只是原始数据的value,绘制的时候需要将其转换成px值,转换的过程就是根据前面介绍的Transformer,以y轴绘制labels为例:

@Override
    public void renderAxisLabels(Canvas c) {

        if (!mYAxis.isEnabled() || !mYAxis.isDrawLabelsEnabled())
            return;
        //1.获取position,转换在这个方法里
        float[] positions = getTransformedPositions();
        //2.设置paint属性
        mAxisLabelPaint.setTextSize(mYAxis.getTextSize());
        mAxisLabelPaint.setColor(mYAxis.getTextColor());
        //3.获取xOffset yOffset,计算出xPos(label的x起始绘制坐标)
        float xoffset = mYAxis.getXOffset();
        float yoffset = mYAxis.getYOffset() +Utils.calcTextHeight(mAxisLabelPaint, "A") / 2.5f;
        float xPos = 0f;
        mAxisLabelPaint.setTextAlign(Paint.Align.RIGHT);
        xPos =  mViewPortHandler.offsetLeft() - xoffset;
        //4.绘制labels

        drawYLabels(c,xPos,positions,yoffset);
    }

 protected float[] getTransformedPositions() {
        //使用buffer时因为不建议在onDraw方法中频繁的创建对象,*2 是每个点都是由x,y组成
        if (mGetTransformedPositionsBuffer.length != mYAxis.mEntryCount * 2) {
            mGetTransformedPositionsBuffer = new float[mYAxis.mEntryCount * 2];
        }
        float[] positions = mGetTransformedPositionsBuffer;
        for (int i = 0; i < positions.length; i += 2) {
            positions[i + 1] = mYAxis.mEntries[i / 2];
        }
        //映射
        mTrans.pointValuesToPixel(positions);
        return positions;
    }

我们再看一下pointValuesToPixel这个方法

/**
注意矩阵的顺序是:value-touch-offset,原因已经在前面的工具类中讲过了
**/
public void pointValuesToPixel(float[] pts) {

        mMatrixValueToPx.mapPoints(pts);
        mViewPortHandler.getMatrixTouch().mapPoints(pts);
        long s = System.currentTimeMillis();
        mMatrixOffset.mapPoints(pts);
        Log.e("mapPoints","count: "+ pts.length+",mapPoints time:" +(System.currentTimeMillis() - s));
    }

3.整个绘制流程

以BarChart为例简述坐标轴的整个绘制流程

3.1 初始化

在Chart 的 init方法中会初始化 Axis,Transform,Renderer,ViewPortHandler的实例

3.2 在notifyDataSetChanged方法中计算最大最小值,偏移量等

 @Override
    public void notifyDataSetChanged() {
        if (mData == null) {
            return;
        }
        if (mRenderer != null)
            mRenderer.initBuffers();
        //计算最大最小值
        calcMinMax();
        //计算坐标轴的labels
        mAxisRendererLeft.computeAxis(mAxisLeft.mAxisMinimum, mAxisLeft.mAxisMaximum, false);
        mXAxisRenderer.computeAxis(mXAxis.mAxisMinimum, mXAxis.mAxisMaximum, false);
        //计算偏移量
        calculateOffsets();
    }

computeAxis 方法上面已经介绍过了,下面看一下其他的两个方法

    protected void calcMinMax() {
        mXAxis.calculate(mData.getXMin(), mData.getXMax());
        // calculate axis range (min / max) according to provided data
        mAxisLeft.calculate(mData.getYMin(), mData.getYMax());
        //mAxisRight.calculate(mData.getYMin(), mData.getYMax());
        );
    }
//计算的content也就是data的offsets
public void calculateOffsets() {
    //1.开始都为0
        float offsetLeft = 0f, offsetRight = 0f, offsetTop = 0f, offsetBottom = 0f;
    //2.bottomOffset加上x轴的在y方向上的offset和labelHeight,y轴也类似,不过加的是labelWidth
        if (mXAxis.isEnabled() && mXAxis.isDrawLabelsEnabled()) {
            float xLabelHeight = mXAxis.mLabelRotatedHeight + mXAxis.getYOffset();
            offsetBottom += xLabelHeight;
        }

        if (mAxisLeft.isEnabled() && mAxisLeft.isDrawLabelsEnabled()) {
            float longestYLabelWidth = mAxisLeft.getRequiredWidthSpace(mAxisRendererLeft.getPaintAxisLabels());
            offsetLeft += longestYLabelWidth;
        }
//3.加上chart设置的offsets
        offsetTop += getExtraTopOffset();
        offsetRight += getExtraRightOffset();
        offsetBottom += getExtraBottomOffset();
        offsetLeft += getExtraLeftOffset();

        float minOffset = Utils.convertDpToPixel(mMinOffset);
//4.重置ViewPortHandler的可绘制区域
        mViewPortHandler.restrainViewPort(
                Math.max(minOffset, offsetLeft),
                Math.max(minOffset, offsetTop),
                Math.max(minOffset, offsetRight),
                Math.max(minOffset, offsetBottom));
//5.初始化Transformer得offsetMartrix和ValuePxMatrix,第一个matrix记录offsets,第二个matrix记录
        prepareOffsetMatrix();
        prepareValuePxMatrix();
    }

prepareOffsetMatrix 和prepareValuePxMatrix还是比较重要的,如下可以看到依次调用了Transformer的对应的方法,这两个方法已经在前面介绍Util的时候介绍过了

protected void prepareOffsetMatrix() {
        mLeftAxisTransformer.prepareMatrixOffset(false);
    }

 protected void prepareValuePxMatrix() {

        mLeftAxisTransformer.prepareMatrixValuePx(mXAxis.mAxisMinimum,
                mXAxis.mAxisRange,
                mAxisLeft.mAxisRange,
                mAxisLeft.mAxisMinimum);
    }

3.3 在onDraw方法中开始绘制坐标轴

 protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
     
        if (mData == null) return;
        long startTime = System.currentTimeMillis();
        
      //1.绘制X轴
        if (mXAxis.isEnabled()) {
            mXAxisRenderer.computeAxis(mXAxis.mAxisMinimum, mXAxis.mAxisMaximum, false);
        }
        
        mXAxisRenderer.renderAxisLine(canvas);
        mXAxisRenderer.renderGridLines(canvas);
        //绘制Y轴
        if (mAxisLeft.isEnabled()) {
            mAxisRendererLeft.computeAxis(mAxisLeft.mAxisMinimum, mAxisLeft.mAxisMaximum, false);
        }

        mAxisRendererLeft.renderAxisLine(canvas);
        mAxisRendererLeft.renderGridLines(canvas);
      ...
       //绘制X,Y的labels
        mXAxisRenderer.renderAxisLabels(canvas);
        mAxisRendererLeft.renderAxisLabels(canvas);
    }

可以看到,onDraw方法中每次都调用了Aixs的computeAxis方法重新计算了坐标轴,这是因为动画和手势操作后需要重新计算数据。

4.总结:

通过坐标轴的绘制流程,可以清晰的看出这个开源库的思路:


image.png

就是先计算出原始数据的坐标,然后利用三个矩阵映射成屏幕上的像素坐标,三个矩阵的作用重新总结一下:

1.mMatrixValueTopx:根据原始数据的最大最小值和屏幕上可绘制区域的rect计算出缩放比例,平移距离等,这样可以轻松的将一组原始数据经过缩放和平移就变成了屏幕上的数据。

2.mMatrixTouch:保存了手势操作造成的影响:缩放比例和拖拽距离。

3.mMatrixOffset:保存了根据设置的offsets以及绘制坐标轴所需要的空间计算出来的content区域的offsets。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容