1use crate::{
2 pos2, vec2, Align2, Color32, Context, CursorIcon, Id, NumExt, Rect, Response, Sense, Shape, Ui,
3 UiBuilder, UiKind, UiStackInfo, Vec2, Vec2b,
4};
5
6#[derive(Clone, Copy, Debug)]
7#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
8pub(crate) struct State {
9 pub(crate) desired_size: Vec2,
14
15 last_content_size: Vec2,
17
18 pub(crate) requested_size: Option<Vec2>,
20}
21
22impl State {
23 pub fn load(ctx: &Context, id: Id) -> Option<Self> {
24 ctx.data_mut(|d| d.get_persisted(id))
25 }
26
27 pub fn store(self, ctx: &Context, id: Id) {
28 ctx.data_mut(|d| d.insert_persisted(id, self));
29 }
30}
31
32#[derive(Clone, Copy, Debug)]
34#[must_use = "You should call .show()"]
35pub struct Resize {
36 id: Option<Id>,
37 id_salt: Option<Id>,
38
39 resizable: Vec2b,
41
42 pub(crate) min_size: Vec2,
43 pub(crate) max_size: Vec2,
44
45 default_size: Vec2,
46
47 with_stroke: bool,
48}
49
50impl Default for Resize {
51 fn default() -> Self {
52 Self {
53 id: None,
54 id_salt: None,
55 resizable: Vec2b::TRUE,
56 min_size: Vec2::splat(16.0),
57 max_size: Vec2::splat(f32::INFINITY),
58 default_size: vec2(320.0, 128.0), with_stroke: true,
60 }
61 }
62}
63
64impl Resize {
65 #[inline]
67 pub fn id(mut self, id: Id) -> Self {
68 self.id = Some(id);
69 self
70 }
71
72 #[inline]
74 #[deprecated = "Renamed id_salt"]
75 pub fn id_source(self, id_salt: impl std::hash::Hash) -> Self {
76 self.id_salt(id_salt)
77 }
78
79 #[inline]
81 pub fn id_salt(mut self, id_salt: impl std::hash::Hash) -> Self {
82 self.id_salt = Some(Id::new(id_salt));
83 self
84 }
85
86 #[inline]
93 pub fn default_width(mut self, width: f32) -> Self {
94 self.default_size.x = width;
95 self
96 }
97
98 #[inline]
106 pub fn default_height(mut self, height: f32) -> Self {
107 self.default_size.y = height;
108 self
109 }
110
111 #[inline]
112 pub fn default_size(mut self, default_size: impl Into<Vec2>) -> Self {
113 self.default_size = default_size.into();
114 self
115 }
116
117 #[inline]
119 pub fn min_size(mut self, min_size: impl Into<Vec2>) -> Self {
120 self.min_size = min_size.into();
121 self
122 }
123
124 #[inline]
126 pub fn min_width(mut self, min_width: f32) -> Self {
127 self.min_size.x = min_width;
128 self
129 }
130
131 #[inline]
133 pub fn min_height(mut self, min_height: f32) -> Self {
134 self.min_size.y = min_height;
135 self
136 }
137
138 #[inline]
140 pub fn max_size(mut self, max_size: impl Into<Vec2>) -> Self {
141 self.max_size = max_size.into();
142 self
143 }
144
145 #[inline]
147 pub fn max_width(mut self, max_width: f32) -> Self {
148 self.max_size.x = max_width;
149 self
150 }
151
152 #[inline]
154 pub fn max_height(mut self, max_height: f32) -> Self {
155 self.max_size.y = max_height;
156 self
157 }
158
159 #[inline]
165 pub fn resizable(mut self, resizable: impl Into<Vec2b>) -> Self {
166 self.resizable = resizable.into();
167 self
168 }
169
170 #[inline]
171 pub fn is_resizable(&self) -> Vec2b {
172 self.resizable
173 }
174
175 pub fn auto_sized(self) -> Self {
178 self.min_size(Vec2::ZERO)
179 .default_size(Vec2::splat(f32::INFINITY))
180 .resizable(false)
181 }
182
183 #[inline]
184 pub fn fixed_size(mut self, size: impl Into<Vec2>) -> Self {
185 let size = size.into();
186 self.default_size = size;
187 self.min_size = size;
188 self.max_size = size;
189 self.resizable = Vec2b::FALSE;
190 self
191 }
192
193 #[inline]
194 pub fn with_stroke(mut self, with_stroke: bool) -> Self {
195 self.with_stroke = with_stroke;
196 self
197 }
198}
199
200struct Prepared {
201 id: Id,
202 corner_id: Option<Id>,
203 state: State,
204 content_ui: Ui,
205}
206
207impl Resize {
208 fn begin(&self, ui: &mut Ui) -> Prepared {
209 let position = ui.available_rect_before_wrap().min;
210 let id = self.id.unwrap_or_else(|| {
211 let id_salt = self.id_salt.unwrap_or_else(|| Id::new("resize"));
212 ui.make_persistent_id(id_salt)
213 });
214
215 let mut state = State::load(ui.ctx(), id).unwrap_or_else(|| {
216 ui.ctx().request_repaint(); let default_size = self
219 .default_size
220 .at_least(self.min_size)
221 .at_most(self.max_size)
222 .at_most(
223 ui.ctx().screen_rect().size() - ui.spacing().window_margin.sum(), );
225
226 State {
227 desired_size: default_size,
228 last_content_size: vec2(0.0, 0.0),
229 requested_size: None,
230 }
231 });
232
233 state.desired_size = state
234 .desired_size
235 .at_least(self.min_size)
236 .at_most(self.max_size);
237
238 let mut user_requested_size = state.requested_size.take();
239
240 let corner_id = self.resizable.any().then(|| id.with("__resize_corner"));
241
242 if let Some(corner_id) = corner_id {
243 if let Some(corner_response) = ui.ctx().read_response(corner_id) {
244 if let Some(pointer_pos) = corner_response.interact_pointer_pos() {
245 user_requested_size =
247 Some(pointer_pos - position + 0.5 * corner_response.rect.size());
248 }
249 }
250 }
251
252 if let Some(user_requested_size) = user_requested_size {
253 state.desired_size = user_requested_size;
254 } else {
255 state.desired_size = state.desired_size.max(state.last_content_size);
259 }
260
261 state.desired_size = state
262 .desired_size
263 .at_least(self.min_size)
264 .at_most(self.max_size);
265
266 let inner_rect = Rect::from_min_size(position, state.desired_size);
269
270 let mut content_clip_rect = inner_rect.expand(ui.visuals().clip_rect_margin);
271
272 content_clip_rect.max = content_clip_rect.max.max(
278 inner_rect.min + state.last_content_size + Vec2::splat(ui.visuals().clip_rect_margin),
279 );
280
281 content_clip_rect = content_clip_rect.intersect(ui.clip_rect()); let mut content_ui = ui.new_child(
284 UiBuilder::new()
285 .ui_stack_info(UiStackInfo::new(UiKind::Resize))
286 .max_rect(inner_rect),
287 );
288 content_ui.set_clip_rect(content_clip_rect);
289
290 Prepared {
291 id,
292 corner_id,
293 state,
294 content_ui,
295 }
296 }
297
298 pub fn show<R>(self, ui: &mut Ui, add_contents: impl FnOnce(&mut Ui) -> R) -> R {
299 let mut prepared = self.begin(ui);
300 let ret = add_contents(&mut prepared.content_ui);
301 self.end(ui, prepared);
302 ret
303 }
304
305 fn end(self, ui: &mut Ui, prepared: Prepared) {
306 let Prepared {
307 id,
308 corner_id,
309 mut state,
310 content_ui,
311 } = prepared;
312
313 state.last_content_size = content_ui.min_size();
314
315 let mut size = state.last_content_size;
318 for d in 0..2 {
319 if self.with_stroke || self.resizable[d] {
320 state.desired_size[d] = state.desired_size[d].max(state.last_content_size[d]);
324
325 size[d] = state.desired_size[d];
327 } else {
328 size[d] = state.last_content_size[d];
330 }
331 }
332 ui.advance_cursor_after_rect(Rect::from_min_size(content_ui.min_rect().min, size));
333
334 let corner_response = if let Some(corner_id) = corner_id {
337 let corner_size = Vec2::splat(ui.visuals().resize_corner_size);
339 let corner_rect = Rect::from_min_size(
340 content_ui.min_rect().left_top() + size - corner_size,
341 corner_size,
342 );
343 Some(ui.interact(corner_rect, corner_id, Sense::drag()))
344 } else {
345 None
346 };
347
348 if self.with_stroke && corner_response.is_some() {
351 let rect = Rect::from_min_size(content_ui.min_rect().left_top(), state.desired_size);
352 let rect = rect.expand(2.0); ui.painter().add(Shape::rect_stroke(
354 rect,
355 3.0,
356 ui.visuals().widgets.noninteractive.bg_stroke,
357 ));
358 }
359
360 if let Some(corner_response) = corner_response {
361 paint_resize_corner(ui, &corner_response);
362
363 if corner_response.hovered() || corner_response.dragged() {
364 ui.ctx().set_cursor_icon(CursorIcon::ResizeNwSe);
365 }
366 }
367
368 state.store(ui.ctx(), id);
369
370 #[cfg(debug_assertions)]
371 if ui.ctx().style().debug.show_resize {
372 ui.ctx().debug_painter().debug_rect(
373 Rect::from_min_size(content_ui.min_rect().left_top(), state.desired_size),
374 Color32::GREEN,
375 "desired_size",
376 );
377 ui.ctx().debug_painter().debug_rect(
378 Rect::from_min_size(content_ui.min_rect().left_top(), state.last_content_size),
379 Color32::LIGHT_BLUE,
380 "last_content_size",
381 );
382 }
383 }
384}
385
386use epaint::Stroke;
387
388pub fn paint_resize_corner(ui: &Ui, response: &Response) {
389 let stroke = ui.style().interact(response).fg_stroke;
390 paint_resize_corner_with_style(ui, &response.rect, stroke.color, Align2::RIGHT_BOTTOM);
391}
392
393pub fn paint_resize_corner_with_style(
394 ui: &Ui,
395 rect: &Rect,
396 color: impl Into<Color32>,
397 corner: Align2,
398) {
399 let painter = ui.painter();
400 let cp = painter.round_pos_to_pixels(corner.pos_in_rect(rect));
401 let mut w = 2.0;
402 let stroke = Stroke {
403 width: 1.0, color: color.into(),
405 };
406
407 while w <= rect.width() && w <= rect.height() {
408 painter.line_segment(
409 [
410 pos2(cp.x - w * corner.x().to_sign(), cp.y),
411 pos2(cp.x, cp.y - w * corner.y().to_sign()),
412 ],
413 stroke,
414 );
415 w += 4.0;
416 }
417}