wsba-hockey 1.1.9__py3-none-any.whl → 1.2.1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (144) hide show
  1. wsba_hockey/tools/scraping.py +146 -170
  2. wsba_hockey/tools/utils/__init__.py +0 -1
  3. wsba_hockey/tools/utils/shared.py +14 -389
  4. wsba_hockey/tools/xg_model.py +6 -1
  5. wsba_hockey/wsba_main.py +47 -14
  6. {wsba_hockey-1.1.9.dist-info → wsba_hockey-1.2.1.dist-info}/METADATA +16 -15
  7. wsba_hockey-1.2.1.dist-info/RECORD +15 -0
  8. wsba_hockey/api/api/index.py +0 -162
  9. wsba_hockey/data_pipelines.py +0 -247
  10. wsba_hockey/evidence/weakside-breakout/node_modules/duckdb/vendor.py +0 -146
  11. wsba_hockey/evidence/weakside-breakout/node_modules/flatted/python/flatted.py +0 -149
  12. wsba_hockey/evidence/weakside-breakout/node_modules/flatted/python/test.py +0 -63
  13. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/gyp_main.py +0 -45
  14. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSNew.py +0 -367
  15. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSProject.py +0 -206
  16. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSSettings.py +0 -1270
  17. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSSettings_test.py +0 -1547
  18. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSToolFile.py +0 -59
  19. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSUserFile.py +0 -153
  20. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSUtil.py +0 -271
  21. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/MSVSVersion.py +0 -574
  22. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/__init__.py +0 -690
  23. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/common.py +0 -661
  24. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/common_test.py +0 -78
  25. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/easy_xml.py +0 -165
  26. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/easy_xml_test.py +0 -109
  27. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/flock_tool.py +0 -55
  28. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/__init__.py +0 -0
  29. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/analyzer.py +0 -808
  30. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/android.py +0 -1173
  31. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/cmake.py +0 -1321
  32. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/compile_commands_json.py +0 -120
  33. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/dump_dependency_json.py +0 -103
  34. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/eclipse.py +0 -464
  35. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/gypd.py +0 -89
  36. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/gypsh.py +0 -58
  37. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/make.py +0 -2714
  38. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/msvs.py +0 -3981
  39. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/msvs_test.py +0 -44
  40. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/ninja.py +0 -2936
  41. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/ninja_test.py +0 -55
  42. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/xcode.py +0 -1394
  43. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/generator/xcode_test.py +0 -25
  44. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/input.py +0 -3130
  45. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/input_test.py +0 -98
  46. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/mac_tool.py +0 -771
  47. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/msvs_emulation.py +0 -1271
  48. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/ninja_syntax.py +0 -174
  49. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/simple_copy.py +0 -61
  50. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/win_tool.py +0 -374
  51. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/xcode_emulation.py +0 -1939
  52. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/xcode_ninja.py +0 -302
  53. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/xcodeproj_file.py +0 -3197
  54. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/pylib/gyp/xml_fix.py +0 -65
  55. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/test_gyp.py +0 -261
  56. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/tools/graphviz.py +0 -102
  57. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/tools/pretty_gyp.py +0 -156
  58. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/tools/pretty_sln.py +0 -181
  59. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/gyp/tools/pretty_vcproj.py +0 -339
  60. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/test/fixtures/test-charmap.py +0 -31
  61. wsba_hockey/evidence/weakside-breakout/node_modules/node-gyp/update-gyp.py +0 -64
  62. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/gyp_main.py +0 -45
  63. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSNew.py +0 -367
  64. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSProject.py +0 -206
  65. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSSettings.py +0 -1270
  66. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSSettings_test.py +0 -1547
  67. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSToolFile.py +0 -59
  68. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSUserFile.py +0 -153
  69. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSUtil.py +0 -271
  70. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/MSVSVersion.py +0 -574
  71. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/__init__.py +0 -666
  72. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/common.py +0 -654
  73. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/common_test.py +0 -78
  74. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/easy_xml.py +0 -165
  75. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/easy_xml_test.py +0 -109
  76. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/flock_tool.py +0 -55
  77. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/__init__.py +0 -0
  78. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/analyzer.py +0 -808
  79. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/android.py +0 -1173
  80. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/cmake.py +0 -1321
  81. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/compile_commands_json.py +0 -120
  82. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/dump_dependency_json.py +0 -103
  83. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/eclipse.py +0 -464
  84. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/gypd.py +0 -89
  85. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/gypsh.py +0 -58
  86. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/make.py +0 -2518
  87. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/msvs.py +0 -3978
  88. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/msvs_test.py +0 -44
  89. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/ninja.py +0 -2936
  90. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/ninja_test.py +0 -55
  91. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/xcode.py +0 -1394
  92. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/generator/xcode_test.py +0 -25
  93. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/input.py +0 -3137
  94. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/input_test.py +0 -98
  95. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/mac_tool.py +0 -771
  96. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/msvs_emulation.py +0 -1271
  97. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/ninja_syntax.py +0 -174
  98. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/simple_copy.py +0 -61
  99. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/win_tool.py +0 -374
  100. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/xcode_emulation.py +0 -1939
  101. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/xcode_ninja.py +0 -302
  102. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/xcodeproj_file.py +0 -3197
  103. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/pylib/gyp/xml_fix.py +0 -65
  104. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/setup.py +0 -42
  105. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/test_gyp.py +0 -260
  106. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/tools/graphviz.py +0 -102
  107. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/tools/pretty_gyp.py +0 -156
  108. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/tools/pretty_sln.py +0 -181
  109. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/gyp/tools/pretty_vcproj.py +0 -339
  110. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/test/fixtures/test-charmap.py +0 -31
  111. wsba_hockey/evidence/weakside-breakout/node_modules/sqlite3/node_modules/node-gyp/update-gyp.py +0 -46
  112. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/duos/app.py +0 -210
  113. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/duos/calc.py +0 -163
  114. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/game_stats/app.py +0 -401
  115. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/game_stats/name_fix.py +0 -47
  116. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/goalie/app.py +0 -101
  117. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/goalie/plot.py +0 -71
  118. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/goalie/rink_plot.py +0 -245
  119. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/heatmaps/app.py +0 -108
  120. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/heatmaps/plot.py +0 -95
  121. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/heatmaps/rink_plot.py +0 -245
  122. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/line-combos/app.py +0 -245
  123. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/line-combos/plot.py +0 -275
  124. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/line-combos/rink_plot.py +0 -245
  125. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/matchups/app.py +0 -145
  126. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/matchups/plot.py +0 -79
  127. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/matchups/rink_plot.py +0 -245
  128. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/pbp/app.py +0 -406
  129. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/pbp/plot.py +0 -79
  130. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/pbp/rink_plot.py +0 -245
  131. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/skater/app.py +0 -110
  132. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/skater/plot.py +0 -59
  133. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/skater/rink_plot.py +0 -245
  134. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/team_heatmaps/app.py +0 -103
  135. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/team_heatmaps/plot.py +0 -95
  136. wsba_hockey/evidence/weakside-breakout/wsba_nhl_apps/wsba_nhl_apps/team_heatmaps/rink_plot.py +0 -245
  137. wsba_hockey/flask/app.py +0 -77
  138. wsba_hockey/tools/utils/config.py +0 -14
  139. wsba_hockey/tools/utils/save_pages.py +0 -133
  140. wsba_hockey/workspace.py +0 -28
  141. wsba_hockey-1.1.9.dist-info/RECORD +0 -148
  142. {wsba_hockey-1.1.9.dist-info → wsba_hockey-1.2.1.dist-info}/WHEEL +0 -0
  143. {wsba_hockey-1.1.9.dist-info → wsba_hockey-1.2.1.dist-info}/licenses/LICENSE +0 -0
  144. {wsba_hockey-1.1.9.dist-info → wsba_hockey-1.2.1.dist-info}/top_level.txt +0 -0
