Private
Public Access
1
0

Merge branch 'feature/fitnesschart' into develop

This commit is contained in:
Sander Roosendaal
2021-01-04 18:55:13 +01:00
4 changed files with 399 additions and 2 deletions

View File

@@ -123,6 +123,29 @@ def newtestpowerid(x):
return x['id']
def newtestpowerdate(x):
try:
if np.isnan(x['testpower']):
return np.nan
except (AttributeError,TypeError):
return np.nan
return x['date']
def all_goldmedalstandards(workouts,startdate,enddate):
dates = []
testpowers = []
testduration = []
for w in workouts:
goldmedalstandard, goldmedalseconds = dataprep.workout_goldmedalstandard(w)
if goldmedalseconds > 60:
dates.append(arrow.get(w.date).datetime)
testpowers.append(goldmedalstandard)
testduration.append(goldmedalseconds)
return dates,testpowers,testduration
def build_goldmedalstandards(workouts,kfitness):
dates = []
testpower = []
@@ -1778,6 +1801,168 @@ def getfatigues(
return fatigues,fitnesses,dates,testpower,testduration,impulses
def goldmedalscorechart(user,startdate=None,enddate=None):
# to avoid data mess later on
startdate = arrow.get(startdate).datetime.replace(hour=0,minute=0,second=0,microsecond=0)
enddate = enddate+datetime.timedelta(days=1)
enddate = arrow.get(enddate).datetime.replace(hour=0,minute=0,second=0,microsecond=0)
TOOLS = 'save,pan,box_zoom,wheel_zoom,reset,tap,hover,crosshair'
workouts = Workout.objects.filter(user=user.rower,date__gte=startdate,
date__lte=enddate,
workouttype__in=mytypes.rowtypes,
duplicate=False)
# marker workouts
dates,testpower,testduration,fatigues,fitnesses,impulses, outids = build_goldmedalstandards(
workouts,42
)
df = pd.DataFrame({
'id':outids,
'date':dates,
'testpower':testpower,
'testduration':testduration,
})
df.sort_values(['date'],inplace=True)
df['testdup'] = df['testpower'].shift(1)
df['testpower'] = df.apply(lambda x: newtestpower(x),axis=1)
#df['date'] = df.apply(lambda x: newtestpowerdate(x), axis=1)
mask = df['testpower'].isnull()
dates = df.mask(mask)['date'].dropna().values
testpower = df.mask(mask)['testpower'].dropna().values
outids = df.mask(mask)['id'].dropna().unique()
# all workouts
alldates,alltestpower,allduration = all_goldmedalstandards(workouts,startdate,enddate)
nrdays = (enddate-startdate).days
td = []
markerscore = []
score = []
markerduration = []
duration = []
previous = 0
for i in range(len(dates)):
dd = str(dates[i])
td.append(arrow.get(dd).datetime)
markerscore.append(testpower[i])
markerduration.append(testduration[i])
score.append(np.nan)
duration.append(np.nan)
for i in range(len(alldates)):
td.append(arrow.get(alldates[i]).datetime)
markerscore.append(np.nan)
score.append(alltestpower[i])
markerduration.append(np.nan)
duration.append(allduration[i])
for i in range(nrdays+1):
td.append(arrow.get(startdate+datetime.timedelta(days=i)).datetime)
markerscore.append(np.nan)
score.append(np.nan)
markerduration.append(np.nan)
duration.append(np.nan)
df = pd.DataFrame({
'markerscore':markerscore,
'markerduration':markerduration,
'score':score,
'duration':duration,
'date':td,
})
df.sort_values(['date'],inplace=True)
df = df.groupby(['date']).max()
df['date'] = df.index.values
source = ColumnDataSource(
data = dict(
markerscore = df['markerscore'],
score = df['score'],
markerduration = df['markerduration'].apply(lambda x:totaltime_sec_to_string(x,shorten=True)),
duration = df['duration'].apply(lambda x:totaltime_sec_to_string(x,shorten=True)),
date = df['date'],
fdate = df['date'].map(lambda x: x.strftime('%d-%m-%Y')),
)
)
plot = Figure(tools=TOOLS,x_axis_type='datetime',
plot_width=900,plot_height=600,
toolbar_location='above',
toolbar_sticky=False)
# add watermark
watermarkurl = "/static/img/logo7.png"
watermarksource = ColumnDataSource(dict(
url = [watermarkurl],))
watermarkrange = Range1d(start=0,end=1)
watermarkalpha = 0.6
watermarkx = 0.99
watermarky = 0.01
watermarkw = 184
watermarkh = 35
watermarkanchor = 'bottom_right'
plot.extra_y_ranges = {"watermark": watermarkrange}
plot.extra_x_ranges = {"watermark": watermarkrange}
plot.image_url([watermarkurl],watermarkx,watermarky,
watermarkw,watermarkh,
global_alpha=watermarkalpha,
w_units='screen',
h_units='screen',
anchor=watermarkanchor,
dilate=True,
x_range_name = "watermark",
y_range_name = "watermark",
)
plot.xaxis.axis_label = 'Date'
plot.yaxis.axis_label = 'Gold Medal Score'
plot.circle('date','score',source=source,fill_color='blue',
size=10,
legend_label='Workouts')
plot.circle('date','markerscore',source=source,fill_color='red',
size=10,
legend_label='Marker Workouts')
plot.legend.location = "bottom_left"
plot.x_range = Range1d(
startdate,enddate+datetime.timedelta(days=5),
)
hover = plot.select(dict(type=HoverTool))
hover.tooltips = OrderedDict([
('Marker','@markerscore{int}'),
('Test', '@markerduration'),
('Score','@score{int}'),
('Duration', '@duration'),
('Date','@fdate'),
])
script, div = components(plot)
return script, div,outids
def performance_chart(user,startdate=None,enddate=None,kfitness=42,kfatigue=7,
metricchoice='trimp',doform=False,dofatigue=False,
showtests=False):

View File

@@ -0,0 +1,142 @@
{% extends "newbase.html" %}
{% load staticfiles %}
{% load rowerfilters %}
{% block title %}Rowsandall Gold Medal Score {% endblock %}
{% block scripts %}
<script type='text/javascript'
src='https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js'>
</script>
<script>
function submit_form() {
console.log("form changed");
var frm = $("#performanceform");
var data = new FormData(frm[0]);
$.ajax({
url:"/rowers/goldmedalscores/user/{{ rower.user.id }}/",
type: "POST",
contentType: false,
processData: false,
data: data,
dataType: 'json',
success: function(data) {
console.log(data);
// var parsedJSON = $.parseJSON(data); //
$("#id_script").replaceWith('<div id="id_script">'+data.script+'</d'+'iv>');
$("#id_chart").replaceWith('<div id="id_chart">'+data.div+'</d'+'iv>');
console.log('done');
}
});
};
$(document).ready(function() {
var csrftoken = jQuery("[name=csrfmiddlewaretoken]").val();
console.log("CSRF token",csrftoken);
function csrfSafeMethod(method) {
// these HTTP methods do not require CSRF protection
return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));
}
$.ajaxSetup({
beforeSend: function(xhr, settings) {
if (!csrfSafeMethod(settings.type) && !this.crossDomain) {
xhr.setRequestHeader("X-CSRFToken", csrftoken);
}
}
});
$("#performanceform").on('change', function(evt) {
submit_form();
});
});
</script>
{% endblock %}
{% block main %}
<script src="https://cdn.pydata.org/bokeh/release/bokeh-2.2.3.min.js"></script>
<script async="true" type="text/javascript">
Bokeh.set_log_level("info");
</script>
<div id="id_script">
{{ chartscript |safe }}
</div>
{% if rower.user %}
<h1>Gold Medal Scores for {{ rower.user.first_name }} </h1>
{% else %}
<h1>Gold Medal Scores for {{ user.first_name }} </h1>
{% endif %}
<ul class="main-content">
<li class="grid_4">
<div id="id_chart">
{{ the_div|safe }}
</div>
</li>
<li class="grid_1">
<form id="dateform" enctype="multipart/form-data" method="post">
<table>
{{ form.as_table }}
</table>
{% csrf_token %}
<input name='form' class="button" type="submit" value="Submit">
</form>
</li>
<li class="grid_2">
<p>
Explanation
</p>
</li>
{% if bestworkouts %}
<li class="grid_4">
<h2>Marker Workouts</h2>
<table width="100%" class="listtable">
<thead>
<tr>
<th>Date</th>
<th>Workout</th>
<th>Gold Medal Score</th>
<th>Duration</th>
</tr>
</thead>
<tbody>
{% for w in bestworkouts %}
<tr>
<td>{{ w.date }}</td>
<td>
<a href="/rowers/workout/{{ w.id|encode }}/">{{ w.name }}
</td>
<td>
{{ w.goldmedalstandard|floatformat:"0" }} %
</td>
<td>
{{ w.goldmedalseconds|secondstotimestring }}
</td>
</tr>
{% endfor %}
</tbody>
</table>
</li>
{% endif %}
</ul>
{% endblock %}
{% block sidebar %}
{% include 'menu_analytics.html' %}
{% endblock %}

