2024-03-08 19:27:24 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2024, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
|
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <LibWeb/CSS/StyleComputer.h>
|
|
|
|
|
#include <LibWeb/DOM/AdoptedStyleSheets.h>
|
|
|
|
|
#include <LibWeb/DOM/Document.h>
|
|
|
|
|
|
|
|
|
|
namespace Web::DOM {
|
|
|
|
|
|
2025-01-10 20:00:43 +03:00
|
|
|
|
GC::Ref<WebIDL::ObservableArray> create_adopted_style_sheets_list(Node& document_or_shadow_root)
|
2024-03-08 19:27:24 +01:00
|
|
|
|
{
|
2025-01-10 20:00:43 +03:00
|
|
|
|
auto adopted_style_sheets = WebIDL::ObservableArray::create(document_or_shadow_root.realm());
|
|
|
|
|
adopted_style_sheets->set_on_set_an_indexed_value_callback([&document_or_shadow_root](JS::Value& value) -> WebIDL::ExceptionOr<void> {
|
|
|
|
|
auto& vm = document_or_shadow_root.vm();
|
2024-03-08 19:27:24 +01:00
|
|
|
|
if (!value.is_object())
|
|
|
|
|
return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, "CSSStyleSheet");
|
|
|
|
|
auto& object = value.as_object();
|
|
|
|
|
if (!is<CSS::CSSStyleSheet>(object))
|
|
|
|
|
return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, "CSSStyleSheet");
|
|
|
|
|
auto& style_sheet = static_cast<CSS::CSSStyleSheet&>(object);
|
|
|
|
|
|
|
|
|
|
// The set an indexed value algorithm for adoptedStyleSheets, given value and index, is the following:
|
|
|
|
|
// 1. If value’s constructed flag is not set, or its constructor document is not equal to this
|
|
|
|
|
// DocumentOrShadowRoot's node document, throw a "NotAllowedError" DOMException.
|
|
|
|
|
if (!style_sheet.constructed())
|
2025-08-07 19:31:52 -04:00
|
|
|
|
return WebIDL::NotAllowedError::create(document_or_shadow_root.realm(), "StyleSheet's constructed flag is not set."_utf16);
|
2025-01-10 20:00:43 +03:00
|
|
|
|
if (!style_sheet.constructed() || style_sheet.constructor_document().ptr() != &document_or_shadow_root.document())
|
2025-08-07 19:31:52 -04:00
|
|
|
|
return WebIDL::NotAllowedError::create(document_or_shadow_root.realm(), "Sharing a StyleSheet between documents is not allowed."_utf16);
|
2025-01-10 20:00:43 +03:00
|
|
|
|
|
|
|
|
|
style_sheet.add_owning_document_or_shadow_root(document_or_shadow_root);
|
|
|
|
|
document_or_shadow_root.document().style_computer().invalidate_rule_cache();
|
|
|
|
|
document_or_shadow_root.invalidate_style(DOM::StyleInvalidationReason::AdoptedStyleSheetsList);
|
2024-03-08 19:27:24 +01:00
|
|
|
|
return {};
|
|
|
|
|
});
|
2025-01-10 20:00:43 +03:00
|
|
|
|
adopted_style_sheets->set_on_delete_an_indexed_value_callback([&document_or_shadow_root](JS::Value value) -> WebIDL::ExceptionOr<void> {
|
2025-01-10 17:25:42 +03:00
|
|
|
|
VERIFY(value.is_object());
|
|
|
|
|
auto& object = value.as_object();
|
|
|
|
|
VERIFY(is<CSS::CSSStyleSheet>(object));
|
|
|
|
|
auto& style_sheet = static_cast<CSS::CSSStyleSheet&>(object);
|
|
|
|
|
|
2025-01-10 20:00:43 +03:00
|
|
|
|
style_sheet.remove_owning_document_or_shadow_root(document_or_shadow_root);
|
|
|
|
|
document_or_shadow_root.document().style_computer().invalidate_rule_cache();
|
|
|
|
|
document_or_shadow_root.invalidate_style(DOM::StyleInvalidationReason::AdoptedStyleSheetsList);
|
2024-03-08 19:27:24 +01:00
|
|
|
|
return {};
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return adopted_style_sheets;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|