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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
use kas::layout::SpriteDisplay;
use kas::{event, prelude::*};
use std::path::PathBuf;
#[derive(Clone, Debug, Default, Widget)]
#[widget(config = noauto)]
pub struct Image {
#[widget_core]
core: CoreData,
sprite: SpriteDisplay,
path: PathBuf,
do_load: bool,
id: Option<ImageId>,
}
impl Image {
pub fn new<P: Into<PathBuf>>(path: P) -> Self {
Image {
core: Default::default(),
sprite: Default::default(),
path: path.into(),
do_load: true,
id: None,
}
}
#[inline]
pub fn with_scaling(mut self, f: impl FnOnce(SpriteDisplay) -> SpriteDisplay) -> Self {
self.sprite = f(self.sprite);
self
}
#[inline]
pub fn set_scaling(&mut self, f: impl FnOnce(&mut SpriteDisplay)) -> TkAction {
f(&mut self.sprite);
TkAction::RESIZE
}
pub fn set_path<P: Into<PathBuf>>(&mut self, mgr: &mut Manager, path: P) {
self.path = path.into();
self.do_load = false;
let mut size = Size::ZERO;
mgr.draw_shared(|ds| {
if let Some(id) = self.id {
ds.image_free(id);
}
match ds.image_from_path(&self.path) {
Ok(id) => {
self.id = Some(id);
size = ds.image_size(id).unwrap_or(Size::ZERO);
}
Err(error) => self.handle_load_fail(&error),
};
});
mgr.redraw(self.id());
if size != self.sprite.size {
self.sprite.size = size;
*mgr |= TkAction::RESIZE;
}
}
pub fn clear(&mut self, mgr: &mut Manager) {
if let Some(id) = self.id.take() {
self.do_load = false;
mgr.draw_shared(|ds| ds.image_free(id));
}
}
fn handle_load_fail(&mut self, mut error: &(dyn std::error::Error)) {
self.id = None;
log::warn!("Failed to load image: {}", self.path.display());
loop {
log::warn!("Cause: {}", error);
if let Some(source) = error.source() {
error = source;
} else {
break;
}
}
}
}
impl WidgetConfig for Image {
fn configure(&mut self, mgr: &mut Manager) {
if self.do_load {
self.do_load = false;
match mgr.draw_shared(|ds| {
ds.image_from_path(&self.path)
.map(|id| (id, ds.image_size(id).unwrap_or(Size::ZERO)))
}) {
Ok((id, size)) => {
self.id = Some(id);
self.sprite.size = size;
}
Err(error) => self.handle_load_fail(&error),
}
}
}
}
impl Layout for Image {
fn size_rules(&mut self, sh: &mut dyn SizeHandle, axis: AxisInfo) -> SizeRules {
self.sprite.size_rules(sh, axis)
}
fn set_rect(&mut self, _: &mut Manager, rect: Rect, align: AlignHints) {
self.core_data_mut().rect = self.sprite.align_rect(rect, align);
}
fn draw(&self, draw: &mut dyn DrawHandle, _: &event::ManagerState, _: bool) {
if let Some(id) = self.id {
draw.image(id, self.rect());
}
}
}