/* * Copyright (c) 2022, Andreas Kling * Copyright (c) 2024, Jamie Mansfield * Copyright (c) 2026, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include namespace Web::HTML { GC_DEFINE_ALLOCATOR(MessageEvent); GC::Ref MessageEvent::create(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init) { return realm.create(realm, event_name, event_init); } WebIDL::ExceptionOr> MessageEvent::construct_impl(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init) { return create(realm, event_name, event_init); } MessageEvent::MessageEventSourceInternal MessageEvent::to_message_event_source_internal(Optional const& source) { if (!source.has_value()) return Empty {}; return source->visit([](auto const& root) -> MessageEventSourceInternal { return GC::Ref { *root }; }); } MessageEvent::MessageEvent(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init) : DOM::Event(realm, event_name, event_init) , m_data(event_init.data) , m_origin(event_init.origin) , m_last_event_id(event_init.last_event_id) , m_source(to_message_event_source_internal(event_init.source)) { m_ports.ensure_capacity(event_init.ports.size()); for (auto const& port : event_init.ports) { VERIFY(port); m_ports.unchecked_append(static_cast(*port)); } } MessageEvent::~MessageEvent() = default; void MessageEvent::initialize(JS::Realm& realm) { WEB_SET_PROTOTYPE_FOR_INTERFACE(MessageEvent); Base::initialize(realm); } void MessageEvent::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_data); visitor.visit(m_ports_array); visitor.visit(m_ports); m_source.visit( [](Empty) {}, [&](auto const& ref) { visitor.visit(ref); }); } MessageEvent::SourceResult MessageEvent::source() const { return m_source.visit( [](Empty) -> SourceResult { return Empty {}; }, [](auto const& ref) -> SourceResult { return GC::Root { *ref }; }); } GC::Ref MessageEvent::ports() const { if (!m_ports_array) { GC::RootVector port_vector(heap()); for (auto const& port : m_ports) port_vector.append(port); m_ports_array = JS::Array::create_from(realm(), port_vector); MUST(m_ports_array->set_integrity_level(IntegrityLevel::Frozen)); } return *m_ports_array; } // https://html.spec.whatwg.org/multipage/comms.html#dom-messageevent-initmessageevent void MessageEvent::init_message_event(String const& type, bool bubbles, bool cancelable, JS::Value data, String const& origin, String const& last_event_id, Optional source, Vector> const& ports) { // The initMessageEvent(type, bubbles, cancelable, data, origin, lastEventId, source, ports) method must initialize the event in a // manner analogous to the similarly-named initEvent() method. // 1. If this’s dispatch flag is set, then return. if (dispatched()) return; // 2. Initialize this with type, bubbles, and cancelable. initialize_event(type, bubbles, cancelable); // Implementation Defined: Initialise other values. m_data = data; m_origin = origin; m_last_event_id = last_event_id; m_source = to_message_event_source_internal(source); m_ports_array = nullptr; m_ports.clear(); m_ports.ensure_capacity(ports.size()); for (auto const& port : ports) { VERIFY(port); m_ports.unchecked_append(static_cast(*port)); } } // https://html.spec.whatwg.org/multipage/comms.html#the-messageevent-interface:extract-an-origin Optional MessageEvent::extract_an_origin() const { // Objects implementing the MessageEvent interface's extract an origin steps are to return this's relevant settings object's origin. return relevant_settings_object(*this).origin(); } }