1#![forbid(unsafe_code)]
2
3use crate::Widget;
24use crate::layout_debugger::{LayoutDebugger, LayoutRecord};
25use ftui_core::geometry::Rect;
26use ftui_render::buffer::Buffer;
27use ftui_render::cell::{Cell, PackedRgba};
28use ftui_render::drawing::{BorderChars, Draw};
29use ftui_render::frame::Frame;
30
31#[derive(Debug, Clone)]
33pub struct ConstraintOverlayStyle {
34 pub normal_color: PackedRgba,
36 pub overflow_color: PackedRgba,
38 pub underflow_color: PackedRgba,
40 pub requested_color: PackedRgba,
42 pub label_fg: PackedRgba,
44 pub label_bg: PackedRgba,
46 pub show_size_diff: bool,
48 pub show_constraint_bounds: bool,
50 pub show_borders: bool,
52 pub show_labels: bool,
54 pub border_chars: BorderChars,
56}
57
58impl Default for ConstraintOverlayStyle {
59 fn default() -> Self {
60 Self {
61 normal_color: PackedRgba::rgb(100, 200, 100),
62 overflow_color: PackedRgba::rgb(240, 80, 80),
63 underflow_color: PackedRgba::rgb(240, 200, 80),
64 requested_color: PackedRgba::rgb(80, 150, 240),
65 label_fg: PackedRgba::rgb(255, 255, 255),
66 label_bg: PackedRgba::rgb(0, 0, 0),
67 show_size_diff: true,
68 show_constraint_bounds: true,
69 show_borders: true,
70 show_labels: true,
71 border_chars: BorderChars::ASCII,
72 }
73 }
74}
75
76pub struct ConstraintOverlay<'a> {
85 debugger: &'a LayoutDebugger,
86 style: ConstraintOverlayStyle,
87}
88
89impl<'a> ConstraintOverlay<'a> {
90 pub fn new(debugger: &'a LayoutDebugger) -> Self {
92 Self {
93 debugger,
94 style: ConstraintOverlayStyle::default(),
95 }
96 }
97
98 #[must_use]
100 pub fn style(mut self, style: ConstraintOverlayStyle) -> Self {
101 self.style = style;
102 self
103 }
104
105 fn render_record(&self, record: &LayoutRecord, area: Rect, buf: &mut Buffer) {
106 let Some(clipped) = record.area_received.intersection_opt(&area) else {
108 return;
109 };
110 if clipped.is_empty() {
111 return;
112 }
113
114 let constraints = &record.constraints;
116 let received = &record.area_received;
117
118 let is_overflow = (constraints.max_width != 0 && received.width > constraints.max_width)
119 || (constraints.max_height != 0 && received.height > constraints.max_height);
120 let is_underflow =
121 received.width < constraints.min_width || received.height < constraints.min_height;
122
123 let border_color = if is_overflow {
124 self.style.overflow_color
125 } else if is_underflow {
126 self.style.underflow_color
127 } else {
128 self.style.normal_color
129 };
130
131 if self.style.show_borders {
133 let border_cell = Cell::from_char('+').with_fg(border_color);
134 buf.draw_border(clipped, self.style.border_chars, border_cell);
135 }
136
137 if self.style.show_size_diff {
139 let requested = &record.area_requested;
140 if requested != received
141 && let Some(req_clipped) = requested.intersection_opt(&area)
142 && !req_clipped.is_empty()
143 {
144 let req_cell = Cell::from_char('.').with_fg(self.style.requested_color);
146 self.draw_requested_outline(req_clipped, buf, req_cell);
147 }
148 }
149
150 if self.style.show_labels {
152 let label = self.format_label(record, is_overflow, is_underflow);
153 let label_x = clipped.x.saturating_add(1);
154 let label_y = clipped.y;
155 let max_x = clipped.right();
156
157 if label_x < max_x {
158 let label_cell = Cell::from_char(' ')
159 .with_fg(self.style.label_fg)
160 .with_bg(self.style.label_bg);
161 let _ = buf.print_text_clipped(label_x, label_y, &label, label_cell, max_x);
162 }
163 }
164
165 for child in &record.children {
167 self.render_record(child, area, buf);
168 }
169 }
170
171 fn draw_requested_outline(&self, area: Rect, buf: &mut Buffer, cell: Cell) {
172 if area.width >= 1 && area.height >= 1 {
174 buf.set(area.x, area.y, cell);
175 }
176 if area.width >= 2 && area.height >= 1 {
177 buf.set(area.right().saturating_sub(1), area.y, cell);
178 }
179 if area.width >= 1 && area.height >= 2 {
180 buf.set(area.x, area.bottom().saturating_sub(1), cell);
181 }
182 if area.width >= 2 && area.height >= 2 {
183 buf.set(
184 area.right().saturating_sub(1),
185 area.bottom().saturating_sub(1),
186 cell,
187 );
188 }
189 }
190
191 fn format_label(&self, record: &LayoutRecord, is_overflow: bool, is_underflow: bool) -> String {
192 let status = if is_overflow {
193 "!"
194 } else if is_underflow {
195 "?"
196 } else {
197 ""
198 };
199
200 let mut label = format!("{}{}", record.widget_name, status);
201
202 let req = &record.area_requested;
204 let got = &record.area_received;
205 if req.width != got.width || req.height != got.height {
206 label.push_str(&format!(
207 " {}x{}\u{2192}{}x{}",
208 req.width, req.height, got.width, got.height
209 ));
210 } else {
211 label.push_str(&format!(" {}x{}", got.width, got.height));
212 }
213
214 if self.style.show_constraint_bounds {
216 let c = &record.constraints;
217 if c.min_width != 0 || c.min_height != 0 || c.max_width != 0 || c.max_height != 0 {
218 label.push_str(&format!(
219 " [{}..{} x {}..{}]",
220 c.min_width,
221 if c.max_width == 0 {
222 "\u{221E}".to_string()
223 } else {
224 c.max_width.to_string()
225 },
226 c.min_height,
227 if c.max_height == 0 {
228 "\u{221E}".to_string()
229 } else {
230 c.max_height.to_string()
231 }
232 ));
233 }
234 }
235
236 label
237 }
238}
239
240impl Widget for ConstraintOverlay<'_> {
241 fn render(&self, area: Rect, frame: &mut Frame) {
242 if !self.debugger.enabled() {
243 return;
244 }
245
246 for record in self.debugger.records() {
247 self.render_record(record, area, &mut frame.buffer);
248 }
249 }
250}
251
252#[cfg(test)]
253mod tests {
254 use super::*;
255 use crate::layout_debugger::LayoutConstraints;
256 use ftui_render::grapheme_pool::GraphemePool;
257
258 #[test]
259 fn overlay_renders_nothing_when_disabled() {
260 let mut debugger = LayoutDebugger::new();
261 debugger.record(LayoutRecord::new(
263 "Root",
264 Rect::new(0, 0, 10, 4),
265 Rect::new(0, 0, 10, 4),
266 LayoutConstraints::unconstrained(),
267 ));
268
269 let overlay = ConstraintOverlay::new(&debugger);
270 let mut pool = GraphemePool::new();
271 let mut frame = Frame::new(20, 10, &mut pool);
272 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
273
274 assert!(frame.buffer.get(0, 0).unwrap().is_empty());
276 }
277
278 #[test]
279 fn overlay_renders_border_for_valid_constraint() {
280 let mut debugger = LayoutDebugger::new();
281 debugger.set_enabled(true);
282 debugger.record(LayoutRecord::new(
283 "Root",
284 Rect::new(1, 1, 6, 4),
285 Rect::new(1, 1, 6, 4),
286 LayoutConstraints::new(4, 10, 2, 6),
287 ));
288
289 let overlay = ConstraintOverlay::new(&debugger);
290 let mut pool = GraphemePool::new();
291 let mut frame = Frame::new(20, 10, &mut pool);
292 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
293
294 let cell = frame.buffer.get(1, 1).unwrap();
296 assert_eq!(cell.content.as_char(), Some('+'));
297 }
298
299 #[test]
300 fn overlay_uses_overflow_color_when_exceeds_max() {
301 let mut debugger = LayoutDebugger::new();
302 debugger.set_enabled(true);
303 debugger.record(LayoutRecord::new(
305 "Overflow",
306 Rect::new(0, 0, 10, 4),
307 Rect::new(0, 0, 10, 4),
308 LayoutConstraints::new(0, 8, 0, 3),
309 ));
310
311 let style = ConstraintOverlayStyle {
312 overflow_color: PackedRgba::rgb(255, 0, 0),
313 ..Default::default()
314 };
315
316 let overlay = ConstraintOverlay::new(&debugger).style(style);
317 let mut pool = GraphemePool::new();
318 let mut frame = Frame::new(20, 10, &mut pool);
319 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
320
321 let cell = frame.buffer.get(0, 0).unwrap();
322 assert_eq!(cell.fg, PackedRgba::rgb(255, 0, 0));
323 }
324
325 #[test]
326 fn overlay_uses_underflow_color_when_below_min() {
327 let mut debugger = LayoutDebugger::new();
328 debugger.set_enabled(true);
329 debugger.record(LayoutRecord::new(
331 "Underflow",
332 Rect::new(0, 0, 4, 2),
333 Rect::new(0, 0, 4, 2),
334 LayoutConstraints::new(6, 0, 3, 0),
335 ));
336
337 let style = ConstraintOverlayStyle {
338 underflow_color: PackedRgba::rgb(255, 255, 0),
339 ..Default::default()
340 };
341
342 let overlay = ConstraintOverlay::new(&debugger).style(style);
343 let mut pool = GraphemePool::new();
344 let mut frame = Frame::new(20, 10, &mut pool);
345 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
346
347 let cell = frame.buffer.get(0, 0).unwrap();
348 assert_eq!(cell.fg, PackedRgba::rgb(255, 255, 0));
349 }
350
351 #[test]
352 fn overlay_shows_requested_vs_received_diff() {
353 let mut debugger = LayoutDebugger::new();
354 debugger.set_enabled(true);
355 debugger.record(LayoutRecord::new(
357 "Diff",
358 Rect::new(0, 0, 10, 5),
359 Rect::new(0, 0, 8, 4),
360 LayoutConstraints::unconstrained(),
361 ));
362
363 let style = ConstraintOverlayStyle {
364 requested_color: PackedRgba::rgb(0, 0, 255),
365 ..Default::default()
366 };
367
368 let overlay = ConstraintOverlay::new(&debugger).style(style);
369 let mut pool = GraphemePool::new();
370 let mut frame = Frame::new(20, 10, &mut pool);
371 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
372
373 let cell = frame.buffer.get(9, 0).unwrap();
375 assert_eq!(cell.content.as_char(), Some('.'));
376 assert_eq!(cell.fg, PackedRgba::rgb(0, 0, 255));
377 }
378
379 #[test]
380 fn overlay_renders_children() {
381 let mut debugger = LayoutDebugger::new();
382 debugger.set_enabled(true);
383
384 let child = LayoutRecord::new(
385 "Child",
386 Rect::new(2, 2, 4, 2),
387 Rect::new(2, 2, 4, 2),
388 LayoutConstraints::unconstrained(),
389 );
390 let parent = LayoutRecord::new(
391 "Parent",
392 Rect::new(0, 0, 10, 6),
393 Rect::new(0, 0, 10, 6),
394 LayoutConstraints::unconstrained(),
395 )
396 .with_child(child);
397 debugger.record(parent);
398
399 let overlay = ConstraintOverlay::new(&debugger);
400 let mut pool = GraphemePool::new();
401 let mut frame = Frame::new(20, 10, &mut pool);
402 overlay.render(Rect::new(0, 0, 20, 10), &mut frame);
403
404 let parent_cell = frame.buffer.get(0, 0).unwrap();
406 assert_eq!(parent_cell.content.as_char(), Some('+'));
407
408 let child_cell = frame.buffer.get(2, 2).unwrap();
409 assert_eq!(child_cell.content.as_char(), Some('+'));
410 }
411
412 #[test]
413 fn overlay_clips_to_render_area() {
414 let mut debugger = LayoutDebugger::new();
415 debugger.set_enabled(true);
416 debugger.record(LayoutRecord::new(
417 "PartiallyVisible",
418 Rect::new(5, 5, 10, 10),
419 Rect::new(5, 5, 10, 10),
420 LayoutConstraints::unconstrained(),
421 ));
422
423 let overlay = ConstraintOverlay::new(&debugger);
424 let mut pool = GraphemePool::new();
425 let mut frame = Frame::new(10, 10, &mut pool);
426 overlay.render(Rect::new(0, 0, 10, 10), &mut frame);
428
429 let cell = frame.buffer.get(5, 5).unwrap();
431 assert_eq!(cell.content.as_char(), Some('+'));
432
433 let outside = frame.buffer.get(0, 0).unwrap();
435 assert!(outside.is_empty());
436 }
437
438 #[test]
439 fn format_label_includes_status_marker() {
440 let debugger = LayoutDebugger::new();
441 let overlay = ConstraintOverlay::new(&debugger);
442
443 let record = LayoutRecord::new(
445 "Widget",
446 Rect::new(0, 0, 10, 4),
447 Rect::new(0, 0, 10, 4),
448 LayoutConstraints::new(0, 8, 0, 0),
449 );
450 let label = overlay.format_label(&record, true, false);
451 assert!(label.starts_with("Widget!"));
452
453 let label = overlay.format_label(&record, false, true);
455 assert!(label.starts_with("Widget?"));
456
457 let label = overlay.format_label(&record, false, false);
459 assert!(label.starts_with("Widget "));
460 }
461
462 #[test]
463 fn style_can_be_customized() {
464 let debugger = LayoutDebugger::new();
465 let style = ConstraintOverlayStyle {
466 show_borders: false,
467 show_labels: false,
468 show_size_diff: false,
469 ..Default::default()
470 };
471
472 let overlay = ConstraintOverlay::new(&debugger).style(style);
473 assert!(!overlay.style.show_borders);
474 assert!(!overlay.style.show_labels);
475 }
476}