mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge.git
synced 2025-01-25 16:15:55 +01:00
initial version of speed zones tab (#674)
* #673 initial version of speed zones tab * #673 fix copyrights and initial step speed length
This commit is contained in:
parent
b31a6a5db9
commit
7dc9c28c74
@ -17,7 +17,9 @@
|
|||||||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
package nodomain.freeyourgadget.gadgetbridge.activities.charts;
|
package nodomain.freeyourgadget.gadgetbridge.activities.charts;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmount;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmount;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmounts;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmounts;
|
||||||
@ -25,6 +27,20 @@ import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||||
|
|
||||||
class ActivityAnalysis {
|
class ActivityAnalysis {
|
||||||
|
// store raw steps and duration
|
||||||
|
protected HashMap<Integer, Long> stats = new HashMap<Integer, Long>();
|
||||||
|
// normalize steps
|
||||||
|
protected HashMap<Float, Float> statsQuantified = new HashMap<Float, Float>();
|
||||||
|
// store maxSpeed / resolution
|
||||||
|
protected float maxSpeedQuantifier;
|
||||||
|
// store an average of round precision
|
||||||
|
protected float roundPrecision = 0f;
|
||||||
|
|
||||||
|
// max speed determined from samples
|
||||||
|
private int maxSpeed = 0;
|
||||||
|
// number of bars on stats chart
|
||||||
|
private int resolution = 5;
|
||||||
|
|
||||||
ActivityAmounts calculateActivityAmounts(List<? extends ActivitySample> samples) {
|
ActivityAmounts calculateActivityAmounts(List<? extends ActivitySample> samples) {
|
||||||
ActivityAmount deepSleep = new ActivityAmount(ActivityKind.TYPE_DEEP_SLEEP);
|
ActivityAmount deepSleep = new ActivityAmount(ActivityKind.TYPE_DEEP_SLEEP);
|
||||||
ActivityAmount lightSleep = new ActivityAmount(ActivityKind.TYPE_LIGHT_SLEEP);
|
ActivityAmount lightSleep = new ActivityAmount(ActivityKind.TYPE_LIGHT_SLEEP);
|
||||||
@ -53,7 +69,7 @@ class ActivityAnalysis {
|
|||||||
|
|
||||||
int steps = sample.getSteps();
|
int steps = sample.getSteps();
|
||||||
if (steps > 0) {
|
if (steps > 0) {
|
||||||
amount.addSteps(sample.getSteps());
|
amount.addSteps(steps);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (previousSample != null) {
|
if (previousSample != null) {
|
||||||
@ -65,12 +81,56 @@ class ActivityAnalysis {
|
|||||||
previousAmount.addSeconds(sharedTimeDifference);
|
previousAmount.addSeconds(sharedTimeDifference);
|
||||||
amount.addSeconds(sharedTimeDifference);
|
amount.addSeconds(sharedTimeDifference);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// add time
|
||||||
|
if (steps > 0 && sample.getKind() == ActivityKind.TYPE_ACTIVITY) {
|
||||||
|
if (steps > maxSpeed) {
|
||||||
|
maxSpeed = steps;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!stats.containsKey(steps)) {
|
||||||
|
//System.out.println("Adding: " + steps);
|
||||||
|
stats.put(steps, timeDifference);
|
||||||
|
} else {
|
||||||
|
long time = stats.get(steps);
|
||||||
|
//System.out.println("Updating: " + steps + " " + timeDifference + time);
|
||||||
|
stats.put(steps, timeDifference + time);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
previousAmount = amount;
|
previousAmount = amount;
|
||||||
previousSample = sample;
|
previousSample = sample;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
maxSpeedQuantifier = maxSpeed / resolution;
|
||||||
|
for (Map.Entry<Integer, Long> entry : stats.entrySet()) {
|
||||||
|
// 0.1 precision
|
||||||
|
//float keyQuantified = Math.round(entry.getKey() / maxSpeedQuantifier * 10f) / 10f;
|
||||||
|
|
||||||
|
// 1 precision
|
||||||
|
float keyQuantified = entry.getKey() / maxSpeedQuantifier;
|
||||||
|
float keyQuantifiedRounded = Math.round(entry.getKey() / maxSpeedQuantifier);
|
||||||
|
float keyQuantifiedPrecision = keyQuantifiedRounded - keyQuantified;
|
||||||
|
roundPrecision = (roundPrecision + Math.abs(keyQuantifiedPrecision)) / 2;
|
||||||
|
//System.out.println("Precision: " + roundPrecision);
|
||||||
|
|
||||||
|
// no scale
|
||||||
|
//keyQuantified = entry.getKey();
|
||||||
|
|
||||||
|
// scaling to minutes
|
||||||
|
float timeMinutes = entry.getValue() / 60;
|
||||||
|
|
||||||
|
if (!statsQuantified.containsKey(keyQuantifiedRounded)) {
|
||||||
|
//System.out.println("Adding: " + keyQuantified + "/" + timeMinutes);
|
||||||
|
statsQuantified.put(keyQuantifiedRounded, timeMinutes);
|
||||||
|
} else {
|
||||||
|
float previousTime = statsQuantified.get(keyQuantifiedRounded);
|
||||||
|
//System.out.println("Updating: " + keyQuantified + "/" + (timeMinutes + previousTime));
|
||||||
|
statsQuantified.put(keyQuantifiedRounded, (timeMinutes + previousTime));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ActivityAmounts result = new ActivityAmounts();
|
ActivityAmounts result = new ActivityAmounts();
|
||||||
if (deepSleep.getTotalSeconds() > 0) {
|
if (deepSleep.getTotalSeconds() > 0) {
|
||||||
result.addAmount(deepSleep);
|
result.addAmount(deepSleep);
|
||||||
|
@ -333,6 +333,8 @@ public class ChartsActivity extends AbstractGBFragmentActivity implements Charts
|
|||||||
return new WeekStepsChartFragment();
|
return new WeekStepsChartFragment();
|
||||||
case 4:
|
case 4:
|
||||||
return new LiveActivityFragment();
|
return new LiveActivityFragment();
|
||||||
|
case 5:
|
||||||
|
return new StatsChartFragment();
|
||||||
|
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
@ -341,7 +343,7 @@ public class ChartsActivity extends AbstractGBFragmentActivity implements Charts
|
|||||||
@Override
|
@Override
|
||||||
public int getCount() {
|
public int getCount() {
|
||||||
// Show 5 total pages.
|
// Show 5 total pages.
|
||||||
return 5;
|
return 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -357,6 +359,8 @@ public class ChartsActivity extends AbstractGBFragmentActivity implements Charts
|
|||||||
return getString(R.string.weekstepschart_steps_a_week);
|
return getString(R.string.weekstepschart_steps_a_week);
|
||||||
case 4:
|
case 4:
|
||||||
return getString(R.string.liveactivity_live_activity);
|
return getString(R.string.liveactivity_live_activity);
|
||||||
|
case 5:
|
||||||
|
return getString(R.string.stats_title);
|
||||||
}
|
}
|
||||||
return super.getPageTitle(position);
|
return super.getPageTitle(position);
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,217 @@
|
|||||||
|
/* Copyright (C) 2015-2017 0nse, Andreas Shimokawa, Carsten Pfeiffer,
|
||||||
|
Daniele Gobbetti, Vebryn
|
||||||
|
|
||||||
|
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 <http://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.activities.charts;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
|
||||||
|
import com.github.mikephil.charting.animation.Easing;
|
||||||
|
import com.github.mikephil.charting.charts.Chart;
|
||||||
|
import com.github.mikephil.charting.components.LegendEntry;
|
||||||
|
import com.github.mikephil.charting.components.XAxis;
|
||||||
|
import com.github.mikephil.charting.components.YAxis;
|
||||||
|
import com.github.mikephil.charting.data.CombinedData;
|
||||||
|
import com.github.mikephil.charting.data.Entry;
|
||||||
|
import com.github.mikephil.charting.data.PieEntry;
|
||||||
|
import com.github.mikephil.charting.formatter.IValueFormatter;
|
||||||
|
import com.github.mikephil.charting.utils.ViewPortHandler;
|
||||||
|
import com.github.mikephil.charting.charts.HorizontalBarChart;
|
||||||
|
import com.github.mikephil.charting.data.BarData;
|
||||||
|
import com.github.mikephil.charting.data.BarDataSet;
|
||||||
|
import com.github.mikephil.charting.data.BarEntry;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.activities.HeartRateUtils;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmount;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityAmounts;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.DateTimeUtils;
|
||||||
|
|
||||||
|
|
||||||
|
public class StatsChartFragment extends AbstractChartFragment {
|
||||||
|
protected static final Logger LOG = LoggerFactory.getLogger(ActivitySleepChartFragment.class);
|
||||||
|
|
||||||
|
private HorizontalBarChart mStatsChart;
|
||||||
|
|
||||||
|
private int mSmartAlarmFrom = -1;
|
||||||
|
private int mSmartAlarmTo = -1;
|
||||||
|
private int mTimestampFrom = -1;
|
||||||
|
private int mSmartAlarmGoneOff = -1;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ChartsData refreshInBackground(ChartsHost chartsHost, DBHandler db, GBDevice device) {
|
||||||
|
List<? extends ActivitySample> samples = getSamples(db, device);
|
||||||
|
|
||||||
|
MySleepChartsData mySleepChartsData = refreshSleepAmounts(device, samples);
|
||||||
|
DefaultChartsData chartsData = refresh(device, samples);
|
||||||
|
|
||||||
|
return new MyChartsData(mySleepChartsData, chartsData);
|
||||||
|
}
|
||||||
|
|
||||||
|
private MySleepChartsData refreshSleepAmounts(GBDevice mGBDevice, List<? extends ActivitySample> samples) {
|
||||||
|
ActivityAnalysis analysis = new ActivityAnalysis();
|
||||||
|
analysis.calculateActivityAmounts(samples);
|
||||||
|
BarData data = new BarData();
|
||||||
|
List<BarEntry> entries = new ArrayList<>();
|
||||||
|
XAxisValueFormatter customXAxis = new XAxisValueFormatter();
|
||||||
|
|
||||||
|
for (Map.Entry<Float, Float> entry : analysis.statsQuantified.entrySet()) {
|
||||||
|
entries.add(new BarEntry(entry.getKey(), entry.getValue()));
|
||||||
|
/*float realValue = entry.getKey() * analysis.maxSpeedQuantifier;
|
||||||
|
String customLabel = Math.round(realValue * (1 - analysis.roundPrecision) * 10f) / 10f + " - " + Math.round(realValue * (1 + analysis.roundPrecision) * 10f) / 10f;*/
|
||||||
|
customXAxis.add("" + entry.getKey() * analysis.maxSpeedQuantifier);
|
||||||
|
}
|
||||||
|
|
||||||
|
BarDataSet set = new BarDataSet(entries, "");
|
||||||
|
set.setColors(getColorFor(ActivityKind.TYPE_ACTIVITY));
|
||||||
|
//set.setDrawValues(false);
|
||||||
|
//data.setBarWidth(0.1f);
|
||||||
|
data.addDataSet(set);
|
||||||
|
|
||||||
|
// set X axis
|
||||||
|
customXAxis.sort();
|
||||||
|
XAxis left = mStatsChart.getXAxis();
|
||||||
|
left.setValueFormatter(customXAxis);
|
||||||
|
|
||||||
|
// display precision
|
||||||
|
//mStatsChart.getDescription().setText(Math.round(analysis.roundPrecision * 100) + "%");
|
||||||
|
|
||||||
|
return new MySleepChartsData("", data);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void updateChartsnUIThread(ChartsData chartsData) {
|
||||||
|
MyChartsData mcd = (MyChartsData) chartsData;
|
||||||
|
mStatsChart.setData(mcd.getPieData().getPieData());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTitle() {
|
||||||
|
return getString(R.string.stats_title);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
||||||
|
Bundle savedInstanceState) {
|
||||||
|
View rootView = inflater.inflate(R.layout.fragment_statschart, container, false);
|
||||||
|
|
||||||
|
mStatsChart = (HorizontalBarChart) rootView.findViewById(R.id.statschart);
|
||||||
|
setupStatsChart();
|
||||||
|
|
||||||
|
// refresh immediately instead of use refreshIfVisible(), for perceived performance
|
||||||
|
refresh();
|
||||||
|
|
||||||
|
return rootView;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupStatsChart() {
|
||||||
|
mStatsChart.setBackgroundColor(BACKGROUND_COLOR);
|
||||||
|
mStatsChart.getDescription().setTextColor(DESCRIPTION_COLOR);
|
||||||
|
mStatsChart.setNoDataText("");
|
||||||
|
mStatsChart.getLegend().setEnabled(false);
|
||||||
|
mStatsChart.setTouchEnabled(false);
|
||||||
|
mStatsChart.getDescription().setText("");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void setupLegend(Chart chart) {
|
||||||
|
List<LegendEntry> legendEntries = new ArrayList<>(3);
|
||||||
|
LegendEntry lightSleepEntry = new LegendEntry();
|
||||||
|
lightSleepEntry.label = akLightSleep.label;
|
||||||
|
lightSleepEntry.formColor = akLightSleep.color;
|
||||||
|
legendEntries.add(lightSleepEntry);
|
||||||
|
|
||||||
|
LegendEntry deepSleepEntry = new LegendEntry();
|
||||||
|
deepSleepEntry.label = akDeepSleep.label;
|
||||||
|
deepSleepEntry.formColor = akDeepSleep.color;
|
||||||
|
legendEntries.add(deepSleepEntry);
|
||||||
|
|
||||||
|
if (supportsHeartrate(getChartsHost().getDevice())) {
|
||||||
|
LegendEntry hrEntry = new LegendEntry();
|
||||||
|
hrEntry.label = HEARTRATE_LABEL;
|
||||||
|
hrEntry.formColor = HEARTRATE_COLOR;
|
||||||
|
legendEntries.add(hrEntry);
|
||||||
|
}
|
||||||
|
chart.getLegend().setCustom(legendEntries);
|
||||||
|
chart.getLegend().setTextColor(LEGEND_TEXT_COLOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected List<? extends ActivitySample> getSamples(DBHandler db, GBDevice device, int tsFrom, int tsTo) {
|
||||||
|
// temporary fix for totally wrong sleep amounts
|
||||||
|
// return super.getSleepSamples(db, device, tsFrom, tsTo);
|
||||||
|
return super.getAllSamples(db, device, tsFrom, tsTo);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void renderCharts() {
|
||||||
|
mStatsChart.invalidate();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MySleepChartsData extends ChartsData {
|
||||||
|
private String totalSleep;
|
||||||
|
private final BarData pieData;
|
||||||
|
|
||||||
|
public MySleepChartsData(String totalSleep, BarData pieData) {
|
||||||
|
this.totalSleep = totalSleep;
|
||||||
|
this.pieData = pieData;
|
||||||
|
}
|
||||||
|
|
||||||
|
public BarData getPieData() {
|
||||||
|
return pieData;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CharSequence getTotalSleep() {
|
||||||
|
return totalSleep;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MyChartsData extends ChartsData {
|
||||||
|
private final DefaultChartsData<CombinedData> chartsData;
|
||||||
|
private final MySleepChartsData pieData;
|
||||||
|
|
||||||
|
public MyChartsData(MySleepChartsData pieData, DefaultChartsData<CombinedData> chartsData) {
|
||||||
|
this.pieData = pieData;
|
||||||
|
this.chartsData = chartsData;
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySleepChartsData getPieData() {
|
||||||
|
return pieData;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DefaultChartsData<CombinedData> getChartsData() {
|
||||||
|
return chartsData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,42 @@
|
|||||||
|
package nodomain.freeyourgadget.gadgetbridge.activities.charts;
|
||||||
|
|
||||||
|
import com.github.mikephil.charting.components.AxisBase;
|
||||||
|
import com.github.mikephil.charting.formatter.IAxisValueFormatter;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by nhu on 30/04/17.
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class XAxisValueFormatter implements IAxisValueFormatter {
|
||||||
|
private List<String> mValues = new ArrayList<>();
|
||||||
|
|
||||||
|
public XAxisValueFormatter() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void add(String label) {
|
||||||
|
mValues.add(label);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void sort() {
|
||||||
|
//System.out.println("Sorting " + mValues);
|
||||||
|
Collections.sort(mValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getFormattedValue(float value, AxisBase axis) {
|
||||||
|
String returnString = "N/A";
|
||||||
|
|
||||||
|
try {
|
||||||
|
returnString = mValues.get((int) value).toString();
|
||||||
|
//System.out.println("Asking " + value + ", returning " + returnString);
|
||||||
|
} catch (Exception e) {
|
||||||
|
System.out.println(e.getMessage());
|
||||||
|
}
|
||||||
|
return returnString;
|
||||||
|
}
|
||||||
|
}
|
39
app/src/main/res/layout/fragment_statschart.xml
Normal file
39
app/src/main/res/layout/fragment_statschart.xml
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
tools:context="nodomain.freeyourgadget.gadgetbridge.activities.charts.ChartsActivity$PlaceholderFragment">
|
||||||
|
|
||||||
|
<com.github.mikephil.charting.charts.HorizontalBarChart
|
||||||
|
android:id="@+id/statschart"
|
||||||
|
android:layout_width="fill_parent"
|
||||||
|
android:layout_height="fill_parent"
|
||||||
|
android:layout_weight="20"
|
||||||
|
android:layout_below="@+id/statsXAxisText"
|
||||||
|
android:layout_alignParentStart="true"
|
||||||
|
android:layout_toStartOf="@+id/statsYAxisText"></com.github.mikephil.charting.charts.HorizontalBarChart>
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/statsXAxisText"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_alignParentEnd="true"
|
||||||
|
android:layout_alignParentStart="true"
|
||||||
|
android:layout_alignParentTop="true"
|
||||||
|
android:paddingLeft="10dp"
|
||||||
|
android:text="@string/stats_x_axis_label" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/statsYAxisText"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_alignParentEnd="true"
|
||||||
|
android:layout_centerVertical="true"
|
||||||
|
android:layout_marginStart="-95dp"
|
||||||
|
android:elegantTextHeight="false"
|
||||||
|
android:rotation="90"
|
||||||
|
android:singleLine="true"
|
||||||
|
android:text="@string/stats_y_axis_label"
|
||||||
|
android:translationX="40dp" />
|
||||||
|
|
||||||
|
</RelativeLayout>
|
@ -258,6 +258,10 @@
|
|||||||
<string name="pref_screen_notification_profile_generic_navigation">Navigation</string>
|
<string name="pref_screen_notification_profile_generic_navigation">Navigation</string>
|
||||||
<string name="pref_screen_notification_profile_generic_social">Social Network</string>
|
<string name="pref_screen_notification_profile_generic_social">Social Network</string>
|
||||||
|
|
||||||
|
<string name="stats_title">Speed zones</string>
|
||||||
|
<string name="stats_x_axis_label">Total minutes</string>
|
||||||
|
<string name="stats_y_axis_label">Steps per minute</string>
|
||||||
|
|
||||||
<string name="control_center_find_lost_device">Find lost Device</string>
|
<string name="control_center_find_lost_device">Find lost Device</string>
|
||||||
<string name="control_center_cancel_to_stop_vibration">Cancel to stop vibration.</string>
|
<string name="control_center_cancel_to_stop_vibration">Cancel to stop vibration.</string>
|
||||||
<string name="title_activity_charts">Your Activity</string>
|
<string name="title_activity_charts">Your Activity</string>
|
||||||
|
Loading…
Reference in New Issue
Block a user