1use super::shadow_parts::ShadowParts;
10use crate::color::{parsing::parse_color_keyword, AbsoluteColor};
11use crate::derives::*;
12use crate::properties::PropertyDeclarationBlock;
13use crate::shared_lock::Locked;
14use crate::str::str_join;
15use crate::str::{read_exponent, read_fraction, HTML_SPACE_CHARACTERS};
16use crate::str::{read_numbers, split_commas, split_html_space_chars};
17use crate::values::specified::color::Color;
18use crate::values::specified::Length;
19use crate::values::AtomString;
20use crate::{Atom, LocalName, Namespace, Prefix};
21use app_units::Au;
22use euclid::num::Zero;
23use num_traits::ToPrimitive;
24use selectors::attr::AttrSelectorOperation;
25use servo_arc::Arc;
26use std::str::FromStr;
27
28const UNSIGNED_LONG_MAX: u32 = 2147483647;
30
31#[derive(Clone, Copy, Debug, PartialEq)]
32#[cfg_attr(feature = "servo", derive(MallocSizeOf))]
33pub enum LengthOrPercentageOrAuto {
34 Auto,
35 Percentage(f32),
36 Length(Au),
37}
38
39#[derive(Clone, Debug)]
40#[cfg_attr(feature = "servo", derive(MallocSizeOf))]
41pub enum AttrValue {
42 String(String),
43 TokenList(String, Vec<Atom>),
44 UInt(String, u32),
45 Int(String, i32),
46 Double(String, f64),
47 Atom(Atom),
48 Length(String, Option<Length>),
49 Color(String, Option<AbsoluteColor>),
50 Dimension(String, LengthOrPercentageOrAuto),
51
52 ResolvedUrl(
57 String,
58 #[ignore_malloc_size_of = "Arc"] Option<Arc<url::Url>>,
59 ),
60
61 Declaration(
74 String,
75 #[ignore_malloc_size_of = "Arc"] Arc<Locked<PropertyDeclarationBlock>>,
76 ),
77
78 ShadowParts(String, ShadowParts),
80}
81
82fn do_parse_integer<T: Iterator<Item = char>>(input: T) -> Result<i64, ()> {
86 let mut input = input
87 .skip_while(|c| HTML_SPACE_CHARACTERS.iter().any(|s| s == c))
88 .peekable();
89
90 let sign = match input.peek() {
91 None => return Err(()),
92 Some(&'-') => {
93 input.next();
94 -1
95 },
96 Some(&'+') => {
97 input.next();
98 1
99 },
100 Some(_) => 1,
101 };
102
103 let (value, _) = read_numbers(input);
104
105 value.and_then(|value| value.checked_mul(sign)).ok_or(())
106}
107
108pub fn parse_integer<T: Iterator<Item = char>>(input: T) -> Result<i32, ()> {
111 do_parse_integer(input).and_then(|result| result.to_i32().ok_or(()))
112}
113
114pub fn parse_unsigned_integer<T: Iterator<Item = char>>(input: T) -> Result<u32, ()> {
117 do_parse_integer(input).and_then(|result| result.to_u32().ok_or(()))
118}
119
120pub fn parse_double(string: &str) -> Result<f64, ()> {
123 let trimmed = string.trim_matches(HTML_SPACE_CHARACTERS);
124 let mut input = trimmed.chars().peekable();
125
126 let (value, divisor, chars_skipped) = match input.peek() {
127 None => return Err(()),
128 Some(&'-') => {
129 input.next();
130 (-1f64, -1f64, 1)
131 },
132 Some(&'+') => {
133 input.next();
134 (1f64, 1f64, 1)
135 },
136 _ => (1f64, 1f64, 0),
137 };
138
139 let (value, value_digits) = if let Some(&'.') = input.peek() {
140 (0f64, 0)
141 } else {
142 let (read_val, read_digits) = read_numbers(input);
143 (
144 value * read_val.and_then(|result| result.to_f64()).unwrap_or(1f64),
145 read_digits,
146 )
147 };
148
149 let input = trimmed
150 .chars()
151 .skip(value_digits + chars_skipped)
152 .peekable();
153
154 let (mut value, fraction_digits) = read_fraction(input, divisor, value);
155
156 let input = trimmed
157 .chars()
158 .skip(value_digits + chars_skipped + fraction_digits)
159 .peekable();
160
161 if let Some(exp) = read_exponent(input) {
162 value *= 10f64.powi(exp)
163 };
164
165 Ok(value)
166}
167
168impl AttrValue {
169 pub fn from_serialized_tokenlist(tokens: String) -> AttrValue {
170 let atoms =
171 split_html_space_chars(&tokens)
172 .map(Atom::from)
173 .fold(vec![], |mut acc, atom| {
174 if !acc.contains(&atom) {
175 acc.push(atom)
176 }
177 acc
178 });
179 AttrValue::TokenList(tokens, atoms)
180 }
181
182 pub fn from_comma_separated_tokenlist(tokens: String) -> AttrValue {
183 let atoms = split_commas(&tokens)
184 .map(Atom::from)
185 .fold(vec![], |mut acc, atom| {
186 if !acc.contains(&atom) {
187 acc.push(atom)
188 }
189 acc
190 });
191 AttrValue::TokenList(tokens, atoms)
192 }
193
194 pub fn from_atomic_tokens(atoms: Vec<Atom>) -> AttrValue {
195 let tokens = String::from(str_join(&atoms, "\x20"));
197 AttrValue::TokenList(tokens, atoms)
198 }
199
200 pub fn from_u32(string: String, default: u32) -> AttrValue {
202 let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
203 let result = if result > UNSIGNED_LONG_MAX {
204 default
205 } else {
206 result
207 };
208 AttrValue::UInt(string, result)
209 }
210
211 pub fn from_i32(string: String, default: i32) -> AttrValue {
212 let result = parse_integer(string.chars()).unwrap_or(default);
213 AttrValue::Int(string, result)
214 }
215
216 pub fn from_double(string: String, default: f64) -> AttrValue {
218 let result = parse_double(&string).unwrap_or(default);
219
220 if result.is_normal() {
221 AttrValue::Double(string, result)
222 } else {
223 AttrValue::Double(string, default)
224 }
225 }
226
227 pub fn from_limited_i32(string: String, default: i32) -> AttrValue {
229 let result = parse_integer(string.chars()).unwrap_or(default);
230
231 if result < 0 {
232 AttrValue::Int(string, default)
233 } else {
234 AttrValue::Int(string, result)
235 }
236 }
237
238 pub fn from_limited_u32(string: String, default: u32) -> AttrValue {
240 let result = parse_unsigned_integer(string.chars()).unwrap_or(default);
241 let result = if result == 0 || result > UNSIGNED_LONG_MAX {
242 default
243 } else {
244 result
245 };
246 AttrValue::UInt(string, result)
247 }
248
249 pub fn from_atomic(string: String) -> AttrValue {
250 let value = Atom::from(string);
251 AttrValue::Atom(value)
252 }
253
254 pub fn from_resolved_url(base: &Arc<::url::Url>, url: String) -> AttrValue {
255 let joined = base.join(&url).ok().map(Arc::new);
256 AttrValue::ResolvedUrl(url, joined)
257 }
258
259 pub fn from_legacy_color(string: String) -> AttrValue {
260 let parsed = parse_legacy_color(&string).ok();
261 AttrValue::Color(string, parsed)
262 }
263
264 pub fn from_dimension(string: String) -> AttrValue {
265 let parsed = parse_length(&string);
266 AttrValue::Dimension(string, parsed)
267 }
268
269 pub fn from_nonzero_dimension(string: String) -> AttrValue {
270 let parsed = parse_nonzero_length(&string);
271 AttrValue::Dimension(string, parsed)
272 }
273
274 pub fn from_shadow_parts(string: String) -> AttrValue {
275 let shadow_parts = ShadowParts::parse(&string);
276 AttrValue::ShadowParts(string, shadow_parts)
277 }
278
279 pub fn as_tokens(&self) -> &[Atom] {
285 match *self {
286 AttrValue::TokenList(_, ref tokens) => tokens,
287 _ => panic!("Tokens not found"),
288 }
289 }
290
291 pub fn as_atom(&self) -> &Atom {
297 match *self {
298 AttrValue::Atom(ref value) => value,
299 _ => panic!("Atom not found"),
300 }
301 }
302
303 pub fn as_color(&self) -> Option<&AbsoluteColor> {
309 match *self {
310 AttrValue::Color(_, ref color) => color.as_ref(),
311 _ => panic!("Color not found"),
312 }
313 }
314
315 pub fn as_dimension(&self) -> &LengthOrPercentageOrAuto {
321 match *self {
322 AttrValue::Dimension(_, ref l) => l,
323 _ => panic!("Dimension not found"),
324 }
325 }
326
327 pub fn as_resolved_url(&self) -> Option<&Arc<::url::Url>> {
333 match *self {
334 AttrValue::ResolvedUrl(_, ref url) => url.as_ref(),
335 _ => panic!("Url not found"),
336 }
337 }
338
339 pub fn as_int(&self) -> i32 {
347 if let AttrValue::Int(_, value) = *self {
348 value
349 } else {
350 panic!("Int not found");
351 }
352 }
353
354 pub fn as_uint(&self) -> u32 {
362 if let AttrValue::UInt(_, value) = *self {
363 value
364 } else {
365 panic!("Uint not found");
366 }
367 }
368
369 pub fn as_uint_px_dimension(&self) -> LengthOrPercentageOrAuto {
379 if let AttrValue::UInt(_, value) = *self {
380 LengthOrPercentageOrAuto::Length(Au::from_px(value as i32))
381 } else {
382 panic!("Uint not found");
383 }
384 }
385
386 pub fn as_shadow_parts(&self) -> &ShadowParts {
395 if let AttrValue::ShadowParts(_, value) = &self {
396 value
397 } else {
398 panic!("Not a shadowpart attribute");
399 }
400 }
401
402 pub fn eval_selector(&self, selector: &AttrSelectorOperation<&AtomString>) -> bool {
403 selector.eval_str(self)
407 }
408}
409
410impl ::std::ops::Deref for AttrValue {
411 type Target = str;
412
413 fn deref(&self) -> &str {
414 match *self {
415 AttrValue::String(ref value)
416 | AttrValue::TokenList(ref value, _)
417 | AttrValue::UInt(ref value, _)
418 | AttrValue::Double(ref value, _)
419 | AttrValue::Length(ref value, _)
420 | AttrValue::Color(ref value, _)
421 | AttrValue::Int(ref value, _)
422 | AttrValue::ResolvedUrl(ref value, _)
423 | AttrValue::Declaration(ref value, _)
424 | AttrValue::ShadowParts(ref value, _)
425 | AttrValue::Dimension(ref value, _) => &value,
426 AttrValue::Atom(ref value) => &value,
427 }
428 }
429}
430
431impl PartialEq<Atom> for AttrValue {
432 fn eq(&self, other: &Atom) -> bool {
433 match *self {
434 AttrValue::Atom(ref value) => value == other,
435 _ => other == &**self,
436 }
437 }
438}
439
440pub fn parse_nonzero_length(value: &str) -> LengthOrPercentageOrAuto {
442 match parse_length(value) {
443 LengthOrPercentageOrAuto::Length(x) if x == Au::zero() => LengthOrPercentageOrAuto::Auto,
444 LengthOrPercentageOrAuto::Percentage(x) if x == 0. => LengthOrPercentageOrAuto::Auto,
445 x => x,
446 }
447}
448
449pub fn parse_legacy_color(mut input: &str) -> Result<AbsoluteColor, ()> {
453 if input.is_empty() {
455 return Err(());
456 }
457
458 input = input.trim_matches(HTML_SPACE_CHARACTERS);
460
461 if input.eq_ignore_ascii_case("transparent") {
463 return Err(());
464 }
465
466 if let Ok(Color::Absolute(ref absolute)) = parse_color_keyword(input) {
468 return Ok(absolute.color);
469 }
470
471 if input.len() == 4 {
473 if let (b'#', Ok(r), Ok(g), Ok(b)) = (
474 input.as_bytes()[0],
475 hex(input.as_bytes()[1] as char),
476 hex(input.as_bytes()[2] as char),
477 hex(input.as_bytes()[3] as char),
478 ) {
479 return Ok(AbsoluteColor::srgb_legacy(r * 17, g * 17, b * 17, 1.0));
480 }
481 }
482
483 let mut new_input = String::new();
485 for ch in input.chars() {
486 if ch as u32 > 0xffff {
487 new_input.push_str("00")
488 } else {
489 new_input.push(ch)
490 }
491 }
492 let mut input = &*new_input;
493
494 for (char_count, (index, _)) in input.char_indices().enumerate() {
496 if char_count == 128 {
497 input = &input[..index];
498 break;
499 }
500 }
501
502 if input.as_bytes()[0] == b'#' {
504 input = &input[1..]
505 }
506
507 let mut new_input = Vec::new();
509 for ch in input.chars() {
510 if hex(ch).is_ok() {
511 new_input.push(ch as u8)
512 } else {
513 new_input.push(b'0')
514 }
515 }
516 let mut input = new_input;
517
518 while input.is_empty() || (input.len() % 3) != 0 {
520 input.push(b'0')
521 }
522
523 let mut length = input.len() / 3;
525 let (mut red, mut green, mut blue) = (
526 &input[..length],
527 &input[length..length * 2],
528 &input[length * 2..],
529 );
530
531 if length > 8 {
533 red = &red[length - 8..];
534 green = &green[length - 8..];
535 blue = &blue[length - 8..];
536 length = 8
537 }
538
539 while length > 2 && red[0] == b'0' && green[0] == b'0' && blue[0] == b'0' {
541 red = &red[1..];
542 green = &green[1..];
543 blue = &blue[1..];
544 length -= 1
545 }
546
547 return Ok(AbsoluteColor::srgb_legacy(
549 hex_string(red).unwrap(),
550 hex_string(green).unwrap(),
551 hex_string(blue).unwrap(),
552 1.0,
553 ));
554
555 fn hex(ch: char) -> Result<u8, ()> {
556 match ch {
557 '0'..='9' => Ok((ch as u8) - b'0'),
558 'a'..='f' => Ok((ch as u8) - b'a' + 10),
559 'A'..='F' => Ok((ch as u8) - b'A' + 10),
560 _ => Err(()),
561 }
562 }
563
564 fn hex_string(string: &[u8]) -> Result<u8, ()> {
565 match string.len() {
566 0 => Err(()),
567 1 => hex(string[0] as char),
568 _ => {
569 let upper = hex(string[0] as char)?;
570 let lower = hex(string[1] as char)?;
571 Ok((upper << 4) | lower)
572 },
573 }
574 }
575}
576
577pub fn parse_length(mut value: &str) -> LengthOrPercentageOrAuto {
582 value = value.trim_start_matches(HTML_SPACE_CHARACTERS);
586
587 match value.chars().nth(0) {
589 Some('0'..='9') => {},
590 _ => return LengthOrPercentageOrAuto::Auto,
591 }
592
593 let mut end_index = value.len();
601 let (mut found_full_stop, mut found_percent) = (false, false);
602 for (i, ch) in value.chars().enumerate() {
603 match ch {
604 '0'..='9' => continue,
605 '%' => {
606 found_percent = true;
607 end_index = i;
608 break;
609 },
610 '.' if !found_full_stop => {
611 found_full_stop = true;
612 continue;
613 },
614 _ => {
615 end_index = i;
616 break;
617 },
618 }
619 }
620 value = &value[..end_index];
621
622 if found_percent {
623 let result: Result<f32, _> = FromStr::from_str(value);
624 match result {
625 Ok(number) => return LengthOrPercentageOrAuto::Percentage((number as f32) / 100.0),
626 Err(_) => return LengthOrPercentageOrAuto::Auto,
627 }
628 }
629
630 match FromStr::from_str(value) {
631 Ok(number) => LengthOrPercentageOrAuto::Length(Au::from_f64_px(number)),
632 Err(_) => LengthOrPercentageOrAuto::Auto,
633 }
634}
635
636#[derive(Clone, Debug)]
638#[cfg_attr(feature = "servo", derive(MallocSizeOf))]
639pub struct AttrIdentifier {
640 pub local_name: LocalName,
641 pub name: LocalName,
642 pub namespace: Namespace,
643 pub prefix: Option<Prefix>,
644}