@@ -1,245 +0,0 @@
1
-
2
- import numpy as np
3
- import plotly.graph_objects as go
4
- import io
5
- import base64
6
- import requests as rs
7
- from PIL import Image
8
-
9
- def rink(setting = "full", vertical = False):
10
- '''
11
- Function to plot rink in Plotly. Takes 2 arguments :
12
-
13
- setting : full (default) for full ice, offense positive half of the ice, ozone positive quarter of ice, defense for negative half of the ice, dzone for negative quarter of the ice, and neutral for the neutral zone
14
- vertical : True if you want a vertical rink, False (default) is for an horizontal rink
15
-
16
- '''
17
-
18
- def faceoff_circle(x, y, outer=True):
19
- segments = []
20
- theta = np.linspace(0, 2*np.pi, 300)
21
- if outer:
22
- # Outer circle
23
- x_outer = x + 15*np.cos(theta)
24
- y_outer = y + 15*np.sin(theta)
25
- outer_circle = go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(width=2, color='red'), showlegend=False, hoverinfo='skip')
26
-
27
- segments.append(outer_circle)
28
-
29
- # Inner circle
30
- x_inner = x + np.cos(theta)
31
- y_inner = y + np.sin(theta)
32
- inner_circle = go.Scatter(x=x_inner, y=y_inner, mode='lines', fill='toself', fillcolor='rgba(255, 0, 0, 0.43)', line=dict(color='rgba(255, 0, 0, 1)', width=2), showlegend=False, hoverinfo='skip')
33
-
34
- segments.append(inner_circle)
35
-
36
- return segments #segments
37
-
38
- fig = go.Figure()
39
-
40
- if vertical :
41
- setting_dict = {
42
- "full" : [-101, 101],
43
- "offense" : [0, 101],
44
- "ozone" : [25, 101],
45
- "defense" : [-101, 0],
46
- "dzone" : [-101, -25],
47
- "neutral" : [-25,25]
48
- }
49
- fig.update_layout(xaxis=dict(range=[-42.6, 42.6], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"), yaxis=dict(range=setting_dict[setting], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"),
50
- showlegend=False, autosize=True, template="plotly_white")
51
- fig.update_yaxes(
52
- scaleanchor="x",
53
- scaleratio=1,
54
- )
55
- def goal_crease(flip=1):
56
- x_seq = np.linspace(-4, 4, 100)
57
- x_goal = np.concatenate(([-4], x_seq, [4]))
58
- y_goal = flip * np.concatenate(([89], 83 + x_seq**2/4**2*1.5, [89]))
59
- goal_crease = go.Scatter(x=x_goal, y=y_goal, fill='toself', fillcolor='rgba(173, 216, 230, 0.3)', line=dict(color='red'))
60
- return goal_crease
61
-
62
- # Outer circle
63
- theta = np.linspace(0, 2*np.pi, 300)
64
- x_outer = 15 * np.cos(theta)
65
- y_outer = 15 * np.sin(theta)
66
- fig.add_trace(go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(color='royalblue', width=2), showlegend=False, hoverinfo='skip'))
67
- # Inner circle
68
- theta2 = np.linspace(np.pi/2, 3*np.pi/2, 300)
69
- x_inner = 42.5 + 10 * np.cos(theta2)
70
- y_inner = 10 * np.sin(theta2)
71
- fig.add_trace(go.Scatter(x=x_inner, y=y_inner, mode='lines', line=dict(color='red', width=2), showlegend=False, hoverinfo='skip'))
72
- # Rink boundaries
73
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=25, x1=42.5, y1=26, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
74
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=-25, x1=42.5, y1=-26, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
75
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=-0.5, x1=42.5, y1=0.5, line=dict(color='red', width=2), fillcolor='red')
76
-
77
- # Goal crease
78
- fig.add_trace(goal_crease())
79
- fig.add_trace(goal_crease(-1))
80
- # Goal lines
81
- goal_line_extreme = 42.5 - 28 + np.sqrt(28**2 - (28-11)**2)
82
- fig.add_shape(type='line', xref='x', yref='y', x0=-goal_line_extreme, y0=89, x1=goal_line_extreme, y1=89, line=dict(color='red', width=2))
83
- fig.add_shape(type='line', xref='x', yref='y', x0=-goal_line_extreme, y0=-89, x1=goal_line_extreme, y1=-89, line=dict(color='red', width=2))
84
-
85
- # Faceoff circles
86
- fig.add_traces(faceoff_circle(-22, 69))
87
- fig.add_traces(faceoff_circle(22, 69))
88
- fig.add_traces(faceoff_circle(-22, -69))
89
- fig.add_traces(faceoff_circle(22, -69))
90
- fig.add_traces(faceoff_circle(-22, -20, False))
91
- fig.add_traces(faceoff_circle(22, -20, False))
92
- fig.add_traces(faceoff_circle(-22, 20, False))
93
- fig.add_traces(faceoff_circle(22, 20, False))
94
-
95
- # Sidelines
96
- theta_lines = np.linspace(0, np.pi/2, 20)
97
- x_lines1 = np.concatenate(([-42.5], -42.5 + 28 - 28*np.cos(theta_lines), 42.5 - 28 + 28*np.cos(np.flip(theta_lines))))
98
- y_lines1 = np.concatenate(([15], 72 + 28*np.sin(theta_lines), 72 + 28*np.sin(np.flip(theta_lines))))
99
- x_lines2 = np.concatenate(([-42.5], -42.5 + 28 - 28*np.cos(theta_lines), 42.5 - 28 + 28*np.cos(np.flip(theta_lines))))
100
- y_lines2 = np.concatenate(([15], -72 - 28*np.sin(theta_lines), -72 - 28*np.sin(np.flip(theta_lines))))
101
- fig.add_trace(go.Scatter(x=x_lines1, y=y_lines1, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
102
- fig.add_trace(go.Scatter(x=x_lines2, y=y_lines2, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
103
- fig.add_shape(type='line', xref='x', yref='y', x0=42.5, y0=-72.5, x1=42.5, y1=72.5, line=dict(color='white', width=2))
104
- fig.add_shape(type='line', xref='x', yref='y', x0=-42.5, y0=-72.5, x1=-42.5, y1=72.5, line=dict(color='white', width=2))
105
-
106
- # Add goals
107
- goal_width = 6 # feet
108
- goal_depth = 4 # feet
109
-
110
- # Top goal
111
- fig.add_shape(
112
- type="rect",
113
- xref="x",
114
- yref="y",
115
- x0=-goal_width / 2,
116
- y0=89,
117
- x1=goal_width / 2,
118
- y1=89 + goal_depth,
119
- line=dict(color="red", width=2),
120
- )
121
- # Bottom goal
122
- fig.add_shape(
123
- type="rect",
124
- xref="x",
125
- yref="y",
126
- x0=-goal_width / 2,
127
- y0=-89 - goal_depth,
128
- x1=goal_width / 2,
129
- y1=-89,
130
- line=dict(color="red", width=2),
131
- )
132
-
133
- else :
134
- setting_dict = {
135
- "full" : [-101, 101],
136
- "offense" : [0, 101],
137
- "ozone" : [25, 101],
138
- "defense" : [-101, 0],
139
- "dzone" : [-101, -25]
140
- }
141
- fig.update_layout(xaxis=dict(range=setting_dict[setting], showgrid=False, zeroline=False, showticklabels=False), yaxis=dict(range=[-42.6, 42.6], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"),
142
- showlegend=True, autosize =True, template="plotly_white")
143
- fig.update_yaxes(
144
- scaleanchor="x",
145
- scaleratio=1,
146
- )
147
- def goal_crease(flip=1):
148
- y_seq = np.linspace(-4, 4, 100)
149
- y_goal = np.concatenate(([-4], y_seq, [4]))
150
- x_goal = flip * np.concatenate(([89], 83 + y_seq**2/4**2*1.5, [89]))
151
- goal_crease = go.Scatter(x=x_goal, y=y_goal, fill='toself', fillcolor='rgba(173, 216, 230, 0.3)', line=dict(color='red'), showlegend=False, hoverinfo='skip')
152
- return goal_crease
153
-
154
- # Outer circle
155
- theta = np.linspace(0, 2 * np.pi, 300)
156
- x_outer = 15 * np.sin(theta)
157
- y_outer = 15 * np.cos(theta)
158
- fig.add_trace(go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(color='royalblue', width=2), showlegend=False, hoverinfo='skip'))
159
- # Inner circle
160
- theta2 = np.linspace(3 * np.pi / 2, np.pi / 2, 300) # Update theta2 to rotate the plot by 180 degrees
161
- x_inner = 10 * np.sin(theta2) # Update x_inner to rotate the plot by 180 degrees
162
- y_inner = -42.5 - 10 * np.cos(theta2) # Update y_inner to rotate the plot by 180 degrees
163
- fig.add_trace(go.Scatter(x=x_inner, y=y_inner, mode='lines', line=dict(color='red', width=2), showlegend=False, hoverinfo='skip'))
164
-
165
- # Rink boundaries
166
- fig.add_shape(type='rect', xref='x', yref='y', x0=25, y0=-42.5, x1=26, y1=42.5, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
167
- fig.add_shape(type='rect', xref='x', yref='y', x0=-25, y0=-42.5, x1=-26, y1=42.5, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
168
- fig.add_shape(type='rect', xref='x', yref='y', x0=-0.5, y0=-42.5, x1=0.5, y1=42.5, line=dict(color='red', width=2), fillcolor='red')
169
- # Goal crease
170
- fig.add_trace(goal_crease())
171
- fig.add_trace(goal_crease(-1))
172
- # Goal lines
173
- goal_line_extreme = 42.5 - 28 + np.sqrt(28 ** 2 - (28 - 11) ** 2)
174
- fig.add_shape(type='line', xref='x', yref='y', x0=89, y0=-goal_line_extreme, x1=89, y1=goal_line_extreme, line=dict(color='red', width=2))
175
- fig.add_shape(type='line', xref='x', yref='y', x0=-89, y0=-goal_line_extreme, x1=-89, y1=goal_line_extreme, line=dict(color='red', width=2))
176
- # Faceoff circles
177
- fig.add_traces(faceoff_circle(-69, -22))
178
- fig.add_traces(faceoff_circle(-69, 22))
179
- fig.add_traces(faceoff_circle(69, -22))
180
- fig.add_traces(faceoff_circle(69, 22))
181
- fig.add_traces(faceoff_circle(-20, -22, False))
182
- fig.add_traces(faceoff_circle(-20, 22, False))
183
- fig.add_traces(faceoff_circle(20, -22, False))
184
- fig.add_traces(faceoff_circle(20, 22, False))
185
-
186
- # Sidelines
187
- theta_lines = np.linspace(0, np.pi / 2, 20)
188
- x_lines1 = np.concatenate(([15], 72 + 28 * np.sin(theta_lines), 72 + 28 * np.sin(np.flip(theta_lines))))
189
- y_lines1 = np.concatenate(([-42.5], -42.5 + 28 - 28 * np.cos(theta_lines), 42.5 - 28 + 28 * np.cos(np.flip(theta_lines))))
190
- x_lines2 = np.concatenate(([15], -72 - 28 * np.sin(theta_lines), -72 - 28 * np.sin(np.flip(theta_lines))))
191
- y_lines2 = np.concatenate(([-42.5], -42.5 + 28 - 28 * np.cos(theta_lines), 42.5 - 28 + 28 * np.cos(np.flip(theta_lines))))
192
- fig.add_trace(go.Scatter(x=x_lines1, y=y_lines1, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
193
- fig.add_trace(go.Scatter(x=x_lines2, y=y_lines2, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
194
- fig.add_shape(type='line', xref='x', yref='y', x0=-72.5, y0=-42.5, x1=72.5, y1=-42.5, line=dict(color='white', width=2))
195
- fig.add_shape(type='line', xref='x', yref='y', x0=-72.5, y0=42.5, x1=72.5, y1=42.5, line=dict(color='white', width=2))
196
-
197
- # Add goals
198
- goal_width = 6 # feet
199
- goal_depth = 4 # feet
200
-
201
- # Right goal
202
- fig.add_shape(
203
- type="rect",
204
- xref="x",
205
- yref="y",
206
- x0=89,
207
- y0=-goal_width / 2,
208
- x1=89 + goal_depth,
209
- y1=goal_width / 2,
210
- line=dict(color="red", width=2),
211
- )
212
- # Left goal
213
- fig.add_shape(
214
- type="rect",
215
- xref="x",
216
- yref="y",
217
- x0=-89 - goal_depth,
218
- y0=-goal_width / 2,
219
- x1=-89,
220
- y1=goal_width / 2,
221
- line=dict(color="red", width=2),
222
- )
223
-
224
- # Add logo
225
- logo = Image.open(rs.get('https://weakside-breakout.s3.us-east-2.amazonaws.com/utils/wsba.png',stream=True).raw)
226
-
227
- fig.add_layout_image(
228
- dict(
229
- source=logo,
230
- xref="x",
231
- yref="y",
232
- x=-12,
233
- y=12,
234
- sizex=24,
235
- sizey=24,
236
- sizing="stretch",
237
- opacity=1)
238
- )
239
-
240
- #Set background to transparent
241
- fig.update_layout(
242
- paper_bgcolor="rgba(0,0,0,0)",
243
- plot_bgcolor="rgba(0,0,0,0)"
244
- )
245
- return fig
@@ -1,110 +0,0 @@
1
- import pandas as pd
2
- import plotly.express as px
3
- import plot as wsba_plt
4
- import numpy as np
5
- from urllib.parse import *
6
- from shiny import *
7
- from shinywidgets import output_widget, render_widget
8
-
9
- app_ui = ui.page_fluid(
10
- ui.tags.style(
11
- "body {background:#09090b"
12
- "}"
13
- ),
14
- output_widget("plot_skater"),
15
- )
16
-
17
- def server(input, output, session):
18
- @output()
19
- @render_widget
20
- def plot_skater():
21
- #Retreive query parameters
22
- search = session.input[".clientdata_url_search"]()
23
- query = parse_qs(urlparse(search).query)
24
-
25
- print(query)
26
- #If no input data is provided automatically provide a select skater and plot all 5v5 fenwick shots
27
- defaults = {
28
- 'skater':['8473419'],
29
- 'season':['20182019'],
30
- 'team':['BOS'],
31
- 'event_type':['missed-shot,shot-on-goal,goal'],
32
- 'strength_state':['5v5'],
33
- 'season_type':['2']
34
- }
35
-
36
- for key in defaults.keys():
37
- if key not in query.keys():
38
- query.update({key:defaults[key]})
39
-
40
- #Iterate through query and parse params with multiple selections
41
- for param in query.keys():
42
- q_string = query[param][0]
43
- query[param] = q_string.split(',')
44
-
45
- print(query)
46
- #Determine which season to load based on the input
47
- season = query['season'][0]
48
- #Load appropriate dataframe
49
- df = pd.read_parquet(f'https://weakside-breakout.s3.us-east-2.amazonaws.com/pbp/{season}.parquet')
50
-
51
- #Prepare dataframe for plotting based on URL parameters
52
- df = df.loc[(df['event_player_1_id'].astype(str).str.replace('.0','').isin(query['skater']))&(df['season'].astype(str).isin(query['season']))&(df['event_team_abbr'].astype(str).isin(query['team']))&(df['season_type'].astype(str).isin(query['season_type']))].replace({np.nan: None})
53
- df = wsba_plt.prep(df,events=query['event_type'],strengths=query['strength_state'])
54
-
55
- #Return empty rink if no data exists else continue
56
- if df.empty:
57
- return wsba_plt.wsba_rink(setting='offense',vertical=True)
58
- else:
59
- colors = wsba_plt.colors(df)
60
- rink = wsba_plt.wsba_rink(setting='offense',vertical=True)
61
-
62
- plot = px.scatter(df,
63
- x='x', y='y',
64
- size='size',
65
- color='Team',
66
- color_discrete_map=colors,
67
- hover_name='Description',
68
- hover_data=['Event Num.', 'Period', 'Time (in seconds)',
69
- 'Strength',
70
- 'Away Score', 'Home Score', 'x', 'y',
71
- 'Event Distance from Attacking Net',
72
- 'Event Angle to Attacking Net',
73
- 'xG'])
74
-
75
- for trace in plot.data:
76
- rink.add_trace(trace)
77
-
78
- player = df['event_player_1_name'].to_list()[0]
79
- season = int(season[0:4])
80
- team = df['event_team_abbr'].to_list()[0]
81
- strengths = 'All Situations' if len(query['strength_state']) == 4 else query['strength_state']
82
- span = 'Regular Season' if query['season_type'][0]=='2' else 'Playoffs'
83
-
84
- rink.add_annotation(
85
- text=f'{season}-{season+1}, {span}, {team}',
86
- xref="paper",
87
- yref="paper",
88
- xanchor='center',
89
- yanchor='top',
90
- font=dict(color='white'),
91
- x=0.5,
92
- y=0,
93
- showarrow=False
94
- )
95
-
96
- return rink.update_layout(
97
- title=dict(
98
- text=f'{player} Fenwick Shots at {strengths}',
99
- x=0.5, y=0.96,
100
- xanchor='center',
101
- yanchor='top',
102
- font=dict(color='white')
103
- ),
104
-
105
- hoverlabel=dict(
106
- font_size=10
107
- )
108
- )
109
-
110
- app = App(app_ui, server)
@@ -1,59 +0,0 @@
1
- import pandas as pd
2
- import numpy as np
3
- import matplotlib.pyplot as plt
4
- import plotly.graph_objects as go
5
- import rink_plot
6
-
7
- event_markers = {
8
- 'faceoff':'X',
9
- 'hit':'P',
10
- 'blocked-shot':'v',
11
- 'missed-shot':'o',
12
- 'shot-on-goal':'D',
13
- 'goal':'*',
14
- 'giveaway':'1',
15
- 'takeaway':'2',
16
- 'penalty':''
17
- }
18
-
19
- def wsba_rink(setting='full', vertical=False):
20
- return rink_plot.rink(setting=setting, vertical=vertical)
21
-
22
- def colors(df):
23
- team = list(df['event_team_abbr'])[0]
24
- season = list(df['season'])[0]
25
- team_data = pd.read_csv('https://weakside-breakout.s3.us-east-2.amazonaws.com/info/nhl_teaminfo.csv')
26
-
27
- team_info ={
28
- team: list(team_data.loc[team_data['WSBA']==f'{team}{season}','Primary Color'])[0],
29
- }
30
-
31
- return team_info
32
-
33
- def prep(df,events,strengths):
34
- df = df.loc[(df['event_type'].isin(events))]
35
-
36
- df['strength_state'] = np.where(df['strength_state'].isin(['5v5','5v4','4v5']),df['strength_state'],'Other')
37
- if strengths != 'all':
38
- df = df.loc[((df['strength_state'].isin(strengths)))]
39
-
40
- df = df.fillna(0)
41
- df['size'] = np.where(df['xG']<=0,40,df['xG']*400)
42
-
43
- df['marker'] = df['event_type'].replace(event_markers)
44
-
45
- df['Description'] = df['description']
46
- df['Team'] = df['event_team_abbr']
47
- df['Event Num.'] = df['event_num']
48
- df['Period'] = df['period']
49
- df['Time (in seconds)'] = df['seconds_elapsed']
50
- df['Strength'] = df['strength_state']
51
- df['Away Score'] = df['away_score']
52
- df['Home Score'] = df['home_score']
53
- df['x'] = np.where(df['x_adj']<0,df['y_adj'],-df['y_adj'])
54
- df['y'] = abs(df['x_adj'])
55
- df['Event Distance from Attacking Net'] = df['event_distance']
56
- df['Event Angle to Attacking Net'] = df['event_angle']
57
- df['xG'] = df['xG']*100
58
-
59
- return df
@@ -1,245 +0,0 @@
1
-
2
- import numpy as np
3
- import plotly.graph_objects as go
4
- import io
5
- import base64
6
- import requests as rs
7
- from PIL import Image
8
-
9
- def rink(setting = "full", vertical = False):
10
- '''
11
- Function to plot rink in Plotly. Takes 2 arguments :
12
-
13
- setting : full (default) for full ice, offense positive half of the ice, ozone positive quarter of ice, defense for negative half of the ice, dzone for negative quarter of the ice, and neutral for the neutral zone
14
- vertical : True if you want a vertical rink, False (default) is for an horizontal rink
15
-
16
- '''
17
-
18
- def faceoff_circle(x, y, outer=True):
19
- segments = []
20
- theta = np.linspace(0, 2*np.pi, 300)
21
- if outer:
22
- # Outer circle
23
- x_outer = x + 15*np.cos(theta)
24
- y_outer = y + 15*np.sin(theta)
25
- outer_circle = go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(width=2, color='red'), showlegend=False, hoverinfo='skip')
26
-
27
- segments.append(outer_circle)
28
-
29
- # Inner circle
30
- x_inner = x + np.cos(theta)
31
- y_inner = y + np.sin(theta)
32
- inner_circle = go.Scatter(x=x_inner, y=y_inner, mode='lines', fill='toself', fillcolor='rgba(255, 0, 0, 0.43)', line=dict(color='rgba(255, 0, 0, 1)', width=2), showlegend=False, hoverinfo='skip')
33
-
34
- segments.append(inner_circle)
35
-
36
- return segments #segments
37
-
38
- fig = go.Figure()
39
-
40
- if vertical :
41
- setting_dict = {
42
- "full" : [-101, 101],
43
- "offense" : [0, 101],
44
- "ozone" : [25, 101],
45
- "defense" : [-101, 0],
46
- "dzone" : [-101, -25],
47
- "neutral" : [-25,25]
48
- }
49
- fig.update_layout(xaxis=dict(range=[-42.6, 42.6], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"), yaxis=dict(range=setting_dict[setting], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"),
50
- showlegend=False, autosize=True, template="plotly_white")
51
- fig.update_yaxes(
52
- scaleanchor="x",
53
- scaleratio=1,
54
- )
55
- def goal_crease(flip=1):
56
- x_seq = np.linspace(-4, 4, 100)
57
- x_goal = np.concatenate(([-4], x_seq, [4]))
58
- y_goal = flip * np.concatenate(([89], 83 + x_seq**2/4**2*1.5, [89]))
59
- goal_crease = go.Scatter(x=x_goal, y=y_goal, fill='toself', fillcolor='rgba(173, 216, 230, 0.3)', line=dict(color='red'))
60
- return goal_crease
61
-
62
- # Outer circle
63
- theta = np.linspace(0, 2*np.pi, 300)
64
- x_outer = 15 * np.cos(theta)
65
- y_outer = 15 * np.sin(theta)
66
- fig.add_trace(go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(color='royalblue', width=2), showlegend=False, hoverinfo='skip'))
67
- # Inner circle
68
- theta2 = np.linspace(np.pi/2, 3*np.pi/2, 300)
69
- x_inner = 42.5 + 10 * np.cos(theta2)
70
- y_inner = 10 * np.sin(theta2)
71
- fig.add_trace(go.Scatter(x=x_inner, y=y_inner, mode='lines', line=dict(color='red', width=2), showlegend=False, hoverinfo='skip'))
72
- # Rink boundaries
73
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=25, x1=42.5, y1=26, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
74
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=-25, x1=42.5, y1=-26, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
75
- fig.add_shape(type='rect', xref='x', yref='y', x0=-42.5, y0=-0.5, x1=42.5, y1=0.5, line=dict(color='red', width=2), fillcolor='red')
76
-
77
- # Goal crease
78
- fig.add_trace(goal_crease())
79
- fig.add_trace(goal_crease(-1))
80
- # Goal lines
81
- goal_line_extreme = 42.5 - 28 + np.sqrt(28**2 - (28-11)**2)
82
- fig.add_shape(type='line', xref='x', yref='y', x0=-goal_line_extreme, y0=89, x1=goal_line_extreme, y1=89, line=dict(color='red', width=2))
83
- fig.add_shape(type='line', xref='x', yref='y', x0=-goal_line_extreme, y0=-89, x1=goal_line_extreme, y1=-89, line=dict(color='red', width=2))
84
-
85
- # Faceoff circles
86
- fig.add_traces(faceoff_circle(-22, 69))
87
- fig.add_traces(faceoff_circle(22, 69))
88
- fig.add_traces(faceoff_circle(-22, -69))
89
- fig.add_traces(faceoff_circle(22, -69))
90
- fig.add_traces(faceoff_circle(-22, -20, False))
91
- fig.add_traces(faceoff_circle(22, -20, False))
92
- fig.add_traces(faceoff_circle(-22, 20, False))
93
- fig.add_traces(faceoff_circle(22, 20, False))
94
-
95
- # Sidelines
96
- theta_lines = np.linspace(0, np.pi/2, 20)
97
- x_lines1 = np.concatenate(([-42.5], -42.5 + 28 - 28*np.cos(theta_lines), 42.5 - 28 + 28*np.cos(np.flip(theta_lines))))
98
- y_lines1 = np.concatenate(([15], 72 + 28*np.sin(theta_lines), 72 + 28*np.sin(np.flip(theta_lines))))
99
- x_lines2 = np.concatenate(([-42.5], -42.5 + 28 - 28*np.cos(theta_lines), 42.5 - 28 + 28*np.cos(np.flip(theta_lines))))
100
- y_lines2 = np.concatenate(([15], -72 - 28*np.sin(theta_lines), -72 - 28*np.sin(np.flip(theta_lines))))
101
- fig.add_trace(go.Scatter(x=x_lines1, y=y_lines1, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
102
- fig.add_trace(go.Scatter(x=x_lines2, y=y_lines2, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
103
- fig.add_shape(type='line', xref='x', yref='y', x0=42.5, y0=-72.5, x1=42.5, y1=72.5, line=dict(color='white', width=2))
104
- fig.add_shape(type='line', xref='x', yref='y', x0=-42.5, y0=-72.5, x1=-42.5, y1=72.5, line=dict(color='white', width=2))
105
-
106
- # Add goals
107
- goal_width = 6 # feet
108
- goal_depth = 4 # feet
109
-
110
- # Top goal
111
- fig.add_shape(
112
- type="rect",
113
- xref="x",
114
- yref="y",
115
- x0=-goal_width / 2,
116
- y0=89,
117
- x1=goal_width / 2,
118
- y1=89 + goal_depth,
119
- line=dict(color="red", width=2),
120
- )
121
- # Bottom goal
122
- fig.add_shape(
123
- type="rect",
124
- xref="x",
125
- yref="y",
126
- x0=-goal_width / 2,
127
- y0=-89 - goal_depth,
128
- x1=goal_width / 2,
129
- y1=-89,
130
- line=dict(color="red", width=2),
131
- )
132
-
133
- else :
134
- setting_dict = {
135
- "full" : [-101, 101],
136
- "offense" : [0, 101],
137
- "ozone" : [25, 101],
138
- "defense" : [-101, 0],
139
- "dzone" : [-101, -25]
140
- }
141
- fig.update_layout(xaxis=dict(range=setting_dict[setting], showgrid=False, zeroline=False, showticklabels=False), yaxis=dict(range=[-42.6, 42.6], showgrid=False, zeroline=False, showticklabels=False, constrain="domain"),
142
- showlegend=True, autosize =True, template="plotly_white")
143
- fig.update_yaxes(
144
- scaleanchor="x",
145
- scaleratio=1,
146
- )
147
- def goal_crease(flip=1):
148
- y_seq = np.linspace(-4, 4, 100)
149
- y_goal = np.concatenate(([-4], y_seq, [4]))
150
- x_goal = flip * np.concatenate(([89], 83 + y_seq**2/4**2*1.5, [89]))
151
- goal_crease = go.Scatter(x=x_goal, y=y_goal, fill='toself', fillcolor='rgba(173, 216, 230, 0.3)', line=dict(color='red'), showlegend=False, hoverinfo='skip')
152
- return goal_crease
153
-
154
- # Outer circle
155
- theta = np.linspace(0, 2 * np.pi, 300)
156
- x_outer = 15 * np.sin(theta)
157
- y_outer = 15 * np.cos(theta)
158
- fig.add_trace(go.Scatter(x=x_outer, y=y_outer, mode='lines', line=dict(color='royalblue', width=2), showlegend=False, hoverinfo='skip'))
159
- # Inner circle
160
- theta2 = np.linspace(3 * np.pi / 2, np.pi / 2, 300) # Update theta2 to rotate the plot by 180 degrees
161
- x_inner = 10 * np.sin(theta2) # Update x_inner to rotate the plot by 180 degrees
162
- y_inner = -42.5 - 10 * np.cos(theta2) # Update y_inner to rotate the plot by 180 degrees
163
- fig.add_trace(go.Scatter(x=x_inner, y=y_inner, mode='lines', line=dict(color='red', width=2), showlegend=False, hoverinfo='skip'))
164
-
165
- # Rink boundaries
166
- fig.add_shape(type='rect', xref='x', yref='y', x0=25, y0=-42.5, x1=26, y1=42.5, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
167
- fig.add_shape(type='rect', xref='x', yref='y', x0=-25, y0=-42.5, x1=-26, y1=42.5, line=dict(color='royalblue', width=1), fillcolor='royalblue', opacity=1)
168
- fig.add_shape(type='rect', xref='x', yref='y', x0=-0.5, y0=-42.5, x1=0.5, y1=42.5, line=dict(color='red', width=2), fillcolor='red')
169
- # Goal crease
170
- fig.add_trace(goal_crease())
171
- fig.add_trace(goal_crease(-1))
172
- # Goal lines
173
- goal_line_extreme = 42.5 - 28 + np.sqrt(28 ** 2 - (28 - 11) ** 2)
174
- fig.add_shape(type='line', xref='x', yref='y', x0=89, y0=-goal_line_extreme, x1=89, y1=goal_line_extreme, line=dict(color='red', width=2))
175
- fig.add_shape(type='line', xref='x', yref='y', x0=-89, y0=-goal_line_extreme, x1=-89, y1=goal_line_extreme, line=dict(color='red', width=2))
176
- # Faceoff circles
177
- fig.add_traces(faceoff_circle(-69, -22))
178
- fig.add_traces(faceoff_circle(-69, 22))
179
- fig.add_traces(faceoff_circle(69, -22))
180
- fig.add_traces(faceoff_circle(69, 22))
181
- fig.add_traces(faceoff_circle(-20, -22, False))
182
- fig.add_traces(faceoff_circle(-20, 22, False))
183
- fig.add_traces(faceoff_circle(20, -22, False))
184
- fig.add_traces(faceoff_circle(20, 22, False))
185
-
186
- # Sidelines
187
- theta_lines = np.linspace(0, np.pi / 2, 20)
188
- x_lines1 = np.concatenate(([15], 72 + 28 * np.sin(theta_lines), 72 + 28 * np.sin(np.flip(theta_lines))))
189
- y_lines1 = np.concatenate(([-42.5], -42.5 + 28 - 28 * np.cos(theta_lines), 42.5 - 28 + 28 * np.cos(np.flip(theta_lines))))
190
- x_lines2 = np.concatenate(([15], -72 - 28 * np.sin(theta_lines), -72 - 28 * np.sin(np.flip(theta_lines))))
191
- y_lines2 = np.concatenate(([-42.5], -42.5 + 28 - 28 * np.cos(theta_lines), 42.5 - 28 + 28 * np.cos(np.flip(theta_lines))))
192
- fig.add_trace(go.Scatter(x=x_lines1, y=y_lines1, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
193
- fig.add_trace(go.Scatter(x=x_lines2, y=y_lines2, mode='lines', line=dict(color='white', width=2), showlegend=False, hoverinfo='skip'))
194
- fig.add_shape(type='line', xref='x', yref='y', x0=-72.5, y0=-42.5, x1=72.5, y1=-42.5, line=dict(color='white', width=2))
195
- fig.add_shape(type='line', xref='x', yref='y', x0=-72.5, y0=42.5, x1=72.5, y1=42.5, line=dict(color='white', width=2))
196
-
197
- # Add goals
198
- goal_width = 6 # feet
199
- goal_depth = 4 # feet
200
-
201
- # Right goal
202
- fig.add_shape(
203
- type="rect",
204
- xref="x",
205
- yref="y",
206
- x0=89,
207
- y0=-goal_width / 2,
208
- x1=89 + goal_depth,
209
- y1=goal_width / 2,
210
- line=dict(color="red", width=2),
211
- )
212
- # Left goal
213
- fig.add_shape(
214
- type="rect",
215
- xref="x",
216
- yref="y",
217
- x0=-89 - goal_depth,
218
- y0=-goal_width / 2,
219
- x1=-89,
220
- y1=goal_width / 2,
221
- line=dict(color="red", width=2),
222
- )
223
-
224
- # Add logo
225
- logo = Image.open(rs.get('https://weakside-breakout.s3.us-east-2.amazonaws.com/utils/wsba.png',stream=True).raw)
226
-
227
- fig.add_layout_image(
228
- dict(
229
- source=logo,
230
- xref="x",
231
- yref="y",
232
- x=-12,
233
- y=12,
234
- sizex=24,
235
- sizey=24,
236
- sizing="stretch",
237
- opacity=1)
238
- )
239
-
240
- #Set background to transparent
241
- fig.update_layout(
242
- paper_bgcolor="rgba(0,0,0,0)",
243
- plot_bgcolor="rgba(0,0,0,0)"
244
- )
245
- return fig