android自定义View-折线图

拿到效果图,手机端需要实现的效果是这样:



先分析:x轴是时间轴,长度固定,区间不固定;y轴是数值轴,区间不固定,需要根据传入数组的max和min定义区间;

先放撸出来后的基本模样:


还是首先确定哪些元素需要在xml中定义,哪些需要代码中动态set。

这一题,我觉得xml中可设置的不用太多,连颜色都可以在view里写死,重要的就是java代码中设置数据源了:
这四项就是数据源了,valueName是图的名字,X,Y,Z分别是三个图的数据源。

    private int[] valuesZ;
    private int[] valuesY;
    private int[] valuesX;
    private String valueName;

然后onMeasure(),因为这个图的目标是做match_parent的,偷懒抛异常

        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);

        if (widthMode == MeasureSpec.EXACTLY) {
            mWidth = widthSize;
        } else if (widthMode == MeasureSpec.AT_MOST) {
            throw new IllegalArgumentException("width must be EXACTLY,you should set like android:width=\"200dp\"");
        }

        if (heightMode == MeasureSpec.EXACTLY) {
            mHeight = heightSize;
        } else if (widthMeasureSpec == MeasureSpec.AT_MOST) {

            throw new IllegalArgumentException("height must be EXACTLY,you should set like android:height=\"200dp\"");
        }

        setMeasuredDimension(mWidth, mHeight);

下面就是最重要的onDraw()了:

从结果图可以看出来,我们是竖着画的,所以 绘制时,画布的xy轴和我们要展现的xy轴需要区分开来。
那么,三个图 其实基本绘制方式一样,只不过是X图和Y图绘制好之后,需要偏移画布(当然,我没有使用偏移画布,用的比较复杂的方法,后来看的也是一头闷水...)

先来矩形:

        Rect rectX = new Rect();
        rectX.left = 150 + (getWidth() - getPaddingRight() - 50) / 3 * 2;//这里是最右边的图
        rectX.top = mToppadding + getPaddingTop();
        rectX.right = (getWidth() - getPaddingRight() - 50);
        rectX.bottom = getHeight() - getPaddingBottom() - 50;
        drawIng(canvas, "X", valuesX, rectX);//具体绘制图形的方法,提取成一个方法,便于复用

        Rect rectY = new Rect();
        rectY.left = 150 + (getWidth() - getPaddingRight() - 50) / 3;//中间的图
        rectY.top = mToppadding + getPaddingTop();
        rectY.right = (getWidth() - getPaddingRight() - 50) / 3 * 2;
        rectY.bottom = getHeight() - getPaddingBottom() - 50;
        drawIng(canvas, "Y", valuesY, rectY);

        Rect rectZ = new Rect();
        rectZ.left = 150 + getPaddingRight();//初始的图,Z
        rectZ.top = mToppadding + getPaddingTop();
        rectZ.right = (getWidth() - getPaddingRight() - 50) / 3;
        rectZ.bottom = getHeight() - getPaddingBottom() - 50;
        drawIng(canvas, "Z", valuesZ, rectZ);

