use super::*;
use std::default::Default;
use std::fmt;
use std::mem::{forget, transmute};
use crate::private::get_api;
pub struct Variant(pub(crate) sys::godot_variant);
macro_rules! variant_constructors {
    (
        $(
            $(#[$attr:meta])*
            pub fn $ctor:ident($Type:ty) -> Self;
        )*
    ) => (
        $(
            $(#[$attr])*
            pub fn $ctor(val: $Type) -> Variant {
                ToVariant::to_variant(val)
            }
        )*
    )
}
macro_rules! variant_to_type_transmute {
    (
        $(
            $(#[$to_attr:meta])*
            pub fn $to_method:ident(&self) -> $ToType:ident : $to_gd_method:ident;
            $(#[$try_attr:meta])*
            pub fn $try_method:ident(&self) -> Option<$TryType:ident>;
        )*
    ) => (
        $(
            $(#[$to_attr])*
            pub fn $to_method(&self) -> $ToType {
                unsafe {
                    #[allow(clippy::useless_transmute)]
                    transmute((get_api().$to_gd_method)(&self.0))
                }
            }
            $(#[$try_attr])*
            pub fn $try_method(&self) -> Option<$TryType> {
                $TryType::from_variant(self).ok()
            }
        )*
    )
}
macro_rules! variant_to_type_from_sys {
    (
        $(
            $(#[$to_attr:meta])*
            pub fn $to_method:ident(&self) -> $ToType:ident : $to_gd_method:ident;
            $(#[$try_attr:meta])*
            pub fn $try_method:ident(&self) -> Option<$TryType:ident>;
        )*
    ) => (
        $(
            $(#[$to_attr])*
            pub fn $to_method(&self) -> $ToType {
                unsafe {
                    $ToType::from_sys((get_api().$to_gd_method)(&self.0))
                }
            }
            $(#[$try_attr])*
            pub fn $try_method(&self) -> Option<$TryType> {
                $TryType::from_variant(self).ok()
            }
        )*
    )
}
#[repr(u32)]
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum VariantType {
    Nil = sys::godot_variant_type_GODOT_VARIANT_TYPE_NIL as u32,
    Bool = sys::godot_variant_type_GODOT_VARIANT_TYPE_BOOL as u32,
    I64 = sys::godot_variant_type_GODOT_VARIANT_TYPE_INT as u32,
    F64 = sys::godot_variant_type_GODOT_VARIANT_TYPE_REAL as u32,
    GodotString = sys::godot_variant_type_GODOT_VARIANT_TYPE_STRING as u32,
    Vector2 = sys::godot_variant_type_GODOT_VARIANT_TYPE_VECTOR2 as u32,
    Rect2 = sys::godot_variant_type_GODOT_VARIANT_TYPE_RECT2 as u32,
    Vector3 = sys::godot_variant_type_GODOT_VARIANT_TYPE_VECTOR3 as u32,
    Transform2D = sys::godot_variant_type_GODOT_VARIANT_TYPE_TRANSFORM2D as u32,
    Plane = sys::godot_variant_type_GODOT_VARIANT_TYPE_PLANE as u32,
    Quat = sys::godot_variant_type_GODOT_VARIANT_TYPE_QUAT as u32,
    Aabb = sys::godot_variant_type_GODOT_VARIANT_TYPE_AABB as u32,
    Basis = sys::godot_variant_type_GODOT_VARIANT_TYPE_BASIS as u32,
    Transform = sys::godot_variant_type_GODOT_VARIANT_TYPE_TRANSFORM as u32,
    Color = sys::godot_variant_type_GODOT_VARIANT_TYPE_COLOR as u32,
    NodePath = sys::godot_variant_type_GODOT_VARIANT_TYPE_NODE_PATH as u32,
    Rid = sys::godot_variant_type_GODOT_VARIANT_TYPE_RID as u32,
    Object = sys::godot_variant_type_GODOT_VARIANT_TYPE_OBJECT as u32,
    Dictionary = sys::godot_variant_type_GODOT_VARIANT_TYPE_DICTIONARY as u32,
    VariantArray = sys::godot_variant_type_GODOT_VARIANT_TYPE_ARRAY as u32,
    ByteArray = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_BYTE_ARRAY as u32,
    Int32Array = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_INT_ARRAY as u32,
    Float32Array = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_REAL_ARRAY as u32,
    StringArray = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_STRING_ARRAY as u32,
    Vector2Array = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_VECTOR2_ARRAY as u32,
    Vector3Array = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_VECTOR3_ARRAY as u32,
    ColorArray = sys::godot_variant_type_GODOT_VARIANT_TYPE_POOL_COLOR_ARRAY as u32,
}
impl VariantType {
    #[doc(hidden)]
    pub fn from_sys(v: sys::godot_variant_type) -> VariantType {
        unsafe { transmute(v as u32) }
    }
}
#[repr(u32)]
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum CallError {
    InvalidMethod =
        sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_INVALID_METHOD as u32,
    InvalidArgument =
        sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_INVALID_ARGUMENT as u32,
    TooManyArguments =
        sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_TOO_MANY_ARGUMENTS as u32,
    TooFewArguments =
        sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_TOO_FEW_ARGUMENTS as u32,
    InstanceIsNull =
        sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_INSTANCE_IS_NULL as u32,
}
impl CallError {
    fn from_sys(v: sys::godot_variant_call_error_error) -> Result<(), CallError> {
        if v == sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_OK {
            Ok(())
        } else {
            debug_assert!(
                (v as u32) <= sys::godot_variant_call_error_error_GODOT_CALL_ERROR_CALL_ERROR_INSTANCE_IS_NULL as u32,
                "Godot should have passed a known error",
            );
            Err(unsafe { transmute(v as u32) })
        }
    }
}
#[repr(u32)]
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum VariantOperator {
    
    Equal,        
    NotEqual,     
    Less,         
    LessEqual,    
    Greater,      
    GreaterEqual, 
    
    Add,      
    Subtact,  
    Multiply, 
    Divide,   
    Negate,   
    Positive, 
    Module,   
    Concat,   
    
    ShiftLeft,  
    ShiftRight, 
    BitAnd,     
    BitOr,      
    BitXor,     
    BitNegate,  
    
    And, 
    Or,  
    Xor, 
    Not, 
    
    In,  
    Max, 
}
type F64 = f64;
type I64 = i64;
type Bool = bool;
impl Variant {
    variant_constructors!(
        
        pub fn from_vector2(&Vector2) -> Self;
        
        pub fn from_vector3(&Vector3) -> Self;
        
        pub fn from_quat(&Quat) -> Self;
        
        pub fn from_plane(&Plane) -> Self;
        
        pub fn from_rect2(&Rect2) -> Self;
        
        pub fn from_transform(&Transform) -> Self;
        
        pub fn from_transform2d(&Transform2D) -> Self;
        
        pub fn from_basis(&Basis) -> Self;
        
        pub fn from_color(&Color) -> Self;
        
        pub fn from_aabb(&Aabb) -> Self;
        
        pub fn from_rid(&Rid) -> Self;
        
        pub fn from_node_path(&NodePath) -> Self;
        
        pub fn from_godot_string(&GodotString) -> Self;
        
        pub fn from_array(&VariantArray) -> Self;
        
        pub fn from_byte_array(&ByteArray) -> Self;
        
        pub fn from_int32_array(&Int32Array) -> Self;
        
        pub fn from_float32_array(&Float32Array) -> Self;
        
        pub fn from_string_array(&StringArray) -> Self;
        
        pub fn from_vector2_array(&Vector2Array) -> Self;
        
        pub fn from_vector3_array(&Vector3Array) -> Self;
        
        pub fn from_color_array(&ColorArray) -> Self;
        
        pub fn from_dictionary(&Dictionary) -> Self;
    );
    
    pub fn new() -> Self {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            (api.godot_variant_new_nil)(&mut dest);
            Variant(dest)
        }
    }
    
    pub fn from_str<S>(s: S) -> Variant
    where
        S: AsRef<str>,
    {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            let val = s.as_ref();
            let mut godot_s =
                (api.godot_string_chars_to_utf8_with_len)(val.as_ptr() as *const _, val.len() as _);
            (api.godot_variant_new_string)(&mut dest, &godot_s);
            (api.godot_string_destroy)(&mut godot_s);
            Variant(dest)
        }
    }
    
    pub fn from_object<T>(val: &T) -> Variant
    where
        T: GodotObject,
    {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            (api.godot_variant_new_object)(&mut dest, val.to_sys());
            Variant(dest)
        }
    }
    
    pub fn from_i64(v: i64) -> Variant {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            (api.godot_variant_new_int)(&mut dest, v);
            Variant(dest)
        }
    }
    
    pub fn from_u64(v: u64) -> Variant {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            (api.godot_variant_new_uint)(&mut dest, v);
            Variant(dest)
        }
    }
    
    pub fn from_f64(v: f64) -> Variant {
        unsafe {
            let api = get_api();
            let mut ret = sys::godot_variant::default();
            (api.godot_variant_new_real)(&mut ret, v);
            Variant(ret)
        }
    }
    
    pub fn from_bool(v: bool) -> Variant {
        unsafe {
            let api = get_api();
            let mut dest = sys::godot_variant::default();
            (api.godot_variant_new_bool)(&mut dest, v);
            Variant(dest)
        }
    }
    fn try_as_sys_of_type(
        &self,
        expected: VariantType,
    ) -> Result<&sys::godot_variant, FromVariantError> {
        let variant_type = self.get_type();
        if variant_type != expected {
            return Err(FromVariantError::InvalidVariantType {
                expected,
                variant_type,
            });
        }
        Ok(&self.0)
    }
    variant_to_type_transmute!(
        
        pub fn to_vector2(&self) -> Vector2 : godot_variant_as_vector2;
        
        pub fn try_to_vector2(&self) -> Option<Vector2>;
        
        pub fn to_vector3(&self) -> Vector3 : godot_variant_as_vector3;
        
        pub fn try_to_vector3(&self) -> Option<Vector3>;
        
        pub fn to_quat(&self) -> Quat : godot_variant_as_quat;
        
        pub fn try_to_quat(&self) -> Option<Quat>;
        
        pub fn to_rect2(&self) -> Rect2 : godot_variant_as_rect2;
        
        pub fn try_to_rect2(&self) -> Option<Rect2>;
        
        pub fn to_transform2d(&self) -> Transform2D : godot_variant_as_transform2d;
        
        pub fn try_to_transform2d(&self) -> Option<Transform2D>;
        
        pub fn to_f64(&self) -> F64 : godot_variant_as_real;
        
        pub fn try_to_f64(&self) -> Option<F64>;
        
        pub fn to_i64(&self) -> I64 : godot_variant_as_int;
        
        pub fn try_to_i64(&self) -> Option<I64>;
        
        pub fn to_bool(&self) -> Bool : godot_variant_as_bool;
        
        pub fn try_to_bool(&self) -> Option<Bool>;
    );
    
    pub fn to_u64(&self) -> u64 {
        unsafe {
            let api = get_api();
            (api.godot_variant_as_uint)(&self.0)
        }
    }
    
    pub fn try_to_u64(&self) -> Option<u64> {
        unsafe {
            let api = get_api();
            if (api.godot_variant_get_type)(&self.0)
                == sys::godot_variant_type_GODOT_VARIANT_TYPE_INT
            {
                Some((api.godot_variant_as_uint)(&self.0))
            } else {
                None
            }
        }
    }
    variant_to_type_from_sys!(
        
        pub fn to_plane(&self) -> Plane : godot_variant_as_plane;
        
        pub fn try_to_plane(&self) -> Option<Plane>;
        
        pub fn to_transform(&self) -> Transform : godot_variant_as_transform;
        
        pub fn try_to_transform(&self) -> Option<Transform>;
        
        pub fn to_color(&self) -> Color : godot_variant_as_color;
        
        pub fn try_to_color(&self) -> Option<Color>;
        
        pub fn to_basis(&self) -> Basis : godot_variant_as_basis;
        
        pub fn try_to_basis(&self) -> Option<Basis>;
        
        pub fn to_aabb(&self) -> Aabb : godot_variant_as_aabb;
        
        pub fn try_to_aabb(&self) -> Option<Aabb>;
        
        pub fn to_node_path(&self) -> NodePath : godot_variant_as_node_path;
        
        pub fn try_to_node_path(&self) -> Option<NodePath>;
        
        pub fn to_godot_string(&self) -> GodotString : godot_variant_as_string;
        
        pub fn try_to_godot_string(&self) -> Option<GodotString>;
        
        pub fn to_rid(&self) -> Rid : godot_variant_as_rid;
        
        pub fn try_to_rid(&self) -> Option<Rid>;
        
        pub fn to_array(&self) -> VariantArray : godot_variant_as_array;
        
        pub fn try_to_array(&self) -> Option<VariantArray>;
        
        pub fn to_byte_array(&self) -> ByteArray : godot_variant_as_pool_byte_array;
        
        pub fn try_to_byte_array(&self) -> Option<ByteArray>;
        
        pub fn to_int32_array(&self) -> Int32Array : godot_variant_as_pool_int_array;
        
        pub fn try_to_int32_array(&self) -> Option<Int32Array>;
        
        pub fn to_float32_array(&self) -> Float32Array : godot_variant_as_pool_real_array;
        
        pub fn try_to_float32_array(&self) -> Option<Float32Array>;
        
        pub fn to_string_array(&self) -> StringArray : godot_variant_as_pool_string_array;
        
        pub fn try_to_string_array(&self) -> Option<StringArray>;
        
        pub fn to_vector2_array(&self) -> Vector2Array : godot_variant_as_pool_vector2_array;
        
        pub fn try_to_vector2_array(&self) -> Option<Vector2Array>;
        
        pub fn to_vector3_array(&self) -> Vector3Array : godot_variant_as_pool_vector3_array;
        
        pub fn try_to_vector3_array(&self) -> Option<Vector3Array>;
        
        pub fn to_color_array(&self) -> ColorArray : godot_variant_as_pool_color_array;
        
        pub fn try_to_color_array(&self) -> Option<ColorArray>;
        
        pub fn to_dictionary(&self) -> Dictionary : godot_variant_as_dictionary;
        
        pub fn try_to_dictionary(&self) -> Option<Dictionary>;
    );
    pub fn try_to_object<T>(&self) -> Option<T>
    where
        T: GodotObject,
    {
        self.try_to_object_with_error().ok()
    }
    pub fn try_to_object_with_error<T>(&self) -> Result<T, FromVariantError>
    where
        T: GodotObject,
    {
        unsafe {
            let api = get_api();
            let obj = self.try_as_sys_of_type(VariantType::Object)?;
            let obj = Object::from_sys((api.godot_variant_as_object)(obj));
            obj.cast::<T>().ok_or_else(|| FromVariantError::CannotCast {
                class: obj.get_class().to_string(),
                to: T::class_name(),
            })
        }
    }
    pub fn to_string(&self) -> String {
        self.to_godot_string().to_string()
    }
    pub fn try_to_string(&self) -> Option<String> {
        self.try_to_godot_string().map(|s| s.to_string())
    }
    
    pub fn get_type(&self) -> VariantType {
        unsafe { VariantType::from_sys((get_api().godot_variant_get_type)(&self.0)) }
    }
    
    pub fn is_nil(&self) -> bool {
        self.get_type() == VariantType::Nil
    }
    pub fn has_method(&self, method: &GodotString) -> bool {
        unsafe { (get_api().godot_variant_has_method)(&self.0, &method.0) }
    }
    pub fn call(&mut self, method: &GodotString, args: &[Variant]) -> Result<Variant, CallError> {
        unsafe {
            let api = get_api();
            let mut err = sys::godot_variant_call_error::default();
            let mut arg_refs = args.iter().map(Variant::sys).collect::<Vec<_>>();
            let variant = (api.godot_variant_call)(
                &mut self.0,
                &method.0,
                arg_refs.as_mut_ptr(),
                args.len() as i32,
                &mut err,
            );
            CallError::from_sys(err.error).map(|_| Variant::from_sys(variant))
        }
    }
    pub(crate) fn cast_ref<'l>(ptr: *const sys::godot_variant) -> &'l Variant {
        unsafe { transmute(ptr) }
    }
    pub(crate) fn cast_mut_ref<'l>(ptr: *mut sys::godot_variant) -> &'l mut Variant {
        unsafe { transmute(ptr) }
    }
    
    
    
    
    
    
    pub fn forget(self) -> sys::godot_variant {
        let v = self.0;
        forget(self);
        v
    }
    
    
    
    
    #[doc(hidden)]
    pub fn to_sys(&self) -> sys::godot_variant {
        self.0
    }
    #[doc(hidden)]
    pub fn sys(&self) -> *const sys::godot_variant {
        &self.0
    }
    #[doc(hidden)]
    pub fn from_sys(sys: sys::godot_variant) -> Self {
        Variant(sys)
    }
}
impl_basic_traits!(
    for Variant as godot_variant {
        Drop => godot_variant_destroy;
        Clone => godot_variant_new_copy;
        PartialEq => godot_variant_operator_equal;
    }
);
impl Default for Variant {
    fn default() -> Self {
        Variant::new()
    }
}
impl fmt::Debug for Variant {
    fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
        write!(f, "{:?}({})", self.get_type(), self.to_string())
    }
}
macro_rules! variant_from_ref {
    ($(impl From<&$Type:ty> : $ctor:ident;)*) => (
        $(
            impl<'l> From<&'l $Type> for Variant
            {
                fn from(val: &'l $Type) -> Variant {
                    Variant::$ctor(val)
                }
            }
        )*
    );
}
macro_rules! variant_from_val {
    ($(impl From<$Type:ty> : $ctor:ident;)*) => (
        $(
            impl From<$Type> for Variant
            {
                fn from(val: $Type) -> Variant {
                    Variant::$ctor(val)
                }
            }
        )*
    );
}
variant_from_val!(
    impl From<i64> : from_i64;
    impl From<u64> : from_u64;
    impl From<bool> : from_bool;
);
variant_from_ref!(
    impl From<&Vector2> : from_vector2;
    impl From<&Vector3> : from_vector3;
    impl From<&Quat> : from_quat;
    impl From<&Plane> : from_plane;
    impl From<&Rect2> : from_rect2;
    impl From<&Transform> : from_transform;
    impl From<&Transform2D> : from_transform2d;
    impl From<&Basis> : from_basis;
    impl From<&Color> : from_color;
    impl From<&Aabb> : from_aabb;
    impl From<&String> : from_str;
    impl From<&Rid> : from_rid;
    impl From<&NodePath> : from_node_path;
    impl From<&GodotString> : from_godot_string;
    impl From<&Dictionary> : from_dictionary;
    impl From<&VariantArray> : from_array;
    impl From<&ByteArray> : from_byte_array;
    impl From<&Int32Array> : from_int32_array;
    impl From<&Float32Array> : from_float32_array;
    impl From<&Vector2Array> : from_vector2_array;
    impl From<&Vector3Array> : from_vector3_array;
    impl From<&ColorArray> : from_color_array;
);
impl<'l> From<&'l str> for Variant {
    fn from(v: &str) -> Variant {
        Variant::from_str(v)
    }
}
impl<T> From<T> for Variant
where
    T: GodotObject,
{
    fn from(val: T) -> Variant {
        Variant::from_object(&val)
    }
}
godot_test!(
    test_variant_nil {
        let nil = Variant::new();
        assert_eq!(nil.get_type(), VariantType::Nil);
        assert!(nil.is_nil());
        assert!(nil.try_to_array().is_none());
        assert!(nil.try_to_rid().is_none());
        assert!(nil.try_to_i64().is_none());
        assert!(nil.try_to_bool().is_none());
        assert!(nil.try_to_aabb().is_none());
        assert!(nil.try_to_vector2().is_none());
        assert!(nil.try_to_basis().is_none());
        assert!(!nil.has_method(&GodotString::from_str("foo")));
        let clone = nil.clone();
        assert!(clone == nil);
    }
    test_variant_i64 {
        let v_42 = Variant::from_i64(42);
        assert_eq!(v_42.get_type(), VariantType::I64);
        assert!(!v_42.is_nil());
        assert_eq!(v_42.try_to_i64(), Some(42));
        assert!(v_42.try_to_f64().is_none());
        assert!(v_42.try_to_array().is_none());
        let v_m1 = Variant::from_i64(-1);
        assert_eq!(v_m1.get_type(), VariantType::I64);
        assert!(!v_m1.is_nil());
        assert_eq!(v_m1.try_to_i64(), Some(-1));
        assert!(v_m1.try_to_f64().is_none());
        assert!(v_m1.try_to_array().is_none());
    }
    test_variant_bool {
        let v_true = Variant::from_bool(true);
        assert_eq!(v_true.get_type(), VariantType::Bool);
        assert!(!v_true.is_nil());
        assert_eq!(v_true.try_to_bool(), Some(true));
        assert!(v_true.try_to_f64().is_none());
        assert!(v_true.try_to_array().is_none());
        let v_false = Variant::from_bool(false);
        assert_eq!(v_false.get_type(), VariantType::Bool);
        assert!(!v_false.is_nil());
        assert_eq!(v_false.try_to_bool(), Some(false));
        assert!(v_false.try_to_f64().is_none());
        assert!(v_false.try_to_array().is_none());
    }
);
pub trait ToVariant {
    fn to_variant(&self) -> Variant;
}
pub trait FromVariant: Sized {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError>;
}
#[derive(Clone, PartialEq, Eq, Debug)]
pub enum FromVariantError {
    
    Unspecified,
    
    Custom(String),
    
    InvalidNil,
    
    InvalidVariantType {
        variant_type: VariantType,
        expected: VariantType,
    },
    
    CannotCast { class: String, to: &'static str },
    
    InvalidLength { len: usize, expected: usize },
    
    InvalidEnumRepr {
        expected: VariantEnumRepr,
        error: Box<FromVariantError>,
    },
    
    InvalidStructRepr {
        expected: VariantStructRepr,
        error: Box<FromVariantError>,
    },
    
    
    
    
    
    UnknownEnumVariant {
        
        variant: String,
        
        expected: &'static [&'static str],
    },
    
    
    
    
    
    
    
    
    InvalidEnumVariant {
        variant: &'static str,
        error: Box<FromVariantError>,
    },
    
    InvalidInstance { expected: &'static str },
    
    InvalidField {
        field_name: &'static str,
        error: Box<FromVariantError>,
    },
    
    InvalidItem {
        index: usize,
        error: Box<FromVariantError>,
    },
}
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
pub enum VariantEnumRepr {
    ExternallyTagged,
}
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
pub enum VariantStructRepr {
    Unit,
    Tuple,
    Struct,
}
impl FromVariantError {
    
    pub fn custom<T: fmt::Display>(message: T) -> Self {
        FromVariantError::Custom(format!("{}", message))
    }
}
impl fmt::Display for FromVariantError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        use FromVariantError as E;
        match self {
            E::Unspecified => write!(f, "unspecified error"),
            E::Custom(s) => write!(f, "{}", s),
            E::InvalidNil => write!(f, "expected non-nullable type, got null"),
            E::InvalidVariantType {
                variant_type,
                expected,
            } => write!(
                f,
                "invalid variant type: expected {:?}, got {:?}",
                expected, variant_type
            ),
            E::CannotCast { class, to } => {
                write!(f, "cannot cast object of class {} to {}", class, to)
            }
            E::InvalidLength { len, expected } => {
                write!(f, "expected collection of length {}, got {}", expected, len)
            }
            E::InvalidEnumRepr { expected, error } => write!(
                f,
                "invalid enum representation: expected {:?}, {}",
                expected, error
            ),
            E::InvalidStructRepr { expected, error } => write!(
                f,
                "invalid struct representation: expected {:?}, {}",
                expected, error
            ),
            E::UnknownEnumVariant { variant, expected } => {
                write!(
                    f,
                    "unknown enum variant {}, expected variants are: ",
                    variant
                )?;
                let mut first = true;
                for v in *expected {
                    if first {
                        first = false;
                    } else {
                        write!(f, ", ")?;
                    }
                    write!(f, "{}", v)?;
                }
                Ok(())
            }
            E::InvalidEnumVariant { variant, error } => {
                write!(f, "invalid value for variant {}: {}", variant, error)
            }
            E::InvalidInstance { expected } => {
                write!(f, "object is not an instance of NativeClass {}", expected)
            }
            E::InvalidField { field_name, error } => {
                write!(f, "invalid value for field {}", field_name)?;
                let mut next_error = error.as_ref();
                loop {
                    match next_error {
                        E::InvalidField { field_name, error } => {
                            write!(f, ".{}", field_name)?;
                            next_error = error.as_ref();
                        }
                        E::InvalidItem { index, error } => {
                            write!(f, "[{}]", index)?;
                            next_error = error.as_ref();
                        }
                        _ => {
                            write!(f, ": {}", next_error)?;
                            return Ok(());
                        }
                    }
                }
            }
            E::InvalidItem { index, error } => {
                write!(f, "invalid value for item at index {}: {}", index, error)
            }
        }
    }
}
impl ToVariant for () {
    fn to_variant(&self) -> Variant {
        Variant::new()
    }
}
impl FromVariant for () {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        variant.try_as_sys_of_type(VariantType::Nil).map(|_| ())
    }
}
impl<'a, T> ToVariant for &'a T
where
    T: ToVariant,
{
    fn to_variant(&self) -> Variant {
        T::to_variant(*self)
    }
}
impl<'a, T> ToVariant for &'a mut T
where
    T: ToVariant,
{
    fn to_variant(&self) -> Variant {
        T::to_variant(*self)
    }
}
macro_rules! from_variant_direct {
    (
        $(
            impl FromVariant for $TryType:ident : VariantType :: $VarType:ident => $try_gd_method:ident;
        )*
    ) => (
        $(
            impl FromVariant for $TryType {
                fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
                    variant.try_as_sys_of_type(VariantType::$VarType)
                        .map(|v| unsafe { (get_api().$try_gd_method)(v) })
                }
            }
        )*
    );
}
from_variant_direct!(
    impl FromVariant for f64 : VariantType::F64 => godot_variant_as_real;
    impl FromVariant for i64 : VariantType::I64 => godot_variant_as_int;
    impl FromVariant for u64 : VariantType::I64 => godot_variant_as_uint;
    impl FromVariant for bool : VariantType::Bool => godot_variant_as_bool;
);
impl ToVariant for i64 {
    fn to_variant(&self) -> Variant {
        Variant::from_i64(*self)
    }
}
impl ToVariant for u64 {
    fn to_variant(&self) -> Variant {
        Variant::from_u64(*self)
    }
}
impl ToVariant for f64 {
    fn to_variant(&self) -> Variant {
        Variant::from_f64(*self)
    }
}
macro_rules! impl_to_variant_for_num {
    (
        $($ty:ty : $src_ty:ty)*
    ) => {
        $(
            impl ToVariant for $ty {
                fn to_variant(&self) -> Variant {
                    ((*self) as $src_ty).to_variant()
                }
            }
            impl FromVariant for $ty {
                fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
                    <$src_ty>::from_variant(variant).map(|i| i as Self)
                }
            }
        )*
    };
}
impl_to_variant_for_num!(
    i8: i64
    i16: i64
    i32: i64
    isize: i64
    u8: u64
    u16: u64
    u32: u64
    usize: u64
    f32: f64
);
macro_rules! to_variant_transmute {
    (
        $(impl ToVariant for $ty:ident: $ctor:ident;)*
    ) => {
        $(
            impl ToVariant for $ty {
                fn to_variant(&self) -> Variant {
                    unsafe {
                        let api = get_api();
                        let mut dest = sys::godot_variant::default();
                        #[allow(clippy::useless_transmute)]
                        (api.$ctor)(&mut dest, transmute(self));
                        Variant::from_sys(dest)
                    }
                }
            }
        )*
    }
}
to_variant_transmute! {
    impl ToVariant for Vector2 : godot_variant_new_vector2;
    impl ToVariant for Vector3 : godot_variant_new_vector3;
    impl ToVariant for Quat : godot_variant_new_quat;
    impl ToVariant for Rect2 : godot_variant_new_rect2;
    impl ToVariant for Transform2D : godot_variant_new_transform2d;
}
macro_rules! to_variant_as_sys {
    (
        $(impl ToVariant for $ty:ident: $ctor:ident;)*
    ) => {
        $(
            impl ToVariant for $ty {
                fn to_variant(&self) -> Variant {
                    unsafe {
                        let api = get_api();
                        let mut dest = sys::godot_variant::default();
                        (api.$ctor)(&mut dest, self.sys());
                        Variant::from_sys(dest)
                    }
                }
            }
        )*
    }
}
to_variant_as_sys! {
    impl ToVariant for Plane : godot_variant_new_plane;
    impl ToVariant for Transform : godot_variant_new_transform;
    impl ToVariant for Basis : godot_variant_new_basis;
    impl ToVariant for Color : godot_variant_new_color;
    impl ToVariant for Aabb : godot_variant_new_aabb;
    impl ToVariant for Rid : godot_variant_new_rid;
    impl ToVariant for NodePath : godot_variant_new_node_path;
    impl ToVariant for GodotString : godot_variant_new_string;
    impl ToVariant for VariantArray : godot_variant_new_array;
    impl ToVariant for ByteArray : godot_variant_new_pool_byte_array;
    impl ToVariant for Int32Array : godot_variant_new_pool_int_array;
    impl ToVariant for Float32Array : godot_variant_new_pool_real_array;
    impl ToVariant for StringArray : godot_variant_new_pool_string_array;
    impl ToVariant for Vector2Array : godot_variant_new_pool_vector2_array;
    impl ToVariant for Vector3Array : godot_variant_new_pool_vector3_array;
    impl ToVariant for ColorArray : godot_variant_new_pool_color_array;
    impl ToVariant for Dictionary : godot_variant_new_dictionary;
}
macro_rules! from_variant_transmute {
    (
        $(
            impl FromVariant for $TryType:ident : $try_gd_method:ident;
        )*
    ) => (
        $(
            impl FromVariant for $TryType {
                fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
                    unsafe {
                        variant.try_as_sys_of_type(VariantType::$TryType)
                            .map(|v| (get_api().$try_gd_method)(v))
                            .map(|v| transmute(v))
                    }
                }
            }
        )*
    );
}
from_variant_transmute!(
    impl FromVariant for Vector2 : godot_variant_as_vector2;
    impl FromVariant for Vector3 : godot_variant_as_vector3;
    impl FromVariant for Quat : godot_variant_as_quat;
    impl FromVariant for Rect2 : godot_variant_as_rect2;
    impl FromVariant for Transform2D : godot_variant_as_transform2d;
);
macro_rules! from_variant_from_sys {
    (
        $(
            impl FromVariant for $TryType:ident : $try_gd_method:ident;
        )*
    ) => (
        $(
            impl FromVariant for $TryType {
                fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
                    unsafe {
                        variant.try_as_sys_of_type(VariantType::$TryType)
                            .map(|v| (get_api().$try_gd_method)(v))
                            .map($TryType::from_sys)
                    }
                }
            }
        )*
    );
}
from_variant_from_sys!(
    impl FromVariant for Plane : godot_variant_as_plane;
    impl FromVariant for Transform : godot_variant_as_transform;
    impl FromVariant for Basis : godot_variant_as_basis;
    impl FromVariant for Color : godot_variant_as_color;
    impl FromVariant for Aabb : godot_variant_as_aabb;
    impl FromVariant for NodePath : godot_variant_as_node_path;
    impl FromVariant for GodotString : godot_variant_as_string;
    impl FromVariant for Rid : godot_variant_as_rid;
    impl FromVariant for VariantArray : godot_variant_as_array;
    impl FromVariant for ByteArray : godot_variant_as_pool_byte_array;
    impl FromVariant for Int32Array : godot_variant_as_pool_int_array;
    impl FromVariant for Float32Array : godot_variant_as_pool_real_array;
    impl FromVariant for StringArray : godot_variant_as_pool_string_array;
    impl FromVariant for Vector2Array : godot_variant_as_pool_vector2_array;
    impl FromVariant for Vector3Array : godot_variant_as_pool_vector3_array;
    impl FromVariant for ColorArray : godot_variant_as_pool_color_array;
    impl FromVariant for Dictionary : godot_variant_as_dictionary;
);
impl ToVariant for str {
    fn to_variant(&self) -> Variant {
        Variant::from_str(self)
    }
}
impl ToVariant for String {
    fn to_variant(&self) -> Variant {
        Variant::from_str(&self)
    }
}
impl FromVariant for String {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        GodotString::from_variant(variant).map(|s| s.to_string())
    }
}
impl ToVariant for bool {
    fn to_variant(&self) -> Variant {
        Variant::from_bool(*self)
    }
}
impl ToVariant for Variant {
    fn to_variant(&self) -> Variant {
        self.clone()
    }
}
impl FromVariant for Variant {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        Ok(variant.clone())
    }
}
impl<T> ToVariant for std::marker::PhantomData<T> {
    fn to_variant(&self) -> Variant {
        Variant::new()
    }
}
impl<T> FromVariant for std::marker::PhantomData<T> {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        variant
            .try_as_sys_of_type(VariantType::Nil)
            .map(|_| std::marker::PhantomData)
    }
}
impl<T: ToVariant> ToVariant for Option<T> {
    fn to_variant(&self) -> Variant {
        match &self {
            Some(thing) => thing.to_variant(),
            None => Variant::new(),
        }
    }
}
impl<T: FromVariant> FromVariant for Option<T> {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        T::from_variant(variant).map(Some).or_else(
            |e| {
                if variant.is_nil() {
                    Ok(None)
                } else {
                    Err(e)
                }
            },
        )
    }
}
#[derive(Clone, Debug)]
pub struct MaybeNot<T>(Result<T, Variant>);
impl<T: FromVariant> FromVariant for MaybeNot<T> {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        Ok(MaybeNot(
            T::from_variant(variant).map_err(|_| variant.clone()),
        ))
    }
}
impl<T> MaybeNot<T> {
    pub fn into_result(self) -> Result<T, Variant> {
        self.0
    }
    pub fn as_ref(&self) -> Result<&T, &Variant> {
        self.0.as_ref()
    }
    pub fn as_mut(&mut self) -> Result<&mut T, &mut Variant> {
        self.0.as_mut()
    }
    pub fn cloned(&self) -> Result<T, Variant>
    where
        T: Clone,
    {
        self.0.clone()
    }
    pub fn ok(self) -> Option<T> {
        self.0.ok()
    }
}
impl<T: ToVariant, E: ToVariant> ToVariant for Result<T, E> {
    fn to_variant(&self) -> Variant {
        let mut dict = Dictionary::new();
        match &self {
            Ok(val) => dict.set(&"Ok".into(), &val.to_variant()),
            Err(err) => dict.set(&"Err".into(), &err.to_variant()),
        }
        dict.to_variant()
    }
}
impl<T: FromVariant, E: FromVariant> FromVariant for Result<T, E> {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        use FromVariantError as FVE;
        let dict = Dictionary::from_variant(variant).map_err(|err| FVE::InvalidEnumRepr {
            expected: VariantEnumRepr::ExternallyTagged,
            error: Box::new(err),
        })?;
        if dict.len() != 1 {
            return Err(FVE::InvalidEnumRepr {
                expected: VariantEnumRepr::ExternallyTagged,
                error: Box::new(FVE::InvalidLength {
                    expected: 1,
                    len: dict.len() as usize,
                }),
            });
        }
        let keys = dict.keys();
        let key_variant = keys.get_ref(0);
        let key = String::from_variant(key_variant).map_err(|err| FVE::InvalidEnumRepr {
            expected: VariantEnumRepr::ExternallyTagged,
            error: Box::new(err),
        })?;
        match key.as_str() {
            "Ok" => {
                let val = T::from_variant(dict.get_ref(key_variant)).map_err(|err| {
                    FVE::InvalidEnumVariant {
                        variant: "Ok",
                        error: Box::new(err),
                    }
                })?;
                Ok(Ok(val))
            }
            "Err" => {
                let err = E::from_variant(dict.get_ref(key_variant)).map_err(|err| {
                    FVE::InvalidEnumVariant {
                        variant: "Err",
                        error: Box::new(err),
                    }
                })?;
                Ok(Err(err))
            }
            variant => Err(FVE::UnknownEnumVariant {
                variant: variant.to_string(),
                expected: &["Ok", "Err"],
            }),
        }
    }
}
impl<T: ToVariant> ToVariant for &[T] {
    fn to_variant(&self) -> Variant {
        let mut array = VariantArray::new();
        for val in self.iter() {
            
            array.push(&val.to_variant());
        }
        array.to_variant()
    }
}
impl<T: ToVariant> ToVariant for Vec<T> {
    fn to_variant(&self) -> Variant {
        self.as_slice().to_variant()
    }
}
impl<T: FromVariant> FromVariant for Vec<T> {
    fn from_variant(variant: &Variant) -> Result<Self, FromVariantError> {
        use std::convert::TryInto;
        let arr = VariantArray::from_variant(variant)?;
        let len: usize = arr
            .len()
            .try_into()
            .expect("variant array length should fit in usize");
        let mut vec = Vec::with_capacity(len);
        for idx in 0..len as i32 {
            let item =
                T::from_variant(arr.get_ref(idx)).map_err(|e| FromVariantError::InvalidItem {
                    index: idx as usize,
                    error: Box::new(e),
                })?;
            vec.push(item);
        }
        Ok(vec)
    }
}
macro_rules! tuple_length {
    () => { 0usize };
    ($_x:ident, $($xs:ident,)*) => {
        1usize + tuple_length!($($xs,)*)
    };
}
macro_rules! impl_variant_for_tuples_next {
    ($_x:ident, $($xs:ident,)*) => {
        impl_variant_for_tuples!($($xs,)*);
    }
}
macro_rules! impl_variant_for_tuples {
    () => {};
    ( $($name:ident,)+ ) => {
        impl<$($name: ToVariant,)+> ToVariant for ($($name,)+) {
            #[allow(non_snake_case)]
            fn to_variant(&self) -> Variant {
                let mut array = VariantArray::new();
                let ($($name,)+) = self;
                $(
                    array.push(&$name.to_variant());
                )+
                array.to_variant()
            }
        }
        impl<$($name: FromVariant,)+> FromVariant for ($($name,)+) {
            #[allow(non_snake_case, unused_assignments)]
            fn from_variant(v: &Variant) -> Result<Self, FromVariantError> {
                let array = VariantArray::from_variant(v)?;
                let expected = tuple_length!($($name,)+);
                let len = array.len() as usize;
                if len != expected {
                    return Err(FromVariantError::InvalidLength { expected, len });
                }
                let mut iter = array.iter();
                let mut index = 0;
                $(
                    let $name = $name::from_variant(iter.next().unwrap())
                        .map_err(|err| FromVariantError::InvalidItem {
                            index,
                            error: Box::new(err),
                        })?;
                    index += 1;
                )+
                Ok(($($name,)+))
            }
        }
        impl_variant_for_tuples_next!($($name,)+);
    };
}
impl_variant_for_tuples!(T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12,);
godot_test!(
    test_variant_option {
        use std::marker::PhantomData;
        let variant = Some(42 as i64).to_variant();
        assert_eq!(Some(42), variant.try_to_i64());
        let variant = Option::<bool>::None.to_variant();
        assert!(variant.is_nil());
        let variant = Variant::new();
        assert_eq!(Ok(None), Option::<i64>::from_variant(&variant));
        assert_eq!(Ok(None), Option::<bool>::from_variant(&variant));
        assert_eq!(Ok(None), Option::<String>::from_variant(&variant));
        let variant = Variant::from_i64(42);
        assert_eq!(Ok(Some(42)), Option::<i64>::from_variant(&variant));
        assert!(Option::<bool>::from_variant(&variant).is_err());
        assert!(Option::<String>::from_variant(&variant).is_err());
        let variant = Variant::new();
        assert_eq!(Ok(Some(())), Option::<()>::from_variant(&variant));
        assert_eq!(Ok(Some(PhantomData)), Option::<PhantomData<*const u8>>::from_variant(&variant));
        let variant = Variant::from_i64(42);
        assert!(Option::<PhantomData<*const u8>>::from_variant(&variant).is_err());
    }
    test_variant_result {
        let variant = Result::<i64, ()>::Ok(42 as i64).to_variant();
        let dict = variant.try_to_dictionary().expect("should be dic");
        assert_eq!(Some(42), dict.get_ref(&"Ok".into()).try_to_i64());
        let variant = Result::<(), i64>::Err(54 as i64).to_variant();
        let dict = variant.try_to_dictionary().expect("should be dic");
        assert_eq!(Some(54), dict.get_ref(&"Err".into()).try_to_i64());
        let variant = Variant::from_bool(true);
        assert_eq!(
            Err(FromVariantError::InvalidEnumRepr {
                expected: VariantEnumRepr::ExternallyTagged,
                error: Box::new(FromVariantError::InvalidVariantType {
                    expected: VariantType::Dictionary,
                    variant_type: VariantType::Bool,
                }),
            }),
            Result::<(), i64>::from_variant(&variant),
        );
        let mut dict = Dictionary::new();
        dict.set(&"Ok".into(), &Variant::from_i64(42));
        assert_eq!(Ok(Ok(42)), Result::<i64, i64>::from_variant(&dict.to_variant()));
        let mut dict = Dictionary::new();
        dict.set(&"Err".into(), &Variant::from_i64(54));
        assert_eq!(Ok(Err(54)), Result::<i64, i64>::from_variant(&dict.to_variant()));
    }
    test_to_variant_iter {
        let slice: &[i64] = &[0, 1, 2, 3, 4];
        let variant = slice.to_variant();
        let array = variant.try_to_array().expect("should be array");
        assert_eq!(5, array.len());
        for i in 0..5 {
            assert_eq!(Some(i), array.get_ref(i as i32).try_to_i64());
        }
        let vec = Vec::<i64>::from_variant(&variant).expect("should succeed");
        assert_eq!(slice, vec.as_slice());
        let mut het_array = VariantArray::new();
        het_array.push(&Variant::from_i64(42));
        het_array.push(&Variant::new());
        assert_eq!(
            Err(FromVariantError::InvalidItem {
                index: 1,
                error: Box::new(FromVariantError::InvalidVariantType {
                    expected: VariantType::I64,
                    variant_type: VariantType::Nil,
                }),
            }),
            Vec::<i64>::from_variant(&het_array.to_variant()),
        );
        assert_eq!(Ok(vec![Some(42), None]), Vec::<Option<i64>>::from_variant(&het_array.to_variant()));
        het_array.push(&f64::to_variant(&54.0));
        assert_eq!(
            Err(FromVariantError::InvalidItem {
                index: 2,
                error: Box::new(FromVariantError::InvalidVariantType {
                    expected: VariantType::I64,
                    variant_type: VariantType::F64,
                }),
            }),
            Vec::<Option<i64>>::from_variant(&het_array.to_variant()),
        );
        let vec_maybe = Vec::<MaybeNot<i64>>::from_variant(&het_array.to_variant()).expect("should succeed");
        assert_eq!(3, vec_maybe.len());
        assert_eq!(Some(&42), vec_maybe[0].as_ref().ok());
        assert_eq!(Some(&Variant::new()), vec_maybe[1].as_ref().err());
        assert_eq!(Some(&f64::to_variant(&54.0)), vec_maybe[2].as_ref().err());
    }
    test_variant_tuple {
        let variant = (42i64, 54i64).to_variant();
        let arr = variant.try_to_array().expect("should be array");
        assert_eq!(Some(42), arr.get_ref(0).try_to_i64());
        assert_eq!(Some(54), arr.get_ref(1).try_to_i64());
        let tuple = <(i64, i64)>::from_variant(&variant);
        assert_eq!(Ok((42, 54)), tuple);
    }
);