개발/프로젝트
웹대시보드 프로젝트 4일차
웅'jk
2022. 12. 21. 09:31
3일차에서 말했던 대로 멀티셀렉트박스를 이용하여 구성을 바꾸고
선수탭을 작업하기 시작하였습니다.
다만 여기서 데이터컬럼명을 rename하는 과정에서 빼먹은 부분이 있어 그부분을 수정하였습니다.
1. app.py
import streamlit as st
from home import home
from team import team
from player import player
import matplotlib.pyplot as plt
import platform
from matplotlib import font_manager, rc
def main() :
plt.rcParams['axes.unicode_minus'] = False
if platform.system() == 'Linux':
rc('font', family='NanumGothic')
team_name_list = ['(팀을 선택해주세요)','ATL', 'BOS', 'NOP', 'CHI', 'DAL', 'DEN', 'HOU', 'LAC', 'LAL',
'MIA', 'MIL', 'MIN', 'BKN', 'NYK', 'ORL', 'IND', 'PHI', 'PHX',
'POR', 'SAC', 'SAS', 'OKC', 'TOR', 'UTA', 'MEM', 'WAS', 'DET',
'CHA', 'CLE', 'GSW']
st.sidebar.title('메뉴')
choose = st.sidebar.selectbox('카테고리',['홈','팀','선수'])
if choose == '홈' :
home()
if choose == '팀' :
team_name = st.sidebar.selectbox('팀 정보',team_name_list)
if team_name != '(팀을 선택해주세요)' :
team(team_name)
else :
st.title('팀을 선택해주세요.')
if choose =='선수' :
player_name = st.sidebar.text_input('선수명(영어로만)',max_chars=100)
if player_name != '' :
player(player_name)
else :
st.sidebar.error('선수명을 입력해주세요.')
image = ''
st.sidebar.image(image,use_column_width='always')
if __name__ == '__main__' :
main()
2. data_init.py
import pandas as pd
df_teams = pd.read_csv('data/teams.csv')
df_games = pd.read_csv('data/games.csv')
df_players = pd.read_csv('data/players.csv')
df_games_detail = pd.read_csv('data/games_details.csv',low_memory = False)
df_teams = df_teams.loc[:,['TEAM_ID','ABBREVIATION','YEARFOUNDED','CITY','ARENA']]
df_games = df_games.loc[:,['GAME_DATE_EST','GAME_ID','HOME_TEAM_ID','PTS_home','AST_home','REB_home','TEAM_ID_away','PTS_away','AST_away','REB_away','HOME_TEAM_WINS']]
# 문제 없는 데이터프레임
df_games_detail_1 = df_games_detail.loc[:,:'START_POSITION']
# 문제 있는 데이터프레임 짜른 뒤 수정
df_games_detail_2 = df_games_detail.loc[:,'COMMENT':].shift(-1,axis= 1)
# 짜른 두 코드를 합쳤다.
df_games_detail = pd.concat([df_games_detail_1,df_games_detail_2],axis=1)
df_games_detail = df_games_detail.loc[:,['GAME_ID', 'TEAM_ID','PLAYER_ID','PLAYER_NAME','START_POSITION',
'COMMENT','MIN', 'FG_PCT', 'FG3_PCT',
'DREB', 'REB', 'AST', 'STL', 'BLK','TO']]
rename ={ 'GAME_DATE_EST' : '경기날짜' ,'GAME_ID':'경기ID', 'HOME_TEAM_ID':'홈팀ID','PTS_home':'홈팀점수','AST_home':'홈팀어시','REB_home':'홈팀리바운드','TEAM_ID_away':'어웨이팀ID','PTS_away':'어웨이팀점수','AST_away':'어웨이팀어시','REB_away':'어웨이팀리바운드','HOME_TEAM_WINS':'홈팀승리여부' }
df_games = df_games.rename(columns=rename)
df_games=df_games.fillna(0)
rename_detail = { 'GAME_ID':'경기ID','TEAM_ID':'팀ID','PLAYER_ID':'선수ID','PLAYER_NAME':'선수명',
'START_POSITION':'포지션',
'COMMENT':'출전시간','MIN':'2점슛', 'FG_PCT':'3점슛', 'FG3_PCT':'자유투',
'DREB':'리바운드','REB':'어시스트','AST':'가로채기','STL':'블락','BLK':'턴오버','TO':'파울'}
df_games_detail = df_games_detail.rename(columns=rename_detail)
df_games_detail = df_games_detail.fillna(0)
rename_playes = { 'PLAYER_NAME':'선수명','TEAM_ID':'팀ID','PLAYER_ID':'선수ID','SEASON':'시즌' }
df_players=df_players.rename(columns=rename_playes)
rename_team = {'TEAM_ID':'팀ID','ABBREVIATION':'팀약어','YEARFOUNDED':'팀창설해',
'CITY':'연고지','ARENA':'홈구장'}
df_teams = df_teams.rename(columns=rename_team)
# 가공을 편하게 하기위해 df_games에 홈팀명,어웨이팀명 추가
df_teams_home = df_teams.rename(columns={'팀ID':'홈팀ID'})
new_data = pd.merge(df_games,df_teams_home).iloc[:,:-3].rename(columns={'팀약어':'홈팀'})
df_teams_away = df_teams.rename(columns={'팀ID':'어웨이팀ID'})
df_games=pd.merge(new_data,df_teams_away).iloc[:,:-3].rename(columns={'팀약어':'어웨이팀'})
# 가공을 편하게 하기 위해 df_platers에 팀명 추가
df_players=pd.merge(df_players,df_teams).iloc[:,0:4+1]
# 가공을 편하게 하기 위해 df_games_detail 에 날짜 추가
df_games_detail=pd.merge(df_games_detail,df_games).loc[:,:'경기날짜'].sort_values('경기날짜',ascending=False)
# 가공을 편하게 하기위해 df_games_detail에 약어 추가
df_games_detail = pd.merge(df_games_detail,df_teams).iloc[:,:-3]
3.player.py
import streamlit as st
import data_init
import pandas as pd
import matplotlib.pyplot as plt
import plotly.express as px
import platform
from matplotlib import font_manager, rc
def player(player_name) :
plt.rcParams['axes.unicode_minus'] = False
if platform.system() == 'Linux':
rc('font', family='NanumGothic')
elif platform.system() == 'Windows':
path = "c:/Windows/Fonts/malgun.ttf"
font_name = font_manager.FontProperties(fname=path).get_name()
rc('font', family=font_name)
# 데이터를 읽어온다.
# 선수 데이터
df_players = data_init.df_players
# 선수 스탯 데이터
df_games_detail = data_init.df_games_detail
# 팀정보
df_team = data_init.df_teams
# header
st.header('선수 정보')
# 선수명 확인
player_list = df_players[df_players['선수명'].str.contains(player_name,case=False)]['선수명'].unique().tolist()
if len(player_list) > 1 :
st.error('중복되는 선수명을 가진 선수가 있습니다. 내역을 보시고 다시 입력해주세요.')
player = pd.DataFrame(player_list,columns=['선수명'])
st.dataframe(player)
elif len(player_list) == 0 :
st.error('등록된 선수가 없습니다. 다시 입력해주세요.')
elif len(player_list) == 1 :
player = player_list[0]
# 플레이어 데이터 보여주기
df_player = df_players[df_players['선수명'] == player]
st.dataframe(df_player.iloc[:,[0,4,2,3]])
# 스탯 조회하기
# 선수 데이터 가져오기
df_player_data = df_games_detail[df_games_detail['선수ID'] == df_player['선수ID'].unique()[0]].sort_values('경기날짜',ascending=False)
# 1. 기간 선택하기
st.subheader('선수 스탯 조회하기')
radio_status = st.radio('검색',['날짜별','소속팀별'])
if radio_status == '날짜별' :
start_date = st.date_input('시작 날짜')
end_date = st.date_input('종료 날짜')
start_date = start_date.strftime('%Y-%m-%d')
end_date = end_date.strftime('%Y-%m-%d')
if start_date > end_date :
st.error('시작 날짜를 종료 날짜보다 전으로 설정해주세요.')
else :
if df_player_data['경기날짜'].values[0] < start_date :
st.error('경기정보가 없습니다.')
else :
df_player_data_set=df_player_data[( df_player_data['경기날짜'] >= start_date ) & (df_player_data['경기날짜'] <= end_date) ]
multiselectbox_list = ['상세','공격지표','수비지표']
selected_player_data = st.multiselect('데이터 선택',multiselectbox_list)
if '상세' in selected_player_data :
st.dataframe(df_player_data_set.iloc[:,[15,16,3,4,5,6,7,8,9,10,11,12,13,14,]])
if '공격지표' in selected_player_data :
offense_data_date = df_player_data_set.iloc[:,[6,7,8,-2]]
chart_offense_data_date = px.bar(offense_data_date,x='경기날짜',y=['2점슛','3점슛','자유투'] ,barmode='group')
st.plotly_chart(chart_offense_data_date)
if '수비지표' in selected_player_data :
defense_data_date = df_player_data_set.iloc[:,[9,10,11,12,13,14,-2]]
chart_defense_data_date = px.bar(defense_data_date,x='경기날짜',y=['리바운드','어시스트','가로채기','블락','턴오버','파울'] ,barmode='group')
st.plotly_chart(chart_defense_data_date)
elif radio_status =='소속팀별' :
pass
4. team.py
import streamlit as st
import pandas as pd
import team_url as tu
import data_init
import matplotlib.pyplot as plt
import numpy as np
import platform
from matplotlib import font_manager, rc
import plotly.express as px
def team(team_name) :
plt.rcParams['axes.unicode_minus'] = False
if platform.system() == 'Linux':
rc('font', family='NanumGothic')
elif platform.system() == 'Windows':
path = "c:/Windows/Fonts/malgun.ttf"
font_name = font_manager.FontProperties(fname=path).get_name()
rc('font', family=font_name)
# 팀의 데이터를 읽어온다.
df_teams = data_init.df_teams
df_games = data_init.df_games
df_games_detail = data_init.df_games_detail
# 팀정보 헤더 지정
st.header(team_name)
# 팀의 맞는 이미지 생성
st.subheader('로고')
st.image(tu.team_url(team_name),width=500)
# 사이드바에서 선택한 팀으로 정보
st.subheader('정보')
df_teams_info = df_teams.loc[df_teams['팀약어'] == team_name,'팀약어':]
st.dataframe(df_teams_info)
# 시즌별 성적보기
st.header('일자별 성적보기')
team_id = df_teams[df_teams['팀약어'] == team_name]['팀ID'].values[0]
# 시작과 끝 종료날짜를 설정하여 데이터 가져오기
start_date = st.date_input('시작 날짜')
end_date = st.date_input('종료 날짜')
start_date = start_date.strftime('%Y-%m-%d')
end_date = end_date.strftime('%Y-%m-%d')
if start_date > end_date :
st.error('시작 날짜를 종료 날짜보다 전으로 설정해주세요.')
# 전적을 비교하기 위한 list 생성
team_name_list = ['','ATL', 'BOS', 'NOP', 'CHI', 'DAL', 'DEN', 'HOU', 'LAC', 'LAL',
'MIA', 'MIL', 'MIN', 'BKN', 'NYK', 'ORL', 'IND', 'PHI', 'PHX',
'POR', 'SAC', 'SAS', 'OKC', 'TOR', 'UTA', 'MEM', 'WAS', 'DET',
'CHA', 'CLE', 'GSW']
if df_games['경기날짜'][0] < start_date :
st.error('경기정보가 없습니다. 날짜를 조정해주세요.')
else :
# 홈 / 어웨이 선택
selected = st.selectbox('홈/어웨이 선택',['홈','어웨이'])
list_selected_box = ['상세','득점','어시스트','리바운드','역대전적','선수 스탯']
if selected == '홈' :
# 홈성적
st.subheader('홈 성적')
home_multi = st.multiselect('데이터 선택',list_selected_box)
df_games_home = df_games.loc[ (df_games['홈팀ID']==team_id) & (start_date<=df_games['경기날짜']) & (df_games['경기날짜']<=end_date),:]
df_games_home = df_games_home.sort_values('경기날짜',ascending=False)
if '상세' in home_multi :
st.subheader('상세')
st.dataframe(df_games_home.iloc[:,[11,12,0,3,4,5,7,8,9]])
if '득점' in home_multi :
#득점 데이터
st.subheader('득점')
recently_game = st.slider('홈 경기 득점 경기 수',min_value=3,max_value=int(df_games_home['경기ID'].count()),value=7)
recently_point_pig = px.bar(df_games_home.head(recently_game),x='경기날짜',y=['홈팀점수','어웨이팀점수'],barmode='group')
st.plotly_chart(recently_point_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '어시스트' in home_multi :
#어시스트 데이터
st.subheader('어시스트')
recently_game = st.slider('홈 경기 어시스트 경기 수',min_value=3,max_value=int(df_games_home['경기ID'].count()),value=7)
recently_assist_pig = px.bar(df_games_home.head(recently_game),x='경기날짜',y=['홈팀어시','어웨이팀어시'],barmode='group')
st.plotly_chart(recently_assist_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '리바운드' in home_multi :
#리바운드 데이터
st.subheader('리바운드')
recently_game = st.slider('최근 경기 리바운드 경기 수',min_value=3,max_value=int(df_games_home['경기ID'].count()),value=7)
recently_rebound_pig = px.bar(df_games_home.head(recently_game),x='경기날짜',y=['홈팀리바운드','어웨이팀리바운드'],barmode='group')
st.plotly_chart(recently_rebound_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '역대전적' in home_multi:
st.subheader('역대 홈 전적비교')
compare_home_team = st.selectbox('팀선택',team_name_list)
print(compare_home_team+' '+team_name)
if compare_home_team != '' and compare_home_team != team_name :
compare_team_id_home = df_teams[df_teams['팀약어'] == compare_home_team]['팀ID'].values[0]
data_home = df_games[(df_games['홈팀ID'] ==team_id) & (df_games['어웨이팀ID'] ==compare_team_id_home)]
st.dataframe(data_home.iloc[:,[11,0,3,4,5,12,7,8,9,10]].sort_values('경기날짜',ascending=False))
if data_home.empty == False :
data_home = data_home.mean(numeric_only=None)
data_home['어웨이승리'] = 1 - data_home['홈팀승리여부']
fig_home = plt.figure()
plt.subplot(2,2,1)
plt.pie(data_home[[9,10]],autopct='%.1f',startangle = 90)
plt.title('홈팀 승률')
plt.legend(['홈','어웨이'])
plt.subplot(2,2,2)
plt.pie(data_home[[2,6]],autopct='%.1f',startangle=90)
plt.title('평균 득점')
plt.legend(['홈','어웨이'])
plt.subplot(2,2,3)
plt.pie(data_home[[3,7]],autopct='%.1f',startangle=90)
plt.title('평균 어시')
plt.legend(['홈','어웨이'])
plt.subplot(2,2,4)
plt.pie(data_home[[4,8]],autopct='%.1f',startangle=90)
plt.title('평균 리바운드')
plt.legend(['홈','어웨이'])
st.pyplot(fig_home)
else :
st.title('')
elif compare_home_team == team_name :
st.error('같은 팀 입니다. 다른 팀을 선택해주세요.')
else :
st.error('팀을 선택해주세요')
if '선수 스탯' in home_multi :
st.subheader('홈경기 선수 스탯')
home_game_id = df_games_home['경기ID']
print(team_id)
home_team_player_info = df_games_detail[(df_games_detail['경기ID'].isin(home_game_id)) & (df_games_detail['팀ID'] == team_id)].iloc[:,3:].groupby('선수명').mean()
st.dataframe(home_team_player_info)
st.info('선수의 자세한 스탯은 카테고리 - 선수 를 이용해주세요.')
else :
st.write('')
elif selected =='어웨이' :
st.subheader('어웨이 성적')
away_multi = st.multiselect('데이터 선택',list_selected_box)
df_games_away = df_games.loc[ (df_games['어웨이팀ID']==team_id) & (start_date<=df_games['경기날짜']) & (df_games['경기날짜']<=end_date),:]
df_games_away = df_games_away.sort_values('경기날짜',ascending=False)
if '상세' in away_multi :
st.subheader('상세')
st.dataframe(df_games_away.iloc[:,[11,12,0,3,4,5,7,8,9]])
if '득점' in away_multi :
#득점 데이터
st.subheader('득점')
recently_game = st.slider('어웨이 경기 득점 경기 수',min_value=3,max_value=int(df_games_away['경기ID'].count()),value=7)
recently_point_pig = px.bar(df_games_away.head(recently_game),x='경기날짜',y=['홈팀점수','어웨이팀점수'],barmode='group')
st.plotly_chart(recently_point_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '어시스트' in away_multi :
#어시스트 데이터
st.subheader('어시스트')
recently_game = st.slider('어웨이 경기 어시스트 경기 수',min_value=3,max_value=int(df_games_away['경기ID'].count()),value=7)
recently_assist_pig = px.bar(df_games_away.head(recently_game),x='경기날짜',y=['홈팀어시','어웨이팀어시'],barmode='group')
st.plotly_chart(recently_assist_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '리바운드' in away_multi :
#리바운드 데이터
st.subheader('리바운드')
recently_game = st.slider('어웨이 경기 리바운드 경기 수',min_value=3,max_value=int(df_games_away['경기ID'].count()),value=7)
recently_rebound_pig = px.bar(df_games_away.head(recently_game),x='경기날짜',y=['홈팀리바운드','어웨이팀리바운드'],barmode='group')
st.plotly_chart(recently_rebound_pig)
st.info('자세히 보고싶은 부분을 좌클릭으로 영역 지정 해주세요.')
if '역대전적' in away_multi :
st.subheader('역대 어웨이 전적비교')
compare_away_team = st.selectbox('팀 선택',team_name_list)
if compare_away_team != '' and compare_away_team != team_name:
compare_team_id_away = df_teams[df_teams['팀약어'] == compare_away_team]['팀ID'].values[0]
data_away=df_games[(df_games['홈팀ID'] == compare_team_id_away) & (df_games['어웨이팀ID'] ==team_id)]
st.dataframe(data_away.iloc[:,[11,0,3,4,5,12,7,8,9,10]].sort_values('경기날짜',ascending=False))
if data_away.empty == False :
data_away = data_away.mean(numeric_only=None)
data_away['어웨이승리'] = 1 - data_away['홈팀승리여부']
fig_away = plt.figure()
plt.subplot(2,2,1)
plt.pie(data_away[[10,9]],autopct='%.1f',startangle = 90)
plt.title('어웨이팀 승률')
plt.legend(['어웨이','홈'])
plt.subplot(2,2,2)
plt.pie(data_away[[6,2]],autopct='%.1f',startangle=90)
plt.title('평균 득점')
plt.legend(['어웨이','홈'])
plt.subplot(2,2,3)
plt.pie(data_away[[7,3]],autopct='%.1f',startangle=90)
plt.title('평균 어시')
plt.legend(['어웨이','홈'])
plt.subplot(2,2,4)
plt.pie(data_away[[8,4]],autopct='%.1f',startangle=90)
plt.title('평균 리바운드')
plt.legend(['어웨이','홈'])
st.pyplot(fig_away)
else :
st.title('')
elif compare_away_team == team_name :
st.error('같은 팀 입니다. 다른 팀을 선택해주세요.')
else :
st.error('팀을 선택해주세요')
if '선수 스탯' in away_multi :
st.subheader('어웨이경기 선수 스탯')
home_away_id = df_games_away['경기ID']
away_team_player_info = df_games_detail[(df_games_detail['경기ID'].isin(home_away_id)) & (df_games_detail['팀ID'] == team_id)].iloc[:,3:].groupby('선수명').mean()
st.dataframe(away_team_player_info)
st.info('선수의 자세한 스탯은 카테고리 - 선수 를 이용해주세요.')
else :
st.write('')