2023-07-26 03:37:02 +00:00
|
|
|
# %%
|
|
|
|
import dash, requests, math
|
|
|
|
from dash import dcc
|
|
|
|
from dash import html
|
|
|
|
from dash.dependencies import Output, State, Input
|
|
|
|
import pandas as pd
|
2023-07-27 04:59:04 +00:00
|
|
|
import numpy as np
|
2023-07-26 03:37:02 +00:00
|
|
|
import plotly.express as px
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
|
|
|
|
|
|
# %%
|
|
|
|
|
2023-07-27 04:07:07 +00:00
|
|
|
app = dash.Dash(__name__)
|
2023-07-26 03:37:02 +00:00
|
|
|
server = app.server
|
|
|
|
|
|
|
|
app.layout = html.Div(children=[
|
2023-07-27 02:35:10 +00:00
|
|
|
|
2023-07-26 03:37:02 +00:00
|
|
|
html.Div(style={
|
|
|
|
'display': 'flex',
|
|
|
|
'align-items': 'center',
|
|
|
|
'justify-content': 'center',
|
|
|
|
'gap': '20px',
|
|
|
|
'margin': 'auto',
|
2023-07-27 04:07:07 +00:00
|
|
|
'flex-wrap': 'wrap',
|
|
|
|
'margin-top': '30px'
|
2023-07-26 03:37:02 +00:00
|
|
|
},children=[
|
|
|
|
dcc.DatePickerRange(
|
|
|
|
id='my-date-picker-range',
|
|
|
|
minimum_nights=30,
|
|
|
|
max_date_allowed=datetime.today().date() - timedelta(days=1),
|
|
|
|
min_date_allowed=datetime.today().date() - timedelta(days=720),
|
|
|
|
end_date=datetime.today().date() - timedelta(days=1),
|
|
|
|
start_date=datetime.today().date() - timedelta(days=365)
|
|
|
|
),
|
|
|
|
dcc.Input(id='input-on-submit', value="", placeholder='API ACCESS TOKEN', type='text'),
|
2023-07-27 04:07:07 +00:00
|
|
|
html.Button(id='submit-button', type='submit', children='Submit', n_clicks=0, className="button-primary"),
|
2023-07-26 03:37:02 +00:00
|
|
|
]),
|
|
|
|
|
|
|
|
html.Div(id='loading-div', style={'margin-top': '40px'}, children=[
|
|
|
|
dcc.Loading(
|
|
|
|
id="loading-progress",
|
|
|
|
type="default",
|
|
|
|
children=html.Div(id="loading-output-1")
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
|
|
|
|
html.Div(id='output_div', children=[
|
|
|
|
|
|
|
|
dcc.Graph(
|
|
|
|
id='graph_RHR',
|
|
|
|
figure=px.line(),
|
|
|
|
config= {'displaylogo': False}
|
|
|
|
),
|
|
|
|
|
|
|
|
dcc.Graph(
|
|
|
|
id='graph_steps',
|
|
|
|
figure=px.bar(),
|
|
|
|
config= {'displaylogo': False}
|
|
|
|
),
|
|
|
|
|
|
|
|
dcc.Graph(
|
|
|
|
id='graph_activity_minutes',
|
|
|
|
figure=px.bar(),
|
|
|
|
config= {'displaylogo': False}
|
|
|
|
),
|
|
|
|
|
|
|
|
dcc.Graph(
|
|
|
|
id='graph_weight',
|
|
|
|
figure=px.line(),
|
|
|
|
config= {'displaylogo': False}
|
|
|
|
),
|
|
|
|
|
|
|
|
dcc.Graph(
|
|
|
|
id='graph_spo2',
|
|
|
|
figure=px.line(),
|
|
|
|
config= {'displaylogo': False}
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
])
|
2023-07-27 02:35:10 +00:00
|
|
|
|
|
|
|
# Limits the date range to one year max
|
|
|
|
@app.callback(Output('my-date-picker-range', 'max_date_allowed'), Output('my-date-picker-range', 'end_date'),
|
|
|
|
[Input('my-date-picker-range', 'start_date')])
|
|
|
|
def set_max_date_allowed(start_date):
|
|
|
|
start = datetime.strptime(start_date, "%Y-%m-%d")
|
|
|
|
max_end_date = start + timedelta(days=365)
|
|
|
|
return max_end_date, max_end_date
|
|
|
|
|
2023-07-27 04:07:07 +00:00
|
|
|
# Disables the button after click and starts calculations
|
|
|
|
@app.callback(Output('submit-button', 'disabled'), Output('my-date-picker-range', 'disabled'), Output('input-on-submit', 'disabled'), Input('submit-button', 'n_clicks'), prevent_initial_call=True)
|
2023-07-27 02:35:10 +00:00
|
|
|
def disable_button_and_calculate(n_clicks):
|
2023-07-27 04:07:07 +00:00
|
|
|
return True, True, True
|
2023-07-27 02:35:10 +00:00
|
|
|
|
|
|
|
# fetch data and update graphs on click of submit
|
2023-07-26 03:37:02 +00:00
|
|
|
@app.callback(Output('graph_RHR', 'figure'), Output('graph_steps', 'figure'), Output('graph_activity_minutes', 'figure'), Output('graph_weight', 'figure'), Output('graph_spo2', 'figure'), Output("loading-output-1", "children"),
|
2023-07-27 02:35:10 +00:00
|
|
|
Input('submit-button', 'disabled'),
|
|
|
|
State('input-on-submit', 'value'), State('my-date-picker-range', 'start_date'), State('my-date-picker-range', 'end_date'),
|
2023-07-26 03:37:02 +00:00
|
|
|
prevent_initial_call=True
|
|
|
|
)
|
|
|
|
def update_output(n_clicks, value, start_date, end_date):
|
|
|
|
|
|
|
|
start_date = datetime.fromisoformat(start_date).strftime("%Y-%m-%d")
|
|
|
|
end_date = datetime.fromisoformat(end_date).strftime("%Y-%m-%d")
|
|
|
|
|
|
|
|
headers = {
|
|
|
|
"Authorization": "Bearer " + value,
|
|
|
|
"Accept": "application/json"
|
|
|
|
}
|
|
|
|
|
|
|
|
# Collecting data-----------------------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
response_heartrate = requests.get("https://api.fitbit.com/1/user/-/activities/heart/date/"+ start_date +"/"+ end_date +".json", headers=headers).json()
|
|
|
|
response_steps = requests.get("https://api.fitbit.com/1/user/-/activities/steps/date/"+ start_date +"/"+ end_date +".json", headers=headers).json()
|
|
|
|
response_weight = requests.get("https://api.fitbit.com/1/user/-/body/weight/date/"+ start_date +"/"+ end_date +".json", headers=headers).json()
|
|
|
|
response_spo2 = requests.get("https://api.fitbit.com/1/user/-/spo2/date/"+ start_date +"/"+ end_date +".json", headers=headers).json()
|
|
|
|
|
|
|
|
# Processing data-----------------------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
dates_list = []
|
|
|
|
dates_str_list = []
|
|
|
|
rhr_list = []
|
|
|
|
steps_list = []
|
|
|
|
weight_list = []
|
|
|
|
spo2_list = []
|
|
|
|
fat_burn_minutes_list, cardio_minutes_list, peak_minutes_list = [], [], []
|
|
|
|
|
|
|
|
for entry in response_heartrate['activities-heart']:
|
|
|
|
dates_str_list.append(entry['dateTime'])
|
|
|
|
dates_list.append(datetime.strptime(entry['dateTime'], '%Y-%m-%d'))
|
|
|
|
try:
|
|
|
|
fat_burn_minutes_list.append(entry["value"]["heartRateZones"][1]["minutes"])
|
|
|
|
cardio_minutes_list.append(entry["value"]["heartRateZones"][2]["minutes"])
|
|
|
|
peak_minutes_list.append(entry["value"]["heartRateZones"][3]["minutes"])
|
|
|
|
except KeyError as E:
|
|
|
|
fat_burn_minutes_list.append(None)
|
|
|
|
cardio_minutes_list.append(None)
|
|
|
|
peak_minutes_list.append(None)
|
|
|
|
if 'restingHeartRate' in entry['value']:
|
|
|
|
rhr_list.append(entry['value']['restingHeartRate'])
|
|
|
|
else:
|
|
|
|
rhr_list.append(None)
|
|
|
|
|
|
|
|
for entry in response_steps['activities-steps']:
|
|
|
|
steps_list.append(int(entry['value']))
|
|
|
|
|
|
|
|
for entry in response_weight["body-weight"]:
|
|
|
|
weight_list.append(float(entry['value']))
|
|
|
|
|
|
|
|
for entry in response_spo2:
|
|
|
|
spo2_list += [None]*(dates_str_list.index(entry["dateTime"])-len(spo2_list))
|
|
|
|
spo2_list.append(entry["value"]["avg"])
|
2023-07-26 04:00:00 +00:00
|
|
|
spo2_list += [None]*(len(dates_str_list)-len(spo2_list))
|
2023-07-26 03:51:55 +00:00
|
|
|
|
2023-07-26 03:37:02 +00:00
|
|
|
df_merged = pd.DataFrame({
|
|
|
|
"Date": dates_list,
|
|
|
|
"Resting Heart Rate": rhr_list,
|
|
|
|
"Steps Count": steps_list,
|
|
|
|
"Fat Burn Minutes": fat_burn_minutes_list,
|
|
|
|
"Cardio Minutes": cardio_minutes_list,
|
|
|
|
"Peak Minutes": peak_minutes_list,
|
|
|
|
"weight": weight_list,
|
|
|
|
"SPO2": spo2_list
|
|
|
|
})
|
|
|
|
|
2023-07-27 04:59:04 +00:00
|
|
|
non_zero_steps_df = df_merged[df_merged["Steps Count"] != 0]
|
|
|
|
|
2023-07-26 03:37:02 +00:00
|
|
|
# Plotting data-----------------------------------------------------------------------------------------------------------------------
|
|
|
|
|
2023-07-27 04:07:07 +00:00
|
|
|
fig_rhr = px.line(df_merged, x="Date", y="Resting Heart Rate", line_shape="spline", color_discrete_sequence=["#d30f1c"], title="Daily Resting Heart Rate")
|
|
|
|
fig_rhr.add_annotation(x=df_merged.iloc[df_merged["Resting Heart Rate"].idxmax()]["Date"], y=df_merged["Resting Heart Rate"].max(), text=str(df_merged["Resting Heart Rate"].max()), showarrow=False, arrowhead=0, bgcolor="#5f040a", opacity=0.80, yshift=15, borderpad=5, font=dict(family="Helvetica, monospace", size=12, color="#ffffff"), )
|
|
|
|
fig_rhr.add_annotation(x=df_merged.iloc[df_merged["Resting Heart Rate"].idxmin()]["Date"], y=df_merged["Resting Heart Rate"].min(), text=str(df_merged["Resting Heart Rate"].min()), showarrow=False, arrowhead=0, bgcolor="#0b2d51", opacity=0.80, yshift=-15, borderpad=5, font=dict(family="Helvetica, monospace", size=12, color="#ffffff"), )
|
|
|
|
fig_rhr.add_hline(y=df_merged["Resting Heart Rate"].mean(), line_dash="dot",annotation_text="Average : " + str(round(df_merged["Resting Heart Rate"].mean(), 1)), annotation_position="bottom right", annotation_bgcolor="#6b3908", annotation_opacity=0.6, annotation_borderpad=5, annotation_font=dict(family="Helvetica, monospace", size=14, color="#ffffff"))
|
|
|
|
fig_rhr.add_hrect(y0=62, y1=68, fillcolor="green", opacity=0.15, line_width=0)
|
2023-07-26 03:37:02 +00:00
|
|
|
fig_steps = px.bar(df_merged, x="Date", y="Steps Count", title="Daily Steps Count")
|
2023-07-27 04:59:04 +00:00
|
|
|
fig_steps.add_annotation(x=df_merged.iloc[df_merged["Steps Count"].idxmax()]["Date"], y=df_merged["Steps Count"].max(), text=str(df_merged["Steps Count"].max())+" steps", showarrow=False, arrowhead=0, bgcolor="#5f040a", opacity=0.80, yshift=15, borderpad=5, font=dict(family="Helvetica, monospace", size=12, color="#ffffff"), )
|
|
|
|
fig_steps.add_annotation(x=non_zero_steps_df.iloc[non_zero_steps_df["Steps Count"].idxmin()]["Date"], y=non_zero_steps_df["Steps Count"].min(), text=str(non_zero_steps_df["Steps Count"].min())+" steps", showarrow=False, arrowhead=0, bgcolor="#0b2d51", opacity=0.80, yshift=-15, borderpad=5, font=dict(family="Helvetica, monospace", size=12, color="#ffffff"), )
|
|
|
|
fig_steps.add_hline(y=non_zero_steps_df["Steps Count"].mean(), line_dash="dot",annotation_text="Average : " + str(round(df_merged["Steps Count"].mean(), 1)), annotation_position="bottom right", annotation_bgcolor="#6b3908", annotation_opacity=0.8, annotation_borderpad=5, annotation_font=dict(family="Helvetica, monospace", size=14, color="#ffffff"))
|
2023-07-26 03:37:02 +00:00
|
|
|
fig_activity_minutes = px.bar(df_merged, x="Date", y=["Fat Burn Minutes", "Cardio Minutes", "Peak Minutes"], title="Activity Minutes")
|
|
|
|
fig_activity_minutes.update_layout(yaxis_title='Active Minutes')
|
|
|
|
fig_weight = px.line(df_merged, x="Date", y="weight", line_shape="spline", color_discrete_sequence=["#6b3908"], title="Weight")
|
|
|
|
fig_spo2 = px.bar(df_merged, x="Date", y="SPO2", title="SPO2 Percentage", range_y=(80,100))
|
|
|
|
|
|
|
|
|
|
|
|
return fig_rhr, fig_steps, fig_activity_minutes, fig_weight, fig_spo2, ""
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
app.run_server(debug=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# %%
|