AndroidPlot自定义关节

时间:2014-06-18 11:08:19

标签: android androidplot

我目前试图使我的情节看起来像这样:

desired

但我不知道如何自定义点在橙色填充周围有黑色笔划,所以我的点现在“粘在”线条上。

actual

或者,至少如何使它看起来像这样(与线相同颜色的外圈)。

androplot

有任何帮助吗?

1 个答案:

答案 0 :(得分:5)

我相信底部图片使用自定义LineAndPointRenderer,这也是您需要用来重现最顶层图像的内容。

这是一个快速而又肮脏的例子,说明如何做到这一点。首先创建一个自定义Formatter,它将保存所需的新格式值:

    /**
     * A LineAndPointFormatter with the addition of paint to be used to "stroke" vertices.
     */
    class MyLineAndPointFormatter extends LineAndPointFormatter{
        private Paint strokePaint;

        /**
         * Some quick and dirty hard-coded params
         */
        public MyLineAndPointFormatter() {
            super(Color.RED, Color.RED, null, null);
            strokePaint = new Paint();
            strokePaint.setColor(Color.BLACK);
            strokePaint.setStrokeWidth(PixelUtils.dpToPix(2));
            strokePaint.setStyle(Paint.Style.STROKE);
            strokePaint.setAntiAlias(true);
        }

        public Paint getStrokePaint() {
            return strokePaint;
        }

        @Override
        public Class<? extends SeriesRenderer> getRendererClass() {
            return MyLineAndPointRenderer.class;
        }

        @Override
        public SeriesRenderer getRendererInstance(XYPlot plot) {
            return new MyLineAndPointRenderer(plot);
        }
    }

接下来,自定义渲染器:

    /**
     * A LineAndPointRenderer that can stroke vertices.
     */
    class MyLineAndPointRenderer extends LineAndPointRenderer<MyLineAndPointFormatter> {

        public MyLineAndPointRenderer(XYPlot plot) {
            super(plot);
        }

        /**
         * Overridden draw method to get the "vertex stroke" effect.  99% of this is copy/pasted from
         * the super class' implementation. 
         * @param canvas
         * @param plotArea
         * @param series
         * @param formatter
         */
        @Override
        protected void drawSeries(Canvas canvas, RectF plotArea, XYSeries series, LineAndPointFormatter formatter) {
            PointF thisPoint;
            PointF lastPoint = null;
            PointF firstPoint = null;
            Paint  linePaint = formatter.getLinePaint();

            Path path = null;
            ArrayList<Pair<PointF, Integer>> points = new ArrayList<Pair<PointF, Integer>>(series.size());
            for (int i = 0; i < series.size(); i++) {
                Number y = series.getY(i);
                Number x = series.getX(i);

                if (y != null && x != null) {
                    thisPoint = ValPixConverter.valToPix(
                            x,
                            y,
                            plotArea,
                            getPlot().getCalculatedMinX(),
                            getPlot().getCalculatedMaxX(),
                            getPlot().getCalculatedMinY(),
                            getPlot().getCalculatedMaxY());
                    points.add(new Pair<PointF, Integer>(thisPoint, i));
                } else {
                    thisPoint = null;
                }

                if(linePaint != null && thisPoint != null) {

                    // record the first point of the new Path
                    if(firstPoint == null) {
                        path = new Path();
                        firstPoint = thisPoint;
                        // create our first point at the bottom/x position so filling
                        // will look good
                        path.moveTo(firstPoint.x, firstPoint.y);
                    }

                    if(lastPoint != null) {
                        appendToPath(path, thisPoint, lastPoint);
                    }

                    lastPoint = thisPoint;
                } else {
                    if(lastPoint != null) {
                        renderPath(canvas, plotArea, path, firstPoint, lastPoint, formatter);
                    }
                    firstPoint = null;
                    lastPoint = null;
                }
            }
            if(linePaint != null && firstPoint != null) {
                renderPath(canvas, plotArea, path, firstPoint, lastPoint, formatter);
            }

            Paint vertexPaint = formatter.getVertexPaint();
            Paint strokePaint = ((MyLineAndPointFormatter)formatter).getStrokePaint();
            PointLabelFormatter plf = formatter.getPointLabelFormatter();
            if (vertexPaint != null || plf != null) {
                for (Pair<PointF, Integer> p : points) {
                    PointLabeler pointLabeler = formatter.getPointLabeler();

                    // if vertexPaint is available, draw vertex:
                    if(vertexPaint != null) {
                        canvas.drawPoint(p.first.x, p.first.y, vertexPaint);
                    }

                    // if stroke is available, draw stroke:
                    if(strokePaint != null) {
                        // you'll probably want to make the radius a configurable parameter
                        // instead of hard-coded like it is here.
                        canvas.drawCircle(p.first.x, p.first.y, 4, strokePaint);
                    }

                    // if textPaint and pointLabeler are available, draw point's text label:
                    if(plf != null && pointLabeler != null) {
                        canvas.drawText(pointLabeler.getLabel(series, p.second), p.first.x + plf.hOffset, p.first.y + plf.vOffset, plf.getTextPaint());
                    }
                }
            }
        }
    }

最后,在您的活动中使用这些新作品:

MyLineAndPointFormatter format = new MyLineAndPointFormatter();
plot.addSeries(series, format);

以下是与SimpleXYPlot示例一起使用时的样子:

enter image description here

通过加粗线条,选择更好的背景颜色等可能更漂亮,但你明白了。