//! Generic values for four sided properties. use crate::error::{ParserError, PrinterError}; use crate::printer::Printer; use crate::traits::{IsCompatible, Parse, ToCss}; #[cfg(feature = "visitor")] use crate::visitor::Visit; use cssparser::*; /// A generic value that represents a value for four sides of a box, /// e.g. border-width, margin, padding, etc. /// /// When serialized, as few components as possible are written when /// there are duplicate values. #[derive(Clone, Debug, PartialEq, Eq)] #[cfg_attr(feature = "visitor", derive(Visit))] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "jsonschema", derive(schemars::JsonSchema))] #[cfg_attr(feature = "into_owned", derive(static_self::IntoOwned))] pub struct Rect( /// The top component. pub T, /// The right component. pub T, /// The bottom component. pub T, /// The left component. pub T, ); impl Rect { /// Returns a new `Rect` value. pub fn new(first: T, second: T, third: T, fourth: T) -> Self { Rect(first, second, third, fourth) } } impl Default for Rect { fn default() -> Rect { Rect::all(T::default()) } } impl Rect where T: Clone, { /// Returns a rect with all the values equal to `v`. pub fn all(v: T) -> Self { Rect::new(v.clone(), v.clone(), v.clone(), v) } /// Parses a new `Rect` value with the given parse function. pub fn parse_with<'i, 't, Parse>( input: &mut Parser<'i, 't>, parse: Parse, ) -> Result>> where Parse: Fn(&mut Parser<'i, 't>) -> Result>>, { let first = parse(input)?; let second = if let Ok(second) = input.try_parse(|i| parse(i)) { second } else { // return Ok(Self::new(first.clone(), first.clone(), first.clone(), first)); }; let third = if let Ok(third) = input.try_parse(|i| parse(i)) { third } else { // return Ok(Self::new(first.clone(), second.clone(), first, second)); }; let fourth = if let Ok(fourth) = input.try_parse(|i| parse(i)) { fourth } else { // return Ok(Self::new(first, second.clone(), third, second)); }; // Ok(Self::new(first, second, third, fourth)) } } impl<'i, T> Parse<'i> for Rect where T: Clone + PartialEq + Parse<'i>, { fn parse<'t>(input: &mut Parser<'i, 't>) -> Result>> { Self::parse_with(input, T::parse) } } impl ToCss for Rect where T: PartialEq + ToCss, { fn to_css(&self, dest: &mut Printer) -> Result<(), PrinterError> where W: std::fmt::Write, { self.0.to_css(dest)?; let same_vertical = self.0 == self.2; let same_horizontal = self.1 == self.3; if same_vertical && same_horizontal && self.0 == self.1 { return Ok(()); } dest.write_str(" ")?; self.1.to_css(dest)?; if same_vertical && same_horizontal { return Ok(()); } dest.write_str(" ")?; self.2.to_css(dest)?; if same_horizontal { return Ok(()); } dest.write_str(" ")?; self.3.to_css(dest) } } impl IsCompatible for Rect { fn is_compatible(&self, browsers: crate::targets::Browsers) -> bool { self.0.is_compatible(browsers) && self.1.is_compatible(browsers) && self.2.is_compatible(browsers) && self.3.is_compatible(browsers) } }