File size: 5,492 Bytes
d7745f2
bcf020e
c46cec1
 
 
 
 
84b4aa4
c46cec1
d7745f2
17976c1
d7745f2
fb11f75
b8df987
35e393c
 
 
bdf1ad8
6706c02
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
75b2bb3
84b4aa4
20ee985
75b2bb3
 
84b4aa4
beb64c6
6706c02
beb64c6
 
 
 
db1622b
 
beb64c6
75b2bb3
beb64c6
 
 
 
75b2bb3
beb64c6
 
 
 
41aaaa6
db1622b
41aaaa6
75b2bb3
e662b95
 
bb00581
 
75b2bb3
c46cec1
bb00581
db1622b
 
 
6706c02
db1622b
6706c02
db1622b
 
 
 
 
 
 
bcf020e
 
6706c02
 
 
c46cec1
6706c02
 
 
 
 
 
 
 
 
c46cec1
6706c02
 
 
 
 
 
 
 
 
 
 
 
beb64c6
 
 
 
 
 
 
c46cec1
bcf020e
b3344b3
87aae5e
6706c02
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
ed4ba7d
 
0d0ff81
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
import os
from shiny import App, ui, render, reactive
import fastf1 as ff1
import matplotlib.pyplot as plt
from matplotlib.collections import LineCollection
from matplotlib import cm
import numpy as np
import shinyswatch

# Define cache folder path
cache_path = os.getcwd() + "/cache"
print(f"Cache path: {cache_path}")

ff1.Cache.enable_cache(cache_path)

# Offline mode to prevent F1 API crashes on Hugging Face
ff1.Cache.offline_mode(enabled=True)

# Define drivers
drivers_2023 = {'Fastest driver': 'Fastest driver', 'VER': 'Max Verstappen', 
                'NOR': 'Lando Norris', 'GAS': 'Pierre Gasly', 'PER': 'Sergio Perez', 
                'ALO': 'Fernando Alonso', 'LEC': 'Charles Leclerc', 'STR': 'Lance Stroll', 
                'MAG': 'Kevin Magnussen', 'TSU': 'Yuki Tsunoda', 'ALB': 'Alexander Albon',
                'ZHO': 'Guanyu Zhou', 'HUL': 'Nico Hülkenberg', 'OCO': 'Esteban Ocon',
                'HAM': 'Lewis Hamilton', 'SAI': 'Carlos Sainz', 'RUS': 'George Russel',
                'BOT': 'Valteri Bottas', 'PIA': 'Oscar Piastri', 'VRI': 'Nyck de Vries',
                'SAR': 'Logan Sargeant', 'RIC': 'Daniel Ricciardo'}

drivers_2022 = {'Fastest driver': 'Fastest driver', 'VER': 'Max Verstappen', 
                'NOR': 'Lando Norris', 'GAS': 'Pierre Gasly', 'PER': 'Sergio Perez', 
                'ALO': 'Fernando Alonso', 'LEC': 'Charles Leclerc', 'STR': 'Lance Stroll', 
                'MAG': 'Kevin Magnussen', 'TSU': 'Yuki Tsunoda', 'ALB': 'Alexander Albon',
                'ZHO': 'Guanyu Zhou', 'HUL': 'Nico Hülkenberg', 'OCO': 'Esteban Ocon',
                'HAM': 'Lewis Hamilton', 'SAI': 'Carlos Sainz', 'RUS': 'George Russel',
                'BOT': 'Valteri Bottas', 'VRI': 'Nyck de Vries', 'VET': 'Sebastian Vettel',
                'RIC': 'Daniel Ricciardo', 'MSC': 'Mick Schumacher',
                'LAT': 'Nicolas Latifi'}

app_ui = ui.page_fluid(
    shinyswatch.theme.minty(),
    ui.panel_title("Gear usage in fastest lap"),
    ui.layout_sidebar(
        ui.panel_sidebar(
            ui.input_select(
                "track_select", "Select track:",
                choices = ["Austria", "Hungary", "Spain", "Bahrain", "Great-Brittain"],
                selected = "Austria"
            ),
            ui.input_select(
                "driver_select", label="Select driver:",
                choices = ["Fastest driver"],
                selected = "Fastest driver"
            ),
            ui.input_radio_buttons(
                "session_type", "Session type:", 
                choices = {"R": "Race", "Q": "Qualification"}, 
                selected = "R"
            ),
            ui.input_radio_buttons(
                "year", "Year:", 
                choices = ["2023", "2022"], 
                selected = "2023"
            ),
            width=2
        ),
        
        ui.panel_main(
            ui.output_plot("gear"),
            ui.output_text("fastest_driver")
        ),
    ),
)   

def server(input, output, session):
    @reactive.Effect()
    def _():
        if input.year() == "2023":
            driver_options = drivers_2023
        elif input.year() == "2022":
            driver_options = drivers_2022
            
        ui.update_select("driver_select",
            label="Select driver:",
            choices=driver_options,
            selected=input.driver_select()
        )
    
    @reactive.Calc
    def get_data():
        try:
            f1_session = ff1.get_session(int(input.year()), input.track_select(), input.session_type())
            f1_session.load()

            # Check if user input == fastest driver
            if input.driver_select() == "Fastest driver":
                lap = f1_session.laps.pick_fastest()
            else:
                laps_driver = f1_session.laps.pick_driver(input.driver_select())
                lap = laps_driver.pick_fastest()
            
            tel = lap.get_telemetry()
            driver = lap['Driver']

            #converting data to numpy data tables
            x = np.array(tel['X'].values)
            y = np.array(tel['Y'].values)

            points = np.array([x, y]).T.reshape(-1, 1, 2)
            segments = np.concatenate([points[:-1], points[1:]], axis=1)
            gear = tel['nGear'].to_numpy().astype(float)
            return segments, gear, driver

        except Exception:
            pass
        
    
    @output
    @render.text
    def fastest_driver():
        segments, gear, driver = get_data()
        #print(f"The driver of the fastest lap this session is: {driver}")
        return f"Fastest lap this session of: {driver}"

    @output
    @render.plot
    def gear():
        try:
            segments, gear, driver = get_data()
            
            cmap = cm.get_cmap('Paired')
            lc_comp = LineCollection(segments, norm=plt.Normalize(1, cmap.N+1), cmap=cmap)
            lc_comp.set_array(gear)
            lc_comp.set_linewidth(4)

            plt.gca().add_collection(lc_comp)
            plt.axis('equal')
            plt.tick_params(labelleft=False, left=False, labelbottom=False, bottom=False)

            cbar = plt.colorbar(mappable=lc_comp, label="Gear", boundaries=np.arange(1, 10))
            cbar.set_ticks(np.arange(1.5, 9.5))
            cbar.set_ticklabels(np.arange(1, 9))
    
            plt

        except Exception:
            ui.notification_show("Data not available. Select another track or driver.", duration=10, type = 'error')


app = App(app_ui, server)