MessageEvent.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. /*
  2. * Copyright (c) 2022, Andreas Kling <andreas@ladybird.org>
  3. * Copyright (c) 2024, Jamie Mansfield <jmansfield@cadixdev.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibJS/Runtime/Array.h>
  8. #include <LibWeb/Bindings/Intrinsics.h>
  9. #include <LibWeb/Bindings/MessageEventPrototype.h>
  10. #include <LibWeb/HTML/MessageEvent.h>
  11. #include <LibWeb/HTML/MessagePort.h>
  12. namespace Web::HTML {
  13. JS_DEFINE_ALLOCATOR(MessageEvent);
  14. JS::NonnullGCPtr<MessageEvent> MessageEvent::create(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init)
  15. {
  16. return realm.heap().allocate<MessageEvent>(realm, realm, event_name, event_init);
  17. }
  18. WebIDL::ExceptionOr<JS::NonnullGCPtr<MessageEvent>> MessageEvent::construct_impl(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init)
  19. {
  20. return create(realm, event_name, event_init);
  21. }
  22. MessageEvent::MessageEvent(JS::Realm& realm, FlyString const& event_name, MessageEventInit const& event_init)
  23. : DOM::Event(realm, event_name, event_init)
  24. , m_data(event_init.data)
  25. , m_origin(event_init.origin)
  26. , m_last_event_id(event_init.last_event_id)
  27. , m_source(event_init.source)
  28. {
  29. m_ports.ensure_capacity(event_init.ports.size());
  30. for (auto const& port : event_init.ports) {
  31. VERIFY(port);
  32. m_ports.unchecked_append(static_cast<JS::Object&>(*port));
  33. }
  34. }
  35. MessageEvent::~MessageEvent() = default;
  36. void MessageEvent::initialize(JS::Realm& realm)
  37. {
  38. Base::initialize(realm);
  39. WEB_SET_PROTOTYPE_FOR_INTERFACE(MessageEvent);
  40. }
  41. void MessageEvent::visit_edges(Cell::Visitor& visitor)
  42. {
  43. Base::visit_edges(visitor);
  44. visitor.visit(m_data);
  45. visitor.visit(m_ports_array);
  46. visitor.visit(m_ports);
  47. }
  48. Variant<JS::Handle<WindowProxy>, JS::Handle<MessagePort>, Empty> MessageEvent::source() const
  49. {
  50. if (!m_source.has_value())
  51. return Empty {};
  52. return m_source.value().downcast<JS::Handle<WindowProxy>, JS::Handle<MessagePort>>();
  53. }
  54. JS::NonnullGCPtr<JS::Object> MessageEvent::ports() const
  55. {
  56. if (!m_ports_array) {
  57. JS::MarkedVector<JS::Value> port_vector(heap());
  58. for (auto const& port : m_ports)
  59. port_vector.append(port);
  60. m_ports_array = JS::Array::create_from(realm(), port_vector);
  61. MUST(m_ports_array->set_integrity_level(IntegrityLevel::Frozen));
  62. }
  63. return *m_ports_array;
  64. }
  65. // https://html.spec.whatwg.org/multipage/comms.html#dom-messageevent-initmessageevent
  66. void MessageEvent::init_message_event(String const& type, bool bubbles, bool cancelable, JS::Value data, String const& origin, String const& last_event_id, Optional<MessageEventSource> source, Vector<JS::Handle<MessagePort>> const& ports)
  67. {
  68. // The initMessageEvent(type, bubbles, cancelable, data, origin, lastEventId, source, ports) method must initialize the event in a
  69. // manner analogous to the similarly-named initEvent() method.
  70. // 1. If this’s dispatch flag is set, then return.
  71. if (dispatched())
  72. return;
  73. // 2. Initialize this with type, bubbles, and cancelable.
  74. initialize_event(type, bubbles, cancelable);
  75. // Implementation Defined: Initialise other values.
  76. m_data = data;
  77. m_origin = origin;
  78. m_last_event_id = last_event_id;
  79. m_source = source;
  80. m_ports.clear();
  81. m_ports.ensure_capacity(ports.size());
  82. for (auto const& port : ports) {
  83. VERIFY(port);
  84. m_ports.unchecked_append(static_cast<JS::Object&>(*port));
  85. }
  86. }
  87. }