import folium
import pandas as pd

from folium.plugins import TimestampedGeoJson

def draw_multiship_routes(csv_path, output_html_path):
    df = pd.read_csv(csv_path, parse_dates=['postime'])
    required_cols = ['lat', 'lon', 'postime', 'mmsi', "yt"]
    map_center = [df['lat'].mean(), df['lon'].mean()]
    m = folium.Map(location=map_center, zoom_start=6)

    folium.TileLayer('CartoDB positron', name='Simple Map', attr='default').add_to(m)
    folium.TileLayer('Stamen Terrain', name='Terrain Map', attr='default').add_to(m)
    folium.TileLayer(
        tiles='https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}',
        attr='Esri', name='Satellite Imagery'
    ).add_to(m)

    colors = ['blue', 'red', 'green', 'purple', 'orange', 'darkred', 'cadetblue', 'darkpurple']
    all_features = []
    grouped = df.groupby('mmsi')

    for i, (ship_id, group_df) in enumerate(grouped):
        color = [colors[i] for i in group_df["type"]]
        ship_layer = folium.FeatureGroup(name=f"Static Route: {ship_id}")

        group_df = group_df.sort_values(by='postime').reset_index(drop=True)

        points = list(zip(group_df['lat'], group_df['lon']))

        folium.PolyLine(
            locations=points,
            color=[colors[i] for i in group_df["type"]],
            weight=3,
            opacity=0.7
        ).add_to(ship_layer)

        ship_layer.add_to(m)


        point_features = []
        for _, row in group_df.iterrows():
            point_features.append({
                'type': 'Feature',
                'geometry': {'type': 'Point', 'coordinates': [row['lon'], row['lat']]},
                'properties': {
                    'time': row['postime'].isoformat(),
                    'icon': 'circle',
                    'iconstyle': {'fillColor': color, 'fillOpacity': 0.8, 'stroke': 'true', 'radius': 7},
                    'popup': f"<b>{row.get('mmsi', ship_id)}</b><br>{row['postime'].strftime('%Y-%m-%d %H:%M')}<br>Lat: {row['lat']} <br>Lon: {row['lon']}"
                }
            })


        line_coordinates = list(zip(group_df['lon'], group_df['lat']))
        line_times = [dt.isoformat() for dt in group_df['postime']]
        line_feature = {
            'type': 'Feature',
            'geometry': {'type': 'LineString', 'coordinates': line_coordinates},
            'properties': {'times': line_times, 'style': {'color': color, 'weight': 5}}
        }

        all_features.extend(point_features)
        all_features.append(line_feature)

    if all_features:
        TimestampedGeoJson(
            {'type': 'FeatureCollection', 'features': all_features},
            period='PT5M',
            add_last_point=True,
            auto_play=False,
            loop=False,
            max_speed=5,
            loop_button=True,
            time_slider_drag_update=True,
        ).add_to(m)

    folium.LayerControl().add_to(m)

    m.save(output_html_path)

def draw_single_color_traj(df, m, color, feature_group_name):
    """
    Draw a single-color trajectory on an existing map
    
    Parameters:
    df: DataFrame, containing trajectory data
    m: folium.Map object
    color: str, trajectory color
    feature_group_name: str, layer name
    
    Returns:
    list, all added features
    """
    ship_layer = folium.FeatureGroup(name=feature_group_name)
    
    # Sort by time
    df = df.sort_values(by='postime').reset_index(drop=True)
    
    # Draw static route
    points = list(zip(df['lat'], df['lon']))
    folium.PolyLine(
        locations=points,
        color=color,
        weight=3,
        opacity=0.7
    ).add_to(ship_layer)
    
    ship_layer.add_to(m)
    
    # Prepare timestamp animation data
    all_features = []
    point_features = []
    
    # Add point features
    for _, row in df.iterrows():
        point_features.append({
            'type': 'Feature',
            'geometry': {'type': 'Point', 'coordinates': [row['lon'], row['lat']]},
            'properties': {
                'time': row['postime'].isoformat(),
                'icon': 'circle',
                'iconstyle': {'fillColor': color, 'fillOpacity': 0.8, 'stroke': 'true', 'radius': 7},
                'popup': f"<b>{row.get('mmsi', 'Unknown')}</b><br>{row['postime'].strftime('%Y-%m-%d %H:%M')}<br>Lat: {row['lat']} <br>Lon: {row['lon']}"
            }
        })
    
    # Add line feature
    line_coordinates = list(zip(df['lon'], df['lat']))
    line_times = [dt.isoformat() for dt in df['postime']]
    line_feature = {
        'type': 'Feature',
        'geometry': {'type': 'LineString', 'coordinates': line_coordinates},
        'properties': {'times': line_times, 'style': {'color': color, 'weight': 5}}
    }
    
    all_features.extend(point_features)
    all_features.append(line_feature)
    
    return all_features

def draw_single_traj_route(df, output_html_path):
    """
    Draw a single trajectory with different colors based on the type field
    
    Parameters:
    df: DataFrame, containing trajectory data, must include lat, lon, postime, type, mmsi columns
    output_html_path: str, output HTML file path
    """
    # Validate required columns
    required_cols = ['lat', 'lon', 'postime', 'type', 'mmsi']
    for col in required_cols:
        if col not in df.columns:
            raise ValueError(f"DataFrame missing required column: {col}")
    
    # Ensure postime is datetime type
    if not pd.api.types.is_datetime64_any_dtype(df['postime']):
        df['postime'] = pd.to_datetime(df['postime'])
    
    # Create map
    map_center = [df['lat'].mean(), df['lon'].mean()]
    m = folium.Map(location=map_center, zoom_start=10)
    
    # Add different base map layers
    folium.TileLayer('CartoDB positron', name='Simple Map', attr='default').add_to(m)
    folium.TileLayer('Stamen Terrain', name='Terrain Map', attr='default').add_to(m)
    folium.TileLayer(
        tiles='https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}',
        attr='Esri', name='Satellite Imagery'
    ).add_to(m)
    
    # Collect all features
    all_features = []
    
    # Process type=0 trajectory (blue) - according to requirements, blue trajectory must exist
    blue_df = df[df['type'] == 0].copy()
    blue_df = blue_df.sort_values(by='postime').reset_index(drop=True)
    blue_features = draw_single_color_traj(blue_df, m, 'blue', 'Static Route: type=0')
    all_features.extend(blue_features)
    
    # Process type=1 trajectory (red)
    red_df = df[df['type'] == 1].copy()
    red_df = red_df.sort_values(by='postime').reset_index(drop=True)
    
    # Get the last point of the blue trajectory and add it to the beginning of the red trajectory
    blue_last_point = blue_df.iloc[-1:].copy()
    
    # Merge blue last point and red trajectory
    red_df = pd.concat([blue_last_point, red_df], ignore_index=True)
    
    red_features = draw_single_color_traj(red_df, m, 'red', 'Static Route: type=1 (Predicted)')
    all_features.extend(red_features)
    
    # Process type=2 trajectory (green - actual values)
    if 2 in df['type'].values:
        green_df = df[df['type'] == 2].copy()
        green_df = green_df.sort_values(by='postime').reset_index(drop=True)
        
        # Get the last point of the blue trajectory and add it to the beginning of the green trajectory
        green_df = pd.concat([blue_last_point, green_df], ignore_index=True)
        
        green_features = draw_single_color_traj(green_df, m, 'green', 'Static Route: type=2 (Actual)')
        all_features.extend(green_features)

    # Add timestamp animation
    if all_features:
        TimestampedGeoJson(
            {'type': 'FeatureCollection', 'features': all_features},
            period='PT5M',
            add_last_point=True,
            auto_play=False,
            loop=False,
            max_speed=5,
            loop_button=True,
            time_slider_drag_update=True,
        ).add_to(m)
    
    # Add layer control
    folium.LayerControl().add_to(m)
    
    # Save map
    m.save(output_html_path)
    return output_html_path
