raui-core 0.70.17

RAUI application layer
Documentation
use crate::{
    PrefabValue, Scalar,
    layout::CoordsMappingScaling,
    props::Props,
    widget::{
        WidgetId,
        node::WidgetNode,
        unit::WidgetUnitData,
        utils::{Color, Rect, Transform, Vec2},
    },
};
use serde::{Deserialize, Serialize};
use std::{collections::HashMap, convert::TryFrom, sync::Arc};

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBoxFrame {
    #[serde(default)]
    pub source: Rect,
    #[serde(default)]
    pub destination: Rect,
    #[serde(default)]
    pub frame_only: bool,
    #[serde(default)]
    pub frame_keep_aspect_ratio: bool,
}

impl From<Scalar> for ImageBoxFrame {
    fn from(v: Scalar) -> Self {
        Self {
            source: v.into(),
            destination: v.into(),
            frame_only: false,
            frame_keep_aspect_ratio: false,
        }
    }
}

impl From<(Scalar, bool)> for ImageBoxFrame {
    fn from((v, fo): (Scalar, bool)) -> Self {
        Self {
            source: v.into(),
            destination: v.into(),
            frame_only: fo,
            frame_keep_aspect_ratio: false,
        }
    }
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub enum ImageBoxImageScaling {
    #[default]
    Stretch,
    Frame(ImageBoxFrame),
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBoxColor {
    #[serde(default)]
    pub color: Color,
    #[serde(default)]
    pub scaling: ImageBoxImageScaling,
}

#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ImageBoxImage {
    #[serde(default)]
    pub id: String,
    #[serde(default)]
    #[serde(skip_serializing_if = "Option::is_none")]
    pub source_rect: Option<Rect>,
    #[serde(default)]
    pub scaling: ImageBoxImageScaling,
    #[serde(default = "ImageBoxImage::default_tint")]
    pub tint: Color,
}

impl Default for ImageBoxImage {
    fn default() -> Self {
        Self {
            id: Default::default(),
            source_rect: Default::default(),
            scaling: Default::default(),
            tint: Self::default_tint(),
        }
    }
}

impl ImageBoxImage {
    fn default_tint() -> Color {
        Color {
            r: 1.0,
            g: 1.0,
            b: 1.0,
            a: 1.0,
        }
    }
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBoxProceduralVertex {
    #[serde(default)]
    pub position: Vec2,
    #[serde(default)]
    pub page: Scalar,
    #[serde(default)]
    pub tex_coord: Vec2,
    #[serde(default)]
    pub color: Color,
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBoxProceduralMeshData {
    #[serde(default)]
    #[serde(skip_serializing_if = "Vec::is_empty")]
    pub vertices: Vec<ImageBoxProceduralVertex>,
    #[serde(default)]
    #[serde(skip_serializing_if = "Vec::is_empty")]
    pub triangles: Vec<[u32; 3]>,
}

#[derive(Clone, Serialize, Deserialize)]
pub enum ImageBoxProceduralMesh {
    Owned(ImageBoxProceduralMeshData),
    Shared(Arc<ImageBoxProceduralMeshData>),
    /// fn(widget local space rect, procedural image parameters map) -> mesh data
    #[serde(skip)]
    #[allow(clippy::type_complexity)]
    Generator(
        Arc<
            dyn Fn(Rect, &HashMap<String, Scalar>) -> ImageBoxProceduralMeshData
                + 'static
                + Send
                + Sync,
        >,
    ),
}

impl std::fmt::Debug for ImageBoxProceduralMesh {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Self::Owned(data) => write!(f, "Owned({data:?})"),
            Self::Shared(data) => write!(f, "Shared({data:?})"),
            Self::Generator(_) => write!(f, "Generator(...)"),
        }
    }
}

impl Default for ImageBoxProceduralMesh {
    fn default() -> Self {
        Self::Owned(Default::default())
    }
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBoxProcedural {
    #[serde(default)]
    pub id: String,
    #[serde(default)]
    #[serde(skip_serializing_if = "HashMap::is_empty")]
    pub parameters: HashMap<String, Scalar>,
    #[serde(default)]
    #[serde(skip_serializing_if = "Vec::is_empty")]
    pub images: Vec<String>,
    #[serde(default)]
    pub mesh: ImageBoxProceduralMesh,
    #[serde(default)]
    pub vertex_mapping: CoordsMappingScaling,
}

impl ImageBoxProcedural {
    pub fn new(id: impl ToString) -> Self {
        Self {
            id: id.to_string(),
            parameters: Default::default(),
            images: Default::default(),
            mesh: Default::default(),
            vertex_mapping: Default::default(),
        }
    }

    pub fn param(mut self, id: impl ToString, value: Scalar) -> Self {
        self.parameters.insert(id.to_string(), value);
        self
    }

    pub fn image(mut self, id: impl ToString) -> Self {
        self.images.push(id.to_string());
        self
    }

