freya-core 0.3.4

Internal core funcionatilies for Freya.
Documentation
use std::sync::{
    Arc,
    Mutex,
};

use freya_native_core::{
    attributes::AttributeName,
    exports::shipyard::Component,
    node::{
        NodeType,
        OwnedAttributeValue,
    },
    node_ref::NodeView,
    prelude::{
        AttributeMaskBuilder,
        Dependancy,
        NodeMaskBuilder,
        State,
    },
    tags::TagName,
    NodeId,
    SendAnyMap,
};
use freya_native_core_macro::partial_derive_state;
use torin::torin::Torin;

use crate::{
    custom_attributes::{
        AttributesBytes,
        CustomAttributeValues,
        ImageReference,
    },
    dom::{
        CompositorDirtyNodes,
        ImageCacheKey,
        ImagesCache,
    },
    parsing::{
        Parse,
        ParseAttribute,
        ParseError,
    },
    values::{
        AspectRatio,
        ImageCover,
        SamplingMode,
    },
};

#[derive(Default, Debug, Clone, PartialEq, Component)]
pub struct ImageState {
    pub image_sampling: SamplingMode,
    pub image_data: Option<AttributesBytes>,
    pub image_cache_key: Option<ImageCacheKey>,
    pub aspect_ratio: AspectRatio,
    pub image_cover: ImageCover,
    pub image_ref: Option<ImageReference>,
}

impl ParseAttribute for ImageState {
    fn parse_attribute(
        &mut self,
        attr: freya_native_core::prelude::OwnedAttributeView<CustomAttributeValues>,
    ) -> Result<(), ParseError> {
        match attr.attribute {
            AttributeName::Sampling => {
                if let Some(value) = attr.value.as_text() {
                    self.image_sampling = SamplingMode::parse(value)?;
                }
            }
            AttributeName::ImageData => {
                if let OwnedAttributeValue::Custom(CustomAttributeValues::Bytes(bytes)) = attr.value
                {
                    self.image_data = Some(bytes.clone());
                }
            }
            AttributeName::ImageCacheKey => {
                if let OwnedAttributeValue::Text(key) = attr.value {
                    self.image_cache_key = Some(ImageCacheKey(key.clone()));
                }
            }
            AttributeName::AspectRatio => {
                if let Some(value) = attr.value.as_text() {
                    self.aspect_ratio = AspectRatio::parse(value).map_err(|_| ParseError)?;
                }
            }
            AttributeName::ImageCover => {
                if let Some(value) = attr.value.as_text() {
                    self.image_cover = ImageCover::parse(value).map_err(|_| ParseError)?;
                }
            }
            AttributeName::ImageReference => {
                if let OwnedAttributeValue::Custom(CustomAttributeValues::ImageReference(
                    reference,
                )) = attr.value
                {
                    self.image_ref = Some(reference.clone());
                }
            }
            _ => {}
        }

        Ok(())
    }
}

#[partial_derive_state]
impl State<CustomAttributeValues> for ImageState {
    type ParentDependencies = ();

    type ChildDependencies = ();

    type NodeDependencies = ();

    const NODE_MASK: NodeMaskBuilder<'static> =
        NodeMaskBuilder::new().with_attrs(AttributeMaskBuilder::Some(&[
            AttributeName::Sampling,
            AttributeName::ImageData,
            AttributeName::ImageCacheKey,
            AttributeName::AspectRatio,
            AttributeName::ImageCover,
            AttributeName::ImageReference,
        ]));

    fn allow_node(node_type: &NodeType<CustomAttributeValues>) -> bool {
        node_type.tag() == Some(&TagName::Image)
    }

    fn update<'a>(
        &mut self,
        node_view: NodeView<CustomAttributeValues>,
        _node: <Self::NodeDependencies as Dependancy>::ElementBorrowed<'a>,
        _parent: Option<<Self::ParentDependencies as Dependancy>::ElementBorrowed<'a>>,
        _children: Vec<<Self::ChildDependencies as Dependancy>::ElementBorrowed<'a>>,
        context: &SendAnyMap,
    ) -> bool {
        let mut image = ImageState::default();

        if let Some(attributes) = node_view.attributes() {
            for attr in attributes {
                image.parse_safe(attr)
            }
        }

        let changed = &image != self;

        if changed {
            let compositor_dirty_nodes = context.get::<Arc<Mutex<CompositorDirtyNodes>>>().unwrap();
            compositor_dirty_nodes
                .lock()
                .unwrap()
                .invalidate(node_view.node_id());

            if let Some(image_cache_key) = &self.image_cache_key {
                let images_cache = context.get::<Arc<Mutex<ImagesCache>>>().unwrap();
                images_cache.lock().unwrap().remove(image_cache_key);
            }

            let torin_layout = context.get::<Arc<Mutex<Torin<NodeId>>>>().unwrap();
            torin_layout.lock().unwrap().invalidate(node_view.node_id());
        }

        *self = image;
        changed
    }
}