Back to project page customhellochartdemo.
The source code is released under:
Apache License
If you think the Android project customhellochartdemo listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.
package lecho.lib.hellocharts.samples; // w w w . ja va 2 s .c om import java.util.ArrayList; import java.util.List; import lecho.lib.hellocharts.model.Axis; import lecho.lib.hellocharts.model.AxisValue; import lecho.lib.hellocharts.model.Line; import lecho.lib.hellocharts.model.LineChartData; import lecho.lib.hellocharts.model.PointValue; import lecho.lib.hellocharts.model.SimpleValueFormatter; import lecho.lib.hellocharts.model.Viewport; import lecho.lib.hellocharts.util.Utils; import lecho.lib.hellocharts.view.LineChartView; import android.graphics.Color; import android.os.Bundle; import android.support.v4.app.Fragment; import android.support.v7.app.ActionBarActivity; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; public class TempoChartActivity extends ActionBarActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_tempo_chart); if (savedInstanceState == null) { getSupportFragmentManager().beginTransaction().add(R.id.container, new PlaceholderFragment()).commit(); } } public static class PlaceholderFragment extends Fragment { private LineChartView chart; private LineChartData data; public PlaceholderFragment() { } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View rootView = inflater.inflate(R.layout.fragment_tempo_chart, container, false); chart = (LineChartView) rootView.findViewById(R.id.chart); generateTempoData(); return rootView; } private void generateTempoData() { // I got speed in range (0-50) and height in meters in range(200 - 300). I want this chart to display both // information. Differences between speed and height values are large and chart doesn't look good so I need // to modify height values to be in range of speed values. // The same for displaying Tempo/Height chart. float minHeight = 200; float maxHeight = 300; float tempoRange = 15; // from 0min/km to 15min/km float scale = tempoRange / maxHeight; float sub = (minHeight * scale) / 2; int numValues = 52; Line line; List<PointValue> values; List<Line> lines = new ArrayList<Line>(); // Height line, add it as first line to be drawn in the background. values = new ArrayList<PointValue>(); for (int i = 0; i < numValues; ++i) { // Some random height values, add +200 to make line a little more natural float rawHeight = (float) (Math.random() * 100 + 200); float normalizedHeight = rawHeight * scale - sub; values.add(new PointValue(i, normalizedHeight)); } line = new Line(values); line.setColor(Color.GRAY); line.setHasPoints(false); line.setFilled(true); line.setStrokeWidth(1); lines.add(line); // Tempo line is a little tricky because worse tempo means bigger value for example 11min per km is worse // than 2min per km but the second should be higher on the chart. So you need to know max tempo and // tempoRange and set // chart values to minTempo - realTempo. values = new ArrayList<PointValue>(); for (int i = 0; i < numValues; ++i) { // Some random raw tempo values. float realTempo = (float) Math.random() * 6 + 2; float revertedTempo = tempoRange - realTempo; values.add(new PointValue(i, revertedTempo)); } line = new Line(values); line.setColor(Utils.COLOR_RED); line.setHasPoints(false); line.setStrokeWidth(3); lines.add(line); // Data and axes data = new LineChartData(lines); // Distance axis(bottom X) with formatter that will ad [km] to values, remember to modify max label charts // value. Axis distanceAxis = new Axis(); distanceAxis.setName("Distance"); distanceAxis.setTextColor(Utils.COLOR_ORANGE); distanceAxis.setMaxLabelChars(4); distanceAxis.setFormatter(new SimpleValueFormatter(0, false, null, "km".toCharArray())); distanceAxis.setHasLines(true); data.setAxisXBottom(distanceAxis); // Tempo uses minutes so I can't use auto-generated axis because auto-generation works only for decimal // system. So generate custom axis values for example every 15 seconds and set custom labels in format // minutes:seconds(00:00), you could do it in formatter but here will be faster. List<AxisValue> axisValues = new ArrayList<AxisValue>(); for (float i = 0; i < tempoRange; i += 0.25f) { // I'am translating float to minutes because I don't have data in minutes, if You store some time data // you may skip translation. axisValues.add(new AxisValue(i, formatMinutes(tempoRange - i))); } Axis tempoAxis = new Axis(axisValues).setName("Tempo [min/km]").setHasLines(true).setMaxLabelChars(4) .setTextColor(Utils.COLOR_RED); data.setAxisYLeft(tempoAxis); // *** Same as in Speed/Height chart. // Height axis, this axis need custom formatter that will translate values back to real height values. data.setAxisYRight(new Axis().setName("Height [m]").setMaxLabelChars(3) .setFormatter(new HeightValueFormater(scale, sub, 0, null, null))); // Set data chart.setLineChartData(data); // Important: adjust viewport, you could skip this step but in this case it will looks better with custom // viewport. Set // viewport with Y range 0-12; Viewport v = chart.getMaximumViewport(); v.set(v.left, tempoRange, v.right, 0); chart.setMaximumViewport(v); chart.setCurrentViewport(v, false); } private char[] formatMinutes(float value) { StringBuilder sb = new StringBuilder(); // translate value to seconds, for example int valueInSeconds = (int) (value * 60); int minutes = (int) Math.floor(valueInSeconds / 60); int seconds = (int) valueInSeconds % 60; sb.append(String.valueOf(minutes)).append(':'); if (seconds < 10) { sb.append('0'); } sb.append(String.valueOf(seconds)); return sb.toString().toCharArray(); } /** * Recalculated height values to display on axis. For this example I use auto-generated height axis so I * override only formatAutoValue method. */ private static class HeightValueFormater extends SimpleValueFormatter { private float scale; private float sub; public HeightValueFormater(float scale, float sub, int digits, char[] prependedText, char[] apendedText) { // Don't use auto digits for auto-generated axes, instead use digits number passed in parameter. super(digits, true, prependedText, apendedText); this.scale = scale; this.sub = sub; } @Override public int formatAutoValue(char[] formattedValue, float[] values, int digits) { int index = values.length - 1;// I just need last value from this array because SimpleValueFormatter // uses only last value and that is enough for axis labels. values[index] = (values[index] + sub) / scale; return super.formatAutoValue(formattedValue, values, digits); } } } }