Rozeeeee's picture
Update app.py
7001d25 verified
raw
history blame
4.56 kB
import streamlit as st
import pandas as pd
import requests
import plotly.express as px
import io
# 設置顏色主題
theme = px.colors.qualitative.Bold
# 下載並載入 CSV 數據
def download_and_load_csv(url):
response = requests.get(url)
response.encoding = 'utf-8'
df = pd.read_csv(io.StringIO(response.text), encoding='utf-8')
df = df.fillna(0) # 避免 NaN 錯誤
return df
# 美化圖表
def beautify_chart(fig):
fig.update_layout(
font_family="Arial",
font_color="#444",
title_font_family="Arial",
title_font_color="#000",
legend_title_font_color="#000",
plot_bgcolor='rgba(0,0,0,0)',
paper_bgcolor='rgba(0,0,0,0)',
)
fig.update_xaxes(showline=True, linewidth=2, linecolor='lightgray', gridcolor='lightgray')
fig.update_yaxes(showline=True, linewidth=2, linecolor='lightgray', gridcolor='lightgray')
return fig
# **生成 Plotly 圖表**
def generate_plots(df, df_name, selected_columns, chart_type):
selected_columns = list(selected_columns) # 確保是列表
if not selected_columns:
st.error("❌ 沒有選擇數據欄位,請至少選擇一個!")
return
if "公司名稱" not in df.columns:
st.error("❌ 缺少 公司名稱 欄位,請確認數據格式!")
return
# ✅ **確保所有 y 軸數據與 x 軸 (公司名稱) 長度一致**
df = df.dropna(subset=["公司名稱"] + selected_columns) # **移除 NaN 確保長度相同**
df[selected_columns] = df[selected_columns].apply(pd.to_numeric, errors='coerce') # **轉換數據類型**
valid_columns = [col for col in selected_columns if col in df.columns and len(df[col]) == len(df["公司名稱"])]
if not valid_columns:
st.error(f"❌ 選擇的欄位長度與 公司名稱 不匹配,請檢查數據!\n"
f"📊 公司名稱 長度: {len(df['公司名稱'])}\n"
f"🟢 可用欄位: {valid_columns}")
return
with st.expander(f"📊 顯示 {df_name} 圖表"):
st.subheader(f"{df_name} - {chart_type} 圖")
# **根據選擇的圖表類型來繪製**
if chart_type == "折線圖":
fig = px.line(df, x="公司名稱", y=valid_columns, title=f"{df_name} - 折線圖", color_discrete_sequence=theme)
elif chart_type == "散點圖":
fig = px.scatter(df, x="公司名稱", y=valid_columns, title=f"{df_name} - 散點圖", color_discrete_sequence=theme)
elif chart_type == "長條圖":
fig = px.bar(df, x="公司名稱", y=valid_columns, title=f"{df_name} - 長條圖", color_discrete_sequence=theme)
elif chart_type == "餅圖":
total_emissions = df.groupby("公司名稱")[valid_columns].sum().reset_index()
total_emissions = total_emissions.melt(id_vars=["公司名稱"], value_vars=valid_columns, var_name="排放類型", value_name="總排放量")
fig = px.pie(total_emissions, values='總排放量', names='公司名稱', title=f"{df_name} - 餅圖", color_discrete_sequence=theme, hole=0.3)
fig.update_traces(textposition='inside', textinfo='percent+label')
fig = beautify_chart(fig)
st.plotly_chart(fig, use_container_width=True) # ✅ **讓圖表自適應畫面**
# **下載數據**
urls = [
"https://mopsfin.twse.com.tw/opendata/t187ap46_L_1.csv",
"https://mopsfin.twse.com.tw/opendata/t187ap46_O_2.csv",
"https://mopsfin.twse.com.tw/opendata/t187ap46_L_6.csv"
]
dfs = [download_and_load_csv(url) for url in urls]
combined_df = pd.concat(dfs, ignore_index=True).fillna(0) # 合併數據並填充 NaN 值
# **再次填充 NaN 值**
combined_df = combined_df.fillna(0)
# **只顯示前300筆數據**
combined_df = combined_df.head(300)
# **Streamlit UI**
st.title("📊 台灣企業 ESG 數據分析")
st.subheader("📂 數據預覽")
st.dataframe(combined_df)
# **選擇數據欄位**
emission_columns = ["範疇一排放量(噸CO2e)", "範疇二排放量(噸CO2e)", "範疇三排放量(噸CO2e)"]
selected_columns = st.multiselect("選擇要顯示的排放類別", emission_columns, default=emission_columns[:1])
# **選擇圖表類型**
chart_type = st.selectbox("選擇圖表類型", ["折線圖", "散點圖", "長條圖", "餅圖"])
# **生成圖表**
if selected_columns:
generate_plots(combined_df, "綜合數據", selected_columns, chart_type)
else:
st.write("⚠️ 請選擇至少一個排放類別來顯示圖表。")