use std::{
    io::Cursor,
    sync::Arc,
};
use freya_core::plugins::{
    FreyaPlugin,
    PluginsManager,
};
use freya_engine::prelude::Color;
use freya_node_state::Parse;
use image::io::Reader;
use winit::window::{
    Icon,
    Window,
    WindowAttributes,
};
pub type WindowBuilderHook = Box<dyn Fn(WindowAttributes) -> WindowAttributes>;
pub type EmbeddedFonts<'a> = Vec<(&'a str, &'a [u8])>;
pub struct WindowConfig {
    pub width: f64,
    pub height: f64,
    pub min_width: Option<f64>,
    pub min_height: Option<f64>,
    pub max_width: Option<f64>,
    pub max_height: Option<f64>,
    pub decorations: bool,
    pub title: &'static str,
    pub transparent: bool,
    pub background: Color,
    pub icon: Option<Icon>,
    pub on_setup: Option<WindowCallback>,
    pub on_exit: Option<WindowCallback>,
    pub window_builder_hook: Option<WindowBuilderHook>,
}
impl Default for WindowConfig {
    fn default() -> Self {
        Self {
            width: 600.0,
            height: 600.0,
            min_width: None,
            min_height: None,
            max_height: None,
            max_width: None,
            decorations: true,
            title: "Freya app",
            transparent: false,
            background: Color::WHITE,
            icon: None,
            on_setup: None,
            on_exit: None,
            window_builder_hook: None,
        }
    }
}
pub struct LaunchConfig<'a, T: Clone = ()> {
    pub state: Option<T>,
    pub window_config: WindowConfig,
    pub embedded_fonts: EmbeddedFonts<'a>,
    pub plugins: PluginsManager,
    pub default_fonts: Vec<String>,
}
impl<'a, T: Clone> Default for LaunchConfig<'a, T> {
    fn default() -> Self {
        Self {
            state: None,
            window_config: Default::default(),
            embedded_fonts: Default::default(),
            plugins: Default::default(),
            default_fonts: vec!["Fira Sans".to_string()],
        }
    }
}
impl<'a, T: Clone> LaunchConfig<'a, T> {
    pub fn new() -> LaunchConfig<'a, T> {
        LaunchConfig::default()
    }
}
impl LaunchConfig<'_, ()> {
    pub fn load_icon(icon: &[u8]) -> Icon {
        let reader = Reader::new(Cursor::new(icon))
            .with_guessed_format()
            .expect("Cursor io never fails");
        let image = reader
            .decode()
            .expect("Failed to open icon path")
            .into_rgba8();
        let (width, height) = image.dimensions();
        let rgba = image.into_raw();
        Icon::from_rgba(rgba, width, height).expect("Failed to open icon")
    }
}
pub type WindowCallback = Arc<Box<fn(&mut Window)>>;
impl<'a, T: Clone> LaunchConfig<'a, T> {
    pub fn with_width(mut self, width: f64) -> Self {
        self.window_config.width = width;
        self
    }
    pub fn with_height(mut self, height: f64) -> Self {
        self.window_config.height = height;
        self
    }
    pub fn with_min_width(mut self, min_width: f64) -> Self {
        self.window_config.min_width = Some(min_width);
        self
    }
    pub fn with_min_height(mut self, min_height: f64) -> Self {
        self.window_config.min_height = Some(min_height);
        self
    }
    pub fn with_max_width(mut self, max_width: f64) -> Self {
        self.window_config.max_width = Some(max_width);
        self
    }
    pub fn with_max_height(mut self, max_height: f64) -> Self {
        self.window_config.max_height = Some(max_height);
        self
    }
    pub fn with_decorations(mut self, decorations: bool) -> Self {
        self.window_config.decorations = decorations;
        self
    }
    pub fn with_title(mut self, title: &'static str) -> Self {
        self.window_config.title = title;
        self
    }
    pub fn with_transparency(mut self, transparency: bool) -> Self {
        self.window_config.transparent = transparency;
        self
    }
    pub fn with_state(mut self, state: T) -> Self {
        self.state = Some(state);
        self
    }
    pub fn with_background(mut self, background: &str) -> Self {
        self.window_config.background = Color::parse(background).unwrap_or(Color::WHITE);
        self
    }
    pub fn with_font(mut self, font_name: &'a str, font: &'a [u8]) -> Self {
        self.embedded_fonts.push((font_name, font));
        self
    }
    pub fn without_default_fonts(mut self) -> Self {
        self.default_fonts.clear();
        self
    }
    pub fn with_default_font(mut self, font_name: &str) -> Self {
        self.default_fonts.push(font_name.to_string());
        self
    }
    pub fn with_icon(mut self, icon: Icon) -> Self {
        self.window_config.icon = Some(icon);
        self
    }
    pub fn on_setup(mut self, callback: fn(&mut Window)) -> Self {
        self.window_config.on_setup = Some(Arc::new(Box::new(callback)));
        self
    }
    pub fn on_exit(mut self, callback: fn(&mut Window)) -> Self {
        self.window_config.on_exit = Some(Arc::new(Box::new(callback)));
        self
    }
    pub fn with_plugin(mut self, plugin: impl FreyaPlugin + 'static) -> Self {
        self.plugins.add_plugin(plugin);
        self
    }
    pub fn with_window_attributes(
        mut self,
        window_builder_hook: impl Fn(WindowAttributes) -> WindowAttributes + 'static,
    ) -> Self {
        self.window_config.window_builder_hook = Some(Box::new(window_builder_hook));
        self
    }
}