use std::collections::HashMap;

use serde::{Deserialize, Deserializer};

#[derive(Debug, Deserialize)]
pub struct InfoJson {
    pub id: String,
    pub title: String,
    pub formats: Vec<Format>,
    pub thumbnails: Vec<ThumbNail>,
    pub thumbnail: String,
    pub description: String,
    pub channel_id: String,
    pub channel_url: String,
    pub duration: u32,
    pub view_count: u32,
    pub age_limit: u32,
    pub webpage_url: String,
    pub categories: Vec<String>,
    pub tags: Vec<String>,
    pub playable_in_embed: bool,
    pub live_status: String,
    _format_sort_fields: Vec<String>,
    pub automatic_captions: HashMap<String, Vec<Caption>>,
    pub subtitles: Subtitles,
    pub comment_count: u32,
    pub like_count: u32,
    pub channel: String,
    pub channel_follower_count: u32,
    pub channel_is_verified: Option<bool>,
    pub uploader: String,
    pub uploader_id: String,
    pub uploader_url: String,
    pub upload_date: String,
    pub availability: String,
    pub webpage_url_basename: String,
    pub webpage_url_domain: String,
    pub extractor: String,
    pub extractor_key: String,
    pub display_id: String,
    pub fulltitle: String,
    pub duration_string: String,
    pub is_live: bool,
    pub was_live: bool,
    pub epoch: u32,
    pub comments: Vec<Comment>,
    pub sponsorblock_chapters: Option<Vec<SponsorblockChapter>>,
    pub format: String,
    pub format_id: String,
    pub ext: String,
    pub protocol: String,
    pub language: Option<String>,
    pub format_note: String,
    pub filesize_approx: u64,
    pub tbr: f64,
    pub width: u32,
    pub height: u32,
    pub resolution: String,
    pub fps: f64,
    pub dynamic_range: String,
    pub vcodec: String,
    pub vbr: f64,
    pub aspect_ratio: f64,
    pub acodec: String,
    pub abr: f64,
    pub asr: u32,
    pub audio_channels: u32,
    _type: String,
    _version: Version,
}

#[derive(Debug, Deserialize)]
pub struct Subtitles {}

#[derive(Debug, Deserialize)]
pub struct Version {
    pub version: String,
    pub release_git_head: String,
    pub repository: String,
}

#[derive(Debug, Deserialize)]
pub struct SponsorblockChapter {}

#[derive(Debug, Deserialize, Clone)]
#[serde(from = "String")]
pub enum Parent {
    Root,
    Id(String),
}

impl Parent {
    pub fn id(&self) -> Option<&str> {
        if let Self::Id(id) = self {
            Some(id)
        } else {
            None
        }
    }
}

impl From<String> for Parent {
    fn from(value: String) -> Self {
        if value == "root" {
            Self::Root
        } else {
            Self::Id(value)
        }
    }
}

#[derive(Debug, Deserialize, Clone)]
#[serde(from = "String")]
pub struct Id {
    pub id: String,
}
impl From<String> for Id {
    fn from(value: String) -> Self {
        Self {
            // Take the last element if the string is split with dots, otherwise take the full id
            id: value.split('.').last().unwrap_or(&value).to_owned(),
        }
    }
}

#[derive(Debug, Deserialize, Clone)]
pub struct Comment {
    pub id: Id,
    pub text: String,
    #[serde(default = "zero")]
    pub like_count: u32,
    pub author_id: String,
    #[serde(default = "unknown")]
    pub author: String,
    pub author_thumbnail: String,
    pub parent: Parent,
    #[serde(deserialize_with = "edited_from_time_text", alias = "_time_text")]
    pub edited: bool,
    // Can't also be deserialized, as it's already used in 'edited'
    // _time_text: String,
    pub timestamp: i64,
    pub author_url: String,
    pub author_is_uploader: bool,
    pub is_favorited: bool,
}
fn unknown() -> String {
    "<Unknown>".to_string()
}
fn zero() -> u32 {
    0
}
fn edited_from_time_text<'de, D>(d: D) -> Result<bool, D::Error>
where
    D: Deserializer<'de>,
{
    let s = String::deserialize(d)?;
    if s.contains(" (edited)") {
        Ok(true)
    } else {
        Ok(false)
    }
}

#[derive(Debug, Deserialize)]
pub struct Caption {
    pub ext: String,
    pub url: String,
    pub name: Option<String>,
    pub protocol: Option<String>,
}

#[derive(Debug, Deserialize)]
pub struct ThumbNail {
    pub url: String,
    pub preference: i32,
    pub id: String,
    pub height: Option<u32>,
    pub width: Option<u32>,
    pub resolution: Option<String>,
}

#[derive(Debug, Deserialize)]
pub struct Format {
    pub format_id: String,
    pub format_note: Option<String>,
    pub ext: String,
    pub protocol: String,
    pub acodec: Option<String>,
    pub vcodec: String,
    pub url: String,
    pub width: Option<u32>,
    pub height: Option<u32>,
    pub fps: Option<f64>,
    pub rows: Option<u32>,
    pub columns: Option<u32>,
    pub fragments: Option<Vec<Fragment>>,
    pub resolution: String,
    pub aspect_ratio: Option<f64>,
    pub http_headers: HttpHeader,
    pub audio_ext: String,
    pub video_ext: String,
    pub vbr: Option<f64>,
    pub abr: Option<f64>,
    pub format: String,
}

#[derive(Debug, Deserialize)]
pub struct HttpHeader {
    #[serde(alias = "User-Agent")]
    pub user_agent: String,
    #[serde(alias = "Accept")]
    pub accept: String,
    #[serde(alias = "Accept-Language")]
    pub accept_language: String,
    #[serde(alias = "Sec-Fetch-Mode")]
    pub sec_fetch_mode: String,
}

#[derive(Debug, Deserialize)]
pub struct Fragment {
    pub url: String,
    pub duration: f64,
}