fix graph library bug
This commit is contained in:
parent
0942ffe0fa
commit
a20e49669a
2 changed files with 410 additions and 8 deletions
|
@ -88,6 +88,7 @@ import info.nightscout.androidaps.plugins.Overview.events.EventDismissNotificati
|
||||||
import info.nightscout.androidaps.plugins.Overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.Overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.Overview.graphExtensions.AreaGraphSeries;
|
import info.nightscout.androidaps.plugins.Overview.graphExtensions.AreaGraphSeries;
|
||||||
import info.nightscout.androidaps.plugins.Overview.graphExtensions.DoubleDataPoint;
|
import info.nightscout.androidaps.plugins.Overview.graphExtensions.DoubleDataPoint;
|
||||||
|
import info.nightscout.androidaps.plugins.Overview.graphExtensions.FixedLineGraphSeries;
|
||||||
import info.nightscout.androidaps.plugins.Overview.graphExtensions.PointsWithLabelGraphSeries;
|
import info.nightscout.androidaps.plugins.Overview.graphExtensions.PointsWithLabelGraphSeries;
|
||||||
import info.nightscout.androidaps.plugins.Overview.graphExtensions.TimeAsXAxisLabelFormatter;
|
import info.nightscout.androidaps.plugins.Overview.graphExtensions.TimeAsXAxisLabelFormatter;
|
||||||
import info.nightscout.androidaps.plugins.SourceXdrip.SourceXdripPlugin;
|
import info.nightscout.androidaps.plugins.SourceXdrip.SourceXdripPlugin;
|
||||||
|
@ -1041,11 +1042,10 @@ public class OverviewFragment extends Fragment implements View.OnClickListener,
|
||||||
}
|
}
|
||||||
|
|
||||||
// **** IOB COB graph ****
|
// **** IOB COB graph ****
|
||||||
LineGraphSeries<DataPoint> iobSeries;
|
FixedLineGraphSeries<DataPoint> iobSeries;
|
||||||
LineGraphSeries<DataPoint> cobSeries;
|
FixedLineGraphSeries<DataPoint> cobSeries;
|
||||||
Double maxIobValueFound = 0d;
|
Double maxIobValueFound = 0d;
|
||||||
Double maxCobValueFound = 0d;
|
Double maxCobValueFound = 0d;
|
||||||
Long lastTime = 0L;
|
|
||||||
|
|
||||||
if (showIobView.isChecked() || showCobView.isChecked()) {
|
if (showIobView.isChecked() || showCobView.isChecked()) {
|
||||||
List<DataPoint> iobArray = new ArrayList<>();
|
List<DataPoint> iobArray = new ArrayList<>();
|
||||||
|
@ -1061,13 +1061,10 @@ public class OverviewFragment extends Fragment implements View.OnClickListener,
|
||||||
//cobArray.add(new DataPoint(time, mealData.mealCOB));
|
//cobArray.add(new DataPoint(time, mealData.mealCOB));
|
||||||
//maxCobValueFound = Math.max(maxCobValueFound, mealData.mealCOB);
|
//maxCobValueFound = Math.max(maxCobValueFound, mealData.mealCOB);
|
||||||
}
|
}
|
||||||
lastTime = time;
|
|
||||||
}
|
}
|
||||||
iobArray.add(new DataPoint(lastTime, 0)); // close the path
|
|
||||||
iobArray.add(new DataPoint(fromTime, 0)); // close the path
|
|
||||||
DataPoint[] iobData = new DataPoint[iobArray.size()];
|
DataPoint[] iobData = new DataPoint[iobArray.size()];
|
||||||
iobData = iobArray.toArray(iobData);
|
iobData = iobArray.toArray(iobData);
|
||||||
iobSeries = new LineGraphSeries<>(iobData);
|
iobSeries = new FixedLineGraphSeries<>(iobData);
|
||||||
iobSeries.setDrawBackground(true);
|
iobSeries.setDrawBackground(true);
|
||||||
iobSeries.setBackgroundColor(0x80FFFFFF & MainApp.sResources.getColor(R.color.ioborange)); //50%
|
iobSeries.setBackgroundColor(0x80FFFFFF & MainApp.sResources.getColor(R.color.ioborange)); //50%
|
||||||
iobSeries.setColor(MainApp.sResources.getColor(R.color.ioborange));
|
iobSeries.setColor(MainApp.sResources.getColor(R.color.ioborange));
|
||||||
|
@ -1077,7 +1074,7 @@ public class OverviewFragment extends Fragment implements View.OnClickListener,
|
||||||
|
|
||||||
DataPoint[] cobData = new DataPoint[cobArray.size()];
|
DataPoint[] cobData = new DataPoint[cobArray.size()];
|
||||||
cobData = cobArray.toArray(cobData);
|
cobData = cobArray.toArray(cobData);
|
||||||
cobSeries = new LineGraphSeries<>(cobData);
|
cobSeries = new FixedLineGraphSeries<>(cobData);
|
||||||
cobSeries.setDrawBackground(true);
|
cobSeries.setDrawBackground(true);
|
||||||
cobSeries.setBackgroundColor(Color.RED);
|
cobSeries.setBackgroundColor(Color.RED);
|
||||||
cobSeries.setThickness(0);
|
cobSeries.setThickness(0);
|
||||||
|
|
|
@ -0,0 +1,405 @@
|
||||||
|
package info.nightscout.androidaps.plugins.Overview.graphExtensions;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by mike on 24.04.2017.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.graphics.Path;
|
||||||
|
|
||||||
|
import com.jjoe64.graphview.GraphView;
|
||||||
|
import com.jjoe64.graphview.series.BaseSeries;
|
||||||
|
import com.jjoe64.graphview.series.DataPointInterface;
|
||||||
|
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.graphics.Path;
|
||||||
|
|
||||||
|
import com.jjoe64.graphview.GraphView;
|
||||||
|
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Series to plot the data as line.
|
||||||
|
* The line can be styled with many options.
|
||||||
|
*
|
||||||
|
* @author jjoe64
|
||||||
|
*/
|
||||||
|
public class FixedLineGraphSeries<E extends DataPointInterface> extends BaseSeries<E> {
|
||||||
|
/**
|
||||||
|
* wrapped styles regarding the line
|
||||||
|
*/
|
||||||
|
private final class Styles {
|
||||||
|
/**
|
||||||
|
* the thickness of the line.
|
||||||
|
* This option will be ignored if you are
|
||||||
|
* using a custom paint via {@link #setCustomPaint(android.graphics.Paint)}
|
||||||
|
*/
|
||||||
|
private int thickness = 5;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the area under the line to the bottom
|
||||||
|
* of the viewport will be filled with a
|
||||||
|
* specific background color.
|
||||||
|
*
|
||||||
|
* @see #backgroundColor
|
||||||
|
*/
|
||||||
|
private boolean drawBackground = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the data points are highlighted as
|
||||||
|
* a visible point.
|
||||||
|
*
|
||||||
|
* @see #dataPointsRadius
|
||||||
|
*/
|
||||||
|
private boolean drawDataPoints = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* the radius for the data points.
|
||||||
|
*
|
||||||
|
* @see #drawDataPoints
|
||||||
|
*/
|
||||||
|
private float dataPointsRadius = 10f;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* the background color for the filling under
|
||||||
|
* the line.
|
||||||
|
*
|
||||||
|
* @see #drawBackground
|
||||||
|
*/
|
||||||
|
private int backgroundColor = Color.argb(100, 172, 218, 255);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* wrapped styles
|
||||||
|
*/
|
||||||
|
private Styles mStyles;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* internal paint object
|
||||||
|
*/
|
||||||
|
private Paint mPaint;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* paint for the background
|
||||||
|
*/
|
||||||
|
private Paint mPaintBackground;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* path for the background filling
|
||||||
|
*/
|
||||||
|
private Path mPathBackground;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* path to the line
|
||||||
|
*/
|
||||||
|
private Path mPath;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* custom paint that can be used.
|
||||||
|
* this will ignore the thickness and color styles.
|
||||||
|
*/
|
||||||
|
private Paint mCustomPaint;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* creates a series without data
|
||||||
|
*/
|
||||||
|
public FixedLineGraphSeries() {
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* creates a series with data
|
||||||
|
*
|
||||||
|
* @param data data points
|
||||||
|
*/
|
||||||
|
public FixedLineGraphSeries(E[] data) {
|
||||||
|
super(data);
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* do the initialization
|
||||||
|
* creates internal objects
|
||||||
|
*/
|
||||||
|
protected void init() {
|
||||||
|
mStyles = new Styles();
|
||||||
|
mPaint = new Paint();
|
||||||
|
mPaint.setStrokeCap(Paint.Cap.ROUND);
|
||||||
|
mPaint.setStyle(Paint.Style.STROKE);
|
||||||
|
mPaintBackground = new Paint();
|
||||||
|
|
||||||
|
mPathBackground = new Path();
|
||||||
|
mPath = new Path();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* plots the series
|
||||||
|
* draws the line and the background
|
||||||
|
*
|
||||||
|
* @param graphView graphview
|
||||||
|
* @param canvas canvas
|
||||||
|
* @param isSecondScale flag if it is the second scale
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void draw(GraphView graphView, Canvas canvas, boolean isSecondScale) {
|
||||||
|
resetDataPoints();
|
||||||
|
|
||||||
|
// get data
|
||||||
|
double maxX = graphView.getViewport().getMaxX(false);
|
||||||
|
double minX = graphView.getViewport().getMinX(false);
|
||||||
|
|
||||||
|
double maxY;
|
||||||
|
double minY;
|
||||||
|
if (isSecondScale) {
|
||||||
|
maxY = graphView.getSecondScale().getMaxY();
|
||||||
|
minY = graphView.getSecondScale().getMinY();
|
||||||
|
} else {
|
||||||
|
maxY = graphView.getViewport().getMaxY(false);
|
||||||
|
minY = graphView.getViewport().getMinY(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
Iterator<E> values = getValues(minX, maxX);
|
||||||
|
|
||||||
|
// draw background
|
||||||
|
double lastEndY = 0;
|
||||||
|
double lastEndX = 0;
|
||||||
|
|
||||||
|
// draw data
|
||||||
|
mPaint.setStrokeWidth(mStyles.thickness);
|
||||||
|
mPaint.setColor(getColor());
|
||||||
|
mPaintBackground.setColor(mStyles.backgroundColor);
|
||||||
|
|
||||||
|
Paint paint;
|
||||||
|
if (mCustomPaint != null) {
|
||||||
|
paint = mCustomPaint;
|
||||||
|
} else {
|
||||||
|
paint = mPaint;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mStyles.drawBackground) {
|
||||||
|
mPathBackground.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
double diffY = maxY - minY;
|
||||||
|
double diffX = maxX - minX;
|
||||||
|
|
||||||
|
float graphHeight = graphView.getGraphContentHeight();
|
||||||
|
float graphWidth = graphView.getGraphContentWidth();
|
||||||
|
float graphLeft = graphView.getGraphContentLeft();
|
||||||
|
float graphTop = graphView.getGraphContentTop();
|
||||||
|
|
||||||
|
lastEndY = 0;
|
||||||
|
lastEndX = 0;
|
||||||
|
double lastUsedEndX = 0;
|
||||||
|
float firstX = 0;
|
||||||
|
int i=0;
|
||||||
|
while (values.hasNext()) {
|
||||||
|
E value = values.next();
|
||||||
|
|
||||||
|
double valY = value.getY() - minY;
|
||||||
|
double ratY = valY / diffY;
|
||||||
|
double y = graphHeight * ratY;
|
||||||
|
|
||||||
|
double valX = value.getX() - minX;
|
||||||
|
double ratX = valX / diffX;
|
||||||
|
double x = graphWidth * ratX;
|
||||||
|
|
||||||
|
double orgX = x;
|
||||||
|
double orgY = y;
|
||||||
|
|
||||||
|
if (i > 0) {
|
||||||
|
// overdraw
|
||||||
|
if (x > graphWidth) { // end right
|
||||||
|
double b = ((graphWidth - lastEndX) * (y - lastEndY)/(x - lastEndX));
|
||||||
|
y = lastEndY+b;
|
||||||
|
x = graphWidth;
|
||||||
|
}
|
||||||
|
if (y < 0) { // end bottom
|
||||||
|
double b = ((0 - lastEndY) * (x - lastEndX)/(y - lastEndY));
|
||||||
|
x = lastEndX+b;
|
||||||
|
y = 0;
|
||||||
|
}
|
||||||
|
if (y > graphHeight) { // end top
|
||||||
|
double b = ((graphHeight - lastEndY) * (x - lastEndX)/(y - lastEndY));
|
||||||
|
x = lastEndX+b;
|
||||||
|
y = graphHeight;
|
||||||
|
}
|
||||||
|
if (lastEndY < 0) { // start bottom
|
||||||
|
double b = ((0 - y) * (x - lastEndX)/(lastEndY - y));
|
||||||
|
lastEndX = x-b;
|
||||||
|
lastEndY = 0;
|
||||||
|
}
|
||||||
|
if (lastEndX < 0) { // start left
|
||||||
|
double b = ((0 - x) * (y - lastEndY)/(lastEndX - x));
|
||||||
|
lastEndY = y-b;
|
||||||
|
lastEndX = 0;
|
||||||
|
}
|
||||||
|
if (lastEndY > graphHeight) { // start top
|
||||||
|
double b = ((graphHeight - y) * (x - lastEndX)/(lastEndY - y));
|
||||||
|
lastEndX = x-b;
|
||||||
|
lastEndY = graphHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
float startX = (float) lastEndX + (graphLeft + 1);
|
||||||
|
float startY = (float) (graphTop - lastEndY) + graphHeight;
|
||||||
|
float endX = (float) x + (graphLeft + 1);
|
||||||
|
float endY = (float) (graphTop - y) + graphHeight;
|
||||||
|
|
||||||
|
// draw data point
|
||||||
|
if (mStyles.drawDataPoints) {
|
||||||
|
//fix: last value was not drawn. Draw here now the end values
|
||||||
|
canvas.drawCircle(endX, endY, mStyles.dataPointsRadius, mPaint);
|
||||||
|
}
|
||||||
|
registerDataPoint(endX, endY, value);
|
||||||
|
|
||||||
|
mPath.reset();
|
||||||
|
mPath.moveTo(startX, startY);
|
||||||
|
mPath.lineTo(endX, endY);
|
||||||
|
canvas.drawPath(mPath, paint);
|
||||||
|
if (mStyles.drawBackground) {
|
||||||
|
if (i==1) {
|
||||||
|
firstX = startX;
|
||||||
|
mPathBackground.moveTo(startX, startY);
|
||||||
|
}
|
||||||
|
mPathBackground.lineTo(endX, endY);
|
||||||
|
}
|
||||||
|
lastUsedEndX = endX;
|
||||||
|
} else if (mStyles.drawDataPoints) {
|
||||||
|
//fix: last value not drawn as datapoint. Draw first point here, and then on every step the end values (above)
|
||||||
|
float first_X = (float) x + (graphLeft + 1);
|
||||||
|
float first_Y = (float) (graphTop - y) + graphHeight;
|
||||||
|
//TODO canvas.drawCircle(first_X, first_Y, dataPointsRadius, mPaint);
|
||||||
|
}
|
||||||
|
lastEndY = orgY;
|
||||||
|
lastEndX = orgX;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mStyles.drawBackground) {
|
||||||
|
// end / close path
|
||||||
|
mPathBackground.lineTo((float) lastUsedEndX, (float) (graphTop - (-minY / diffY * graphHeight)) + graphHeight);
|
||||||
|
mPathBackground.lineTo(firstX, (float) (graphTop - (-minY / diffY * graphHeight)) + graphHeight);
|
||||||
|
mPathBackground.close();
|
||||||
|
canvas.drawPath(mPathBackground, mPaintBackground);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* the thickness of the line.
|
||||||
|
* This option will be ignored if you are
|
||||||
|
* using a custom paint via {@link #setCustomPaint(android.graphics.Paint)}
|
||||||
|
*
|
||||||
|
* @return the thickness of the line
|
||||||
|
*/
|
||||||
|
public int getThickness() {
|
||||||
|
return mStyles.thickness;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* the thickness of the line.
|
||||||
|
* This option will be ignored if you are
|
||||||
|
* using a custom paint via {@link #setCustomPaint(android.graphics.Paint)}
|
||||||
|
*
|
||||||
|
* @param thickness thickness of the line
|
||||||
|
*/
|
||||||
|
public void setThickness(int thickness) {
|
||||||
|
mStyles.thickness = thickness;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the area under the line to the bottom
|
||||||
|
* of the viewport will be filled with a
|
||||||
|
* specific background color.
|
||||||
|
*
|
||||||
|
* @return whether the background will be drawn
|
||||||
|
* @see #getBackgroundColor()
|
||||||
|
*/
|
||||||
|
public boolean isDrawBackground() {
|
||||||
|
return mStyles.drawBackground;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the area under the line to the bottom
|
||||||
|
* of the viewport will be filled with a
|
||||||
|
* specific background color.
|
||||||
|
*
|
||||||
|
* @param drawBackground whether the background will be drawn
|
||||||
|
* @see #setBackgroundColor(int)
|
||||||
|
*/
|
||||||
|
public void setDrawBackground(boolean drawBackground) {
|
||||||
|
mStyles.drawBackground = drawBackground;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the data points are highlighted as
|
||||||
|
* a visible point.
|
||||||
|
*
|
||||||
|
* @return flag whether the data points are highlighted
|
||||||
|
* @see #setDataPointsRadius(float)
|
||||||
|
*/
|
||||||
|
public boolean isDrawDataPoints() {
|
||||||
|
return mStyles.drawDataPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flag whether the data points are highlighted as
|
||||||
|
* a visible point.
|
||||||
|
*
|
||||||
|
* @param drawDataPoints flag whether the data points are highlighted
|
||||||
|
* @see #setDataPointsRadius(float)
|
||||||
|
*/
|
||||||
|
public void setDrawDataPoints(boolean drawDataPoints) {
|
||||||
|
mStyles.drawDataPoints = drawDataPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the radius for the data points.
|
||||||
|
* @see #setDrawDataPoints(boolean)
|
||||||
|
*/
|
||||||
|
public float getDataPointsRadius() {
|
||||||
|
return mStyles.dataPointsRadius;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param dataPointsRadius the radius for the data points.
|
||||||
|
* @see #setDrawDataPoints(boolean)
|
||||||
|
*/
|
||||||
|
public void setDataPointsRadius(float dataPointsRadius) {
|
||||||
|
mStyles.dataPointsRadius = dataPointsRadius;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the background color for the filling under
|
||||||
|
* the line.
|
||||||
|
* @see #setDrawBackground(boolean)
|
||||||
|
*/
|
||||||
|
public int getBackgroundColor() {
|
||||||
|
return mStyles.backgroundColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param backgroundColor the background color for the filling under
|
||||||
|
* the line.
|
||||||
|
* @see #setDrawBackground(boolean)
|
||||||
|
*/
|
||||||
|
public void setBackgroundColor(int backgroundColor) {
|
||||||
|
mStyles.backgroundColor = backgroundColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* custom paint that can be used.
|
||||||
|
* this will ignore the thickness and color styles.
|
||||||
|
*
|
||||||
|
* @param customPaint the custom paint to be used for rendering the line
|
||||||
|
*/
|
||||||
|
public void setCustomPaint(Paint customPaint) {
|
||||||
|
this.mCustomPaint = customPaint;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue