From 9a3a69a4357676c5ff3d47fd3acde0f88c12279b Mon Sep 17 00:00:00 2001
From: Sander Roosendaal
Date: Fri, 22 Mar 2024 14:57:28 +0100
Subject: [PATCH] itneractive plot v1
---
rowers/interactiveplots.py | 121 +++------------------
rowers/templates/instroke_interactive.html | 11 +-
rowers/templates/workout_view.html | 84 +++++++-------
rowers/tests/testdata/testdata.tcx.gz | Bin 4000 -> 3999 bytes
4 files changed, 61 insertions(+), 155 deletions(-)
diff --git a/rowers/interactiveplots.py b/rowers/interactiveplots.py
index 8551c179..6fa0ae28 100644
--- a/rowers/interactiveplots.py
+++ b/rowers/interactiveplots.py
@@ -2089,10 +2089,6 @@ def course_map(course):
return script, div
-
-
-
-
def leaflet_chart(lat, lon, name="", raceresult=0):
try:
if lat.empty or lon.empty: # pragma: no cover
@@ -3558,110 +3554,23 @@ def interactive_chart(id=0, promember=0, intervaldata={}):
except KeyError: # pragma: no cover
datadf['pace'] = 0
- source = ColumnDataSource(
- datadf
- )
+ data_dict = datadf.to_dict("records")
- plot = figure(x_axis_type="datetime", y_axis_type="datetime",
- width=400,
- height=400,
- toolbar_sticky=False,
- tools=TOOLS)
+ metrics_list = [{'name': name, 'rowingmetrics':d } for name, d in metrics.rowingmetrics]
+
+ chart_data = {
+ 'title': row.name,
+ 'x': "time",
+ 'y1': "pace",
+ 'y2': "spm",
+ 'data': data_dict,
+ 'metrics': metrics_list,
+ }
- # add watermark
- watermarkurl = "/static/img/logo7.png"
- watermarkrange = Range1d(start=0, end=1)
- watermarkalpha = 0.6
- watermarkw = 184
- watermarkh = 35
- plot.extra_y_ranges = {"watermark": watermarkrange}
- plot.extra_x_ranges = {"watermark": watermarkrange}
+ script, div = get_chart("/interactive", chart_data)
- plot.image_url([watermarkurl], 0.01, 0.99,
- 0.5*watermarkw, 0.5*watermarkh,
- global_alpha=watermarkalpha,
- w_units='screen',
- h_units='screen',
- anchor='top_left',
- dilate=True,
- x_range_name="watermark",
- y_range_name="watermark",
- )
+ return script, div
- plot.line('time', 'pace', source=source, legend_label="Pace", name="pace")
-
- try:
- plot.title.text = row.name
- except ValueError: # pragma: no cover
- plot.title.text = ""
- plot.title.text_font_size = "1.0em"
- #plot.sizing_mode = 'stretch_both'
- plot.xaxis.axis_label = "Time"
- plot.yaxis.axis_label = "Pace (/500m)"
- plot.xaxis[0].formatter = DatetimeTickFormatter(
- hours=["%H"],
- minutes=["%M"],
- seconds=["%S"],
- days=["0"],
- months=[""],
- years=[""]
- )
- plot.yaxis[0].formatter = DatetimeTickFormatter(
- seconds=["%S"],
- minutes=["%M"]
- )
- ymax = 90.
- ymin = 150.
-
- if row.workouttype == 'water':
- ymax = 90.
- ymin = 210.
-
- plot.y_range = Range1d(1.e3*ymin, 1.e3*ymax)
-
- plot.extra_y_ranges["spmax"] = Range1d(start=10, end=45)
- plot.line('time', 'spm', source=source, color="red",
- y_range_name="spmax", legend_label="Stroke Rate", name="spm")
- plot.add_layout(LinearAxis(y_range_name="spmax",
- axis_label="SPM"), 'right')
-
- plot.legend.location = "bottom_right"
-
- # add shaded bar chart areas
- if intervaldata != {}:
- intervaldf = pd.DataFrame(intervaldata)
- intervaldf['itime'] = intervaldf['itime']*1.e3
- intervaldf['time'] = intervaldf['itime'].cumsum()
- intervaldf['time'] = intervaldf['time'].shift(1)
- intervaldf.loc[0, 'time'] = 0
- intervaldf['time_r'] = intervaldf['time'] + intervaldf['itime']
- intervaldf['value'] = 100
- mask = intervaldf['itype'] == 3
- intervaldf.loc[mask, 'value'] = 0
- intervaldf['bottom'] = 10
-
- intervalsource = ColumnDataSource(
- intervaldf
- )
-
- plot.quad(left='time', top='value', bottom='bottom',
- right='time_r', source=intervalsource, color='mediumvioletred',
- y_range_name='spmax', fill_alpha=0.2, line_alpha=0.2)
-
- hover = plot.select(dict(type=HoverTool))
-
- hover.tooltips = OrderedDict([
- ('Time', '@ftime'),
- ('Pace', '@fpace'),
- ('HR', '@hr{int}'),
- ('SPM', '@spm{1.1}'),
- ])
-
- hover.mode = 'mouse'
- # hover.name = ["spm", "pace"]
- script, div = components(plot)
-
- return [script, div]
def interactive_chart_video(videodata):
@@ -3948,12 +3857,12 @@ def interactive_flexchart_stacked(id, r, xparam='time',
if rowdata.empty:
- return "", "No valid data", '', '', comment
+ return "", "No valid data"
try:
tseconds = rowdata.loc[:, 'time']
except KeyError: # pragma: no cover
- return '', 'No time data - cannot make flex plot', '', '', comment
+ return '', 'No time data - cannot make flex plot'
try:
rowdata['x1'] = rowdata.loc[:, xparam]
diff --git a/rowers/templates/instroke_interactive.html b/rowers/templates/instroke_interactive.html
index d7d89704..6f3be39c 100644
--- a/rowers/templates/instroke_interactive.html
+++ b/rowers/templates/instroke_interactive.html
@@ -122,9 +122,6 @@ $( function() {
{{ the_script |safe }}
-
- {{ ds |safe }}
-
In Stroke Metrics
@@ -172,10 +169,16 @@ $( function() {
{{ the_div|safe }}
- -
+
-
+
{{ dd|safe }}
+
+
+ {{ ds |safe }}
+
+
diff --git a/rowers/templates/workout_view.html b/rowers/templates/workout_view.html
index af45ac7e..eaa3184c 100644
--- a/rowers/templates/workout_view.html
+++ b/rowers/templates/workout_view.html
@@ -104,59 +104,53 @@
{% endif %}
- {% for course in courses %}
-
- |
- Timed Course:
- |
-
- {{ course }}
- |
-
- {% endfor %}
-
-
-
- Workout Summary
-
-
-
- {{ summary }}
-
-
-
- {% if mapdiv %}
-
-
-
+ {% for course in courses %}
+
+ |
+ Timed Course:
+ |
+
+ {{ course }}
+ |
+
+{% endfor %}
+
+
+
+ Workout Summary
+
+
+
+ {{ summary }}
+
+
+
+{% if mapdiv %}
+
+
+
{{ mapdiv|safe }}
-
+
-
-
- {% endif %}
-
-
-
-
- {{ interactiveplot |safe }}
-
- {{ the_div|safe }}
-
-
- {% for graph in graphs %}
-
-
+
+
+{% endif %}
+
+
+ {{ the_div|safe }}
+ {{ interactiveplot |safe }}
+
+{% for graph in graphs %}
+
+
-
+
- {% endfor %}
+{% endfor %}
{% endblock %}
diff --git a/rowers/tests/testdata/testdata.tcx.gz b/rowers/tests/testdata/testdata.tcx.gz
index 48ef22c8cf5affe8abbb6bff226cc4031e518250..74d4512c54218bb2e5d6196c85237a0e52864745 100644
GIT binary patch
literal 3999
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
F0RT%=7v%r|
literal 4000
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
GfB^utL_q8S