cursive/src/xy.rs

90 lines
2.3 KiB
Rust
Raw Normal View History

use direction::Orientation;
2016-07-14 04:30:30 +00:00
2016-07-13 04:01:11 +00:00
use std::iter;
/// A generic structure with a value for each axis.
#[derive(Debug,Clone,Copy,PartialEq)]
pub struct XY<T> {
/// X-axis value
pub x: T,
/// Y-axis value
pub y: T,
}
2016-07-13 04:33:50 +00:00
impl<T> XY<T> {
2016-07-13 04:01:11 +00:00
/// Creates a new `XY` from the given values.
pub fn new(x: T, y: T) -> Self {
2016-07-13 04:33:50 +00:00
XY { x: x, y: y }
2016-07-13 04:01:11 +00:00
}
2016-07-13 08:19:05 +00:00
/// Creates a new `XY` by applying `f` to `x` and `y`.
pub fn map<U, F: Fn(T) -> U>(self, f: F) -> XY<U> {
XY::new(f(self.x), f(self.y))
}
2016-07-13 04:01:11 +00:00
/// Destructure self into a pair.
pub fn pair(self) -> (T, T) {
(self.x, self.y)
}
2016-07-15 05:52:37 +00:00
/// Return a `XY` with references to this one's values.
2016-07-13 04:01:11 +00:00
pub fn as_ref(&self) -> XY<&T> {
XY::new(&self.x, &self.y)
}
/// Creates an iterator that returns references to `x`, then `y`.
pub fn iter(&self) -> iter::Chain<iter::Once<&T>, iter::Once<&T>> {
iter::once(&self.x).chain(iter::once(&self.y))
}
2016-07-14 04:30:30 +00:00
/// Returns a reference to the value on the given axis.
pub fn get(&self, o: Orientation) -> &T {
match o {
Orientation::Horizontal => &self.x,
Orientation::Vertical => &self.y,
}
}
/// Returns a new `XY` of tuples made by zipping `self` and `other`.
pub fn zip<U>(self, other: XY<U>) -> XY<(T, U)> {
XY::new((self.x, other.x), (self.y, other.y))
}
2016-07-15 05:52:37 +00:00
/// Returns a new `XY` by calling `f` on `self` and `other` for each axis.
pub fn zip_map<U, V, F: Fn(T, U) -> V>(self, other: XY<U>, f: F) -> XY<V> {
XY::new(f(self.x, other.x), f(self.y, other.y))
2016-07-14 04:30:30 +00:00
}
}
impl<T> XY<Option<T>> {
2016-07-15 05:52:37 +00:00
/// Returns a new `XY` by calling `unwrap_or` on each axis.
2016-07-14 04:30:30 +00:00
pub fn unwrap_or(self, other: XY<T>) -> XY<T> {
self.zip_map(other, |s, o| s.unwrap_or(o))
}
2016-07-13 04:01:11 +00:00
}
impl XY<bool> {
/// Returns `true` if any of `x` or `y` is `true`.
pub fn any(&self) -> bool {
self.x || self.y
}
/// Returns `true` if both `x` and `y` are `true`.
pub fn both(&self) -> bool {
self.x && self.y
}
}
2016-07-13 04:33:50 +00:00
impl<T: Copy> XY<T> {
2016-07-13 04:01:11 +00:00
/// Creates a `XY` with both `x` and `y` set to `value`.
pub fn both_from(value: T) -> Self {
2016-07-13 04:01:11 +00:00
XY::new(value, value)
}
}
2016-07-13 04:33:50 +00:00
impl<T> From<(T, T)> for XY<T> {
2016-07-13 04:01:11 +00:00
fn from((x, y): (T, T)) -> Self {
XY::new(x, y)
}
}