//! Implements `buffer_interface::*` traits for enums dispatching to backends

use crate::{backend_interface::*, backends, InitError, Rect, SoftBufferError};

use raw_window_handle::{HasDisplayHandle, HasWindowHandle};
use std::fmt;
use std::num::NonZeroU32;

/// A macro for creating the enum used to statically dispatch to the platform-specific implementation.
macro_rules! make_dispatch {
    (
        <$dgen: ident, $wgen: ident> =>
        $(
            $(#[$attr:meta])*
            $name: ident
            ($context_inner: ty, $surface_inner: ty, $buffer_inner: ty),
        )*
    ) => {
        #[derive(Clone)]
        pub(crate) enum ContextDispatch<$dgen> {
            $(
                $(#[$attr])*
                $name($context_inner),
            )*
        }

        impl<D: HasDisplayHandle> ContextDispatch<D> {
            pub fn variant_name(&self) -> &'static str {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(_) => stringify!($name),
                    )*
                }
            }
        }

        impl<D: HasDisplayHandle> ContextInterface<D> for ContextDispatch<D> {
            fn new(mut display: D) -> Result<Self, InitError<D>>
            where
                D: Sized,
            {
                $(
                    $(#[$attr])*
                    match <$context_inner as ContextInterface<D>>::new(display) {
                        Ok(x) => {
                            return Ok(Self::$name(x));
                        }
                        Err(InitError::Unsupported(d)) => display = d,
                        Err(InitError::Failure(f)) => return Err(InitError::Failure(f)),
                    }
                )*

                Err(InitError::Unsupported(display))
            }
        }

        impl<D: fmt::Debug> fmt::Debug for ContextDispatch<D> {
            fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.fmt(f),
                    )*
                }
            }
        }

        #[allow(clippy::large_enum_variant)] // it's boxed anyways
        pub(crate) enum SurfaceDispatch<$dgen, $wgen> {
            $(
                $(#[$attr])*
                $name($surface_inner),
            )*
        }

        impl<D: HasDisplayHandle, W: HasWindowHandle> SurfaceInterface<D, W> for SurfaceDispatch<D, W> {
            type Context = ContextDispatch<D>;
            type Buffer<'a> = BufferDispatch<'a, D, W> where Self: 'a;

            fn new(window: W, display: &Self::Context) -> Result<Self, InitError<W>>
            where
                W: Sized,
            Self: Sized {
                match display {
                    $(
                        $(#[$attr])*
                        ContextDispatch::$name(inner) => Ok(Self::$name(<$surface_inner>::new(window, inner)?)),
                    )*
                }
            }

            fn window(&self) -> &W {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.window(),
                    )*
                }
            }

            fn resize(&mut self, width: NonZeroU32, height: NonZeroU32) -> Result<(), SoftBufferError> {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.resize(width, height),
                    )*
                }
            }

            fn buffer_mut(&mut self) -> Result<BufferDispatch<'_, D, W>, SoftBufferError> {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => Ok(BufferDispatch::$name(inner.buffer_mut()?)),
                    )*
                }
            }

            fn fetch(&mut self) -> Result<Vec<u32>, SoftBufferError> {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.fetch(),
                    )*
                }
            }
        }

        impl<D: fmt::Debug, W: fmt::Debug> fmt::Debug for SurfaceDispatch<D, W> {
            fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.fmt(f),
                    )*
                }
            }
        }

        pub(crate) enum BufferDispatch<'a, $dgen, $wgen> {
            $(
                $(#[$attr])*
                $name($buffer_inner),
            )*
        }

        impl<'a, D: HasDisplayHandle, W: HasWindowHandle> BufferInterface for BufferDispatch<'a, D, W> {
            #[inline]
            fn width(&self) -> NonZeroU32 {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.width(),
                    )*
                }
            }

            #[inline]
            fn height(&self) -> NonZeroU32 {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.height(),
                    )*
                }
            }

            #[inline]
            fn pixels(&self) -> &[u32] {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.pixels(),
                    )*
                }
            }

            #[inline]
            fn pixels_mut(&mut self) -> &mut [u32] {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.pixels_mut(),
                    )*
                }
            }

            fn age(&self) -> u8 {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.age(),
                    )*
                }
            }

            fn present(self) -> Result<(), SoftBufferError> {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.present(),
                    )*
                }
            }

            fn present_with_damage(self, damage: &[Rect]) -> Result<(), SoftBufferError> {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.present_with_damage(damage),
                    )*
                }
            }
        }

        impl<D: fmt::Debug, W: fmt::Debug> fmt::Debug for BufferDispatch<'_, D, W> {
            fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                match self {
                    $(
                        $(#[$attr])*
                        Self::$name(inner) => inner.fmt(f),
                    )*
                }
            }
        }
    };
}

// XXX empty enum with generic bound is invalid?

make_dispatch! {
    <D, W> =>
    #[cfg(target_os = "android")]
    Android(D, backends::android::AndroidImpl<D, W>, backends::android::BufferImpl<'a, D, W>),
    #[cfg(all(
        feature = "x11",
        not(any(
            target_os = "android",
            target_vendor = "apple",
            target_os = "redox",
            target_family = "wasm",
            target_os = "windows"
        ))
    ))]
    X11(std::sync::Arc<backends::x11::X11DisplayImpl<D>>, backends::x11::X11Impl<D, W>, backends::x11::BufferImpl<'a, D, W>),
    #[cfg(all(
        feature = "wayland",
        not(any(
            target_os = "android",
            target_vendor = "apple",
            target_os = "redox",
            target_family = "wasm",
            target_os = "windows"
        ))
    ))]
    Wayland(std::sync::Arc<backends::wayland::WaylandDisplayImpl<D>>, backends::wayland::WaylandImpl<D, W>, backends::wayland::BufferImpl<'a, D, W>),
    #[cfg(all(
        feature = "kms",
        not(any(
            target_os = "android",
            target_vendor = "apple",
            target_os = "redox",
            target_family = "wasm",
            target_os = "windows"
        ))
    ))]
    Kms(std::sync::Arc<backends::kms::KmsDisplayImpl<D>>, backends::kms::KmsImpl<D, W>, backends::kms::BufferImpl<'a, D, W>),
    #[cfg(target_os = "windows")]
    Win32(D, backends::win32::Win32Impl<D, W>, backends::win32::BufferImpl<'a, D, W>),
    #[cfg(target_vendor = "apple")]
    CoreGraphics(D, backends::cg::CGImpl<D, W>, backends::cg::BufferImpl<'a, D, W>),
    #[cfg(target_family = "wasm")]
    Web(backends::web::WebDisplayImpl<D>, backends::web::WebImpl<D, W>, backends::web::BufferImpl<'a, D, W>),
    #[cfg(target_os = "redox")]
    Orbital(D, backends::orbital::OrbitalImpl<D, W>, backends::orbital::BufferImpl<'a, D, W>),
}