use std::collections::HashMap;
use std::ffi::{CStr, CString};
use std::fmt::{Debug, Formatter};
use std::ops::{Deref, DerefMut};
use std::ptr::{slice_from_raw_parts, NonNull};
use std::slice::from_raw_parts;
use bitflags::bitflags;
use pipewire_wrapper_proc_macro::{RawWrapper, Wrapper};
use crate::core_api::node::Node;
use crate::enum_wrapper;
use crate::spa::dict::DictRef;
use crate::spa::param::{ParamInfo, ParamInfoRef};
use crate::wrapper::{RawWrapper, Wrapper};
bitflags! {
    #[derive(Debug, PartialEq, Eq, Clone, Copy)]
    #[repr(transparent)]
    pub struct ChangeMask: u64 {
        const INPUT_PORTS = pw_sys::PW_NODE_CHANGE_MASK_INPUT_PORTS as u64;
        const OUTPUT_PORTS = pw_sys::PW_NODE_CHANGE_MASK_OUTPUT_PORTS as u64;
        const STATE = pw_sys::PW_NODE_CHANGE_MASK_STATE as u64;
        const PROPS = pw_sys::PW_NODE_CHANGE_MASK_PROPS as u64;
        const PARAMS = pw_sys::PW_NODE_CHANGE_MASK_PARAMS as u64;
        const ALL = pw_sys::PW_NODE_CHANGE_MASK_ALL as u64;
    }
}
enum_wrapper!(
    NodeState,
    pw_sys::pw_node_state,
    ERROR: pw_sys::pw_node_state_PW_NODE_STATE_ERROR,
    CREATING: pw_sys::pw_node_state_PW_NODE_STATE_CREATING,
    SUSPENDED: pw_sys::pw_node_state_PW_NODE_STATE_SUSPENDED,
    IDLE: pw_sys::pw_node_state_PW_NODE_STATE_IDLE,
    RUNNING: pw_sys::pw_node_state_PW_NODE_STATE_RUNNING,
);
#[derive(RawWrapper)]
#[repr(transparent)]
pub struct NodeInfoRef {
    #[raw]
    raw: pw_sys::pw_node_info,
}
impl NodeInfoRef {
    pub fn id(&self) -> u32 {
        self.raw.id
    }
    pub fn max_input_ports(&self) -> u32 {
        self.raw.max_input_ports
    }
    pub fn max_output_ports(&self) -> u32 {
        self.raw.max_output_ports
    }
    pub fn change_mask(&self) -> ChangeMask {
        ChangeMask::from_bits_retain(self.raw.change_mask)
    }
    pub fn n_input_ports(&self) -> u32 {
        self.raw.n_input_ports
    }
    pub fn n_output_ports(&self) -> u32 {
        self.raw.n_output_ports
    }
    pub fn state(&self) -> NodeState {
        NodeState::from_raw(self.raw.state)
    }
    pub fn error(&self) -> Option<&CStr> {
        unsafe { self.raw.error.as_ref().map(|r| CStr::from_ptr(r)) }
    }
    pub fn props(&self) -> &DictRef {
        unsafe { DictRef::from_raw_ptr(self.raw.props) }
    }
    pub fn params(&self) -> &[ParamInfoRef] {
        unsafe {
            from_raw_parts(
                self.raw.params as *mut ParamInfoRef,
                self.raw.n_params as usize,
            )
        }
    }
}
impl Debug for NodeInfoRef {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("NodeInfoRef")
            .field("id", &self.id())
            .field("max_input_ports", &self.max_input_ports())
            .field("max_output_ports", &self.max_output_ports())
            .field("change_mask", &self.change_mask())
            .field("n_input_ports", &self.n_input_ports())
            .field("n_output_ports", &self.n_output_ports())
            .field("state", &self.state())
            .field("error", &self.error())
            .field("props", &self.props())
            .field("params", &self.params())
            .finish()
    }
}
#[derive(Debug, Clone)]
pub struct NodeInfo {
    id: u32,
    max_input_ports: u32,
    max_output_ports: u32,
    change_mask: ChangeMask,
    n_input_ports: u32,
    n_output_ports: u32,
    state: NodeState,
    error: Option<CString>,
    props: HashMap<CString, CString>,
    params: Vec<ParamInfo>,
}
impl NodeInfo {
    pub fn from_ref(ref_: &NodeInfoRef) -> Self {
        let raw = ref_.raw;
        Self {
            id: raw.id,
            max_input_ports: raw.max_input_ports,
            max_output_ports: raw.max_output_ports,
            change_mask: ref_.change_mask(),
            n_input_ports: raw.n_input_ports,
            n_output_ports: raw.n_output_ports,
            state: ref_.state(),
            error: ref_.error().map(CString::from),
            props: ref_.props().into(),
            params: ref_.params().iter().map(ParamInfo::from_ref).collect(),
        }
    }
    pub fn id(&self) -> u32 {
        self.id
    }
    pub fn max_input_ports(&self) -> u32 {
        self.max_input_ports
    }
    pub fn max_output_ports(&self) -> u32 {
        self.max_output_ports
    }
    pub fn change_mask(&self) -> ChangeMask {
        self.change_mask
    }
    pub fn n_input_ports(&self) -> u32 {
        self.n_input_ports
    }
    pub fn n_output_ports(&self) -> u32 {
        self.n_output_ports
    }
    pub fn state(&self) -> NodeState {
        self.state
    }
    pub fn error(&self) -> &Option<CString> {
        &self.error
    }
    pub fn props(&self) -> &HashMap<CString, CString> {
        &self.props
    }
    pub fn params(&self) -> &Vec<ParamInfo> {
        &self.params
    }
}
impl From<&NodeInfoRef> for NodeInfo {
    fn from(value: &NodeInfoRef) -> Self {
        NodeInfo::from_ref(value)
    }
}