use std::ffi::CStr;
use std::fmt::{Debug, Formatter};
use std::os::fd::RawFd;
use std::pin::Pin;
use std::ptr::NonNull;
use derive_builder::Builder;
use pw_sys::{pw_core_events, pw_core_info};
use pipewire_wrapper_proc_macro::{RawWrapper, Wrapper};
use crate::core_api::core::info::CoreInfoRef;
use crate::core_api::core::CoreRef;
use crate::events_builder_build;
use crate::spa::interface::{Hook, HookRef};
use crate::wrapper::RawWrapper;
#[derive(RawWrapper, Debug)]
#[repr(transparent)]
pub struct CoreEventsRef {
    #[raw]
    raw: pw_sys::pw_core_events,
}
pub type InfoCallback<'c> = Box<dyn FnMut(&'c CoreInfoRef) + 'c>;
pub type DoneCallback<'c> = Box<dyn FnMut(u32, i32) + 'c>;
pub type PingCallback<'c> = Box<dyn FnMut(u32, i32) + 'c>;
pub type ErrorCallback<'c> = Box<dyn FnMut(u32, i32, i32, &CStr) + 'c>;
pub type RemoveIdCallback<'c> = Box<dyn FnMut(u32) + 'c>;
pub type BoundIdCallback<'c> = Box<dyn FnMut(u32, u32) + 'c>;
pub type AddMemCallback<'c> = Box<dyn FnMut(u32, u32, RawFd, u32) + 'c>;
pub type RemoveMemCallback<'c> = Box<dyn FnMut(u32) + 'c>;
#[derive(Wrapper, Builder)]
#[builder(setter(skip, strip_option), build_fn(skip), pattern = "owned")]
pub struct CoreEvents<'c> {
    #[raw_wrapper]
    ref_: NonNull<CoreEventsRef>,
    raw: Pin<Box<CoreEventsRef>>,
    hook: Pin<Box<Hook>>,
    #[builder(setter)]
    info: Option<InfoCallback<'c>>,
    #[builder(setter)]
    done: Option<DoneCallback<'c>>,
    #[builder(setter)]
    ping: Option<PingCallback<'c>>,
    #[builder(setter)]
    error: Option<ErrorCallback<'c>>,
    #[builder(setter)]
    remove_id: Option<RemoveIdCallback<'c>>,
    #[builder(setter)]
    bound_id: Option<BoundIdCallback<'c>>,
    #[builder(setter)]
    add_mem: Option<AddMemCallback<'c>>,
    #[builder(setter)]
    remove_mem: Option<RemoveMemCallback<'c>>,
}
impl<'c> CoreEvents<'c> {
    unsafe extern "C" fn info_call(data: *mut ::std::os::raw::c_void, info: *const pw_core_info) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.info {
                callback(CoreInfoRef::from_raw_ptr(info));
            }
        }
    }
    unsafe extern "C" fn done_call(
        data: *mut ::std::os::raw::c_void,
        id: u32,
        seq: ::std::os::raw::c_int,
    ) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.done {
                callback(id, seq);
            }
        }
    }
    unsafe extern "C" fn ping_call(
        data: *mut ::std::os::raw::c_void,
        id: u32,
        seq: ::std::os::raw::c_int,
    ) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.ping {
                callback(id, seq);
            }
        }
    }
    unsafe extern "C" fn error_call(
        data: *mut ::std::os::raw::c_void,
        id: u32,
        seq: i32,
        res: i32,
        message: *const ::std::os::raw::c_char,
    ) {
        if let Some(events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut events.error {
                callback(id, seq, res, CStr::from_ptr(message));
            }
        }
    }
    unsafe extern "C" fn remove_id_call(data: *mut ::std::os::raw::c_void, id: u32) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.remove_id {
                callback(id);
            }
        }
    }
    unsafe extern "C" fn bound_id_call(data: *mut ::std::os::raw::c_void, id: u32, global_id: u32) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.bound_id {
                callback(id, global_id);
            }
        }
    }
    unsafe extern "C" fn add_mem_call(
        data: *mut ::std::os::raw::c_void,
        id: u32,
        type_: u32,
        fd: ::std::os::raw::c_int,
        flags: u32,
    ) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.add_mem {
                callback(id, type_, fd, flags);
            }
        }
    }
    unsafe extern "C" fn remove_mem_call(data: *mut ::std::os::raw::c_void, id: u32) {
        if let Some(core_events) = (data as *mut CoreEvents).as_mut() {
            if let Some(callback) = &mut core_events.remove_mem {
                callback(id);
            }
        }
    }
    pub fn version(&self) -> u32 {
        self.raw.raw.version
    }
    pub fn hook(&self) -> &Pin<Box<Hook>> {
        &self.hook
    }
}
impl<'c> CoreEventsBuilder<'c> {
    events_builder_build! {
        CoreEvents<'c>,
        pw_core_events,
        info => info_call,
        done => done_call,
        ping => ping_call,
        error => error_call,
        remove_id => remove_id_call,
        bound_id => bound_id_call,
        add_mem => add_mem_call,
        remove_mem => remove_mem_call,
    }
}
impl Debug for CoreEvents<'_> {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("CoreEvents")
            .field("raw", &self.raw)
            .finish()
    }
}