    pub fn mesh(mut self, mesh: ImageBoxProceduralMesh) -> Self {
        self.mesh = mesh;
        self
    }

    pub fn triangle(mut self, vertices: [ImageBoxProceduralVertex; 3]) -> Self {
        if let ImageBoxProceduralMesh::Owned(mesh) = &mut self.mesh {
            let count = mesh.vertices.len() as u32;
            mesh.vertices.extend(vertices);
            mesh.triangles.push([count, count + 1, count + 2]);
        }
        self
    }

    pub fn quad(mut self, vertices: [ImageBoxProceduralVertex; 4]) -> Self {
        if let ImageBoxProceduralMesh::Owned(mesh) = &mut self.mesh {
            let count = mesh.vertices.len() as u32;
            mesh.vertices.extend(vertices);
            mesh.triangles.push([count, count + 1, count + 2]);
            mesh.triangles.push([count + 2, count + 3, count]);
        }
        self
    }

    pub fn extend(
        mut self,
        vertices: impl IntoIterator<Item = ImageBoxProceduralVertex>,
        triangles: impl IntoIterator<Item = [u32; 3]>,
    ) -> Self {
        if let ImageBoxProceduralMesh::Owned(mesh) = &mut self.mesh {
            let count = mesh.vertices.len() as u32;
            mesh.vertices.extend(vertices);
            mesh.triangles.extend(
                triangles
                    .into_iter()
                    .map(|[a, b, c]| [a + count, b + count, c + count]),
            );
        }
        self
    }

    pub fn vertex_mapping(mut self, value: CoordsMappingScaling) -> Self {
        self.vertex_mapping = value;
        self
    }
}

#[derive(Debug, Clone, Serialize, Deserialize)]
pub enum ImageBoxMaterial {
    Color(ImageBoxColor),
    Image(ImageBoxImage),
    Procedural(ImageBoxProcedural),
}

impl Default for ImageBoxMaterial {
    fn default() -> Self {
        Self::Color(Default::default())
    }
}

#[derive(Debug, Default, Copy, Clone, Serialize, Deserialize)]
pub enum ImageBoxSizeValue {
    #[default]
    Fill,
    Exact(Scalar),
}

#[derive(Debug, Default, Copy, Clone, Serialize, Deserialize)]
pub struct ImageBoxAspectRatio {
    #[serde(default)]
    pub horizontal_alignment: Scalar,
    #[serde(default)]
    pub vertical_alignment: Scalar,
    #[serde(default)]
    pub outside: bool,
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct ImageBox {
    #[serde(default)]
    pub id: WidgetId,
    #[serde(default)]
    pub width: ImageBoxSizeValue,
    #[serde(default)]
    pub height: ImageBoxSizeValue,
    #[serde(default)]
    #[serde(skip_serializing_if = "Option::is_none")]
    pub content_keep_aspect_ratio: Option<ImageBoxAspectRatio>,
    #[serde(default)]
    pub material: ImageBoxMaterial,
    #[serde(default)]
    pub transform: Transform,
}

impl WidgetUnitData for ImageBox {
    fn id(&self) -> &WidgetId {
        &self.id
    }
}

impl TryFrom<ImageBoxNode> for ImageBox {
    type Error = ();

    fn try_from(node: ImageBoxNode) -> Result<Self, Self::Error> {
        let ImageBoxNode {
            id,
            width,
            height,
            content_keep_aspect_ratio,
            material,
            transform,
            ..
        } = node;
        Ok(Self {
            id,
            width,
            height,
            content_keep_aspect_ratio,
            material,
            transform,
        })
    }
}

#[derive(Debug, Default, Clone)]
pub struct ImageBoxNode {
    pub id: WidgetId,
    pub props: Props,
    pub width: ImageBoxSizeValue,
    pub height: ImageBoxSizeValue,
    pub content_keep_aspect_ratio: Option<ImageBoxAspectRatio>,
    pub material: ImageBoxMaterial,
    pub transform: Transform,
}

impl ImageBoxNode {
    pub fn remap_props<F>(&mut self, mut f: F)
    where
        F: FnMut(Props) -> Props,
    {
        let props = std::mem::take(&mut self.props);
        self.props = (f)(props);
    }
}

impl From<ImageBoxNode> for WidgetNode {
    fn from(data: ImageBoxNode) -> Self {
        Self::Unit(data.into())
    }
}

#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub(crate) struct ImageBoxNodePrefab {
    #[serde(default)]
    pub id: WidgetId,
    #[serde(default)]
    pub props: PrefabValue,
    #[serde(default)]
    pub width: ImageBoxSizeValue,
    #[serde(default)]
    pub height: ImageBoxSizeValue,
    #[serde(default)]
    #[serde(skip_serializing_if = "Option::is_none")]
    pub content_keep_aspect_ratio: Option<ImageBoxAspectRatio>,
    #[serde(default)]
    pub material: ImageBoxMaterial,
    #[serde(default)]
    pub transform: Transform,
}