/* * Copyright (c) 2019-2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include namespace Web::CSS { class CSSStyleSheet final : public StyleSheet { public: using WrapperType = Bindings::CSSStyleSheetWrapper; static NonnullRefPtr create(NonnullRefPtrVector rules) { return adopt_ref(*new CSSStyleSheet(move(rules))); } virtual ~CSSStyleSheet() override; void set_owner_css_rule(CSSRule* rule) { m_owner_css_rule = rule; } virtual String type() const override { return "text/css"; } CSSRuleList const& rules() const { return m_rules; } CSSRuleList& rules() { return m_rules; } void set_rules(NonnullRefPtr rules) { m_rules = move(rules); } CSSRuleList* css_rules() { return m_rules; } CSSRuleList const* css_rules() const { return m_rules; } DOM::ExceptionOr insert_rule(StringView rule, unsigned index); DOM::ExceptionOr remove_rule(unsigned index); DOM::ExceptionOr delete_rule(unsigned index); template void for_each_effective_style_rule(Callback 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); } } template bool for_first_not_loaded_import_rule(Callback 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; } private: explicit CSSStyleSheet(NonnullRefPtrVector); NonnullRefPtr m_rules; // FIXME: Use WeakPtr. CSSRule* m_owner_css_rule { nullptr }; }; } namespace Web::Bindings { CSSStyleSheetWrapper* wrap(JS::GlobalObject&, CSS::CSSStyleSheet&); }