mirror of
https://github.com/servo/servo.git
synced 2025-07-03 05:23:38 +01:00
style: Support calc in cursor.
Bug: 1453258 Reviewed-by: emilio MozReview-Commit-ID: 1ZzxkYboWZg
This commit is contained in:
parent
da43a33fb1
commit
89fba41e1d
3 changed files with 135 additions and 166 deletions
|
@ -6,137 +6,16 @@
|
||||||
//!
|
//!
|
||||||
//! https://drafts.csswg.org/css-ui/#pointing-keyboard
|
//! https://drafts.csswg.org/css-ui/#pointing-keyboard
|
||||||
|
|
||||||
use cssparser::Parser;
|
use values::computed::Number;
|
||||||
use parser::{Parse, ParserContext};
|
|
||||||
use selectors::parser::SelectorParseErrorKind;
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
use std::fmt::{self, Write};
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
use style_traits::{CssWriter, ToCss};
|
|
||||||
use style_traits::ParseError;
|
|
||||||
use style_traits::cursor::CursorKind;
|
|
||||||
use values::computed::color::Color;
|
use values::computed::color::Color;
|
||||||
use values::generics::pointing::CaretColor as GenericCaretColor;
|
use values::computed::url::ComputedImageUrl;
|
||||||
#[cfg(feature = "gecko")]
|
use values::generics::pointing as generics;
|
||||||
use values::specified::url::SpecifiedImageUrl;
|
|
||||||
|
|
||||||
/// The computed value for the `cursor` property.
|
|
||||||
///
|
|
||||||
/// https://drafts.csswg.org/css-ui/#cursor
|
|
||||||
pub use values::specified::pointing::Cursor;
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
pub use values::specified::pointing::CursorImage;
|
|
||||||
|
|
||||||
impl Cursor {
|
|
||||||
/// Set `cursor` to `auto`
|
|
||||||
#[cfg(feature = "servo")]
|
|
||||||
#[inline]
|
|
||||||
pub fn auto() -> Self {
|
|
||||||
Cursor(CursorKind::Auto)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Set `cursor` to `auto`
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
#[inline]
|
|
||||||
pub fn auto() -> Self {
|
|
||||||
Self {
|
|
||||||
images: vec![].into_boxed_slice(),
|
|
||||||
keyword: CursorKind::Auto,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Parse for Cursor {
|
|
||||||
/// cursor: [auto | default | ...]
|
|
||||||
#[cfg(feature = "servo")]
|
|
||||||
fn parse<'i, 't>(
|
|
||||||
context: &ParserContext,
|
|
||||||
input: &mut Parser<'i, 't>,
|
|
||||||
) -> Result<Self, ParseError<'i>> {
|
|
||||||
Ok(Cursor(CursorKind::parse(context, input)?))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// cursor: [<url> [<number> <number>]?]# [auto | default | ...]
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
fn parse<'i, 't>(
|
|
||||||
context: &ParserContext,
|
|
||||||
input: &mut Parser<'i, 't>,
|
|
||||||
) -> Result<Self, ParseError<'i>> {
|
|
||||||
let mut images = vec![];
|
|
||||||
loop {
|
|
||||||
match input.try(|input| CursorImage::parse_image(context, input)) {
|
|
||||||
Ok(image) => images.push(image),
|
|
||||||
Err(_) => break,
|
|
||||||
}
|
|
||||||
input.expect_comma()?;
|
|
||||||
}
|
|
||||||
Ok(Self {
|
|
||||||
images: images.into_boxed_slice(),
|
|
||||||
keyword: CursorKind::parse(context, input)?,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
impl ToCss for Cursor {
|
|
||||||
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
|
||||||
where
|
|
||||||
W: Write,
|
|
||||||
{
|
|
||||||
for url in &*self.images {
|
|
||||||
url.to_css(dest)?;
|
|
||||||
dest.write_str(", ")?;
|
|
||||||
}
|
|
||||||
self.keyword.to_css(dest)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Parse for CursorKind {
|
|
||||||
fn parse<'i, 't>(
|
|
||||||
_context: &ParserContext,
|
|
||||||
input: &mut Parser<'i, 't>,
|
|
||||||
) -> Result<Self, ParseError<'i>> {
|
|
||||||
let location = input.current_source_location();
|
|
||||||
let ident = input.expect_ident()?;
|
|
||||||
CursorKind::from_css_keyword(&ident).map_err(|_| {
|
|
||||||
location.new_custom_error(SelectorParseErrorKind::UnexpectedIdent(ident.clone()))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
impl CursorImage {
|
|
||||||
fn parse_image<'i, 't>(
|
|
||||||
context: &ParserContext,
|
|
||||||
input: &mut Parser<'i, 't>,
|
|
||||||
) -> Result<Self, ParseError<'i>> {
|
|
||||||
Ok(Self {
|
|
||||||
url: SpecifiedImageUrl::parse(context, input)?,
|
|
||||||
// FIXME(emilio): Should use Number::parse to handle calc() correctly.
|
|
||||||
hotspot: match input.try(|input| input.expect_number()) {
|
|
||||||
Ok(number) => Some((number, input.expect_number()?)),
|
|
||||||
Err(_) => None,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
impl ToCss for CursorImage {
|
|
||||||
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
|
||||||
where
|
|
||||||
W: Write,
|
|
||||||
{
|
|
||||||
self.url.to_css(dest)?;
|
|
||||||
if let Some((x, y)) = self.hotspot {
|
|
||||||
dest.write_str(" ")?;
|
|
||||||
x.to_css(dest)?;
|
|
||||||
dest.write_str(" ")?;
|
|
||||||
y.to_css(dest)?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A computed value for the `caret-color` property.
|
/// A computed value for the `caret-color` property.
|
||||||
pub type CaretColor = GenericCaretColor<Color>;
|
pub type CaretColor = generics::CaretColor<Color>;
|
||||||
|
|
||||||
|
/// A computed value for the `cursor` property.
|
||||||
|
pub type Cursor = generics::Cursor<CursorImage>;
|
||||||
|
|
||||||
|
/// A computed value for item of `image cursors`.
|
||||||
|
pub type CursorImage = generics::CursorImage<ComputedImageUrl, Number>;
|
||||||
|
|
|
@ -4,6 +4,10 @@
|
||||||
|
|
||||||
//! Generic values for pointing properties.
|
//! Generic values for pointing properties.
|
||||||
|
|
||||||
|
use std::fmt::{self, Write};
|
||||||
|
use style_traits::{CssWriter, ToCss};
|
||||||
|
use style_traits::cursor::CursorKind;
|
||||||
|
|
||||||
/// A generic value for the `caret-color` property.
|
/// A generic value for the `caret-color` property.
|
||||||
#[derive(Animate, Clone, ComputeSquaredDistance, Copy, Debug, MallocSizeOf, PartialEq,
|
#[derive(Animate, Clone, ComputeSquaredDistance, Copy, Debug, MallocSizeOf, PartialEq,
|
||||||
ToAnimatedValue, ToAnimatedZero, ToComputedValue, ToCss)]
|
ToAnimatedValue, ToAnimatedZero, ToComputedValue, ToCss)]
|
||||||
|
@ -13,3 +17,63 @@ pub enum CaretColor<Color> {
|
||||||
/// The keyword `auto`.
|
/// The keyword `auto`.
|
||||||
Auto,
|
Auto,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A generic value for the `cursor` property.
|
||||||
|
///
|
||||||
|
/// https://drafts.csswg.org/css-ui/#cursor
|
||||||
|
#[derive(Clone, Debug, MallocSizeOf, PartialEq, ToComputedValue)]
|
||||||
|
pub struct Cursor<Image> {
|
||||||
|
/// The parsed images for the cursor.
|
||||||
|
pub images: Box<[Image]>,
|
||||||
|
/// The kind of the cursor [default | help | ...].
|
||||||
|
pub keyword: CursorKind,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Image> Cursor<Image> {
|
||||||
|
/// Set `cursor` to `auto`
|
||||||
|
#[inline]
|
||||||
|
pub fn auto() -> Self {
|
||||||
|
Self {
|
||||||
|
images: vec![].into_boxed_slice(),
|
||||||
|
keyword: CursorKind::Auto,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Image: ToCss> ToCss for Cursor<Image> {
|
||||||
|
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
||||||
|
where
|
||||||
|
W: Write,
|
||||||
|
{
|
||||||
|
for image in &*self.images {
|
||||||
|
image.to_css(dest)?;
|
||||||
|
dest.write_str(", ")?;
|
||||||
|
}
|
||||||
|
self.keyword.to_css(dest)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A generic value for item of `image cursors`.
|
||||||
|
#[derive(Clone, Debug, MallocSizeOf, PartialEq, ToComputedValue)]
|
||||||
|
pub struct CursorImage<ImageUrl, Number> {
|
||||||
|
/// The url to parse images from.
|
||||||
|
pub url: ImageUrl,
|
||||||
|
/// The <x> and <y> coordinates.
|
||||||
|
pub hotspot: Option<(Number, Number)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<ImageUrl: ToCss, Number: ToCss> ToCss for CursorImage<ImageUrl, Number> {
|
||||||
|
fn to_css<W>(&self, dest: &mut CssWriter<W>) -> fmt::Result
|
||||||
|
where
|
||||||
|
W: Write,
|
||||||
|
{
|
||||||
|
self.url.to_css(dest)?;
|
||||||
|
if let Some((ref x, ref y)) = self.hotspot {
|
||||||
|
dest.write_str(" ")?;
|
||||||
|
x.to_css(dest)?;
|
||||||
|
dest.write_str(" ")?;
|
||||||
|
y.to_css(dest)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -8,44 +8,15 @@
|
||||||
|
|
||||||
use cssparser::Parser;
|
use cssparser::Parser;
|
||||||
use parser::{Parse, ParserContext};
|
use parser::{Parse, ParserContext};
|
||||||
use style_traits::ParseError;
|
use style_traits::{ParseError, StyleParseErrorKind};
|
||||||
use style_traits::cursor::CursorKind;
|
use style_traits::cursor::CursorKind;
|
||||||
use values::generics::pointing::CaretColor as GenericCaretColor;
|
use values::generics::pointing as generics;
|
||||||
|
use values::specified::Number;
|
||||||
use values::specified::color::Color;
|
use values::specified::color::Color;
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
use values::specified::url::SpecifiedImageUrl;
|
use values::specified::url::SpecifiedImageUrl;
|
||||||
|
|
||||||
/// The specified value for the `cursor` property.
|
|
||||||
///
|
|
||||||
/// https://drafts.csswg.org/css-ui/#cursor
|
|
||||||
#[cfg(feature = "servo")]
|
|
||||||
#[derive(Clone, Copy, Debug, MallocSizeOf, PartialEq, ToComputedValue, ToCss)]
|
|
||||||
pub struct Cursor(pub CursorKind);
|
|
||||||
|
|
||||||
/// The specified value for the `cursor` property.
|
|
||||||
///
|
|
||||||
/// https://drafts.csswg.org/css-ui/#cursor
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
#[derive(Clone, Debug, MallocSizeOf, PartialEq, ToComputedValue)]
|
|
||||||
pub struct Cursor {
|
|
||||||
/// The parsed images for the cursor.
|
|
||||||
pub images: Box<[CursorImage]>,
|
|
||||||
/// The kind of the cursor [default | help | ...].
|
|
||||||
pub keyword: CursorKind,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// The specified value for the `image cursors`.
|
|
||||||
#[cfg(feature = "gecko")]
|
|
||||||
#[derive(Clone, Debug, MallocSizeOf, PartialEq, ToComputedValue)]
|
|
||||||
pub struct CursorImage {
|
|
||||||
/// The url to parse images from.
|
|
||||||
pub url: SpecifiedImageUrl,
|
|
||||||
/// The <x> and <y> coordinates.
|
|
||||||
pub hotspot: Option<(f32, f32)>,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A specified value for the `caret-color` property.
|
/// A specified value for the `caret-color` property.
|
||||||
pub type CaretColor = GenericCaretColor<Color>;
|
pub type CaretColor = generics::CaretColor<Color>;
|
||||||
|
|
||||||
impl Parse for CaretColor {
|
impl Parse for CaretColor {
|
||||||
fn parse<'i, 't>(
|
fn parse<'i, 't>(
|
||||||
|
@ -53,8 +24,63 @@ impl Parse for CaretColor {
|
||||||
input: &mut Parser<'i, 't>,
|
input: &mut Parser<'i, 't>,
|
||||||
) -> Result<Self, ParseError<'i>> {
|
) -> Result<Self, ParseError<'i>> {
|
||||||
if input.try(|i| i.expect_ident_matching("auto")).is_ok() {
|
if input.try(|i| i.expect_ident_matching("auto")).is_ok() {
|
||||||
return Ok(GenericCaretColor::Auto);
|
return Ok(generics::CaretColor::Auto);
|
||||||
}
|
}
|
||||||
Ok(GenericCaretColor::Color(Color::parse(context, input)?))
|
Ok(generics::CaretColor::Color(Color::parse(context, input)?))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A specified value for the `cursor` property.
|
||||||
|
pub type Cursor = generics::Cursor<CursorImage>;
|
||||||
|
|
||||||
|
/// A specified value for item of `image cursors`.
|
||||||
|
pub type CursorImage = generics::CursorImage<SpecifiedImageUrl, Number>;
|
||||||
|
|
||||||
|
impl Parse for Cursor {
|
||||||
|
/// cursor: [<url> [<number> <number>]?]# [auto | default | ...]
|
||||||
|
fn parse<'i, 't>(
|
||||||
|
context: &ParserContext,
|
||||||
|
input: &mut Parser<'i, 't>,
|
||||||
|
) -> Result<Self, ParseError<'i>> {
|
||||||
|
let mut images = vec![];
|
||||||
|
loop {
|
||||||
|
match input.try(|input| CursorImage::parse(context, input)) {
|
||||||
|
Ok(image) => images.push(image),
|
||||||
|
Err(_) => break,
|
||||||
|
}
|
||||||
|
input.expect_comma()?;
|
||||||
|
}
|
||||||
|
Ok(Self {
|
||||||
|
images: images.into_boxed_slice(),
|
||||||
|
keyword: CursorKind::parse(context, input)?,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parse for CursorKind {
|
||||||
|
fn parse<'i, 't>(
|
||||||
|
_context: &ParserContext,
|
||||||
|
input: &mut Parser<'i, 't>,
|
||||||
|
) -> Result<Self, ParseError<'i>> {
|
||||||
|
let location = input.current_source_location();
|
||||||
|
let ident = input.expect_ident()?;
|
||||||
|
CursorKind::from_css_keyword(&ident).map_err(|_| {
|
||||||
|
location.new_custom_error(StyleParseErrorKind::UnspecifiedError)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parse for CursorImage {
|
||||||
|
fn parse<'i, 't>(
|
||||||
|
context: &ParserContext,
|
||||||
|
input: &mut Parser<'i, 't>,
|
||||||
|
) -> Result<Self, ParseError<'i>> {
|
||||||
|
Ok(Self {
|
||||||
|
url: SpecifiedImageUrl::parse(context, input)?,
|
||||||
|
hotspot: match input.try(|input| Number::parse(context, input)) {
|
||||||
|
Ok(number) => Some((number, Number::parse(context, input)?)),
|
||||||
|
Err(_) => None,
|
||||||
|
},
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue