/* * Copyright (c) 2019-2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include namespace Web::CSS { CSSStyleSheet::CSSStyleSheet(NonnullRefPtrVector rules) : m_rules(CSSRuleList::create(move(rules))) { } CSSStyleSheet::~CSSStyleSheet() { } // https://drafts.csswg.org/cssom/#dom-cssstylesheet-insertrule DOM::ExceptionOr CSSStyleSheet::insert_rule(StringView rule, unsigned index) { // FIXME: 1. If the origin-clean flag is unset, throw a SecurityError exception. // FIXME: 2. If the disallow modification flag is set, throw a NotAllowedError DOMException. // 3. Let parsed rule be the return value of invoking parse a rule with rule. auto parsed_rule = parse_css_rule(CSS::ParsingContext {}, rule); // 4. If parsed rule is a syntax error, return parsed rule. if (!parsed_rule) return DOM::SyntaxError::create("Unable to parse CSS rule."); // FIXME: 5. If parsed rule is an @import rule, and the constructed flag is set, throw a SyntaxError DOMException. // 6. Return the result of invoking insert a CSS rule rule in the CSS rules at index. return m_rules->insert_a_css_rule(parsed_rule.release_nonnull(), index); } // https://drafts.csswg.org/cssom/#dom-cssstylesheet-deleterule DOM::ExceptionOr CSSStyleSheet::delete_rule(unsigned index) { // FIXME: 1. If the origin-clean flag is unset, throw a SecurityError exception. // FIXME: 2. If the disallow modification flag is set, throw a NotAllowedError DOMException. // 3. Remove a CSS rule in the CSS rules at index. return m_rules->remove_a_css_rule(index); } // https://drafts.csswg.org/cssom/#dom-cssstylesheet-removerule DOM::ExceptionOr CSSStyleSheet::remove_rule(unsigned index) { // The removeRule(index) method must run the same steps as deleteRule(). return delete_rule(index); } void CSSStyleSheet::for_each_effective_style_rule(Function const& callback) const { for (auto& rule : *m_rules) if (rule.type() == CSSRule::Type::Style) { callback(verify_cast(rule)); } else if (rule.type() == CSSRule::Type::Import) { const auto& import_rule = verify_cast(rule); if (import_rule.has_import_result()) import_rule.loaded_style_sheet()->for_each_effective_style_rule(callback); } } bool CSSStyleSheet::for_first_not_loaded_import_rule(Function const& callback) { for (auto& rule : *m_rules) if (rule.type() == CSSRule::Type::Import) { auto& import_rule = verify_cast(rule); if (!import_rule.has_import_result()) { callback(import_rule); return true; } if (import_rule.loaded_style_sheet()->for_first_not_loaded_import_rule(callback)) { return true; } } return false; } }