use kurbo::Shape;
use pax_engine::*;
use pax_runtime::api::{borrow_mut, use_RefCell};
use pax_runtime::{api::Property, api::RenderContext, ExpandedNodeIdentifier};
use std::collections::HashSet;
use_RefCell!();
use pax_message::ImagePatch;
use pax_runtime::{
BaseInstance, ExpandedNode, InstanceFlags, InstanceNode, InstantiationArgs, RuntimeContext,
};
use std::rc::Rc;
use crate::common::patch_if_needed;
#[pax]
#[engine_import_path("pax_engine")]
#[primitive("pax_std::core::image::ImageInstance")]
pub struct Image {
pub source: Property<ImageSource>,
pub fit: Property<ImageFit>,
}
#[pax]
#[engine_import_path("pax_engine")]
pub enum ImageSource {
#[default]
Empty,
Url(String),
Data(usize, usize, Vec<u8>),
}
pub struct ImageInstance {
base: BaseInstance,
needs_to_load_data: Rc<RefCell<HashSet<ExpandedNodeIdentifier>>>,
}
impl InstanceNode for ImageInstance {
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: pax_runtime::api::Layer::Canvas,
is_component: false,
},
),
needs_to_load_data: Default::default(),
})
}
fn handle_mount(
self: Rc<Self>,
expanded_node: &Rc<ExpandedNode>,
context: &Rc<RuntimeContext>,
) {
let id = expanded_node.id.to_u32();
let weak_self_ref = Rc::downgrade(&expanded_node);
let context = Rc::clone(context);
let last_patch = Rc::new(RefCell::new(ImagePatch {
id,
..Default::default()
}));
let deps =
[expanded_node.with_properties_unwrapped(|props: &mut Image| props.source.untyped())];
let needs_to_load_data = Rc::clone(&self.needs_to_load_data);
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 = ImagePatch {
id,
..Default::default()
};
expanded_node.with_properties_unwrapped(|props: &mut Image| {
let source = props.source.get();
match source {
ImageSource::Empty => (),
ImageSource::Url(url) => {
let update =
patch_if_needed(&mut old_state.path, &mut patch.path, url);
if update {
context.enqueue_native_message(
pax_message::NativeMessage::ImageLoad(patch),
);
}
}
ImageSource::Data(_width, _height, _data) => {
borrow_mut!(needs_to_load_data).insert(expanded_node.id.clone());
}
}
});
()
},
&deps,
));
}
fn handle_unmount(&self, expanded_node: &Rc<ExpandedNode>, _context: &Rc<RuntimeContext>) {
let id = expanded_node.id.clone();
expanded_node
.native_message_listener
.replace_with(Property::default());
borrow_mut!(self.needs_to_load_data).remove(&id);
}
fn render(
&self,
expanded_node: &ExpandedNode,
_rtc: &Rc<RuntimeContext>,
rc: &mut dyn RenderContext,
) {
let t_and_b = expanded_node.transform_and_bounds.get();
let (container_width, container_height) = t_and_b.bounds;
expanded_node.with_properties_unwrapped(|props: &mut Image| {
let image_size_and_load_path = props.source.read(|source| {
match source {
ImageSource::Empty => return None,
ImageSource::Url(url) => {
let Some((image_width, image_height)) = rc.get_image_size(&url) else {
return None;
};
Some((image_width, image_height, url.to_string()))
}
&ImageSource::Data(width, height, ref data) => {
let mut last_images = borrow_mut!(self.needs_to_load_data);
let unique_ident = format!("raw-image-ref-{}", expanded_node.id.to_u32());
if last_images.contains(&expanded_node.id) {
let mut data_of_correct_len = vec![0; width * height * 4];
data_of_correct_len[0..data.len()].copy_from_slice(&data);
rc.load_image(&unique_ident, &data_of_correct_len, width, height);
last_images.remove(&expanded_node.id);
}
Some((width, height, unique_ident))
}
}
});
let Some((image_width, image_height, path)) = image_size_and_load_path else {
return;
};
let (image_width, image_height) = (image_width as f64, image_height as f64);
let stretch_w = container_width / image_width;
let stretch_h = container_height / image_height;
let (width, height) = match props.fit.get() {
ImageFit::FillVertical => (image_width * stretch_h, image_height * stretch_h),
ImageFit::FillHorizontal => (image_width * stretch_w, image_height * stretch_w),
ImageFit::Fill => {
let stretch = stretch_h.max(stretch_w);
(image_width * stretch, image_height * stretch)
}
ImageFit::Fit => {
let stretch = stretch_h.min(stretch_w);
(image_width * stretch, image_height * stretch)
}
ImageFit::Stretch => (container_width, container_height),
};
let x = (container_width - width) / 2.0;
let y = (container_height - height) / 2.0;
let transformed_bounds = kurbo::Rect::new(x, y, x + width, y + height);
let clip_path = kurbo::Rect::new(0.0, 0.0, container_width, container_height);
let layer_id = format!("{}", expanded_node.occlusion.get().occlusion_layer_id);
rc.save(&layer_id);
rc.transform(&layer_id, t_and_b.transform.into());
rc.clip(&layer_id, clip_path.into_path(0.01));
rc.draw_image(&layer_id, &path, transformed_bounds);
rc.restore(&layer_id);
});
}
fn base(&self) -> &BaseInstance {
&self.base
}
fn resolve_debug(
&self,
f: &mut std::fmt::Formatter,
_expanded_node: Option<&pax_runtime::ExpandedNode>,
) -> std::fmt::Result {
f.debug_struct("Image").finish_non_exhaustive()
}
}
#[pax]
#[engine_import_path("pax_engine")]
pub enum ImageFit {
FillVertical,
FillHorizontal,
Fill,
#[default]
Fit,
Stretch,
}