use std::io::Cursor;
use freya_core::{
    event_loop_messages::EventLoopMessage,
    parsing::Parse,
    plugins::{
        FreyaPlugin,
        PluginsManager,
    },
    style::default_fonts,
};
use freya_engine::prelude::Color;
use image::ImageReader;
use winit::{
    event_loop::EventLoopBuilder,
    window::{
        Icon,
        Window,
        WindowAttributes,
    },
};
pub type WindowCallback = Box<dyn FnOnce(&mut Window)>;
pub type EventLoopBuilderHook = Box<dyn FnOnce(&mut EventLoopBuilder<EventLoopMessage>)>;
pub type WindowBuilderHook = Box<dyn FnOnce(WindowAttributes) -> WindowAttributes>;
pub type EmbeddedFonts<'a> = Vec<(&'a str, &'a [u8])>;
pub struct WindowConfig {
    pub size: (f64, f64),
    pub min_size: Option<(f64, f64)>,
    pub max_size: Option<(f64, f64)>,
    pub decorations: bool,
    pub title: &'static str,
    pub transparent: bool,
    pub background: Color,
    pub visible: bool,
    pub icon: Option<Icon>,
    pub on_setup: Option<WindowCallback>,
    pub on_exit: Option<WindowCallback>,
    pub window_attributes_hook: Option<WindowBuilderHook>,
    pub event_loop_builder_hook: Option<EventLoopBuilderHook>,
}
impl Default for WindowConfig {
    fn default() -> Self {
        Self {
            size: (600.0, 600.0),
            min_size: None,
            max_size: None,
            decorations: true,
            title: "Freya app",
            transparent: false,
            background: Color::WHITE,
            visible: true,
            icon: None,
            on_setup: None,
            on_exit: None,
            window_attributes_hook: None,
            event_loop_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: default_fonts(),
        }
    }
}
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 = ImageReader::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")
    }
}
impl<'a, T: Clone> LaunchConfig<'a, T> {
    pub fn with_size(mut self, width: f64, height: f64) -> Self {
        self.window_config.size = (width, height);
        self
    }
    pub fn with_min_size(mut self, min_width: f64, min_height: f64) -> Self {
        self.window_config.min_size = Some((min_width, min_height));
        self
    }
    pub fn with_max_size(mut self, max_width: f64, max_height: f64) -> Self {
        self.window_config.max_size = Some((max_width, 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_visible(mut self, visible: bool) -> Self {
        self.window_config.visible = visible;
        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: impl FnOnce(&mut Window) + 'static) -> Self {
        self.window_config.on_setup = Some(Box::new(callback));
        self
    }
    pub fn on_exit(mut self, callback: impl FnOnce(&mut Window) + 'static) -> Self {
        self.window_config.on_exit = Some(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_attributes_hook: impl FnOnce(WindowAttributes) -> WindowAttributes + 'static,
    ) -> Self {
        self.window_config.window_attributes_hook = Some(Box::new(window_attributes_hook));
        self
    }
    pub fn with_event_loop_builder(
        mut self,
        event_loop_builder_hook: impl FnOnce(&mut EventLoopBuilder<EventLoopMessage>) + 'static,
    ) -> Self {
        self.window_config.event_loop_builder_hook = Some(Box::new(event_loop_builder_hook));
        self
    }
}