1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
use crate::theme::{ThemeColor, ThemeProps, ThemedWidgetProps};
use raui_core::prelude::*;
use serde::{Deserialize, Serialize};

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

#[derive(PropsData, Debug, Default, Clone, Serialize, Deserialize)]
#[props_data(raui_core::props::PropsData)]
#[prefab(raui_core::Prefab)]
pub struct IconPaperProps {
    #[serde(default)]
    pub image: IconImage,
    #[serde(default)]
    pub size_level: usize,
    #[serde(default)]
    pub transform: Transform,
}

pub fn icon_paper(context: WidgetContext) -> WidgetNode {
    let WidgetContext {
        idref,
        key,
        props,
        shared_props,
        ..
    } = context;

    let themed_props = props.read_cloned_or_default::<ThemedWidgetProps>();
    let tint = match shared_props.read::<ThemeProps>() {
        Ok(props) => match themed_props.color {
            ThemeColor::Default => props.active_colors.contrast.default.main,
            ThemeColor::Primary => props.active_colors.contrast.primary.main,
            ThemeColor::Secondary => props.active_colors.contrast.secondary.main,
        },
        Err(_) => Default::default(),
    };
    let icon_props = props.read_cloned_or_default::<IconPaperProps>();
    let size = match shared_props.read::<ThemeProps>() {
        Ok(props) => props
            .icons_level_sizes
            .get(icon_props.size_level)
            .copied()
            .unwrap_or(24.0),
        Err(_) => 24.0,
    };
    let IconImage {
        id,
        source_rect,
        scaling,
    } = icon_props.image;
    let image = ImageBoxImage {
        id,
        source_rect,
        scaling,
        tint,
    };
    let props = ImageBoxProps {
        width: ImageBoxSizeValue::Exact(size),
        height: ImageBoxSizeValue::Exact(size),
        content_keep_aspect_ratio: Some(ImageBoxAspectRatio {
            horizontal_alignment: 0.5,
            vertical_alignment: 0.5,
        }),
        material: ImageBoxMaterial::Image(image),
        transform: icon_props.transform,
    };

    widget! {
        (#{key} | {idref.cloned()} image_box: {props})
    }
}