这里就是画布的X轴向右加,也算是一种偏移吧哈哈。

        mPaint.setColor(mGridLineColor);
        mPaint.setStyle(Paint.Style.STROKE);
        mPaint.setStrokeWidth(3);
        canvas.drawRect(r, mPaint);
        int rWidth = r.right - r.left;
        int rHeight = r.bottom - r.top;

        int aW = rWidth / 9 * 2;//折线图的y轴,矩形边分为四块,我具体细分为18分,每块占4分,格线到矩形边缘距离1分

        int aH = rHeight / 11;//矩形边分为11分,时间轴部分占10分,细分10部分,前后padding共1分
        int mgH = aH / 2 + mToppadding;//画布原点到时间轴起始位置的距离

        //根据values的值,找出最大值和最小值,确定y轴范围
        for (int i = 0; i < 5; i++) {
            canvas.drawLine(i * aW + r.left + aW / 4, mToppadding - 30, i * aW + r.left + aW / 4, mToppadding, mPaint);
        }
        //时间轴的黑点
        mPaint.setStrokeWidth(20);
        for (int i = 0; i < 11; i++) {
            canvas.drawLine(r.left - 20, mgH + i * aH, r.left, mgH + i * aH, mPaint);
            if (i == 10) continue;
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 , r.left, mgH + i * aH + aH / 7 , mPaint);
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 * 2, r.left, mgH + i * aH + aH / 7 * 2, mPaint);
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 * 3, r.left, mgH + i * aH + aH / 7 * 3, mPaint);
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 * 4, r.left, mgH + i * aH + aH / 7 * 4, mPaint);
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 * 5, r.left, mgH + i * aH + aH / 7 * 5, mPaint);
            canvas.drawLine(r.left - 20, mgH + i * aH + aH / 7 * 6, r.left, mgH + i * aH + aH / 7 * 6, mPaint);
        }
        mPaint.setColor(mGridLineColor);
        mPaint.setTextSize(50);
        mPaint.setStrokeWidth(1);
        mPaint.setStyle(Paint.Style.FILL);

        int[] vL = getMinMaxValue(values);//将传入的数组的最大值最小值中间值提取出来
        String vMin = String.valueOf(vL[0]);
        String vMid = String.valueOf(vL[1]);
        String vMax = String.valueOf(vL[2]);
        String title = valueName + "_" + type;
        canvas.drawText(title, r.left + (r.right - r.left) / 2 - mPaint.measureText(title) / 2, 50, mPaint);
        //画数值轴
        float y = mToppadding - 30 - Math.abs(mPaint.getFontMetrics().bottom - mPaint.getFontMetrics().top) / 2;
        canvas.drawText(vMin, r.left + aW / 4 - mPaint.measureText(vMin) / 2, y, mPaint);
        canvas.drawText(vMid, r.left + aW / 4 + aW * 2 - mPaint.measureText(vMid) / 2, y, mPaint);
        canvas.drawText(vMax, r.left + aW / 4 + aW * 4 - mPaint.measureText(vMax) / 2, y, mPaint);
        //画时间轴
        canvas.rotate(90, 0, 0);//画布旋转90°,渲染时间轴的文字
        for (int i = 0; i < 11; i++) {
            String time = DateUtils.getDateToString(times[times.length / 11 * i] * 1000);
            time = time.substring(time.length() - 5, time.length());
            canvas.drawText(time, mgH + i * aH - mPaint.measureText(time) / 2, +100 - r.left, mPaint);
        }
        canvas.rotate(-90, 0, 0);//别忘了转回来
        //画点和线
        //现在的画布y轴是时间,有规律,递增的  x轴是数值,
        mPaint.setColor(mLineColor);
        mPaint.setStyle(Paint.Style.STROKE);
        mPaint.setStrokeWidth(2);
        //先求出x/y轴上每一个单位占多少dp
        float xx = aW * 4.0f / (vL[2] - vL[0]);
        float yy = aH * 10.0f / times.length;

        Path path = new Path();
        for (int i = 0; i < values.length; i++) {
            //画path
            if (i == 0) {
                path.moveTo((values[i] - vL[0]) * xx + r.left + aW / 4, mgH + yy * i);
            } else {
                path.lineTo((values[i] - vL[0]) * xx + r.left + aW / 4, mgH + yy * i);
            }
        }
        canvas.drawPath(path, mPaint);

下面是提取一个数组的最小值、最大值、中间值的方法

private int[] getMinMaxValue(int[] values) {
        int min = values[0], max = values[0];
        for (int i = 0; i < values.length - 1; i++) {

            if (values[i] > max) max = values[i];
            if (values[i] < min) min = values[i];
        }
        max = Math.round(max * 1.0f / 100.0f) * 100;
        min = Math.round(min * 1.0f / 100.0f) * 100;
        if (max + min == 0)
            return new int[]{min, 0, max};
        else {
            if ((max + min) % 200 == 0)
                return new int[]{min, (min + max) / 2, max};
            else return new int[]{min, (min + max + 100) / 2, max + 100};
        }

好了,然后在xml中调用这个view:

<com.ec.vone.view.LinechartView
        android:id="@+id/linechartView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        custom:background="@color/color_bg"
        custom:gridLineColor="#000"
        custom:lineColor="@color/colorAccent" />

activity中:

        long now = (int) (System.currentTimeMillis() /1000);
        for (int i = 0; i < 300; i++) {
            valueZ[i] = (int) (Math.random() * 6000 - 3000);
            valueX[i] = (int) (Math.random() * 1000) - 500;
            valueY[i] = (int) (Math.random() * -200) - 200;
            times[i] = now;
            now++;
        }
        linechartView.setTimes(times);
        linechartView.setValuesZ(valueZ);
        linechartView.setValuesY(valueY);
        linechartView.setValuesX(valueX);
        linechartView.setValueName("Mag");

即可。

模拟的值看起来很难看,当然运用到项目中,就是和原图差不多的效果了。

当然,如果做成实时的折线,像心率一样,就是要scrollby了,今天的结果不是这个,具体后面有机会再填坑。

具体代码点击查看

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

推荐阅读更多精彩内容