appcui 0.4.8

A feature-rich and cross-platform TUI/CUI framework for Rust, enabling modern terminal-based applications on Windows, Linux, and macOS. Includes built-in UI components like buttons, menus, list views, tree views, checkboxes, and more. Perfect for building fast and interactive CLI tools and text-based interfaces.
Documentation
use std::ops::{Add, AddAssign};

use super::{Point, Size};

/// RectAlignment enum represents the alignment of a rectangle in a 2D space.
/// It is used to specify how a rectangle or other rectangular objects should be positioned relative to a given point.
#[repr(u8)]
#[derive(Copy, Clone, PartialEq, Debug, Eq)]
pub enum RectAlignment {
    /// The top-left corner of the rectangle.
    /// This is the default alignment.
    TopLeft = 0,

    /// The top-center of the rectangle.
    /// This alignment is used to center the rectangle horizontally at the top.
    Top,

    /// The top-right corner of the rectangle.
    /// This alignment is used to position the rectangle at the top-right corner.
    TopRight,

    /// The right-center of the rectangle.
    /// This alignment is used to center the rectangle vertically on the right side.
    Right,

    /// The bottom-right corner of the rectangle.
    /// This alignment is used to position the rectangle at the bottom-right corner.
    BottomRight,

    /// The bottom-center of the rectangle.
    /// This alignment is used to center the rectangle horizontally at the bottom.
    Bottom,

    /// The bottom-left corner of the rectangle.
    /// This alignment is used to position the rectangle at the bottom-left corner.
    BottomLeft,

    /// The left-center of the rectangle.
    /// This alignment is used to center the rectangle vertically on the left side.
    Left,

    /// The center of the rectangle.
    /// This alignment is used to center the rectangle both horizontally and vertically.
    Center,
}

/// A rectangle defined by its left, top, right, and bottom coordinates.
/// The coordinates are automatically adjusted to ensure that left <= right and top <= bottom.
#[derive(Copy, Clone, Debug, Default, Eq, PartialEq)]
pub struct Rect {
    left: i32,
    top: i32,
    right: i32,
    bottom: i32,
}
impl Rect {
    /// Creates a new rectangle with the specified coordinates. The coordinates are automatically
    /// adjusted to ensure that left <= right and top <= bottom.
    pub fn new(left: i32, top: i32, right: i32, bottom: i32) -> Self {
        Self {
            left: left.min(right),
            top: top.min(bottom),
            right: right.max(left),
            bottom: bottom.max(top),
        }
    }

    /// Creates a new rectangle with the specified coordinates and size.
    pub fn with_size(x: i32, y: i32, width: u16, height: u16) -> Self {
        Self {
            left: x,
            top: y,
            right: x + (width as i32).max(1) - 1,
            bottom: y + (height as i32).max(1) - 1,
        }
    }

    /// Creates a new rectangle with the specified coordinates, size and alignment.
    /// The alignment can have the following values:
    /// - `TopLeft`: the x,y coordinates are the top-left corner of the rectangle
    /// - `Top`: the x,y coordinates are the top-center of the rectangle
    /// - `TopRight`: the x,y coordinates are the top-right corner of the rectangle
    /// - `Right`: the x,y coordinates are the right-center of the rectangle
    /// - `BottomRight`: the x,y coordinates are the bottom-right corner of the rectangle
    /// - `Bottom`: the x,y coordinates are the bottom-center of the rectangle
    /// - `BottomLeft`: the x,y coordinates are the bottom-left corner of the rectangle
    /// - `Left`: the x,y coordinates are the left-center of the rectangle
    /// - `Center`: the x,y coordinates are the center of the rectangle
    pub fn with_alignment(x: i32, y: i32, width: u16, height: u16, align: RectAlignment) -> Self {
        let x = match align {
            RectAlignment::TopLeft => x,
            RectAlignment::Top => x - ((width as i32) / 2),
            RectAlignment::TopRight => x - (width as i32) + 1,
            RectAlignment::Right => x - (width as i32) + 1,
            RectAlignment::BottomRight => x - (width as i32) + 1,
            RectAlignment::Bottom => x - ((width as i32) / 2),
            RectAlignment::BottomLeft => x,
            RectAlignment::Left => x,
            RectAlignment::Center => x - ((width as i32) / 2),
        };
        let y = match align {
            RectAlignment::TopLeft => y,
            RectAlignment::Top => y,
            RectAlignment::TopRight => y,
            RectAlignment::Right => y - ((height as i32) / 2),
            RectAlignment::BottomRight => y - (height as i32) + 1,
            RectAlignment::Bottom => y - (height as i32) + 1,
            RectAlignment::BottomLeft => y - (height as i32) + 1,
            RectAlignment::Left => y - ((height as i32) / 2),
            RectAlignment::Center => y - ((height as i32) / 2),
        };
        Self {
            left: x,
            top: y,
            right: x + (width as i32).max(1) - 1,
            bottom: y + (height as i32).max(1) - 1,
        }
    }

    /// Creates a new rectangle with the specified coordinates and size.
    /// The coordonates are represented by a Point and are considered the top-left corner of the rectangle.
    pub fn with_point_and_size(point: Point, size: Size) -> Self {
        Rect {
            left: point.x,
            top: point.y,
            right: point.x + (size.width.saturating_sub(1) as i32),
            bottom: point.y + (size.height.saturating_sub(1) as i32),
        }
    }

    /// Returns the rectangle left coordinate.
    #[inline(always)]
    pub fn left(&self) -> i32 {
        self.left
    }

    /// Returns the rectangle right coordinate.
    #[inline(always)]
    pub fn right(&self) -> i32 {
        self.right
    }

    /// Returns the rectangle top coordinate.
    #[inline(always)]
    pub fn top(&self) -> i32 {
        self.top
    }

    /// Returns the rectangle bottom coordinate.
    #[inline(always)]
    pub fn bottom(&self) -> i32 {
        self.bottom
    }

    /// Returns the rectangle width.
    #[inline(always)]
    pub fn width(&self) -> u32 {
        ((self.right - self.left) as u32) + 1u32
    }

    /// Returns the rectangle height.
    #[inline(always)]
    pub fn height(&self) -> u32 {
        ((self.bottom - self.top) as u32) + 1u32
    }

    /// Returns the size of the Rect
    #[inline(always)]
    pub fn size(&self) -> Size {
        Size::new(((self.right - self.left) as u32) + 1u32, ((self.bottom - self.top) as u32) + 1u32)
    }

    /// Returns the rectangle X-axis center.
    #[inline(always)]
    pub fn center_x(&self) -> i32 {
        (self.right + self.left) / 2
    }

    /// Returns the rectangle Y-axis center.
    #[inline(always)]
    pub fn center_y(&self) -> i32 {
        (self.bottom + self.top) / 2
    }

    /// Returns true if the rectangle contains the given point.
    #[inline(always)]
    pub fn contains(&self, point: Point) -> bool {
        (point.x >= self.left) && (point.x <= self.right) && (point.y >= self.top) && (point.y <= self.bottom)
    }

    /// Returns true if the rectangle fully contains the given rectangle.
    /// If only a part of the rectangle is inside this rectangle, false is returned.
    #[inline(always)]
    pub fn contains_rect(&self, rect: Rect) -> bool {
        (rect.left >= self.left) && (rect.right <= self.right) && (rect.top >= self.top) && (rect.bottom <= self.bottom)
    }

    /// Returns the center point of the rectangle.
    #[inline(always)]
    pub fn center(&self) -> Point {
        Point {
            x: self.center_x(),
            y: self.center_y(),
        }
    }

    /// Returns the top-left corner of the rectangle.
    #[inline(always)]
    pub fn top_left(&self) -> Point {
        Point { x: self.left, y: self.top }
    }

    /// Returns the top-right corner of the rectangle.
    #[inline(always)]
    pub fn top_right(&self) -> Point {
        Point { x: self.right, y: self.top }
    }

    /// Returns the bottom-right corner of the rectangle.
    #[inline(always)]
    pub fn bottom_right(&self) -> Point {
        Point {
            x: self.right,
            y: self.bottom,
        }
    }

    /// Returns the bottom-left corner of the rectangle.
    #[inline(always)]
    pub fn bottom_left(&self) -> Point {
        Point {
            x: self.left,
            y: self.bottom,
        }
    }

    /// Inflates the rectangle by the given amount in left, top, right, bottom.
    /// if the margins cross each other (e.g. there is a right margin becomes smaller than the left margin) the rights margin will be clamped to the left margin.
    #[inline(always)]
    pub fn inflate_width(&mut self, left: i32, top: i32, right: i32, bottom: i32) {
        self.left = self.right.min(self.left - left);
        self.top = self.bottom.min(self.top - top);
        self.right = self.left.max(self.right + right);
        self.bottom = self.top.max(self.bottom + bottom);
    }
    /// Translates this rectangle **in place** by the given offsets.
    ///
    /// The rectangle’s size (width/height) is unchanged; only its position moves.
    /// Positive `dx` moves it right; positive `dy` moves it down.
    #[inline(always)]
    pub fn translate(&mut self, dx: i32, dy: i32) {
        self.left += dx;
        self.top += dy;
        self.right += dx;
        self.bottom += dy;
    }
    /// Sets the **left edge** of the rectangle to `x` value. The parameter `preserve_weight` indicates if the width of the rectangle should be preserved or not.
    /// If `preserve_weight` is true, the rectangle is shifted horizontally so that `left == x` and the `right` edge is updated accordingly, leaving the width unchanged.
    /// If `preserve_weight` is false, the `left` edge is updated to `x` only if `x` is less than or equal to the current `right` edge.  
    /// The vertical position (`top`, `bottom`) is unaffected.
    #[inline]
    pub fn set_left(&mut self, x: i32, preserve_weight: bool) {
        if preserve_weight {
            let w = self.right - self.left;
            self.left = x;
            self.right = x + w;
        } else if x <= self.right {
            self.left = x;
        }
    }

    /// Sets the **right edge** of the rectangle to `x` value. The parameter `preserve_weight` indicates if the width of the rectangle should be preserved or not.
    /// If `preserve_weight` is true, the rectangle is shifted horizontally so that `right == x` and the `left` edge is updated accordingly, leaving the width unchanged.
    /// If `preserve_weight` is false, the `right` edge is updated to `x` only if `x` is greater than or equal to the current `left` edge.  
    /// The vertical position (`top`, `bottom`) is unaffected.
    #[inline]
    pub fn set_right(&mut self, x: i32, preserve_weight: bool) {
        if preserve_weight {
            let w = self.right - self.left;
            self.right = x;
            self.left = x - w;
        } else if x >= self.left {
            self.right = x;
        }
    }

    /// Sets the **top edge** of the rectangle to `y` value. The parameter `preserve_height` indicates if the height of the rectangle should be preserved or not.
    /// If `preserve_height` is true, the rectangle is shifted vertically so that `top == y` and the `bottom` edge is updated accordingly, leaving the height unchanged.
    /// If `preserve_height` is false, the `top` edge is updated to `y` only if `y` is less than or equal to the current `bottom` edge.  
    /// The horizontal position (`left`, `right`) is unaffected.
    #[inline]
    pub fn set_top(&mut self, y: i32, preserve_height: bool) {
        if preserve_height {
            let h = self.bottom - self.top;
            self.top = y;
            self.bottom = y + h;
        } else if y <= self.bottom {
            self.top = y;
        }
    }

    /// Sets the **bottom edge** of the rectangle to `y` value. The parameter `preserve_height` indicates if the height of the rectangle should be preserved or not.
    /// If `preserve_height` is true, the rectangle is shifted vertically so that `bottom == y` and the `top` edge is updated accordingly, leaving the height unchanged.
    /// If `preserve_height` is false, the `bottom` edge is updated to `y` only if `y` is greater than or equal to the current `top` edge.  
    /// The horizontal position (`left`, `right`) is unaffected.
    #[inline]
    pub fn set_bottom(&mut self, y: i32, preserve_height: bool) {
        if preserve_height {
            let h = self.bottom - self.top;
            self.bottom = y;
            self.top = y - h;
        } else if y >= self.top {
            self.bottom = y;
        }
    }
}

/// Adds a `(dx, dy)` offset to this rectangle in place.
impl AddAssign<(i32, i32)> for Rect {
    fn add_assign(&mut self, (dx, dy): (i32, i32)) {
        self.translate(dx, dy);
    }
}

/// Adds a `(dx, dy)` offset to this rectangle and returns a new rectangle.
impl Add<(i32, i32)> for Rect {
    type Output = Rect;

    fn add(self, rhs: (i32, i32)) -> Self::Output {
        Rect {
            left: self.left + rhs.0,
            top: self.top + rhs.1,
            right: self.right + rhs.0,
            bottom: self.bottom + rhs.1,
        }
    }
}