use core::option::Option::Some;
use pax_runtime::api::{borrow, borrow_mut, use_RefCell, Size};
use pax_runtime::{BaseInstance, InstanceFlags, RuntimeContext};
use_RefCell!();
use pax_engine::pax;
use std::rc::Rc;
use pax_message::{AnyCreatePatch, NativeInterrupt, ScrollerPatch};
use pax_runtime::api::{Layer, Property};
use pax_runtime::{ExpandedNode, InstanceNode, InstantiationArgs};
use crate::common::patch_if_needed;
#[pax]
#[engine_import_path("pax_engine")]
#[primitive("pax_std::core::scrollbar::ScrollbarInstance")]
pub struct Scrollbar {
    pub size_inner_pane_x: Property<Size>,
    pub size_inner_pane_y: Property<Size>,
    pub scroll_x: Property<f64>,
    pub scroll_y: Property<f64>,
}
pub struct ScrollbarInstance {
    base: BaseInstance,
}
impl InstanceNode for ScrollbarInstance {
    fn instantiate(args: InstantiationArgs) -> Rc<Self>
    where
        Self: Sized,
    {
        Rc::new(Self {
            base: BaseInstance::new(
                args,
                InstanceFlags {
                    invisible_to_slot: false,
                    invisible_to_raycasting: false,
                    layer: Layer::Native,
                    is_component: false,
                },
            ),
        })
    }
    fn handle_mount(
        self: Rc<Self>,
        expanded_node: &Rc<ExpandedNode>,
        context: &Rc<RuntimeContext>,
    ) {
        let id = expanded_node.id.to_u32();
        context.enqueue_native_message(pax_message::NativeMessage::ScrollerCreate(
            AnyCreatePatch {
                id,
                parent_frame: expanded_node.parent_frame.get().map(|v| v.to_u32()),
                occlusion_layer_id: 0,
            },
        ));
        {
            let cp = expanded_node.get_common_properties();
            let cp = borrow!(cp);
            let (s_x, s_y) =
                expanded_node.with_properties_unwrapped(|properties: &mut Scrollbar| {
                    let s_x = properties.size_inner_pane_x.clone();
                    let s_y = properties.size_inner_pane_y.clone();
                    (s_x, s_y)
                });
            let w = cp.width.clone();
            let h = cp.height.clone();
            let t_and_b = expanded_node.transform_and_bounds.clone();
            let deps = [
                w.untyped(),
                h.untyped(),
                t_and_b.untyped(),
                s_x.untyped(),
                s_y.untyped(),
            ];
            cp._raycastable.replace_with(Property::computed(
                move || {
                    let (width, height) = t_and_b.get().bounds;
                    let outer_w = w.get().unwrap_or(Size::default()).get_pixels(width);
                    let outer_h = h.get().unwrap_or(Size::default()).get_pixels(height);
                    let inner_w = s_x.get().get_pixels(width);
                    let inner_h = s_y.get().get_pixels(height);
                    let res = Some(!(inner_w <= outer_w && inner_h <= outer_h));
                    res
                },
                &deps,
            ));
        }
        let weak_self_ref = Rc::downgrade(&expanded_node);
        let context = Rc::clone(context);
        let last_patch = Rc::new(RefCell::new(ScrollerPatch {
            id,
            ..Default::default()
        }));
        let deps: Vec<_> = borrow!(expanded_node.properties_scope)
            .values()
            .cloned()
            .map(|v| v.get_untyped_property().clone())
            .chain([expanded_node.transform_and_bounds.untyped()])
            .collect();
        expanded_node
            .native_message_listener
            .replace_with(Property::computed(
                move || {
                    let Some(expanded_node) = weak_self_ref.upgrade() else {
                        unreachable!()
                    };
                    let mut old_state = borrow_mut!(last_patch);
                    let mut patch = ScrollerPatch {
                        id,
                        ..Default::default()
                    };
                    expanded_node.with_properties_unwrapped(|properties: &mut Scrollbar| {
                        let computed_tab = expanded_node.transform_and_bounds.get();
                        let (width, height) = computed_tab.bounds;
                        let updates = [
                            patch_if_needed(
                                &mut old_state.size_inner_pane_x,
                                &mut patch.size_inner_pane_x,
                                properties.size_inner_pane_x.get().get_pixels(width),
                            ),
                            patch_if_needed(
                                &mut old_state.size_inner_pane_y,
                                &mut patch.size_inner_pane_y,
                                properties.size_inner_pane_y.get().get_pixels(height),
                            ),
                            patch_if_needed(&mut old_state.size_x, &mut patch.size_x, width),
                            patch_if_needed(&mut old_state.size_y, &mut patch.size_y, height),
                            patch_if_needed(
                                &mut old_state.scroll_x,
                                &mut patch.scroll_x,
                                properties.scroll_x.get(),
                            ),
                            patch_if_needed(
                                &mut old_state.scroll_y,
                                &mut patch.scroll_y,
                                properties.scroll_y.get(),
                            ),
                            patch_if_needed(
                                &mut old_state.transform,
                                &mut patch.transform,
                                computed_tab.transform.coeffs().to_vec(),
                            ),
                        ];
                        if updates.into_iter().any(|v| v == true) {
                            context.enqueue_native_message(
                                pax_message::NativeMessage::ScrollerUpdate(patch),
                            );
                        }
                    });
                    ()
                },
                &deps,
            ));
    }
    fn handle_unmount(&self, expanded_node: &Rc<ExpandedNode>, context: &Rc<RuntimeContext>) {
        let id = expanded_node.id.to_u32();
        expanded_node
            .native_message_listener
            .replace_with(Property::default());
        context.enqueue_native_message(pax_message::NativeMessage::ScrollerDelete(id));
    }
    fn base(&self) -> &BaseInstance {
        &self.base
    }
    fn resolve_debug(
        &self,
        f: &mut std::fmt::Formatter,
        _expanded_node: Option<&ExpandedNode>,
    ) -> std::fmt::Result {
        f.debug_struct("Scrollbar").finish_non_exhaustive()
    }
    fn handle_native_interrupt(
        &self,
        expanded_node: &Rc<ExpandedNode>,
        interrupt: &NativeInterrupt,
    ) {
        if let NativeInterrupt::Scrollbar(args) = interrupt {
            expanded_node.with_properties_unwrapped(|props: &mut Scrollbar| {
                if (props.scroll_x.get() - args.scroll_x).abs() > 1e-4 {
                    props.scroll_x.set(args.scroll_x);
                }
                if (props.scroll_y.get() - args.scroll_y).abs() > 1e-4 {
                    props.scroll_y.set(args.scroll_y);
                }
            });
        }
    }
}