forked from damianboh/pokedex
-
Notifications
You must be signed in to change notification settings - Fork 0
/
app.py
341 lines (295 loc) · 14.4 KB
/
app.py
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
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
import streamlit as st
from PIL import Image
import numpy as np
import plotly.express as px
import pandas as pd
import matplotlib.pyplot as plt
st.set_page_config(page_title = "Pokédex", layout = "wide")
# css file for displaying Pokemon type (fire, water etc.)
def local_css(file_name):
with open(file_name) as f:
st.markdown(f'<style>{f.read()}</style>', unsafe_allow_html=True)
@st.experimental_singleton
def get_data():
# read null values as empty string
# only keep pokemon until generation 6
return pd.read_csv('pokedex.csv', keep_default_na = False).iloc[:847]
# load css file and get data
local_css('style.css')
df = get_data()
# sidebar configuration for searching Pokemon by name
st.sidebar.title('Pokédex')
name = st.sidebar.text_input('Search Name', '').lower() # input name
# find names that matches input and return it in a list
matches = list(df[df['name'].str.lower().str.contains(name)]['name'])
# dropdown menu with names that matches input
if len(matches) >= 1:
name = st.sidebar.selectbox('Pokemon Matches', matches).lower()
else: # if no name matches input
name = st.sidebar.selectbox('Pokemon Matches', ['No match'])
# filter row of data that matches Pokemon selected in dropdown menu
match = df[df['name'].str.lower() == name]
# select information to view
info_list = ['Basic Information', 'Base Stats & Type Defenses', 'Training and Breeding', 'Radar Chart']
selected_info = st.sidebar.multiselect('View Information', info_list, default = info_list)
# search Pokemon using min and max base stats (speed, special defense etc.)
with st.sidebar.form(key="my_form"):
st.subheader('Search Base Stats Range')
min_speed, max_speed = st.select_slider('Speed', range(251), value = [0, 250])
min_sp_def, max_sp_def = st.select_slider('Special Defense', range(251), value = [0, 250])
min_sp_atk, max_sp_atk = st.select_slider('Special Attack', range(251), value = [0, 250])
min_def, max_def = st.select_slider('Defense', range(251), value = [0, 250])
min_atk, max_atk = st.select_slider('Attack', range(251), value = [0, 250])
min_hp, max_hp = st.select_slider('HP', range(251), value = [0, 250])
# pressed is a Boolean that becomes True when the "Search Pokemon" button is pressed
# code to handle button pressing is all the way at the bottom
pressed = st.form_submit_button("Search Pokemon")
# display credits on sidebar
st.sidebar.subheader('Credits')
st.sidebar.write("Of course, I do not claim to own any of the Pokemon data or images in any way. :full_moon_with_face:")
st.sidebar.write("The Kaggle links below show where I've obtained the data. Note that there are further acknowledgements within the Kaggle pages themselves, as the authors got their data from more original sources.")
st.sidebar.markdown('Pokemon dataset taken from <a href="https://www.kaggle.com/datasets/mariotormo/complete-pokemon-dataset-updated-090420?select=pokedex_%28Update_04.21%29.csv">this Kaggle link</a>.', unsafe_allow_html = True)
st.sidebar.markdown('Pokemon images taken from <a href="https://www.kaggle.com/datasets/kvpratama/pokemon-images-dataset">this Kaggle link</a>.', unsafe_allow_html = True)
# use Pokemon name and id to get image path, refer to 'pokemon_images' folder to see how images are named
def get_image_path(name, id):
if name.startswith('Mega'):
if name.endswith(' X'):
path = 'pokemon_images/' + str(id) + '-mega-x.png'
elif name.endswith(' Y'):
path = 'pokemon_images/' + str(id) + '-mega-y.png'
else:
path = 'pokemon_images/' + str(id) + '-mega.png'
elif name.endswith(' Rotom'):
rotom_type = name.split()[0].lower()
path = 'pokemon_images/' + str(id) + '-' + rotom_type + '.png'
elif name.endswith(' Forme') or name.endswith(' Cloak') or name.endswith(' Form'):
if 'Zygarde' in name: # only 1 image present for Zygarde
path = 'pokemon_images/' + str(id) + '.png'
else:
type = name.split()[1].lower()
path = 'pokemon_images/' + str(id) + '-' + type + '.png'
elif name.startswith('Primal '):
type = name.split()[0].lower()
path = 'pokemon_images/' + str(id) + '-' + type + '.png'
elif name.startswith('Arceus'):
path = 'pokemon_images/' + str(id) + '-normal.png' # this is just how Arceus is named in the image file
else:
path = 'pokemon_images/' + str(id) + '.png'
return path
def display_basic_info(match):
# get basic info data
name = match['name'].iloc[0]
id = match['pokedex_number'].iloc[0]
height = str(match['height_m'].iloc[0])
weight = str(match['weight_kg'].iloc[0])
species = ' '.join(match['species'].iloc[0].split(' ')[:-1])
type1 = match['type_1'].iloc[0]
type2 = match['type_2'].iloc[0]
type_number = match['type_number'].iloc[0]
ability1 = match['ability_1'].iloc[0]
ability2 = match['ability_2'].iloc[0]
ability_hidden = match['ability_hidden'].iloc[0]
st.title(name + ' #' + str(id).zfill(3))
col1, col2, col3 = st.columns(3)
# leftmost column col1 displays pokemon image
try:
path = get_image_path(name, id)
image = Image.open(path)
col1.image(image)
except: # output 'Image not available' instead of crashing the program when image not found
col1.write('Image not available.')
# middle column col2 displays nicely formatted Pokemon type using css loaded earlier
with col2.container():
col2.write('Type')
# html code that loads the class defined in css, each Pokemon type has a different style color
type_text = f'<span class="icon type-{type1.lower()}">{type1}</span>'
if type_number == 2:
type_text += f' <span class="icon type-{type2.lower()}">{type2}</span>'
# markdown displays html code directly
col2.markdown(type_text, unsafe_allow_html=True)
col2.metric("Height", height + " m")
col2.metric("Weight", weight + " kg")
# rightmost column col3 displays Pokemon abilities
with col3.container():
col3.metric("Species", species)
col3.write('Abilities')
if ability1 != '':
col3.subheader(ability1)
if ability2 != '':
col3.subheader(ability2)
if ability_hidden != '':
col3.subheader(ability_hidden + ' (Hidden)')
def display_base_stats_type_defenses(match):
# list to gather all type weaknesses and resistances
weakness_2_types = []
weakness_4_types = []
resistance_half_types = []
resistance_quarter_types = []
# dataset only shows damage (x4, x2, x0.25, x0.5) of each type towards the Pokemon
# manually classify the damages into weaknesses and resistances list
for i, j in match.iterrows():
for column, value in j.iteritems():
if column.startswith('against_'):
type = column.split('_')[1]
if value == 0.5:
resistance_half_types.append(type)
elif value == 0.25:
resistance_quarter_types.append(type)
elif value == 2:
weakness_2_types.append(type)
elif value == 4:
weakness_4_types.append(type)
with st.container():
col1, col2 = st.columns(2)
# left column col1 displays horizontal bar chart of base stats
col1.subheader('Base Stats')
# get base stats of Pokemon and rename columns nicely
df_stats = match[['hp', 'attack', 'defense', 'sp_attack', 'sp_defense', 'speed']]
df_stats = df_stats.rename(columns={'hp': 'HP', 'attack': 'Attack', 'defense': 'Defense', 'sp_attack': 'Special Attack', 'sp_defense': 'Special Defense', 'speed': 'Speed'}).T
df_stats.columns=['stats']
# plot horizontal bar chart using matplotlib.pyplot
fig, ax = plt.subplots()
ax.barh(y = df_stats.index, width = df_stats.stats)
plt.xlim([0, 250])
col1.pyplot(fig)
# right column col2 displays the weaknesses and resistances
# the displayed types are nicely formatted using css (same as earlier)
col2.subheader('Type Defenses')
col2.write('Strong Weaknesses (x4)')
weakness_text = ''
for type in weakness_4_types:
weakness_text += f' <span class="icon type-{type}">{type}</span>'
col2.markdown(weakness_text, unsafe_allow_html=True)
col2.write('Weaknesses (x2)')
weakness_text = ''
for type in weakness_2_types:
weakness_text += f' <span class="icon type-{type}">{type}</span>'
col2.markdown(weakness_text, unsafe_allow_html=True)
col2.write('Resistances (x0.5)')
resistance_half_text = ''
for type in resistance_half_types:
resistance_half_text += f' <span class="icon type-{type}">{type}</span>'
col2.markdown(resistance_half_text, unsafe_allow_html=True)
col2.write('Strong Resistances (x0.25)')
resistance_quarter_text = ''
for type in resistance_quarter_types:
resistance_quarter_text += f' <span class="icon type-{type}">{type}</span>'
col2.markdown(resistance_quarter_text, unsafe_allow_html=True)
def display_training_breeding(match):
# get training data
catch_rate = match['catch_rate'].iloc[0]
base_friendship = match['base_friendship'].iloc[0]
base_experience = match['base_experience'].iloc[0]
growth_rate = match['growth_rate'].iloc[0]
# get breeding data
egg_type_number = match['egg_type_number'].iloc[0]
egg_type_1 = match['egg_type_1'].iloc[0]
egg_type_2 = match['egg_type_2'].iloc[0]
percentage_male = match['percentage_male'].iloc[0]
egg_cycles = match['egg_cycles'].iloc[0]
with st.container():
col1, col2 = st.columns(2)
# left column col1 displays training data
col1.subheader('Training')
col1.metric('Catch Rate', catch_rate)
col1.metric('Base Friendship', base_friendship)
col1.metric('Base Experience', base_experience)
col1.metric('Growth Rate', growth_rate)
# right column col2 displays breeding data
col2.subheader('Breeding')
if egg_type_number == 2: # some Pokemon have 2 egg types
col2.metric('Egg Types', egg_type_1 + ', ' + egg_type_2)
else:
col2.metric('Egg Types', egg_type_1)
if percentage_male != '':
percentage_female = str(100 - float(match['percentage_male'].iloc[0]))
col2.metric('Percentage Male/Female', percentage_male + '% / ' + percentage_female + '%' )
else:
# this metric is not available for Pokemon without eggs, e.g. Mewtwo
col2.metric('Percentage Male/Female', 'NA')
col2.metric('Egg Cycles', egg_cycles)
def display_radar_chart(match):
st.header('Radar Chart of Base Stats')
# get base stats of Pokemon and rename columns nicely
df_stats = match[['hp', 'attack', 'defense', 'sp_attack', 'sp_defense', 'speed']]
df_stats = df_stats.rename(columns={'hp': 'HP', 'attack': 'Attack', 'defense': 'Defense', 'sp_attack': 'Special Attack', 'sp_defense': 'Special Defense', 'speed': 'Speed'}).T
df_stats.columns=['stats']
# use plotly express to plot out radar char of stats
fig = px.line_polar(df_stats, r='stats', theta=df_stats.index, line_close=True, range_r=[0, 250])
st.plotly_chart(fig)
if st.button('Search for Pokemons with Similar Base Stats'):
display_similar_pokemons(match)
def display_similar_pokemons(match):
# get base stats of Pokemon and rename columns nicely
df_stats = match[['hp', 'attack', 'defense', 'sp_attack', 'sp_defense', 'speed']]
df_stats = df_stats.rename(columns={'hp': 'HP', 'attack': 'Attack', 'defense': 'Defense', 'sp_attack': 'Special Attack', 'sp_defense': 'Special Defense', 'speed': 'Speed'})
# get stats of all other Pokemon in the full dataframe
df_stats_all = df[['name', 'hp', 'attack', 'defense', 'sp_attack', 'sp_defense', 'speed']].set_index('name')
df_stats_all = df_stats_all.rename(columns={'hp': 'HP', 'attack': 'Attack', 'defense': 'Defense', 'sp_attack': 'Special Attack', 'sp_defense': 'Special Defense', 'speed': 'Speed'})
# find difference between stat of Pokemon and each of the other Pokemons
diff_df = pd.DataFrame(df_stats_all.values - df_stats.values, index = df_stats_all.index)
# find norm 'distance' between this Pokemon and all other Pokemon
norm_df = diff_df.apply(np.linalg.norm, axis=1)
# find 20 other Pokemon with smallest distance, i.e. with most similar base stats to this Pokemon
similar_pokemons = norm_df.nsmallest(21)[1:22].index # index [1:22] so it does not show itself
# store all similar Pokemon with their stats in df
similar_pokemons_df = df_stats_all.loc[similar_pokemons]
# display name, image, radar chart of each similar Pokemon
for row in similar_pokemons_df.iterrows():
name = row[0]
st.subheader(name) # display Pokemon name
id = df[df.name == name]['pokedex_number'].iloc[0]
# display Pokemon image
try:
path = get_image_path(name, id)
image = Image.open(path)
st.image(image)
except:
st.write('Image not available.')
# display radar chart
fig = px.line_polar(row[1], r=name, theta=row[1].index, line_close=True, range_r=[0, 255])
st.plotly_chart(fig)
# display full table of all 20 similar Pokemons and their stats
st.subheader('20 Most Similar Pokemons')
st.table(similar_pokemons_df)
# if "Search Pokemon" button is not pressed,
# i.e. we are searching Pokemon by name instead of by base stats slider section
if not pressed:
if len(match) == 0:
st.write('Enter name to search for details.')
# display information of Pokemon according to the information the user selects in the sidebar multiselector
elif len(match) == 1:
if 'Basic Information' in selected_info:
display_basic_info(match)
if 'Base Stats & Type Defenses' in selected_info:
display_base_stats_type_defenses(match)
if 'Training and Breeding' in selected_info:
display_training_breeding(match)
if 'Radar Chart' in selected_info:
display_radar_chart(match)
# if "Search Pokemon" button IS PRESSED,
# filter Pokemon according to the base stats in the slider
# display all the matched Pokemon and their stats in a table
# we do not display every bit of information about the search Pokemons as the list may be huge and the app can hang
else:
# get base stats of all Pokemon
df_stats_all = df[['name', 'hp', 'attack', 'defense', 'sp_attack', 'sp_defense', 'speed']].set_index('name')
df_stats_all = df_stats_all.rename(columns={'hp': 'HP', 'attack': 'Attack', 'defense': 'Defense', 'sp_attack': 'Special Attack', 'sp_defense': 'Special Defense', 'speed': 'Speed'})
# filter stats according to search criteria from the sliders
searched_pokemons_df = df_stats_all[
(df_stats_all['HP'] >= min_hp) & (df_stats_all['HP'] <= max_hp) &
(df_stats_all['Attack'] >= min_atk) & (df_stats_all['Attack'] <= max_atk) &
(df_stats_all['Defense'] >= min_def) & (df_stats_all['Defense'] <= max_def) &
(df_stats_all['Special Attack'] >= min_sp_atk) & (df_stats_all['Special Attack'] <= max_sp_atk) &
(df_stats_all['Special Defense'] >= min_sp_def) & (df_stats_all['Special Defense'] <= max_sp_def) &
(df_stats_all['Speed'] >= min_speed) & (df_stats_all['Speed'] <= max_speed)
]
st.header('Pokemon Search Using Base Stats')
st.table(searched_pokemons_df)
hide_streamlit_style = """
<style>
#MainMenu {visibility: hidden;}
footer {visibility: hidden;}
</style>
"""
st.markdown(hide_streamlit_style, unsafe_allow_html=True)