forked from parcel-bundler/lightningcss
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathurl.rs
More file actions
122 lines (104 loc) · 3.5 KB
/
url.rs
File metadata and controls
122 lines (104 loc) · 3.5 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
//! CSS url() values.
use crate::dependencies::{Dependency, Location, UrlDependency};
use crate::error::{ParserError, PrinterError};
use crate::printer::Printer;
use crate::traits::{Parse, ToCss};
use crate::values::string::CowArcStr;
use cssparser::*;
/// A CSS [url()](https://www.w3.org/TR/css-values-4/#urls) value and its source location.
#[derive(Debug, Clone, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct Url<'i> {
/// The url string.
#[cfg_attr(feature = "serde", serde(borrow))]
pub url: CowArcStr<'i>,
/// The location where the `url()` was seen in the CSS source file.
pub loc: Location,
}
impl<'i> Parse<'i> for Url<'i> {
fn parse<'t>(input: &mut Parser<'i, 't>) -> Result<Self, ParseError<'i, ParserError<'i>>> {
let loc = input.current_source_location();
let url = input.expect_url()?.into();
Ok(Url { url, loc: loc.into() })
}
}
impl<'i> ToCss for Url<'i> {
fn to_css<W>(&self, dest: &mut Printer<W>) -> Result<(), PrinterError>
where
W: std::fmt::Write,
{
let dep = if dest.dependencies.is_some() {
Some(UrlDependency::new(self, dest.filename()))
} else {
None
};
// If adding dependencies, always write url() with quotes so that the placeholder can
// be replaced without escaping more easily. Quotes may be removed later during minification.
if let Some(dep) = dep {
dest.write_str("url(")?;
serialize_string(&dep.placeholder, dest)?;
dest.write_char(')')?;
if let Some(dependencies) = &mut dest.dependencies {
dependencies.push(Dependency::Url(dep))
}
return Ok(());
}
use cssparser::ToCss;
if dest.minify {
let mut buf = String::new();
Token::UnquotedUrl(CowRcStr::from(self.url.as_ref())).to_css(&mut buf)?;
// If the unquoted url is longer than it would be quoted (e.g. `url("...")`)
// then serialize as a string and choose the shorter version.
if buf.len() > self.url.len() + 7 {
let mut buf2 = String::new();
serialize_string(&self.url, &mut buf2)?;
if buf2.len() + 5 < buf.len() {
dest.write_str("url(")?;
dest.write_str(&buf2)?;
return dest.write_char(')');
}
}
dest.write_str(&buf)?;
} else {
dest.write_str("url(")?;
serialize_string(&self.url, dest)?;
dest.write_char(')')?;
}
Ok(())
}
}
impl<'i> Url<'i> {
/// Returns whether the URL is absolute, and not relative.
pub fn is_absolute(&self) -> bool {
let url = self.url.as_ref();
// Quick checks. If the url starts with '.', it is relative.
if url.starts_with('.') {
return false;
}
// If the url starts with '/' it is absolute.
if url.starts_with('/') {
return true;
}
// If the url starts with '#' we have a fragment URL.
// These are resolved relative to the document rather than the CSS file.
// https://drafts.csswg.org/css-values-4/#local-urls
if url.starts_with('#') {
return true;
}
// Otherwise, we might have a scheme. These must start with an ascii alpha character.
// https://url.spec.whatwg.org/#scheme-start-state
if !url.starts_with(|c| matches!(c, 'a'..='z' | 'A'..='Z')) {
return false;
}
// https://url.spec.whatwg.org/#scheme-state
for b in url.as_bytes() {
let c = *b as char;
match c {
'a'..='z' | 'A'..='Z' | '0'..='9' | '+' | '-' | '.' => {}
':' => return true,
_ => break,
}
}
false
}
}