mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-04 07:10:57 +00:00 
			
		
		
		
	url() has some limitations because of allowing unquoted URLs as its contents. For example, it can't use `var()`. To get around this, there's an alternative `src()` function which behaves the same as `url()` except that it is parsed as a regular function, which makes `var()` and friends work properly. There's no WPT test for this as far as I can tell, so I added our own.
		
			
				
	
	
		
			144 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						||
 * Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
 | 
						||
 *
 | 
						||
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						||
 */
 | 
						||
 | 
						||
#include <LibWeb/CSS/Serialize.h>
 | 
						||
#include <LibWeb/CSS/URL.h>
 | 
						||
#include <LibWeb/Fetch/Infrastructure/HTTP/Requests.h>
 | 
						||
 | 
						||
namespace Web::CSS {
 | 
						||
 | 
						||
URL::URL(String url, Type type, Vector<RequestURLModifier> request_url_modifiers)
 | 
						||
    : m_type(type)
 | 
						||
    , m_url(move(url))
 | 
						||
    , m_request_url_modifiers(move(request_url_modifiers))
 | 
						||
{
 | 
						||
}
 | 
						||
 | 
						||
// https://drafts.csswg.org/cssom-1/#serialize-a-url
 | 
						||
String URL::to_string() const
 | 
						||
{
 | 
						||
    // To serialize a URL means to create a string represented by "url(", followed by the serialization of the URL as a
 | 
						||
    // string, followed by ")".
 | 
						||
    // AD-HOC: Serialize as src() if it was declared as that.
 | 
						||
    StringBuilder builder;
 | 
						||
    switch (m_type) {
 | 
						||
    case Type::Url:
 | 
						||
        builder.append("url("sv);
 | 
						||
        break;
 | 
						||
    case Type::Src:
 | 
						||
        builder.append("src("sv);
 | 
						||
        break;
 | 
						||
    }
 | 
						||
    serialize_a_string(builder, m_url);
 | 
						||
 | 
						||
    // AD-HOC: Serialize the RequestURLModifiers
 | 
						||
    // Spec issue: https://github.com/w3c/csswg-drafts/issues/12057
 | 
						||
    for (auto const& modifier : m_request_url_modifiers)
 | 
						||
        builder.appendff(" {}", modifier.to_string());
 | 
						||
 | 
						||
    builder.append(')');
 | 
						||
 | 
						||
    return builder.to_string_without_validation();
 | 
						||
}
 | 
						||
 | 
						||
bool URL::operator==(URL const&) const = default;
 | 
						||
 | 
						||
RequestURLModifier RequestURLModifier::create_cross_origin(CrossOriginModifierValue value)
 | 
						||
{
 | 
						||
    return RequestURLModifier { Type::CrossOrigin, value };
 | 
						||
}
 | 
						||
 | 
						||
RequestURLModifier RequestURLModifier::create_integrity(FlyString value)
 | 
						||
{
 | 
						||
    return RequestURLModifier { Type::Integrity, move(value) };
 | 
						||
}
 | 
						||
 | 
						||
RequestURLModifier RequestURLModifier::create_referrer_policy(ReferrerPolicyModifierValue value)
 | 
						||
{
 | 
						||
    return RequestURLModifier { Type::ReferrerPolicy, value };
 | 
						||
}
 | 
						||
 | 
						||
RequestURLModifier::RequestURLModifier(Type type, Value value)
 | 
						||
    : m_type(type)
 | 
						||
    , m_value(move(value))
 | 
						||
{
 | 
						||
}
 | 
						||
 | 
						||
void RequestURLModifier::modify_request(GC::Ref<Fetch::Infrastructure::Request> request) const
 | 
						||
{
 | 
						||
    switch (m_type) {
 | 
						||
    case Type::CrossOrigin: {
 | 
						||
        // https://drafts.csswg.org/css-values-5/#typedef-request-url-modifier-crossorigin-modifier
 | 
						||
        // The URL request modifier steps for this modifier given request req are:
 | 
						||
 | 
						||
        // 1. Set req’s mode to "cors".
 | 
						||
        request->set_mode(Fetch::Infrastructure::Request::Mode::CORS);
 | 
						||
 | 
						||
        // 2. If the given value is use-credentials, set req’s credentials mode to "include".
 | 
						||
        if (m_value == CrossOriginModifierValue::UseCredentials) {
 | 
						||
            request->set_credentials_mode(Fetch::Infrastructure::Request::CredentialsMode::Include);
 | 
						||
        }
 | 
						||
        // 3. Otherwise, set req’s credentials mode to "same-origin".
 | 
						||
        else {
 | 
						||
            request->set_credentials_mode(Fetch::Infrastructure::Request::CredentialsMode::SameOrigin);
 | 
						||
        }
 | 
						||
        break;
 | 
						||
    }
 | 
						||
    case Type::Integrity: {
 | 
						||
        // https://drafts.csswg.org/css-values-5/#typedef-request-url-modifier-integrity-modifier
 | 
						||
 | 
						||
        // The URL request modifier steps for this modifier given request req are to set request’s integrity metadata to
 | 
						||
        // the given <string>.
 | 
						||
        request->set_integrity_metadata(m_value.get<FlyString>().to_string());
 | 
						||
        break;
 | 
						||
    }
 | 
						||
    case Type::ReferrerPolicy: {
 | 
						||
        // https://drafts.csswg.org/css-values-5/#typedef-request-url-modifier-referrerpolicy-modifier
 | 
						||
        // The URL request modifier steps for this modifier given request req are to set request’s referrer policy to the
 | 
						||
        // ReferrerPolicy that matches the given value.
 | 
						||
        auto referrer_policy = [](ReferrerPolicyModifierValue value) {
 | 
						||
            switch (value) {
 | 
						||
            case ReferrerPolicyModifierValue::NoReferrer:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::NoReferrer;
 | 
						||
            case ReferrerPolicyModifierValue::NoReferrerWhenDowngrade:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::NoReferrerWhenDowngrade;
 | 
						||
            case ReferrerPolicyModifierValue::SameOrigin:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::SameOrigin;
 | 
						||
            case ReferrerPolicyModifierValue::Origin:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::Origin;
 | 
						||
            case ReferrerPolicyModifierValue::StrictOrigin:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::StrictOrigin;
 | 
						||
            case ReferrerPolicyModifierValue::OriginWhenCrossOrigin:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::OriginWhenCrossOrigin;
 | 
						||
            case ReferrerPolicyModifierValue::StrictOriginWhenCrossOrigin:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::StrictOriginWhenCrossOrigin;
 | 
						||
            case ReferrerPolicyModifierValue::UnsafeUrl:
 | 
						||
                return ReferrerPolicy::ReferrerPolicy::UnsafeURL;
 | 
						||
            }
 | 
						||
            VERIFY_NOT_REACHED();
 | 
						||
        }(m_value.get<ReferrerPolicyModifierValue>());
 | 
						||
        request->set_referrer_policy(referrer_policy);
 | 
						||
        break;
 | 
						||
    }
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
String RequestURLModifier::to_string() const
 | 
						||
{
 | 
						||
    switch (m_type) {
 | 
						||
    case Type::CrossOrigin:
 | 
						||
        return MUST(String::formatted("crossorigin({})", CSS::to_string(m_value.get<CrossOriginModifierValue>())));
 | 
						||
    case Type::Integrity:
 | 
						||
        return MUST(String::formatted("integrity({})", serialize_a_string(m_value.get<FlyString>())));
 | 
						||
    case Type::ReferrerPolicy:
 | 
						||
        return MUST(String::formatted("referrerpolicy({})", CSS::to_string(m_value.get<ReferrerPolicyModifierValue>())));
 | 
						||
    }
 | 
						||
    VERIFY_NOT_REACHED();
 | 
						||
}
 | 
						||
 | 
						||
bool RequestURLModifier::operator==(RequestURLModifier const&) const = default;
 | 
						||
 | 
						||
}
 |