View File

@@ -355,6 +355,9 @@ urlpatterns = [
re_path(r'^fitness-fit/$',views.fitness_from_cp_view,name='fitness_from_cp_view'),
re_path(r'^fitness-fit/user/(?P<userid>\d+)/$',views.fitness_from_cp_view,name='fitness_from_cp_view'),
re_path(r'^fitness-fit/user/(?P<userid>\d+)/(?P<mode>\w+.*)/$',views.fitness_from_cp_view,name='fitness_from_cp_view'),
re_path(r'^goldmedalscores/$',views.goldmedalscores_view,name='goldmedalscores_view'),
re_path(r'^goldmedalscores/user/(?P<userid>\d+)/$',views.goldmedalscores_view,name='goldmedalscores_view'),
re_path(r'^goldmedalscores/user/(?P<userid>\d+)/(?P<mode>\w+.*)/$',views.goldmedalscores_view,name='goldmedalscores_view'),
re_path(r'^performancemanager/$',views.performancemanager_view,name='performancemanager_view'),
re_path(r'^performancemanager/user/(?P<userid>\d+)/$',views.performancemanager_view,name='performancemanager_view'),
re_path(r'^performancemanager/user/(?P<userid>\d+)/(?P<mode>\w+.*)/$',views.performancemanager_view,name='performancemanager_view'),

View File

@@ -1543,6 +1543,73 @@ def fitnessmetric_view(request,userid=0,mode='rower',
'form':form,
})
@user_passes_test(ispromember, login_url="/rowers/paidplans",
message="This functionality requires a Pro plan or higher. If you are already a Pro user, please log in to access this functionality",
redirect_field_name=None)
def goldmedalscores_view(request,userid=0,
startdate=timezone.now()-timezone.timedelta(days=365),
enddate=timezone.now()):
is_ajax = False
if request.is_ajax():
is_ajax = True
therower = getrequestrower(request,userid=userid)
theuser = therower.user
if request.method == 'POST':
form = DateRangeForm(request.POST)
if form.is_valid():
startdate = form.cleaned_data['startdate']
enddate = form.cleaned_data['enddate']
if startdate > enddate:
s = enddate
enddate = startdate
startdate = s
else:
form = DateRangeForm(initial={
'startdate':startdate,
'enddate':enddate,
})
script, div, ids = goldmedalscorechart(
theuser,startdate=startdate,enddate=enddate,
)
bestworkouts = Workout.objects.filter(id__in=ids).order_by('date')
breadcrumbs = [
{
'url':'/rower/analysis',
'name':'Analysis',
},
{
'url':reverse(goldmedalscores_view),
'name': 'Gold Medal Scores'
}
]
if is_ajax:
response = json.dumps({
'script':script,
'div':div,
})
return(HttpResponse(response,content_type='application/json'))
return render(request,'goldmedalscores.html',
{
'rower':therower,
'active':'nav-analysis',
'chartscript':script,
'breadcrumbs':breadcrumbs,
'the_div':div,
'form':form,
'bestworkouts':bestworkouts,
})
@user_passes_test(ispromember, login_url="/rowers/paidplans",
message="This functionality requires a Pro plan or higher. If you are already a Pro user, please log in to access this functionality",
redirect_field_name=None)
@@ -1606,8 +1673,8 @@ def performancemanager_view(request,userid=0,mode='rower',
'name':'Analysis'
},
{
'url':reverse('fitnessmetric_view'),
'name': 'Power Progress'
'url':reverse('performancemanager_view'),
'name': 'Performance Manager'
}
]