From b2adb9f00062b660a96915ea665c25845eb768a4 Mon Sep 17 00:00:00 2001 From: Rene Saarsoo Date: Mon, 26 Dec 2022 20:41:09 +0200 Subject: [PATCH] Updates --- asset-manifest.json | 8 ++++---- index.html | 2 +- ... precache-manifest.719b7202eae437868e1d9a53892cf7af.js | 8 ++++---- service-worker.js | 2 +- static/js/main.8dd5c635.chunk.js | 2 -- static/js/main.8dd5c635.chunk.js.map | 1 - static/js/main.d9d6cdce.chunk.js | 2 ++ static/js/main.d9d6cdce.chunk.js.map | 1 + 8 files changed, 13 insertions(+), 13 deletions(-) rename precache-manifest.42466d82a651f42506b6f930e1a34b6b.js => precache-manifest.719b7202eae437868e1d9a53892cf7af.js (73%) delete mode 100644 static/js/main.8dd5c635.chunk.js delete mode 100644 static/js/main.8dd5c635.chunk.js.map create mode 100644 static/js/main.d9d6cdce.chunk.js create mode 100644 static/js/main.d9d6cdce.chunk.js.map diff --git a/asset-manifest.json b/asset-manifest.json index 224ad38..204d669 100644 --- a/asset-manifest.json +++ b/asset-manifest.json @@ -1,14 +1,14 @@ { "files": { "main.css": "/workout-editor/static/css/main.20aa8553.chunk.css", - "main.js": "/workout-editor/static/js/main.8dd5c635.chunk.js", - "main.js.map": "/workout-editor/static/js/main.8dd5c635.chunk.js.map", + "main.js": "/workout-editor/static/js/main.d9d6cdce.chunk.js", + "main.js.map": "/workout-editor/static/js/main.d9d6cdce.chunk.js.map", "runtime-main.js": "/workout-editor/static/js/runtime-main.766fd1ba.js", "runtime-main.js.map": "/workout-editor/static/js/runtime-main.766fd1ba.js.map", "static/js/2.317bd3cc.chunk.js": "/workout-editor/static/js/2.317bd3cc.chunk.js", "static/js/2.317bd3cc.chunk.js.map": "/workout-editor/static/js/2.317bd3cc.chunk.js.map", "index.html": "/workout-editor/index.html", - "precache-manifest.42466d82a651f42506b6f930e1a34b6b.js": "/workout-editor/precache-manifest.42466d82a651f42506b6f930e1a34b6b.js", + "precache-manifest.719b7202eae437868e1d9a53892cf7af.js": "/workout-editor/precache-manifest.719b7202eae437868e1d9a53892cf7af.js", "service-worker.js": "/workout-editor/service-worker.js", "static/css/main.20aa8553.chunk.css.map": "/workout-editor/static/css/main.20aa8553.chunk.css.map", "static/js/2.317bd3cc.chunk.js.LICENSE.txt": "/workout-editor/static/js/2.317bd3cc.chunk.js.LICENSE.txt" @@ -17,6 +17,6 @@ "static/js/runtime-main.766fd1ba.js", "static/js/2.317bd3cc.chunk.js", "static/css/main.20aa8553.chunk.css", - "static/js/main.8dd5c635.chunk.js" + "static/js/main.d9d6cdce.chunk.js" ] } \ No newline at end of file diff --git a/index.html b/index.html index ce26ed9..eacf49d 100644 --- a/index.html +++ b/index.html @@ -1 +1 @@ -Workout editorFork me on GitHub
\ No newline at end of file +Workout editorFork me on GitHub
\ No newline at end of file diff --git a/precache-manifest.42466d82a651f42506b6f930e1a34b6b.js b/precache-manifest.719b7202eae437868e1d9a53892cf7af.js similarity index 73% rename from precache-manifest.42466d82a651f42506b6f930e1a34b6b.js rename to precache-manifest.719b7202eae437868e1d9a53892cf7af.js index 4da2f6e..4272829 100644 --- a/precache-manifest.42466d82a651f42506b6f930e1a34b6b.js +++ b/precache-manifest.719b7202eae437868e1d9a53892cf7af.js @@ -1,10 +1,10 @@ self.__precacheManifest = (self.__precacheManifest || []).concat([ { - "revision": "f3da0e65c0dccc8f34a6f1f0da9d4e15", + "revision": "abc953eebe662776e9f71300b7c3abf8", "url": "/workout-editor/index.html" }, { - "revision": "672b819b8cff43b65f45", + "revision": "4dc054f245084734635e", "url": "/workout-editor/static/css/main.20aa8553.chunk.css" }, { @@ -16,8 +16,8 @@ self.__precacheManifest = (self.__precacheManifest || []).concat([ "url": "/workout-editor/static/js/2.317bd3cc.chunk.js.LICENSE.txt" }, { - "revision": "672b819b8cff43b65f45", - "url": "/workout-editor/static/js/main.8dd5c635.chunk.js" + "revision": "4dc054f245084734635e", + "url": "/workout-editor/static/js/main.d9d6cdce.chunk.js" }, { "revision": "b5e368ec906fbe804156", diff --git a/service-worker.js b/service-worker.js index b7c71f4..3a125b2 100644 --- a/service-worker.js +++ b/service-worker.js @@ -14,7 +14,7 @@ importScripts("https://storage.googleapis.com/workbox-cdn/releases/4.3.1/workbox-sw.js"); importScripts( - "/workout-editor/precache-manifest.42466d82a651f42506b6f930e1a34b6b.js" + "/workout-editor/precache-manifest.719b7202eae437868e1d9a53892cf7af.js" ); self.addEventListener('message', (event) => { diff --git a/static/js/main.8dd5c635.chunk.js b/static/js/main.8dd5c635.chunk.js deleted file mode 100644 index e8ef912..0000000 --- a/static/js/main.8dd5c635.chunk.js +++ /dev/null @@ -1,2 +0,0 @@ -(this["webpackJsonpworkout-editor"]=this["webpackJsonpworkout-editor"]||[]).push([[0],{121:function(n,e,t){"use strict";t.r(e);var r=t(0),a=t.n(r),o=t(56),c=t.n(o),i=(t(67),t(15)),u=t(1),l=t(2),s=t(3);function d(){var n=Object(u.a)(["\n border-bottom-left-radius: 10px;\n border-bottom-right-radius: 10px;\n height: ","%;\n clip-path: ",";\n background: linear-gradient(\n to right,\n ",",\n ","\n );\n"]);return d=function(){return n},n}function f(){var n=Object(u.a)(["\n border-radius: 10px;\n height: ","%;\n background: ",";\n"]);return f=function(){return n},n}function g(){var n=Object(u.a)(["\n display: inline-block;\n vertical-align: bottom;\n margin-right: 0.1%;\n /* exclude 0.1% margin from bar width */\n width: ","%;\n transition: width 0.1s, height 0.1s, background-color 0.1s;\n"]);return g=function(){return n},n}var b=l.a.div(g(),(function(n){return n.durationPercentage-.1})),m={free:"linear-gradient(to top, rgba(204,204,204,1), rgba(255,255,255,0))",Z1:"#7f7f7f",Z2:"#338cff",Z3:"#59bf59",Z4:"#ffcc3f",Z5:"#ff6639",Z6:"#ff330c"},h=Object(l.a)(b)(f(),(function(n){return"free"===n.zone?100:n.intensityPercentage}),(function(n){return m[n.zone]})),p=Object(l.a)(b)(d(),(function(n){return n.maxIntensityPercentage}),(function(n){var e=n.direction,t=n.relativeMinIntensityPercentage;return"up"===e?"polygon(0% 100%, 100% 100%, 100% 0%, 0% ".concat(t,"%)"):function(n){return"polygon(0% 0%, 0% 100%, 100% 100%, 100% ".concat(n,"%)")}(t)}),(function(n){return m[n.startZone]}),(function(n){return m[n.endZone]}));function v(){var n=Object(u.a)(["\n white-space: nowrap;\n overflow: hidden;\n height: 200px;\n border-radius: 5px;\n padding: 5px;\n margin: 10px 0;\n"]);return v=function(){return n},n}var x=l.a.div(v()),E=function(n){var e=n.intervals,t=Object(s.totalDuration)(e),r=Object(s.maximumIntensity)(e);return a.a.createElement(x,null,e.map((function(n,e){return n.intensity instanceof s.RangeIntensity?a.a.createElement(p,Object.assign({key:e},function(n,e,t){var r=Math.min(n.intensity.start,n.intensity.end)/t.value*100,a=Math.max(n.intensity.start,n.intensity.end)/t.value*100;return{durationPercentage:n.duration.seconds/e.seconds*100,maxIntensityPercentage:a,relativeMinIntensityPercentage:100-r/a*100,direction:n.intensity.start0?"".concat(t,"h ").concat(r,"min"):r>0?"".concat(r,"min"):a>0?"".concat(a,"sec"):"-"};function w(){var n=Object(u.a)(["\n display: grid;\n grid-template-columns: 1fr 1fr 1fr;\n"]);return w=function(){return n},n}function y(){var n=Object(u.a)(["\n margin: 1m 0;\n padding: 0;\n list-style: none;\n"]);return y=function(){return n},n}function j(){var n=Object(u.a)(["\n margin: 0;\n font-size: 14px;\n font-weight: normal;\n"]);return j=function(){return n},n}function C(){var n=Object(u.a)(["\n grid-area: xp;\n"]);return C=function(){return n},n}function Q(){var n=Object(u.a)(["\n grid-area: zones;\n"]);return Q=function(){return n},n}function U(){var n=Object(u.a)(["\n grid-area: summary;\n"]);return U=function(){return n},n}function D(){var n=Object(u.a)(['\n display: grid;\n grid-template-columns: 1fr 3fr;\n grid-template-areas:\n "summary zones"\n "xp xp ";\n border: 1px solid #bbb;\n border-radius: 10px;\n padding: 10px;\n font-size: 12px;\n background: rgba(255, 255, 255, 0.6);\n']);return D=function(){return n},n}var k=function(n){return"".concat(Math.round(100*n.value),"%")},T=function(n){var e=n.label,t=n.value;return a.a.createElement(a.a.Fragment,null,a.a.createElement("strong",null,e)," ",t)},M=function(n){return a.a.createElement("li",null,a.a.createElement(T,n))},I=l.a.div(D()),X=l.a.section(U()),F=l.a.section(Q()),Y=l.a.section(C()),G=l.a.h2(j()),Z=l.a.ul(y()),L=Object(l.a)(Z)(w()),K=function(n,e){var t=Math.ceil(n/20),r=e.seconds/60/60,a=Math.round(t/r);return"equivalent to riding ".concat(t," km at ").concat(a," km/h")},R=function(n){var e=n.workout,t=Object(s.stats)(e),r=t.totalDuration,o=t.averageIntensity,c=t.normalizedIntensity,i=t.tss,u=t.xp,l=t.zones;return a.a.createElement(I,null,a.a.createElement(X,null,a.a.createElement(G,null,"Summary"),a.a.createElement(Z,null,a.a.createElement(M,{label:"Duration:",value:O(r)}),a.a.createElement(M,{label:"Average intensity:",value:k(o)}),a.a.createElement(M,{label:"Normalized intensity:",value:k(c)}),a.a.createElement(M,{label:"TSS:",value:Math.round(i)}))),a.a.createElement(F,null,a.a.createElement(G,null,"Zone distribution"),a.a.createElement(L,null,l.map((function(n){return a.a.createElement(M,{key:n.name,label:n.name,value:O(n.duration)})})))),a.a.createElement(Y,null,a.a.createElement(T,{label:"Zwift XP:",value:"".concat(u," (").concat(K(u,r),")")}),";"))};function B(){var n=Object(u.a)(["\n color: red;\n background: #fee;\n border-radius: 10px;\n border: 2px solid red;\n padding: 5px;\n margin: 10px 0;\n"]);return B=function(){return n},n}var z=l.a.p(B()),S=t(60),H=t.n(S);function J(){var n=Object(u.a)(['\n font-family: source-code-pro, Menlo, Monaco, Consolas, "Courier New", monospace;\n font-size: 14px;\n line-height: 1.3;\n border: 1px inset #bbb;\n border-radius: 3px;\n background: #fff;\n\n code.keyword {\n font-weight: bold;\n }\n code.duration {\n color: #681caf;\n }\n code.intensity {\n color: #af391c;\n }\n code.cadence {\n color: #86af1c;\n }\n code.range {\n color: #888;\n }\n code.comment {\n font-style: italic;\n color: #888;\n .duration {\n color: #8d67af;\n }\n }\n code.comment-start {\n font-weight: bold;\n font-style: italic;\n color: #888;\n }\n code.parse-error {\n background-color: rgba(252, 152, 152, 0.5);\n border-radius: 4px;\n }\n code.validation-error {\n background-color: rgba(252, 209, 152, 0.5);\n border-radius: 4px;\n }\n']);return J=function(){return n},n}var N=Object(l.a)(H.a).attrs({padding:10})(J()),P=function(n){var e=n.value,t=n.onValueChange,o=n.error,c=Object(r.useCallback)((function(n){return function(n){return n.replace(/^(Name|Author|Description|Warmup|Rest|Interval|Cooldown|FreeRide):/gm,"$&").replace(/-?(\d{1,2}:)?\d{1,2}:\d{1,2}/g,"$&").replace(/\d+%/g,"$&").replace(/\d+rpm/g,"$&").replace(/\.\./g,"$&").replace(/@(.*?)$/gm,"@$1")}(o?function(n,e){var t=new RegExp("^((?:[^\\n]*?\\n){".concat(e.loc.row,"})([^\\n]*?)\\n")),r=e instanceof s.ValidationError?"validation-error":"parse-error";return n.replace(t,"$1$2\n"))}(n,o):n)}),[o]);return a.a.createElement(N,{value:e,onValueChange:t,highlight:c})};function W(){var n=Object(u.a)(["\n border: 1px solid #bbb;\n border-radius: 3px;\n padding: 10px;\n"]);return W=function(){return n},n}function V(){var n=Object(u.a)(["\n font-weight: normal;\n font-size: 16px;\n"]);return V=function(){return n},n}var q=l.a.h2(V()),A=l.a.pre(W()),$=function(n){var e=n.workout;return a.a.createElement("div",null,a.a.createElement(q,null,"Generated Zwift workout file (.zwo):"),a.a.createElement(A,null,Object(s.generateZwo)(e)))},_=t(61),nn=t.n(_);function en(){var n=Object(u.a)(["\n display: inline-block;\n color: #cc2222;\n opacity: 0.7;\n font-size: 20px;\n font-weight: bold;\n transform: rotate(20deg) translate(-15px, -8px);\n"]);return en=function(){return n},n}function tn(){var n=Object(u.a)(["\n margin-right: 0.5em;\n vertical-align: bottom;\n"]);return tn=function(){return n},n}function rn(){var n=Object(u.a)(["\n font-weight: normal;\n"]);return rn=function(){return n},n}var an=l.a.h1(rn()),on=l.a.img.attrs({src:nn.a,width:45,height:45})(tn()),cn=l.a.span(en()),un=function(){return a.a.createElement(an,null,a.a.createElement(on,null),"Workout editor ",a.a.createElement(cn,null,"beta"))};function ln(){var n=Object(u.a)(["\n font-size: 12px;\n text-align: center;\n margin-top: 3em;\n border-top: 1px solid #eee;\n padding-top: 1em;\n color: gray;\n"]);return ln=function(){return n},n}var sn=l.a.p(ln()),dn=function(){return a.a.createElement(sn,null,"Built by Rene Saarsoo. \xb7 Graphics inspired by ",a.a.createElement("a",{href:"https://whatsonzwift.com/workouts/"},"What's on Zwift?"),"\xa0\xb7 Sweat provided by ",a.a.createElement("a",{href:"https://zwift.com"},"Zwift")," :-)")};function fn(){var n=Object(u.a)(["\n max-width: 800px;\n margin: 0 auto;\n"]);return fn=function(){return n},n}var gn=l.a.div(fn());function bn(){var n=Object(r.useState)(!1),e=Object(i.a)(n,2),t=e[0],o=e[1],c=Object(r.useState)(""),u=Object(i.a)(c,2),l=u[0],d=u[1],f=Object(r.useState)(Object(s.parse)("")),g=Object(i.a)(f,2),b=g[0],m=g[1],h=Object(r.useState)(void 0),p=Object(i.a)(h,2),v=p[0],x=p[1],O=Object(r.useCallback)((function(n){d(n);try{m(Object(s.parse)(n)),x(void 0)}catch(e){if(!(e instanceof s.ParseError||e instanceof s.ValidationError))throw e;x(e)}}),[d,m,x]);return Object(r.useEffect)((function(){t?function(n){localStorage.setItem("workout-editor-text",n)}(l):(o(!0),O(function(){var n=localStorage.getItem("workout-editor-text");return n&&""!==n.trim()?n:"Name: Sample workout\nDescription: Try changing it, and see what happens below.\n\nWarmup: 10:00 30%..75%\nInterval: 15:00 100% 90rpm\n @ 00:00 Start off easy\n @ 01:00 Settle into rhythm\n @ 07:30 Half way through\n @ 14:00 Final minute, stay strong!\nRest: 10:00 75%\nFreeRide: 20:00\n @ 00:10 Just have some fun, riding as you wish\nCooldown: 10:00 70%..30%\n"}()))}),[l,O,o,t]),a.a.createElement(gn,null,a.a.createElement(un,null),a.a.createElement(P,{onValueChange:O,value:l,error:v}),a.a.createElement(E,{intervals:b.intervals}),v&&a.a.createElement(z,null,v.message),a.a.createElement(R,{workout:b}),a.a.createElement($,{workout:b}),a.a.createElement(dn,null))}c.a.render(a.a.createElement(a.a.StrictMode,null,a.a.createElement(bn,null)),document.getElementById("root"))},61:function(n,e){n.exports=""},62:function(n,e,t){n.exports=t(121)},67:function(n,e,t){},77:function(n,e){},79:function(n,e){}},[[62,1,2]]]); -//# sourceMappingURL=main.8dd5c635.chunk.js.map \ No newline at end of file diff --git a/static/js/main.8dd5c635.chunk.js.map b/static/js/main.8dd5c635.chunk.js.map deleted file mode 100644 index bab2999..0000000 --- a/static/js/main.8dd5c635.chunk.js.map +++ /dev/null @@ -1 +0,0 @@ -{"version":3,"sources":["components/Bar.tsx","components/WorkoutPlot.tsx","components/formatDuration.ts","components/WorkoutStats.tsx","components/ErrorMessage.tsx","components/CodeEditor.tsx","components/ZwoOutput.tsx","components/AppTitle.tsx","components/Credits.tsx","App.tsx","storage.ts","index.tsx","logo.png"],"names":["BaseBar","styled","div","props","durationPercentage","zoneColorsMap","free","Z1","Z2","Z3","Z4","Z5","Z6","Bar","zone","intensityPercentage","RangeBar","maxIntensityPercentage","direction","relativeMinIntensityPercentage","middle","createDownPolygon","startZone","endZone","Plot","WorkoutPlot","intervals","workoutDuration","totalDuration","maxIntensity","maximumIntensity","map","interval","i","intensity","RangeIntensity","key","minIntensityPercentage","Math","min","start","end","value","max","duration","seconds","intensityValueToZoneType","toRangeBarProps","toBarProps","formatDuration","hours","floor","minutes","splitDuration","formatIntensity","round","StatsLine","label","StatsListItem","Container","SummarySection","section","ZonesSection","XpSection","Header","h2","List","ul","ZoneList","xpEquivalent","xp","distanceInKm","ceil","durationInHours","speed","WorkoutStats","workout","stats","averageIntensity","normalizedIntensity","tss","zones","name","ErrorMessage","p","BaseCodeEditor","Editor","attrs","padding","CodeEditor","onValueChange","error","highlightFn","useCallback","code","replace","highlight","regex","RegExp","loc","row","className","ValidationError","highlightErrorLine","ZwoCode","pre","ZwoOutput","generateZwo","Title","h1","Logo","img","src","logo","width","height","Beta","span","AppTitle","Text","Credits","href","AppContainer","App","useState","loaded","setLoaded","text","setText","parse","setWorkout","undefined","setError","onChange","e","ParseError","useEffect","localStorage","setItem","saveWorkout","getItem","trim","loadWorkout","message","ReactDOM","render","StrictMode","document","getElementById","module","exports"],"mappings":"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","file":"static/js/main.8dd5c635.chunk.js","sourcesContent":["import styled from \"styled-components\";\nimport { ZoneType } from \"zwiftout\";\n\ntype BaseBarProps = {\n // Percentage of total workout length\n durationPercentage: number;\n};\n\nconst BaseBar = styled.div`\n display: inline-block;\n vertical-align: bottom;\n margin-right: 0.1%;\n /* exclude 0.1% margin from bar width */\n width: ${(props) => props.durationPercentage - 0.1}%;\n transition: width 0.1s, height 0.1s, background-color 0.1s;\n`;\n\nexport type BarProps = BaseBarProps & {\n // Percentage of maximum intensity in workout\n intensityPercentage: number;\n zone: ZoneType;\n};\n\nconst zoneColorsMap: Record = {\n free: \"linear-gradient(to top, rgba(204,204,204,1), rgba(255,255,255,0))\",\n Z1: \"#7f7f7f\",\n Z2: \"#338cff\",\n Z3: \"#59bf59\",\n Z4: \"#ffcc3f\",\n Z5: \"#ff6639\",\n Z6: \"#ff330c\",\n};\n\nexport const Bar = styled(BaseBar)`\n border-radius: 10px;\n height: ${(props) => (props.zone === \"free\" ? 100 : props.intensityPercentage)}%;\n background: ${(props) => zoneColorsMap[props.zone]};\n`;\n\nexport type RangeBarProps = BaseBarProps & {\n // Percentage of maximum intensity in workout\n maxIntensityPercentage: number;\n // Percentage relative to `maxIntensityPercentage`\n relativeMinIntensityPercentage: number;\n startZone: ZoneType;\n endZone: ZoneType;\n direction: \"up\" | \"down\";\n};\n\nconst createUpPolygon = (middle: number) => `polygon(0% 100%, 100% 100%, 100% 0%, 0% ${middle}%)`;\nconst createDownPolygon = (middle: number) => `polygon(0% 0%, 0% 100%, 100% 100%, 100% ${middle}%)`;\n\nexport const RangeBar = styled(BaseBar)`\n border-bottom-left-radius: 10px;\n border-bottom-right-radius: 10px;\n height: ${(props) => props.maxIntensityPercentage}%;\n clip-path: ${({ direction, relativeMinIntensityPercentage }) =>\n direction === \"up\"\n ? createUpPolygon(relativeMinIntensityPercentage)\n : createDownPolygon(relativeMinIntensityPercentage)};\n background: linear-gradient(\n to right,\n ${(props) => zoneColorsMap[props.startZone]},\n ${(props) => zoneColorsMap[props.endZone]}\n );\n`;\n","import React from \"react\";\nimport styled from \"styled-components\";\nimport {\n Interval,\n Intensity,\n Duration,\n totalDuration,\n maximumIntensity,\n RangeIntensity,\n intensityValueToZoneType,\n} from \"zwiftout\";\nimport { BarProps, Bar, RangeBar, RangeBarProps } from \"./Bar\";\n\nconst Plot = styled.div`\n white-space: nowrap;\n overflow: hidden;\n height: 200px;\n border-radius: 5px;\n padding: 5px;\n margin: 10px 0;\n`;\n\nconst toBarProps = (interval: Interval, workoutDuration: Duration, maxIntensity: Intensity): BarProps => ({\n zone: interval.intensity.zone,\n durationPercentage: (interval.duration.seconds / workoutDuration.seconds) * 100,\n intensityPercentage: (interval.intensity.value / maxIntensity.value) * 100,\n});\n\nconst toRangeBarProps = (interval: Interval, workoutDuration: Duration, maxIntensity: Intensity): RangeBarProps => {\n const minIntensityPercentage =\n (Math.min(interval.intensity.start, interval.intensity.end) / maxIntensity.value) * 100;\n const maxIntensityPercentage =\n (Math.max(interval.intensity.start, interval.intensity.end) / maxIntensity.value) * 100;\n\n return {\n durationPercentage: (interval.duration.seconds / workoutDuration.seconds) * 100,\n maxIntensityPercentage: maxIntensityPercentage,\n relativeMinIntensityPercentage: 100 - (minIntensityPercentage / maxIntensityPercentage) * 100,\n direction: interval.intensity.start < interval.intensity.end ? \"up\" : \"down\",\n startZone: intensityValueToZoneType(interval.intensity.start),\n endZone: intensityValueToZoneType(interval.intensity.end),\n };\n};\n\nexport const WorkoutPlot: React.FC<{ intervals: Interval[] }> = ({ intervals }) => {\n const workoutDuration = totalDuration(intervals);\n const maxIntensity = maximumIntensity(intervals);\n\n return (\n \n {intervals.map((interval, i) => {\n if (interval.intensity instanceof RangeIntensity) {\n return ;\n } else {\n return ;\n }\n })}\n \n );\n};\n","import { Duration } from \"zwiftout\";\n\nconst splitDuration = (duration: Duration) => ({\n hours: Math.floor(duration.seconds / 60 / 60),\n minutes: Math.floor(duration.seconds / 60) % 60,\n seconds: duration.seconds % 60,\n});\n\nexport const formatDuration = (duration: Duration): string => {\n const { hours, minutes, seconds } = splitDuration(duration);\n if (hours > 0) {\n return `${hours}h ${minutes}min`;\n } else if (minutes > 0) {\n return `${minutes}min`;\n } else if (seconds > 0) {\n return `${seconds}sec`;\n } else {\n return `-`;\n }\n};\n","import React from \"react\";\nimport { stats, Workout, Intensity, Duration } from \"zwiftout\";\nimport { formatDuration } from \"./formatDuration\";\nimport styled from \"styled-components\";\n\nconst formatIntensity = (intensity: Intensity): string => `${Math.round(intensity.value * 100)}%`;\n\ntype StatsProps = { label: string; value: string | number };\n\nconst StatsLine: React.FC = ({ label, value }) => (\n <>\n {label} {value}\n \n);\n\nconst StatsListItem: React.FC = (props) => (\n
  • \n \n
  • \n);\n\nconst Container = styled.div`\n display: grid;\n grid-template-columns: 1fr 3fr;\n grid-template-areas:\n \"summary zones\"\n \"xp xp \";\n border: 1px solid #bbb;\n border-radius: 10px;\n padding: 10px;\n font-size: 12px;\n background: rgba(255, 255, 255, 0.6);\n`;\nconst SummarySection = styled.section`\n grid-area: summary;\n`;\nconst ZonesSection = styled.section`\n grid-area: zones;\n`;\nconst XpSection = styled.section`\n grid-area: xp;\n`;\n\nconst Header = styled.h2`\n margin: 0;\n font-size: 14px;\n font-weight: normal;\n`;\nconst List = styled.ul`\n margin: 1m 0;\n padding: 0;\n list-style: none;\n`;\nconst ZoneList = styled(List)`\n display: grid;\n grid-template-columns: 1fr 1fr 1fr;\n`;\n\nconst xpEquivalent = (xp: number, totalDuration: Duration): string => {\n const distanceInKm = Math.ceil(xp / 20);\n const durationInHours = totalDuration.seconds / 60 / 60;\n const speed = Math.round(distanceInKm / durationInHours);\n return `equivalent to riding ${distanceInKm} km at ${speed} km/h`;\n};\n\nexport const WorkoutStats: React.FC<{ workout: Workout }> = ({ workout }) => {\n const { totalDuration, averageIntensity, normalizedIntensity, tss, xp, zones } = stats(workout);\n\n return (\n \n \n
    Summary
    \n \n \n \n \n \n \n
    \n \n
    Zone distribution
    \n \n {zones.map((zone) => (\n \n ))}\n \n
    \n \n ;\n \n
    \n );\n};\n","import styled from \"styled-components\";\n\nexport const ErrorMessage = styled.p`\n color: red;\n background: #fee;\n border-radius: 10px;\n border: 2px solid red;\n padding: 5px;\n margin: 10px 0;\n`;\n","import React, { useCallback } from \"react\";\nimport Editor from \"react-simple-code-editor\";\nimport styled from \"styled-components\";\nimport { ValidationError, ZwiftoutException } from \"zwiftout\";\n\nconst highlight = (code: string): string => {\n return code\n .replace(/^(Name|Author|Description|Warmup|Rest|Interval|Cooldown|FreeRide):/gm, \"$&\")\n .replace(/-?(\\d{1,2}:)?\\d{1,2}:\\d{1,2}/g, \"$&\")\n .replace(/\\d+%/g, \"$&\")\n .replace(/\\d+rpm/g, \"$&\")\n .replace(/\\.\\./g, \"$&\")\n .replace(/@(.*?)$/gm, \"@$1\");\n};\n\nexport const BaseCodeEditor = styled(Editor).attrs({ padding: 10 })`\n font-family: source-code-pro, Menlo, Monaco, Consolas, \"Courier New\", monospace;\n font-size: 14px;\n line-height: 1.3;\n border: 1px inset #bbb;\n border-radius: 3px;\n background: #fff;\n\n code.keyword {\n font-weight: bold;\n }\n code.duration {\n color: #681caf;\n }\n code.intensity {\n color: #af391c;\n }\n code.cadence {\n color: #86af1c;\n }\n code.range {\n color: #888;\n }\n code.comment {\n font-style: italic;\n color: #888;\n .duration {\n color: #8d67af;\n }\n }\n code.comment-start {\n font-weight: bold;\n font-style: italic;\n color: #888;\n }\n code.parse-error {\n background-color: rgba(252, 152, 152, 0.5);\n border-radius: 4px;\n }\n code.validation-error {\n background-color: rgba(252, 209, 152, 0.5);\n border-radius: 4px;\n }\n`;\n\nconst highlightErrorLine = (code: string, error: ZwiftoutException): string => {\n const regex = new RegExp(`^((?:[^\\\\n]*?\\\\n){${error.loc.row}})([^\\\\n]*?)\\\\n`);\n const className = error instanceof ValidationError ? \"validation-error\" : \"parse-error\";\n return code.replace(regex, `$1$2\\n`);\n};\n\ninterface CodeEditorProps {\n value: string;\n onValueChange: (value: string) => void;\n error: ZwiftoutException | undefined;\n}\n\nexport const CodeEditor: React.FC = ({ value, onValueChange, error }) => {\n const highlightFn = useCallback((code: string) => highlight(error ? highlightErrorLine(code, error) : code), [error]);\n return ;\n};\n","import React from \"react\";\nimport { Workout, generateZwo } from \"zwiftout\";\nimport styled from \"styled-components\";\n\nconst Header = styled.h2`\n font-weight: normal;\n font-size: 16px;\n`;\n\nconst ZwoCode = styled.pre`\n border: 1px solid #bbb;\n border-radius: 3px;\n padding: 10px;\n`;\n\nexport const ZwoOutput: React.FC<{ workout: Workout }> = ({ workout }) => {\n return (\n
    \n
    Generated Zwift workout file (.zwo):
    \n {generateZwo(workout)}\n
    \n );\n};\n","import React from \"react\";\nimport styled from \"styled-components\";\nimport logo from \"../logo.png\";\n\nconst Title = styled.h1`\n font-weight: normal;\n`;\n\nconst Logo = styled.img.attrs({ src: logo, width: 45, height: 45 })`\n margin-right: 0.5em;\n vertical-align: bottom;\n`;\n\nconst Beta = styled.span`\n display: inline-block;\n color: #cc2222;\n opacity: 0.7;\n font-size: 20px;\n font-weight: bold;\n transform: rotate(20deg) translate(-15px, -8px);\n`;\n\nexport const AppTitle: React.FC<{}> = () => (\n \n <Logo />\n Workout editor <Beta>beta</Beta>\n \n);\n","import React from \"react\";\nimport styled from \"styled-components\";\n\nconst Text = styled.p`\n font-size: 12px;\n text-align: center;\n margin-top: 3em;\n border-top: 1px solid #eee;\n padding-top: 1em;\n color: gray;\n`;\n\nexport const Credits: React.FC<{}> = () => (\n \n Built by Rene Saarsoo. · Graphics inspired by What's on Zwift?\n  · Sweat provided by Zwift :-)\n \n);\n","import React, { useState, useCallback, useEffect } from \"react\";\nimport { WorkoutPlot } from \"./components/WorkoutPlot\";\nimport { WorkoutStats } from \"./components/WorkoutStats\";\nimport { parse, ParseError, ValidationError, ZwiftoutException } from \"zwiftout\";\nimport { ErrorMessage } from \"./components/ErrorMessage\";\nimport styled from \"styled-components\";\nimport { CodeEditor } from \"./components/CodeEditor\";\nimport { ZwoOutput } from \"./components/ZwoOutput\";\nimport { AppTitle } from \"./components/AppTitle\";\nimport { Credits } from \"./components/Credits\";\nimport { loadWorkout, saveWorkout } from \"./storage\";\n\nconst AppContainer = styled.div`\n max-width: 800px;\n margin: 0 auto;\n`;\n\nexport function App() {\n const [loaded, setLoaded] = useState(false);\n const [text, setText] = useState(\"\");\n const [workout, setWorkout] = useState(parse(\"\"));\n const [error, setError] = useState(undefined);\n\n const onChange = useCallback(\n (value: string) => {\n setText(value);\n try {\n setWorkout(parse(value));\n setError(undefined);\n } catch (e) {\n if (e instanceof ParseError || e instanceof ValidationError) {\n setError(e);\n } else {\n throw e;\n }\n }\n },\n [setText, setWorkout, setError],\n );\n\n useEffect(() => {\n if (loaded) {\n saveWorkout(text);\n } else {\n setLoaded(true);\n onChange(loadWorkout());\n }\n }, [text, onChange, setLoaded, loaded]);\n\n return (\n \n \n \n \n {error && {error.message}}\n \n \n \n \n );\n}\n","const defaultWorkout = `Name: Sample workout\nDescription: Try changing it, and see what happens below.\n\nWarmup: 10:00 30%..75%\nInterval: 15:00 100% 90rpm\n @ 00:00 Start off easy\n @ 01:00 Settle into rhythm\n @ 07:30 Half way through\n @ 14:00 Final minute, stay strong!\nRest: 10:00 75%\nFreeRide: 20:00\n @ 00:10 Just have some fun, riding as you wish\nCooldown: 10:00 70%..30%\n`;\n\nexport const loadWorkout = (): string => {\n const text = localStorage.getItem(\"workout-editor-text\");\n return (!text || text.trim() === \"\") ? defaultWorkout : text;\n};\n\nexport const saveWorkout = (text: string) => localStorage.setItem(\"workout-editor-text\", text);\n","import React from \"react\";\nimport ReactDOM from \"react-dom\";\nimport \"./index.css\";\nimport { App } from \"./App\";\n\nReactDOM.render(\n \n \n ,\n document.getElementById(\"root\"),\n);\n","module.exports = \"\""],"sourceRoot":""} \ No newline at end of file diff --git a/static/js/main.d9d6cdce.chunk.js b/static/js/main.d9d6cdce.chunk.js new file mode 100644 index 0000000..73d0ec2 --- /dev/null +++ b/static/js/main.d9d6cdce.chunk.js @@ -0,0 +1,2 @@ +(this["webpackJsonpworkout-editor"]=this["webpackJsonpworkout-editor"]||[]).push([[0],{121:function(n,e,t){"use strict";t.r(e);var r=t(0),a=t.n(r),o=t(56),c=t.n(o),i=(t(67),t(15)),u=t(1),l=t(2),s=t(3);function d(){var n=Object(u.a)(["\n border-bottom-left-radius: 10px;\n border-bottom-right-radius: 10px;\n height: ","%;\n clip-path: ",";\n background: linear-gradient(\n to right,\n ",",\n ","\n );\n"]);return d=function(){return n},n}function f(){var n=Object(u.a)(["\n border-radius: 10px;\n height: ","%;\n background: ",";\n"]);return f=function(){return n},n}function g(){var n=Object(u.a)(["\n display: inline-block;\n vertical-align: bottom;\n margin-right: 0.1%;\n /* exclude 0.1% margin from bar width */\n width: ","%;\n transition: width 0.1s, height 0.1s, background-color 0.1s;\n"]);return g=function(){return n},n}var b=l.a.div(g(),(function(n){return n.durationPercentage-.1})),m={free:"linear-gradient(to top, rgba(204,204,204,1), rgba(255,255,255,0))",Z1:"#7f7f7f",Z2:"#338cff",Z3:"#59bf59",Z4:"#ffcc3f",Z5:"#ff6639",Z6:"#ff330c"},h=Object(l.a)(b)(f(),(function(n){return"free"===n.zone?100:n.intensityPercentage}),(function(n){return m[n.zone]})),p=Object(l.a)(b)(d(),(function(n){return n.maxIntensityPercentage}),(function(n){var e=n.direction,t=n.relativeMinIntensityPercentage;return"up"===e?"polygon(0% 100%, 100% 100%, 100% 0%, 0% ".concat(t,"%)"):function(n){return"polygon(0% 0%, 0% 100%, 100% 100%, 100% ".concat(n,"%)")}(t)}),(function(n){return m[n.startZone]}),(function(n){return m[n.endZone]}));function v(){var n=Object(u.a)(["\n white-space: nowrap;\n overflow: hidden;\n height: 200px;\n border-radius: 5px;\n padding: 5px;\n margin: 10px 0;\n"]);return v=function(){return n},n}var x=l.a.div(v()),E=function(n){var e=n.intervals,t=Object(s.totalDuration)(e),r=Object(s.maximumIntensity)(e);return a.a.createElement(x,null,e.map((function(n,e){return n.intensity instanceof s.RangeIntensity?a.a.createElement(p,Object.assign({key:e},function(n,e,t){var r=Math.min(n.intensity.start,n.intensity.end)/t.value*100,a=Math.max(n.intensity.start,n.intensity.end)/t.value*100;return{durationPercentage:n.duration.seconds/e.seconds*100,maxIntensityPercentage:a,relativeMinIntensityPercentage:100-r/a*100,direction:n.intensity.start0?"".concat(t,"h ").concat(r,"min"):r>0?"".concat(r,"min"):a>0?"".concat(a,"sec"):"-"};function w(){var n=Object(u.a)(["\n display: grid;\n grid-template-columns: 1fr 1fr 1fr;\n"]);return w=function(){return n},n}function y(){var n=Object(u.a)(["\n margin: 1m 0;\n padding: 0;\n list-style: none;\n"]);return y=function(){return n},n}function j(){var n=Object(u.a)(["\n margin: 0;\n font-size: 14px;\n font-weight: normal;\n"]);return j=function(){return n},n}function C(){var n=Object(u.a)(["\n grid-area: xp;\n"]);return C=function(){return n},n}function Q(){var n=Object(u.a)(["\n grid-area: zones;\n"]);return Q=function(){return n},n}function U(){var n=Object(u.a)(["\n grid-area: summary;\n"]);return U=function(){return n},n}function D(){var n=Object(u.a)(['\n display: grid;\n grid-template-columns: 1fr 3fr;\n grid-template-areas:\n "summary zones"\n "xp xp ";\n border: 1px solid #bbb;\n border-radius: 10px;\n padding: 10px;\n font-size: 12px;\n background: rgba(255, 255, 255, 0.6);\n']);return D=function(){return n},n}var k=function(n){return"".concat(Math.round(100*n.value),"%")},T=function(n){var e=n.label,t=n.value;return a.a.createElement(a.a.Fragment,null,a.a.createElement("strong",null,e)," ",t)},M=function(n){return a.a.createElement("li",null,a.a.createElement(T,n))},I=l.a.div(D()),X=l.a.section(U()),F=l.a.section(Q()),Y=l.a.section(C()),G=l.a.h2(j()),Z=l.a.ul(y()),L=Object(l.a)(Z)(w()),R=function(n,e){var t=Math.ceil(n/20),r=e.seconds/60/60,a=Math.round(t/r);return"equivalent to riding ".concat(t," km at ").concat(a," km/h")},K=function(n){var e=n.workout,t=Object(s.stats)(e),r=t.totalDuration,o=t.averageIntensity,c=t.normalizedIntensity,i=t.tss,u=t.xp,l=t.zones;return a.a.createElement(I,null,a.a.createElement(X,null,a.a.createElement(G,null,"Summary"),a.a.createElement(Z,null,a.a.createElement(M,{label:"Duration:",value:O(r)}),a.a.createElement(M,{label:"Average intensity:",value:k(o)}),a.a.createElement(M,{label:"Normalized intensity:",value:k(c)}),a.a.createElement(M,{label:"TSS:",value:Math.round(i)}))),a.a.createElement(F,null,a.a.createElement(G,null,"Zone distribution"),a.a.createElement(L,null,l.map((function(n){return a.a.createElement(M,{key:n.name,label:n.name,value:O(n.duration)})})))),a.a.createElement(Y,null,a.a.createElement(T,{label:"Zwift XP:",value:"".concat(u," (").concat(R(u,r),")")}),";"))};function B(){var n=Object(u.a)(["\n color: red;\n background: #fee;\n border-radius: 10px;\n border: 2px solid red;\n padding: 5px;\n margin: 10px 0;\n"]);return B=function(){return n},n}var z=l.a.p(B()),S=t(60),H=t.n(S);function J(){var n=Object(u.a)(['\n font-family: source-code-pro, Menlo, Monaco, Consolas, "Courier New", monospace;\n font-size: 14px;\n line-height: 1.3;\n border: 1px inset #bbb;\n border-radius: 3px;\n background: #fff;\n\n code.keyword {\n font-weight: bold;\n }\n code.duration {\n color: #681caf;\n }\n code.intensity {\n color: #af391c;\n }\n code.cadence {\n color: #86af1c;\n }\n code.range {\n color: #888;\n }\n code.comment {\n font-style: italic;\n color: #888;\n .duration {\n color: #8d67af;\n }\n }\n code.comment-start {\n font-weight: bold;\n font-style: italic;\n color: #888;\n }\n code.parse-error {\n background-color: rgba(252, 152, 152, 0.5);\n border-radius: 4px;\n }\n code.validation-error {\n background-color: rgba(252, 209, 152, 0.5);\n border-radius: 4px;\n }\n']);return J=function(){return n},n}var N=Object(l.a)(H.a).attrs({padding:10})(J()),P=function(n){var e=n.value,t=n.onValueChange,o=n.error,c=Object(r.useCallback)((function(n){return function(n){return n.replace(/^(Name|Author|Description|Warmup|Rest|Interval|Cooldown|FreeRide|Ramp):/gm,"$&").replace(/-?(\d{1,2}:)?\d{1,2}:\d{1,2}/g,"$&").replace(/\d+%/g,"$&").replace(/\d+rpm/g,"$&").replace(/\.\./g,"$&").replace(/@(.*?)$/gm,"@$1")}(o?function(n,e){var t=new RegExp("^((?:[^\\n]*?\\n){".concat(e.loc.row,"})([^\\n]*?)\\n")),r=e instanceof s.ValidationError?"validation-error":"parse-error";return n.replace(t,"$1$2\n"))}(n,o):n)}),[o]);return a.a.createElement(N,{value:e,onValueChange:t,highlight:c})};function W(){var n=Object(u.a)(["\n border: 1px solid #bbb;\n border-radius: 3px;\n padding: 10px;\n"]);return W=function(){return n},n}function V(){var n=Object(u.a)(["\n font-weight: normal;\n font-size: 16px;\n"]);return V=function(){return n},n}var q=l.a.h2(V()),A=l.a.pre(W()),$=function(n){var e=n.workout;return a.a.createElement("div",null,a.a.createElement(q,null,"Generated Zwift workout file (.zwo):"),a.a.createElement(A,null,Object(s.generateZwo)(e)))},_=t(61),nn=t.n(_);function en(){var n=Object(u.a)(["\n display: inline-block;\n color: #cc2222;\n opacity: 0.7;\n font-size: 20px;\n font-weight: bold;\n transform: rotate(20deg) translate(-15px, -8px);\n"]);return en=function(){return n},n}function tn(){var n=Object(u.a)(["\n margin-right: 0.5em;\n vertical-align: bottom;\n"]);return tn=function(){return n},n}function rn(){var n=Object(u.a)(["\n font-weight: normal;\n"]);return rn=function(){return n},n}var an=l.a.h1(rn()),on=l.a.img.attrs({src:nn.a,width:45,height:45})(tn()),cn=l.a.span(en()),un=function(){return a.a.createElement(an,null,a.a.createElement(on,null),"Workout editor ",a.a.createElement(cn,null,"beta"))};function ln(){var n=Object(u.a)(["\n font-size: 12px;\n text-align: center;\n margin-top: 3em;\n border-top: 1px solid #eee;\n padding-top: 1em;\n color: gray;\n"]);return ln=function(){return n},n}var sn=l.a.p(ln()),dn=function(){return a.a.createElement(sn,null,"Built by Rene Saarsoo. \xb7 Graphics inspired by ",a.a.createElement("a",{href:"https://whatsonzwift.com/workouts/"},"What's on Zwift?"),"\xa0\xb7 Sweat provided by ",a.a.createElement("a",{href:"https://zwift.com"},"Zwift")," :-)")},fn=function(){var n,e;return null!==(n=null!==(e=function(){if(document.location.hash)return decodeURIComponent(document.location.hash.slice(1))}())&&void 0!==e?e:function(){var n=localStorage.getItem("workout-editor-text");return n&&""!==n.trim()?n:void 0}())&&void 0!==n?n:"Name: Sample workout\nDescription: Try changing it, and see what happens below.\n\nWarmup: 10:00 30%..75%\nInterval: 15:00 100% 90rpm\n @ 00:00 Start off easy\n @ 01:00 Settle into rhythm\n @ 07:30 Half way through\n @ 14:00 Final minute, stay strong!\nRest: 10:00 75%\nFreeRide: 20:00\n @ 00:10 Just have some fun, riding as you wish\nCooldown: 10:00 70%..30%\n"};function gn(){var n=Object(u.a)(["\n max-width: 800px;\n margin: 0 auto;\n"]);return gn=function(){return n},n}var bn=l.a.div(gn());function mn(){var n=Object(r.useState)(!1),e=Object(i.a)(n,2),t=e[0],o=e[1],c=Object(r.useState)(""),u=Object(i.a)(c,2),l=u[0],d=u[1],f=Object(r.useState)(Object(s.parse)("")),g=Object(i.a)(f,2),b=g[0],m=g[1],h=Object(r.useState)(void 0),p=Object(i.a)(h,2),v=p[0],x=p[1],O=Object(r.useCallback)((function(n){d(n);try{m(Object(s.parse)(n)),x(void 0)}catch(e){if(!(e instanceof s.ParseError||e instanceof s.ValidationError))throw e;x(e)}}),[d,m,x]);return Object(r.useEffect)((function(){t?function(n){localStorage.setItem("workout-editor-text",n)}(l):(o(!0),O(fn()))}),[l,O,o,t]),a.a.createElement(bn,null,a.a.createElement(un,null),a.a.createElement(P,{onValueChange:O,value:l,error:v}),a.a.createElement(E,{intervals:b.intervals}),v&&a.a.createElement(z,null,v.message),a.a.createElement(K,{workout:b}),a.a.createElement($,{workout:b}),a.a.createElement(dn,null))}c.a.render(a.a.createElement(a.a.StrictMode,null,a.a.createElement(mn,null)),document.getElementById("root"))},61:function(n,e){n.exports=""},62:function(n,e,t){n.exports=t(121)},67:function(n,e,t){},77:function(n,e){},79:function(n,e){}},[[62,1,2]]]); +//# sourceMappingURL=main.d9d6cdce.chunk.js.map \ No newline at end of file diff --git a/static/js/main.d9d6cdce.chunk.js.map b/static/js/main.d9d6cdce.chunk.js.map new file mode 100644 index 0000000..4b74d8e --- /dev/null +++ b/static/js/main.d9d6cdce.chunk.js.map @@ -0,0 +1 @@ +{"version":3,"sources":["components/Bar.tsx","components/WorkoutPlot.tsx","components/formatDuration.ts","components/WorkoutStats.tsx","components/ErrorMessage.tsx","components/CodeEditor.tsx","components/ZwoOutput.tsx","components/AppTitle.tsx","components/Credits.tsx","storage.ts","App.tsx","index.tsx","logo.png"],"names":["BaseBar","styled","div","props","durationPercentage","zoneColorsMap","free","Z1","Z2","Z3","Z4","Z5","Z6","Bar","zone","intensityPercentage","RangeBar","maxIntensityPercentage","direction","relativeMinIntensityPercentage","middle","createDownPolygon","startZone","endZone","Plot","WorkoutPlot","intervals","workoutDuration","totalDuration","maxIntensity","maximumIntensity","map","interval","i","intensity","RangeIntensity","key","minIntensityPercentage","Math","min","start","end","value","max","duration","seconds","intensityValueToZoneType","toRangeBarProps","toBarProps","formatDuration","hours","floor","minutes","splitDuration","formatIntensity","round","StatsLine","label","StatsListItem","Container","SummarySection","section","ZonesSection","XpSection","Header","h2","List","ul","ZoneList","xpEquivalent","xp","distanceInKm","ceil","durationInHours","speed","WorkoutStats","workout","stats","averageIntensity","normalizedIntensity","tss","zones","name","ErrorMessage","p","BaseCodeEditor","Editor","attrs","padding","CodeEditor","onValueChange","error","highlightFn","useCallback","code","replace","highlight","regex","RegExp","loc","row","className","ValidationError","highlightErrorLine","ZwoCode","pre","ZwoOutput","generateZwo","Title","h1","Logo","img","src","logo","width","height","Beta","span","AppTitle","Text","Credits","href","loadWorkout","document","location","hash","decodeURIComponent","slice","loadFromUrlHash","text","localStorage","getItem","trim","undefined","loadFromLocalStorage","AppContainer","App","useState","loaded","setLoaded","setText","parse","setWorkout","setError","onChange","e","ParseError","useEffect","setItem","saveWorkout","message","ReactDOM","render","StrictMode","getElementById","module","exports"],"mappings":"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","file":"static/js/main.d9d6cdce.chunk.js","sourcesContent":["import styled from \"styled-components\";\nimport { ZoneType } from \"zwiftout\";\n\ntype BaseBarProps = {\n // Percentage of total workout length\n durationPercentage: number;\n};\n\nconst BaseBar = styled.div`\n display: inline-block;\n vertical-align: bottom;\n margin-right: 0.1%;\n /* exclude 0.1% margin from bar width */\n width: ${(props) => props.durationPercentage - 0.1}%;\n transition: width 0.1s, height 0.1s, background-color 0.1s;\n`;\n\nexport type BarProps = BaseBarProps & {\n // Percentage of maximum intensity in workout\n intensityPercentage: number;\n zone: ZoneType;\n};\n\nconst zoneColorsMap: Record = {\n free: \"linear-gradient(to top, rgba(204,204,204,1), rgba(255,255,255,0))\",\n Z1: \"#7f7f7f\",\n Z2: \"#338cff\",\n Z3: \"#59bf59\",\n Z4: \"#ffcc3f\",\n Z5: \"#ff6639\",\n Z6: \"#ff330c\",\n};\n\nexport const Bar = styled(BaseBar)`\n border-radius: 10px;\n height: ${(props) => (props.zone === \"free\" ? 100 : props.intensityPercentage)}%;\n background: ${(props) => zoneColorsMap[props.zone]};\n`;\n\nexport type RangeBarProps = BaseBarProps & {\n // Percentage of maximum intensity in workout\n maxIntensityPercentage: number;\n // Percentage relative to `maxIntensityPercentage`\n relativeMinIntensityPercentage: number;\n startZone: ZoneType;\n endZone: ZoneType;\n direction: \"up\" | \"down\";\n};\n\nconst createUpPolygon = (middle: number) => `polygon(0% 100%, 100% 100%, 100% 0%, 0% ${middle}%)`;\nconst createDownPolygon = (middle: number) => `polygon(0% 0%, 0% 100%, 100% 100%, 100% ${middle}%)`;\n\nexport const RangeBar = styled(BaseBar)`\n border-bottom-left-radius: 10px;\n border-bottom-right-radius: 10px;\n height: ${(props) => props.maxIntensityPercentage}%;\n clip-path: ${({ direction, relativeMinIntensityPercentage }) =>\n direction === \"up\"\n ? createUpPolygon(relativeMinIntensityPercentage)\n : createDownPolygon(relativeMinIntensityPercentage)};\n background: linear-gradient(\n to right,\n ${(props) => zoneColorsMap[props.startZone]},\n ${(props) => zoneColorsMap[props.endZone]}\n );\n`;\n","import React from \"react\";\nimport styled from \"styled-components\";\nimport {\n Interval,\n Intensity,\n Duration,\n totalDuration,\n maximumIntensity,\n RangeIntensity,\n intensityValueToZoneType,\n} from \"zwiftout\";\nimport { BarProps, Bar, RangeBar, RangeBarProps } from \"./Bar\";\n\nconst Plot = styled.div`\n white-space: nowrap;\n overflow: hidden;\n height: 200px;\n border-radius: 5px;\n padding: 5px;\n margin: 10px 0;\n`;\n\nconst toBarProps = (interval: Interval, workoutDuration: Duration, maxIntensity: Intensity): BarProps => ({\n zone: interval.intensity.zone,\n durationPercentage: (interval.duration.seconds / workoutDuration.seconds) * 100,\n intensityPercentage: (interval.intensity.value / maxIntensity.value) * 100,\n});\n\nconst toRangeBarProps = (interval: Interval, workoutDuration: Duration, maxIntensity: Intensity): RangeBarProps => {\n const minIntensityPercentage =\n (Math.min(interval.intensity.start, interval.intensity.end) / maxIntensity.value) * 100;\n const maxIntensityPercentage =\n (Math.max(interval.intensity.start, interval.intensity.end) / maxIntensity.value) * 100;\n\n return {\n durationPercentage: (interval.duration.seconds / workoutDuration.seconds) * 100,\n maxIntensityPercentage: maxIntensityPercentage,\n relativeMinIntensityPercentage: 100 - (minIntensityPercentage / maxIntensityPercentage) * 100,\n direction: interval.intensity.start < interval.intensity.end ? \"up\" : \"down\",\n startZone: intensityValueToZoneType(interval.intensity.start),\n endZone: intensityValueToZoneType(interval.intensity.end),\n };\n};\n\nexport const WorkoutPlot: React.FC<{ intervals: Interval[] }> = ({ intervals }) => {\n const workoutDuration = totalDuration(intervals);\n const maxIntensity = maximumIntensity(intervals);\n\n return (\n \n {intervals.map((interval, i) => {\n if (interval.intensity instanceof RangeIntensity) {\n return ;\n } else {\n return ;\n }\n })}\n \n );\n};\n","import { Duration } from \"zwiftout\";\n\nconst splitDuration = (duration: Duration) => ({\n hours: Math.floor(duration.seconds / 60 / 60),\n minutes: Math.floor(duration.seconds / 60) % 60,\n seconds: duration.seconds % 60,\n});\n\nexport const formatDuration = (duration: Duration): string => {\n const { hours, minutes, seconds } = splitDuration(duration);\n if (hours > 0) {\n return `${hours}h ${minutes}min`;\n } else if (minutes > 0) {\n return `${minutes}min`;\n } else if (seconds > 0) {\n return `${seconds}sec`;\n } else {\n return `-`;\n }\n};\n","import React from \"react\";\nimport { stats, Workout, Intensity, Duration } from \"zwiftout\";\nimport { formatDuration } from \"./formatDuration\";\nimport styled from \"styled-components\";\n\nconst formatIntensity = (intensity: Intensity): string => `${Math.round(intensity.value * 100)}%`;\n\ntype StatsProps = { label: string; value: string | number };\n\nconst StatsLine: React.FC = ({ label, value }) => (\n <>\n {label} {value}\n \n);\n\nconst StatsListItem: React.FC = (props) => (\n
  • \n \n
  • \n);\n\nconst Container = styled.div`\n display: grid;\n grid-template-columns: 1fr 3fr;\n grid-template-areas:\n \"summary zones\"\n \"xp xp \";\n border: 1px solid #bbb;\n border-radius: 10px;\n padding: 10px;\n font-size: 12px;\n background: rgba(255, 255, 255, 0.6);\n`;\nconst SummarySection = styled.section`\n grid-area: summary;\n`;\nconst ZonesSection = styled.section`\n grid-area: zones;\n`;\nconst XpSection = styled.section`\n grid-area: xp;\n`;\n\nconst Header = styled.h2`\n margin: 0;\n font-size: 14px;\n font-weight: normal;\n`;\nconst List = styled.ul`\n margin: 1m 0;\n padding: 0;\n list-style: none;\n`;\nconst ZoneList = styled(List)`\n display: grid;\n grid-template-columns: 1fr 1fr 1fr;\n`;\n\nconst xpEquivalent = (xp: number, totalDuration: Duration): string => {\n const distanceInKm = Math.ceil(xp / 20);\n const durationInHours = totalDuration.seconds / 60 / 60;\n const speed = Math.round(distanceInKm / durationInHours);\n return `equivalent to riding ${distanceInKm} km at ${speed} km/h`;\n};\n\nexport const WorkoutStats: React.FC<{ workout: Workout }> = ({ workout }) => {\n const { totalDuration, averageIntensity, normalizedIntensity, tss, xp, zones } = stats(workout);\n\n return (\n \n \n
    Summary
    \n \n \n \n \n \n \n
    \n \n
    Zone distribution
    \n \n {zones.map((zone) => (\n \n ))}\n \n
    \n \n ;\n \n
    \n );\n};\n","import styled from \"styled-components\";\n\nexport const ErrorMessage = styled.p`\n color: red;\n background: #fee;\n border-radius: 10px;\n border: 2px solid red;\n padding: 5px;\n margin: 10px 0;\n`;\n","import React, { useCallback } from \"react\";\nimport Editor from \"react-simple-code-editor\";\nimport styled from \"styled-components\";\nimport { ValidationError, ZwiftoutException } from \"zwiftout\";\n\nconst highlight = (code: string): string => {\n return code\n .replace(\n /^(Name|Author|Description|Warmup|Rest|Interval|Cooldown|FreeRide|Ramp):/gm,\n \"$&\",\n )\n .replace(/-?(\\d{1,2}:)?\\d{1,2}:\\d{1,2}/g, \"$&\")\n .replace(/\\d+%/g, \"$&\")\n .replace(/\\d+rpm/g, \"$&\")\n .replace(/\\.\\./g, \"$&\")\n .replace(/@(.*?)$/gm, \"@$1\");\n};\n\nexport const BaseCodeEditor = styled(Editor).attrs({ padding: 10 })`\n font-family: source-code-pro, Menlo, Monaco, Consolas, \"Courier New\", monospace;\n font-size: 14px;\n line-height: 1.3;\n border: 1px inset #bbb;\n border-radius: 3px;\n background: #fff;\n\n code.keyword {\n font-weight: bold;\n }\n code.duration {\n color: #681caf;\n }\n code.intensity {\n color: #af391c;\n }\n code.cadence {\n color: #86af1c;\n }\n code.range {\n color: #888;\n }\n code.comment {\n font-style: italic;\n color: #888;\n .duration {\n color: #8d67af;\n }\n }\n code.comment-start {\n font-weight: bold;\n font-style: italic;\n color: #888;\n }\n code.parse-error {\n background-color: rgba(252, 152, 152, 0.5);\n border-radius: 4px;\n }\n code.validation-error {\n background-color: rgba(252, 209, 152, 0.5);\n border-radius: 4px;\n }\n`;\n\nconst highlightErrorLine = (code: string, error: ZwiftoutException): string => {\n const regex = new RegExp(`^((?:[^\\\\n]*?\\\\n){${error.loc.row}})([^\\\\n]*?)\\\\n`);\n const className = error instanceof ValidationError ? \"validation-error\" : \"parse-error\";\n return code.replace(regex, `$1$2\\n`);\n};\n\ninterface CodeEditorProps {\n value: string;\n onValueChange: (value: string) => void;\n error: ZwiftoutException | undefined;\n}\n\nexport const CodeEditor: React.FC = ({ value, onValueChange, error }) => {\n const highlightFn = useCallback((code: string) => highlight(error ? highlightErrorLine(code, error) : code), [error]);\n return ;\n};\n","import React from \"react\";\nimport { Workout, generateZwo } from \"zwiftout\";\nimport styled from \"styled-components\";\n\nconst Header = styled.h2`\n font-weight: normal;\n font-size: 16px;\n`;\n\nconst ZwoCode = styled.pre`\n border: 1px solid #bbb;\n border-radius: 3px;\n padding: 10px;\n`;\n\nexport const ZwoOutput: React.FC<{ workout: Workout }> = ({ workout }) => {\n return (\n
    \n
    Generated Zwift workout file (.zwo):
    \n {generateZwo(workout)}\n
    \n );\n};\n","import React from \"react\";\nimport styled from \"styled-components\";\nimport logo from \"../logo.png\";\n\nconst Title = styled.h1`\n font-weight: normal;\n`;\n\nconst Logo = styled.img.attrs({ src: logo, width: 45, height: 45 })`\n margin-right: 0.5em;\n vertical-align: bottom;\n`;\n\nconst Beta = styled.span`\n display: inline-block;\n color: #cc2222;\n opacity: 0.7;\n font-size: 20px;\n font-weight: bold;\n transform: rotate(20deg) translate(-15px, -8px);\n`;\n\nexport const AppTitle: React.FC<{}> = () => (\n \n <Logo />\n Workout editor <Beta>beta</Beta>\n \n);\n","import React from \"react\";\nimport styled from \"styled-components\";\n\nconst Text = styled.p`\n font-size: 12px;\n text-align: center;\n margin-top: 3em;\n border-top: 1px solid #eee;\n padding-top: 1em;\n color: gray;\n`;\n\nexport const Credits: React.FC<{}> = () => (\n \n Built by Rene Saarsoo. · Graphics inspired by What's on Zwift?\n  · Sweat provided by Zwift :-)\n \n);\n","const defaultWorkout = `Name: Sample workout\nDescription: Try changing it, and see what happens below.\n\nWarmup: 10:00 30%..75%\nInterval: 15:00 100% 90rpm\n @ 00:00 Start off easy\n @ 01:00 Settle into rhythm\n @ 07:30 Half way through\n @ 14:00 Final minute, stay strong!\nRest: 10:00 75%\nFreeRide: 20:00\n @ 00:10 Just have some fun, riding as you wish\nCooldown: 10:00 70%..30%\n`;\n\nconst loadFromUrlHash = (): string | undefined => {\n if (document.location.hash) {\n // Skip the leading '#' when decoding\n return decodeURIComponent(document.location.hash.slice(1));\n }\n return undefined;\n};\n\nconst loadFromLocalStorage = (): string | undefined => {\n const text = localStorage.getItem(\"workout-editor-text\");\n return !text || text.trim() === \"\" ? undefined : text;\n};\n\nexport const loadWorkout = (): string => {\n return loadFromUrlHash() ?? loadFromLocalStorage() ?? defaultWorkout;\n};\n\nexport const saveWorkout = (text: string) => localStorage.setItem(\"workout-editor-text\", text);\n","import React, { useState, useCallback, useEffect } from \"react\";\nimport { WorkoutPlot } from \"./components/WorkoutPlot\";\nimport { WorkoutStats } from \"./components/WorkoutStats\";\nimport { parse, ParseError, ValidationError, ZwiftoutException } from \"zwiftout\";\nimport { ErrorMessage } from \"./components/ErrorMessage\";\nimport styled from \"styled-components\";\nimport { CodeEditor } from \"./components/CodeEditor\";\nimport { ZwoOutput } from \"./components/ZwoOutput\";\nimport { AppTitle } from \"./components/AppTitle\";\nimport { Credits } from \"./components/Credits\";\nimport { loadWorkout, saveWorkout } from \"./storage\";\n\nconst AppContainer = styled.div`\n max-width: 800px;\n margin: 0 auto;\n`;\n\nexport function App() {\n const [loaded, setLoaded] = useState(false);\n const [text, setText] = useState(\"\");\n const [workout, setWorkout] = useState(parse(\"\"));\n const [error, setError] = useState(undefined);\n\n const onChange = useCallback(\n (value: string) => {\n setText(value);\n try {\n setWorkout(parse(value));\n setError(undefined);\n } catch (e) {\n if (e instanceof ParseError || e instanceof ValidationError) {\n setError(e);\n } else {\n throw e;\n }\n }\n },\n [setText, setWorkout, setError],\n );\n\n useEffect(() => {\n if (loaded) {\n saveWorkout(text);\n } else {\n setLoaded(true);\n onChange(loadWorkout());\n }\n }, [text, onChange, setLoaded, loaded]);\n\n return (\n \n \n \n \n {error && {error.message}}\n \n \n \n \n );\n}\n","import React from \"react\";\nimport ReactDOM from \"react-dom\";\nimport \"./index.css\";\nimport { App } from \"./App\";\n\nReactDOM.render(\n \n \n ,\n document.getElementById(\"root\"),\n);\n","module.exports = \"\""],"sourceRoot":""} \ No newline at end of file