Skip to main content

ftui_widgets/
constraint_overlay.rs

1#![forbid(unsafe_code)]
2
3//! Constraint visualization overlay for layout debugging.
4//!
5//! Provides a visual overlay that shows layout constraint violations,
6//! requested vs received sizes, and constraint bounds at widget positions.
7//!
8//! # Example
9//!
10//! ```ignore
11//! use ftui_widgets::{ConstraintOverlay, LayoutDebugger, Widget};
12//!
13//! let mut debugger = LayoutDebugger::new();
14//! debugger.set_enabled(true);
15//!
16//! // Record constraint data during layout...
17//!
18//! // Later, render the overlay
19//! let overlay = ConstraintOverlay::new(&debugger);
20//! overlay.render(area, &mut frame);
21//! ```
22
23use 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/// Visualization style for constraint overlay.
32#[derive(Debug, Clone)]
33pub struct ConstraintOverlayStyle {
34    /// Border color for widgets without constraint violations.
35    pub normal_color: PackedRgba,
36    /// Border color for widgets exceeding max constraints (overflow).
37    pub overflow_color: PackedRgba,
38    /// Border color for widgets below min constraints (underflow).
39    pub underflow_color: PackedRgba,
40    /// Color for the "requested" size outline.
41    pub requested_color: PackedRgba,
42    /// Label foreground color.
43    pub label_fg: PackedRgba,
44    /// Label background color.
45    pub label_bg: PackedRgba,
46    /// Whether to show requested vs received size difference.
47    pub show_size_diff: bool,
48    /// Whether to show constraint bounds in labels.
49    pub show_constraint_bounds: bool,
50    /// Whether to show border outlines.
51    pub show_borders: bool,
52    /// Whether to show labels.
53    pub show_labels: bool,
54    /// Border characters to use.
55    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
76/// Constraint visualization overlay widget.
77///
78/// Renders layout constraint information as a visual overlay:
79/// - Red borders for overflow violations (received > max)
80/// - Yellow borders for underflow violations (received < min)
81/// - Green borders for widgets within constraints
82/// - Blue dashed outline showing requested size vs received size
83/// - Labels showing widget name, sizes, and constraint bounds
84pub struct ConstraintOverlay<'a> {
85    debugger: &'a LayoutDebugger,
86    style: ConstraintOverlayStyle,
87}
88
89impl<'a> ConstraintOverlay<'a> {
90    /// Create a new constraint overlay for the given debugger.
91    pub fn new(debugger: &'a LayoutDebugger) -> Self {
92        Self {
93            debugger,
94            style: ConstraintOverlayStyle::default(),
95        }
96    }
97
98    /// Set custom styling.
99    #[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        // Only render if the received area intersects with our render area
107        let Some(clipped) = record.area_received.intersection_opt(&area) else {
108            return;
109        };
110        if clipped.is_empty() {
111            return;
112        }
113
114        // Determine constraint status
115        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        // Draw received area border
132        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        // Draw requested area outline if different from received
138        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                // Draw dashed corners to indicate requested size
145                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        // Draw label
151        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        // Render children
166        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        // Draw corner dots to indicate requested size boundary
173        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        // Add size info
203        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        // Add constraint bounds if requested
215        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        // Not enabled, so record is ignored
262        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        // Buffer should be unchanged (all default cells)
275        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        // Should have border drawn
295        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        // Received 10x4 but max is 8x3 (overflow)
304        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        // Received 4x2 but min is 6x3 (underflow)
330        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        // Requested 10x5 but got 8x4
356        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        // Corner of requested area (10x5) should have dot marker
374        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        // Both parent and child should have borders
405        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        // Render area is 0,0,10,10 but widget is at 5,5,10,10
427        overlay.render(Rect::new(0, 0, 10, 10), &mut frame);
428
429        // Should render the visible portion
430        let cell = frame.buffer.get(5, 5).unwrap();
431        assert_eq!(cell.content.as_char(), Some('+'));
432
433        // Outside render area should be empty
434        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        // Overflow case
444        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        // Underflow case
454        let label = overlay.format_label(&record, false, true);
455        assert!(label.starts_with("Widget?"));
456
457        // Normal case
458        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}