mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge
synced 2024-09-07 15:05:25 +02:00
442 lines
16 KiB
Java
442 lines
16 KiB
Java
/* Copyright (C) 2017-2024 Alberto, Andreas Shimokawa, Carsten Pfeiffer,
|
|
Daniele Gobbetti, José Rebelo, Pavel Elagin, Petr Vaněk
|
|
|
|
This file is part of Gadgetbridge.
|
|
|
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as published
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Gadgetbridge is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
|
package nodomain.freeyourgadget.gadgetbridge.activities.charts;
|
|
|
|
import android.app.Activity;
|
|
import android.graphics.Color;
|
|
import android.os.Bundle;
|
|
import android.text.format.DateUtils;
|
|
import android.view.LayoutInflater;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
import android.widget.ImageView;
|
|
import android.widget.TextView;
|
|
|
|
import androidx.fragment.app.FragmentManager;
|
|
|
|
import com.github.mikephil.charting.charts.BarChart;
|
|
import com.github.mikephil.charting.charts.PieChart;
|
|
import com.github.mikephil.charting.components.LimitLine;
|
|
import com.github.mikephil.charting.components.XAxis;
|
|
import com.github.mikephil.charting.components.YAxis;
|
|
import com.github.mikephil.charting.data.BarData;
|
|
import com.github.mikephil.charting.data.BarDataSet;
|
|
import com.github.mikephil.charting.data.BarEntry;
|
|
import com.github.mikephil.charting.data.ChartData;
|
|
import com.github.mikephil.charting.data.PieData;
|
|
import com.github.mikephil.charting.data.PieDataSet;
|
|
import com.github.mikephil.charting.data.PieEntry;
|
|
import com.github.mikephil.charting.formatter.ValueFormatter;
|
|
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Calendar;
|
|
import java.util.List;
|
|
import java.util.Locale;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
|
import nodomain.freeyourgadget.gadgetbridge.R;
|
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
|
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmounts;
|
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
|
import nodomain.freeyourgadget.gadgetbridge.util.LimitedQueue;
|
|
|
|
|
|
public abstract class AbstractWeekChartFragment extends AbstractActivityChartFragment<AbstractWeekChartFragment.MyChartsData> {
|
|
protected static final Logger LOG = LoggerFactory.getLogger(AbstractWeekChartFragment.class);
|
|
protected final int TOTAL_DAYS = getRangeDays();
|
|
protected int TOTAL_DAYS_FOR_AVERAGE = 0;
|
|
|
|
private Locale mLocale;
|
|
private int mTargetValue = 0;
|
|
|
|
private PieChart mTodayPieChart;
|
|
private BarChart mWeekChart;
|
|
private TextView mBalanceView;
|
|
|
|
private int mOffsetHours = getOffsetHours();
|
|
ImageView stepsStreaksButton;
|
|
|
|
@Override
|
|
protected MyChartsData refreshInBackground(ChartsHost chartsHost, DBHandler db, GBDevice device) {
|
|
Calendar day = Calendar.getInstance();
|
|
day.setTime(chartsHost.getEndDate());
|
|
//NB: we could have omitted the day, but this way we can move things to the past easily
|
|
DayData dayData = refreshDayPie(db, day, device);
|
|
WeekChartsData<BarData> weekBeforeData = refreshWeekBeforeData(db, mWeekChart, day, device);
|
|
|
|
return new MyChartsData(dayData, weekBeforeData);
|
|
}
|
|
|
|
@Override
|
|
protected void updateChartsnUIThread(MyChartsData mcd) {
|
|
setupLegend(mWeekChart);
|
|
mTodayPieChart.setCenterText(mcd.getDayData().centerText);
|
|
mTodayPieChart.setData(mcd.getDayData().data);
|
|
//set custom renderer for 30days bar charts
|
|
if (GBApplication.getPrefs().getBoolean("charts_range", true)) {
|
|
mWeekChart.setRenderer(new AngledLabelsChartRenderer(mWeekChart, mWeekChart.getAnimator(), mWeekChart.getViewPortHandler()));
|
|
}
|
|
|
|
mWeekChart.setData(null); // workaround for https://github.com/PhilJay/MPAndroidChart/issues/2317
|
|
mWeekChart.setData(mcd.getWeekBeforeData().getData());
|
|
mWeekChart.getXAxis().setValueFormatter(mcd.getWeekBeforeData().getXValueFormatter());
|
|
|
|
mBalanceView.setText(mcd.getWeekBeforeData().getBalanceMessage());
|
|
|
|
//disable the streak FAB once we move away from today
|
|
Calendar day = Calendar.getInstance();
|
|
day.setTime(getChartsHost().getEndDate());
|
|
if (DateUtils.isToday(day.getTimeInMillis()) && enableStepStreaksButton()){
|
|
stepsStreaksButton.setVisibility(View.VISIBLE);
|
|
}else
|
|
{
|
|
stepsStreaksButton.setVisibility(View.GONE);
|
|
}
|
|
}
|
|
|
|
private boolean enableStepStreaksButton(){
|
|
return this.getClass().getSimpleName().equals("WeekStepsChartFragment");
|
|
}
|
|
|
|
@Override
|
|
protected void renderCharts() {
|
|
mWeekChart.invalidate();
|
|
mTodayPieChart.invalidate();
|
|
// mBalanceView.setText(getBalanceMessage(balance));
|
|
}
|
|
|
|
private String getWeeksChartsLabel(Calendar day){
|
|
if (GBApplication.getPrefs().getBoolean("charts_range", true)) {
|
|
//month, show day date
|
|
return String.valueOf(day.get(Calendar.DAY_OF_MONTH));
|
|
}
|
|
else{
|
|
//week, show short day name
|
|
return day.getDisplayName(Calendar.DAY_OF_WEEK, Calendar.SHORT, mLocale);
|
|
}
|
|
}
|
|
|
|
private WeekChartsData<BarData> refreshWeekBeforeData(DBHandler db, BarChart barChart, Calendar day, GBDevice device) {
|
|
day = (Calendar) day.clone(); // do not modify the caller's argument
|
|
day.add(Calendar.DATE, -TOTAL_DAYS);
|
|
List<BarEntry> entries = new ArrayList<>();
|
|
ArrayList<String> labels = new ArrayList<String>();
|
|
|
|
long balance = 0;
|
|
long daily_balance = 0;
|
|
TOTAL_DAYS_FOR_AVERAGE=0;
|
|
|
|
for (int counter = 0; counter < TOTAL_DAYS; counter++) {
|
|
ActivityAmounts amounts = getActivityAmountsForDay(db, day, device);
|
|
daily_balance=calculateBalance(amounts);
|
|
if (daily_balance > 0) {
|
|
TOTAL_DAYS_FOR_AVERAGE++;
|
|
}
|
|
|
|
balance += daily_balance;
|
|
entries.add(new BarEntry(counter, getTotalsForActivityAmounts(amounts)));
|
|
labels.add(getWeeksChartsLabel(day));
|
|
day.add(Calendar.DATE, 1);
|
|
}
|
|
|
|
BarDataSet set = new BarDataSet(entries, "");
|
|
set.setColors(getColors());
|
|
set.setValueFormatter(getBarValueFormatter());
|
|
|
|
BarData barData = new BarData(set);
|
|
barData.setValueTextColor(Color.GRAY); //prevent tearing other graph elements with the black text. Another approach would be to hide the values cmpletely with data.setDrawValues(false);
|
|
barData.setValueTextSize(10f);
|
|
|
|
LimitLine target = new LimitLine(mTargetValue);
|
|
barChart.getAxisLeft().removeAllLimitLines();
|
|
barChart.getAxisLeft().addLimitLine(target);
|
|
|
|
float average = 0;
|
|
if (TOTAL_DAYS_FOR_AVERAGE > 0) {
|
|
average = Math.abs(balance / TOTAL_DAYS_FOR_AVERAGE);
|
|
}
|
|
LimitLine average_line = new LimitLine(average);
|
|
average_line.setLabel(getString(R.string.average, getAverage(average)));
|
|
|
|
if (average > (mTargetValue)) {
|
|
average_line.setLineColor(Color.GREEN);
|
|
average_line.setTextColor(Color.GREEN);
|
|
}
|
|
else {
|
|
average_line.setLineColor(Color.RED);
|
|
average_line.setTextColor(Color.RED);
|
|
}
|
|
if (average > 0) {
|
|
if (GBApplication.getPrefs().getBoolean("charts_show_average", true)) {
|
|
barChart.getAxisLeft().addLimitLine(average_line);
|
|
}
|
|
}
|
|
|
|
return new WeekChartsData(barData, new PreformattedXIndexLabelFormatter(labels), getBalanceMessage(balance, mTargetValue));
|
|
}
|
|
|
|
private DayData refreshDayPie(DBHandler db, Calendar day, GBDevice device) {
|
|
|
|
PieData data = new PieData();
|
|
List<PieEntry> entries = new ArrayList<>();
|
|
PieDataSet set = new PieDataSet(entries, "");
|
|
|
|
ActivityAmounts amounts = getActivityAmountsForDay(db, day, device);
|
|
float totalValues[] = getTotalsForActivityAmounts(amounts);
|
|
String[] pieLabels = getPieLabels();
|
|
float totalValue = 0;
|
|
for (int i = 0; i < totalValues.length; i++) {
|
|
float value = totalValues[i];
|
|
totalValue += value;
|
|
entries.add(new PieEntry(value, pieLabels[i]));
|
|
}
|
|
|
|
set.setColors(getColors());
|
|
|
|
if (totalValues.length < 2) {
|
|
if (totalValue < mTargetValue) {
|
|
entries.add(new PieEntry((mTargetValue - totalValue)));
|
|
set.addColor(Color.GRAY);
|
|
}
|
|
}
|
|
|
|
data.setDataSet(set);
|
|
|
|
if (totalValues.length < 2) {
|
|
data.setDrawValues(false);
|
|
}
|
|
else {
|
|
set.setXValuePosition(PieDataSet.ValuePosition.OUTSIDE_SLICE);
|
|
set.setYValuePosition(PieDataSet.ValuePosition.OUTSIDE_SLICE);
|
|
set.setValueTextColor(DESCRIPTION_COLOR);
|
|
set.setValueTextSize(13f);
|
|
set.setValueFormatter(getPieValueFormatter());
|
|
}
|
|
|
|
return new DayData(data, formatPieValue((long) totalValue));
|
|
}
|
|
|
|
@Override
|
|
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
|
Bundle savedInstanceState) {
|
|
mLocale = getResources().getConfiguration().locale;
|
|
|
|
View rootView = inflater.inflate(R.layout.fragment_weeksteps_chart, container, false);
|
|
|
|
final int goal = getGoal();
|
|
if (goal >= 0) {
|
|
mTargetValue = goal;
|
|
}
|
|
|
|
mTodayPieChart = rootView.findViewById(R.id.todaystepschart);
|
|
mWeekChart = rootView.findViewById(R.id.weekstepschart);
|
|
mBalanceView = rootView.findViewById(R.id.balance);
|
|
|
|
setupWeekChart();
|
|
setupTodayPieChart();
|
|
|
|
stepsStreaksButton = rootView.findViewById(R.id.steps_streaks_button);
|
|
if (enableStepStreaksButton()) {
|
|
stepsStreaksButton.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View v) {
|
|
FragmentManager fm = getActivity().getSupportFragmentManager();
|
|
StepStreaksDashboard stepStreaksDashboard = StepStreaksDashboard.newInstance(getGoal(), getChartsHost().getDevice());
|
|
stepStreaksDashboard.show(fm, "steps_streaks_dashboard");
|
|
}
|
|
});
|
|
}
|
|
|
|
// refresh immediately instead of use refreshIfVisible(), for perceived performance
|
|
refresh();
|
|
|
|
return rootView;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void setupTodayPieChart() {
|
|
mTodayPieChart.setBackgroundColor(BACKGROUND_COLOR);
|
|
mTodayPieChart.getDescription().setTextColor(DESCRIPTION_COLOR);
|
|
mTodayPieChart.setEntryLabelColor(DESCRIPTION_COLOR);
|
|
mTodayPieChart.getDescription().setText(getPieDescription(mTargetValue));
|
|
// mTodayPieChart.setNoDataTextDescription("");
|
|
mTodayPieChart.setNoDataText("");
|
|
mTodayPieChart.getLegend().setEnabled(false);
|
|
}
|
|
|
|
private void setupWeekChart() {
|
|
mWeekChart.setBackgroundColor(BACKGROUND_COLOR);
|
|
mWeekChart.getDescription().setTextColor(DESCRIPTION_COLOR);
|
|
mWeekChart.getDescription().setText("");
|
|
mWeekChart.setFitBars(true);
|
|
|
|
configureBarLineChartDefaults(mWeekChart);
|
|
|
|
XAxis x = mWeekChart.getXAxis();
|
|
x.setDrawLabels(true);
|
|
x.setDrawGridLines(false);
|
|
x.setEnabled(true);
|
|
x.setTextColor(CHART_TEXT_COLOR);
|
|
x.setDrawLimitLinesBehindData(true);
|
|
x.setPosition(XAxis.XAxisPosition.BOTTOM);
|
|
|
|
YAxis y = mWeekChart.getAxisLeft();
|
|
y.setDrawGridLines(false);
|
|
y.setDrawTopYLabelEntry(false);
|
|
y.setTextColor(CHART_TEXT_COLOR);
|
|
y.setDrawZeroLine(true);
|
|
y.setSpaceBottom(0);
|
|
y.setAxisMinimum(0);
|
|
y.setValueFormatter(getYAxisFormatter());
|
|
y.setEnabled(true);
|
|
|
|
YAxis yAxisRight = mWeekChart.getAxisRight();
|
|
yAxisRight.setDrawGridLines(false);
|
|
yAxisRight.setEnabled(false);
|
|
yAxisRight.setDrawLabels(false);
|
|
yAxisRight.setDrawTopYLabelEntry(false);
|
|
yAxisRight.setTextColor(CHART_TEXT_COLOR);
|
|
}
|
|
|
|
private List<? extends ActivitySample> getSamplesOfDay(DBHandler db, Calendar day, int offsetHours, GBDevice device) {
|
|
int startTs;
|
|
int endTs;
|
|
|
|
day = (Calendar) day.clone(); // do not modify the caller's argument
|
|
day.set(Calendar.HOUR_OF_DAY, 0);
|
|
day.set(Calendar.MINUTE, 0);
|
|
day.set(Calendar.SECOND, 0);
|
|
day.add(Calendar.HOUR, offsetHours);
|
|
|
|
startTs = (int) (day.getTimeInMillis() / 1000);
|
|
endTs = startTs + 24 * 60 * 60 - 1;
|
|
|
|
return getSamples(db, device, startTs, endTs);
|
|
}
|
|
|
|
@Override
|
|
protected List<? extends ActivitySample> getSamples(DBHandler db, GBDevice device, int tsFrom, int tsTo) {
|
|
return super.getAllSamples(db, device, tsFrom, tsTo);
|
|
}
|
|
|
|
private static class DayData {
|
|
private final PieData data;
|
|
private final CharSequence centerText;
|
|
|
|
DayData(PieData data, String centerText) {
|
|
this.data = data;
|
|
this.centerText = centerText;
|
|
}
|
|
}
|
|
|
|
protected static class MyChartsData extends ChartsData {
|
|
private final WeekChartsData<BarData> weekBeforeData;
|
|
private final DayData dayData;
|
|
|
|
MyChartsData(DayData dayData, WeekChartsData<BarData> weekBeforeData) {
|
|
this.dayData = dayData;
|
|
this.weekBeforeData = weekBeforeData;
|
|
}
|
|
|
|
DayData getDayData() {
|
|
return dayData;
|
|
}
|
|
|
|
WeekChartsData<BarData> getWeekBeforeData() {
|
|
return weekBeforeData;
|
|
}
|
|
}
|
|
|
|
private ActivityAmounts getActivityAmountsForDay(DBHandler db, Calendar day, GBDevice device) {
|
|
|
|
LimitedQueue<Integer, ActivityAmounts> activityAmountCache = null;
|
|
ActivityAmounts amounts = null;
|
|
|
|
Activity activity = getActivity();
|
|
int key = (int) (day.getTimeInMillis() / 1000) + (mOffsetHours * 3600);
|
|
if (activity != null) {
|
|
activityAmountCache = ((ActivityChartsActivity) activity).mActivityAmountCache;
|
|
amounts = activityAmountCache.lookup(key);
|
|
}
|
|
|
|
if (amounts == null) {
|
|
ActivityAnalysis analysis = new ActivityAnalysis();
|
|
amounts = analysis.calculateActivityAmounts(getSamplesOfDay(db, day, mOffsetHours, device));
|
|
if (activityAmountCache != null) {
|
|
activityAmountCache.add(key, amounts);
|
|
}
|
|
}
|
|
|
|
return amounts;
|
|
}
|
|
|
|
private int getRangeDays(){
|
|
if (GBApplication.getPrefs().getBoolean("charts_range", true)) {
|
|
return 30;}
|
|
else{
|
|
return 7;
|
|
}
|
|
}
|
|
|
|
abstract String getAverage(float value);
|
|
|
|
abstract int getGoal();
|
|
|
|
abstract int getOffsetHours();
|
|
|
|
abstract float[] getTotalsForActivityAmounts(ActivityAmounts activityAmounts);
|
|
|
|
abstract String formatPieValue(long value);
|
|
|
|
abstract String[] getPieLabels();
|
|
|
|
abstract ValueFormatter getPieValueFormatter();
|
|
|
|
abstract ValueFormatter getBarValueFormatter();
|
|
|
|
abstract ValueFormatter getYAxisFormatter();
|
|
|
|
abstract int[] getColors();
|
|
|
|
abstract String getPieDescription(int targetValue);
|
|
|
|
protected abstract long calculateBalance(ActivityAmounts amounts);
|
|
|
|
protected abstract String getBalanceMessage(long balance, int targetValue);
|
|
|
|
private class WeekChartsData<T extends ChartData<?>> extends DefaultChartsData<T> {
|
|
private final String balanceMessage;
|
|
|
|
public WeekChartsData(T data, PreformattedXIndexLabelFormatter xIndexLabelFormatter, String balanceMessage) {
|
|
super(data, xIndexLabelFormatter);
|
|
this.balanceMessage = balanceMessage;
|
|
}
|
|
|
|
public String getBalanceMessage() {
|
|
return balanceMessage;
|
|
}
|
|
}
|
|
}
|