mirror of
https://github.com/servo/servo
synced 2026-05-12 09:56:50 +02:00
This way we always serialize in the shortest form, and take less space. This is useful because when serializing uncomputed values we'd like to compare to the initial value to avoid serializing parts of a shorthand, but with the existing implementation we would generate always a second keyword, which means that we'll never match it. This also matches Chrome and WebKit, incidentally, so I'm pretty confident the behavior change when serializing specified style is web-compatible. Differential Revision: https://phabricator.services.mozilla.com/D11941
140 lines
4.4 KiB
Rust
140 lines
4.4 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
//! Specified types for CSS values related to backgrounds.
|
|
|
|
use crate::parser::{Parse, ParserContext};
|
|
use crate::values::generics::background::BackgroundSize as GenericBackgroundSize;
|
|
use crate::values::specified::length::NonNegativeLengthOrPercentageOrAuto;
|
|
use cssparser::Parser;
|
|
use selectors::parser::SelectorParseErrorKind;
|
|
use std::fmt::{self, Write};
|
|
use style_traits::{CssWriter, ParseError, ToCss};
|
|
|
|
/// A specified value for the `background-size` property.
|
|
pub type BackgroundSize = GenericBackgroundSize<NonNegativeLengthOrPercentageOrAuto>;
|
|
|
|
impl Parse for BackgroundSize {
|
|
fn parse<'i, 't>(
|
|
context: &ParserContext,
|
|
input: &mut Parser<'i, 't>,
|
|
) -> Result<Self, ParseError<'i>> {
|
|
if let Ok(width) = input.try(|i| NonNegativeLengthOrPercentageOrAuto::parse(context, i)) {
|
|
let height = input
|
|
.try(|i| NonNegativeLengthOrPercentageOrAuto::parse(context, i))
|
|
.unwrap_or(NonNegativeLengthOrPercentageOrAuto::auto());
|
|
return Ok(GenericBackgroundSize::Explicit { width, height });
|
|
}
|
|
Ok(try_match_ident_ignore_ascii_case! { input,
|
|
"cover" => GenericBackgroundSize::Cover,
|
|
"contain" => GenericBackgroundSize::Contain,
|
|
})
|
|
}
|
|
}
|
|
|
|
impl BackgroundSize {
|
|
/// Returns `auto auto`.
|
|
pub fn auto() -> Self {
|
|
GenericBackgroundSize::Explicit {
|
|
width: NonNegativeLengthOrPercentageOrAuto::auto(),
|
|
height: NonNegativeLengthOrPercentageOrAuto::auto(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// One of the keywords for `background-repeat`.
|
|
#[derive(
|
|
Clone,
|
|
Copy,
|
|
Debug,
|
|
Eq,
|
|
MallocSizeOf,
|
|
Parse,
|
|
PartialEq,
|
|
SpecifiedValueInfo,
|
|
ToComputedValue,
|
|
ToCss,
|
|
)]
|
|
#[allow(missing_docs)]
|
|
#[value_info(other_values = "repeat-x,repeat-y")]
|
|
pub enum BackgroundRepeatKeyword {
|
|
Repeat,
|
|
Space,
|
|
Round,
|
|
NoRepeat,
|
|
}
|
|
|
|
/// The value of the `background-repeat` property, with `repeat-x` / `repeat-y`
|
|
/// represented as the combination of `no-repeat` and `repeat` in the opposite
|
|
/// axes.
|
|
///
|
|
/// https://drafts.csswg.org/css-backgrounds/#the-background-repeat
|
|
#[derive(Clone, Debug, MallocSizeOf, PartialEq, SpecifiedValueInfo, ToComputedValue)]
|
|
pub struct BackgroundRepeat(pub BackgroundRepeatKeyword, pub BackgroundRepeatKeyword);
|
|
|
|
impl BackgroundRepeat {
|
|
/// Returns the `repeat repeat` value.
|
|
pub fn repeat() -> Self {
|
|
BackgroundRepeat(
|
|
BackgroundRepeatKeyword::Repeat,
|
|
BackgroundRepeatKeyword::Repeat,
|
|
)
|
|
}
|
|
}
|
|
|
|
impl ToCss for BackgroundRepeat {
|
|
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
|
where
|
|
W: Write,
|
|
{
|
|
match (self.0, self.1) {
|
|
(BackgroundRepeatKeyword::Repeat, BackgroundRepeatKeyword::NoRepeat) => {
|
|
dest.write_str("repeat-x")
|
|
},
|
|
(BackgroundRepeatKeyword::NoRepeat, BackgroundRepeatKeyword::Repeat) => {
|
|
dest.write_str("repeat-y")
|
|
},
|
|
(horizontal, vertical) => {
|
|
horizontal.to_css(dest)?;
|
|
if horizontal != vertical {
|
|
dest.write_str(" ")?;
|
|
vertical.to_css(dest)?;
|
|
}
|
|
Ok(())
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Parse for BackgroundRepeat {
|
|
fn parse<'i, 't>(
|
|
_context: &ParserContext,
|
|
input: &mut Parser<'i, 't>,
|
|
) -> Result<Self, ParseError<'i>> {
|
|
let ident = input.expect_ident_cloned()?;
|
|
|
|
match_ignore_ascii_case! { &ident,
|
|
"repeat-x" => {
|
|
return Ok(BackgroundRepeat(BackgroundRepeatKeyword::Repeat, BackgroundRepeatKeyword::NoRepeat));
|
|
},
|
|
"repeat-y" => {
|
|
return Ok(BackgroundRepeat(BackgroundRepeatKeyword::NoRepeat, BackgroundRepeatKeyword::Repeat));
|
|
},
|
|
_ => {},
|
|
}
|
|
|
|
let horizontal = match BackgroundRepeatKeyword::from_ident(&ident) {
|
|
Ok(h) => h,
|
|
Err(()) => {
|
|
return Err(
|
|
input.new_custom_error(SelectorParseErrorKind::UnexpectedIdent(ident.clone()))
|
|
);
|
|
},
|
|
};
|
|
|
|
let vertical = input.try(BackgroundRepeatKeyword::parse).ok();
|
|
Ok(BackgroundRepeat(horizontal, vertical.unwrap_or(horizontal)))
|
|
}
|
|
}
|