use std::pin::Pin;
use std::ptr::null_mut;
use pipewire_wrapper_proc_macro::{interface, proxy_wrapper, RawWrapper};
use crate::core_api::core::Core;
use crate::core_api::device::events::DeviceEvents;
use crate::core_api::proxy::{Proxy, ProxyRef};
use crate::core_api::registry::restricted::RegistryBind;
use crate::i32_as_void_result;
use crate::listeners::{AddListener, Listeners, OwnListeners};
use crate::spa::param::ParamType;
use crate::spa::pod::PodRef;
use crate::spa_interface_call;
use crate::wrapper::{RawWrapper, Wrapper};
pub mod events;
pub mod info;
#[derive(RawWrapper, Debug)]
#[interface(methods=pw_sys::pw_device_methods, interface="Device")]
#[repr(transparent)]
pub struct DeviceRef {
    #[raw]
    raw: pw_sys::pw_device,
}
impl DeviceRef {
    pub fn subscribe_params(&self, param_types: &[ParamType]) -> crate::Result<()> {
        let result = unsafe {
            spa_interface_call!(
                self,
                subscribe_params,
                param_types.as_ptr() as *mut _,
                param_types.len() as u32
            )?
        };
        i32_as_void_result(result)
    }
    pub fn enum_params(
        &self,
        seq: i32,
        id: ParamType,
        start: u32,
        num: u32,
        filter: Option<&PodRef>,
    ) -> crate::Result<()> {
        let result = unsafe {
            spa_interface_call!(
                self,
                enum_params,
                seq,
                *id.as_raw(),
                start,
                num,
                filter.map_or(null_mut(), |f| f as *const _ as *mut _)
            )?
        };
        i32_as_void_result(result)
    }
    pub fn set_param(&self, id: u32, flags: u32, param: &PodRef) -> crate::Result<()> {
        let result =
            unsafe { spa_interface_call!(self, set_param, id, flags, param.as_raw_ptr())? };
        i32_as_void_result(result)
    }
}
impl<'a> AddListener<'a> for DeviceRef {
    type Events = DeviceEvents<'a>;
    fn add_listener(&self, events: Pin<Box<Self::Events>>) -> Pin<Box<Self::Events>> {
        unsafe {
            spa_interface_call!(
                self,
                add_listener,
                events.hook().as_raw_ptr(),
                events.as_raw_ptr(),
                &*events as *const _ as *mut _
            )
        };
        events
    }
}
#[derive(Clone, Debug)]
#[proxy_wrapper(DeviceRef)]
pub struct Device<'c> {
    ref_: Proxy<'c>,
    listeners: Listeners<Pin<Box<DeviceEvents<'c>>>>,
}
impl<'c> RegistryBind<'c> for Device<'c> {
    fn from_ref(core: &'c Core, ref_: &ProxyRef) -> Self {
        Self {
            ref_: Proxy::from_ref(core, ref_),
            listeners: Listeners::default(),
        }
    }
}
impl<'a> OwnListeners<'a> for Device<'a> {
    fn listeners(
        &self,
    ) -> &Listeners<Pin<Box<<<Self as Wrapper>::RawWrapperType as AddListener<'a>>::Events>>> {
        &self.listeners
    }
}