from dash import Dash, dcc, Input, Output, html
import dash_bootstrap_components as dbc
import pandas as pd
import plotly.express as px
df = pd.read_csv('assets/2019.csv')
app = Dash(__name__, external_stylesheets=[dbc.themes.BOOTSTRAP])
app.layout = dbc.Container([
    html.Div(className="app-header", children=[
        html.H1('World Happiness Index Dashboard', className='display-1')
    ]),
    dbc.Row([
        dbc.Col([
            dcc.Dropdown(
                id="metric-dropdown",
                options=[
                    {'label': 'Happiness Score', 'value': 'Score'},
                    {'label': 'GDP per Capita', 'value': 'GDP per capita'},
                    {'label': 'Social Support', 'value': 'Social support'},
                    {'label': 'Healthy Life Expectancy', 'value': 'Healthy life expectancy'},
                    {'label': 'Freedom to make life choices', 'value': 'Freedom to make life choices'},
                    {'label': 'Generosity', 'value': 'Generosity'},
                    {'label': 'Perceptions of Corruption', 'value': 'Perceptions of corruption'},
                ],
                value='Score',
                style={'width': '100%'}
            )
        ], width={'size': 6, 'offset': 3}, className='dropdown-container')
    ]),
    dbc.Row([
        dbc.Col(dcc.Graph(id='world-map'), width=12),
    ]),
    dbc.Row([
        dbc.Col([
            html.Div(id='data-insights', className='data-insights'),
            html.Div(id='top-bottom-countries', className='top-bottom-countries')
        ], width=8),
        dbc.Col([
            html.Div(id='country-details', className='country-details'),
        ], width=4)
    ])
], fluid=True)
# World Map callback
@app.callback(
    Output('world-map', 'figure'),
    [Input('metric-dropdown', 'value'),]
)
def update_world_map(selected_value):
    fig = px.choropleth(
        df,
        locations='Country or region',
        locationmode='country names',
        color=selected_value,
        hover_name='Country or region',
        color_continuous_scale=px.colors.sequential.Aggrnyl,
        title=f'World Happiness Index: {selected_value}'
    )
    fig.update_layout(margin={'t': 40, 'r': 0, 'l': 0, 'b': 40})
    return fig
#  Highest and Lowest Country
@app.callback(
    Output('data-insights', 'children'),
    Input('metric-dropdown', 'value')
)
def update_insights(selected_value):
    highest = df.loc[df[selected_value].idxmax()]
    lowest = df.loc[df[selected_value].idxmin()]
    insights = [
        html.H3(f"Highest {selected_value}: {highest['Country or region']} ({highest[selected_value]})"),
        html.H3(f"Lowest {selected_value}: {lowest['Country or region']} ({lowest[selected_value]})")
    ]
    return insights
# Top and Bottom
@app.callback(
    Output('top-bottom-countries', 'children'),
    Input('metric-dropdown', 'value')
)
def update_top_bottom(selected_value):
    top_countries = df.nlargest(5, selected_value)
    bottom_countries = df.nsmallest(5, selected_value)
    top_countries_list = html.Ul([
        html.Li(f"{row['Country or region']}: {row[selected_value]}") for _, row in top_countries.iterrows()
    ])
    bottom_countries_list = html.Ul([
        html.Li(f"{row['Country or region']}: {row[selected_value]}") for _, row in bottom_countries.iterrows()
    ])
    return html.Div([
        html.Div([
            html.H3("Top 5 Countries"),
            top_countries_list
        ], className='top-bottom-section'),
        html.Div([
            html.H3("Bottom 5 Countries"),
            bottom_countries_list
        ], className='top-bottom-section'),
        ], className='top-bottom-container')
# Country details
@app.callback(
    Output('country-details', 'children'),
    Input('world-map', 'clickData')
)
def display_country_details(clickData):
    if clickData:
        country_name = clickData['points'][0]['location']
        country_data = df[df['Country or region'] == country_name]
        if not country_data.empty:
            country = country_data.iloc[0]
            details = [
                html.H3(f"Details for {country_name}"),
                html.P(f"Overall Rank: {country['Overall rank']}"),
                html.P(f"Score: {country['Score']}"),
                html.P(f"GDP per Capita: {country['GDP per capita']}"),
                html.P(f"Social Support: {country['Social support']}"),
                html.P(f"Healthy Life Expectancy: {country['Healthy life expectancy']}"),
                html.P(f"Freedom to Make Life Choices: {country['Freedom to make life choices']}"),
                html.P(f"Generosity: {country['Generosity']}"),
                html.P(f"Perceptions of Corruption: {country['Perceptions of corruption']}")
            ]
            return html.Div(details, className='country-details-section')
    return html.Div("Click on a country to see details.", className='country-details-section')
if __name__ =="__main__":
    app.run_server(debug=True)