/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::codegen::Bindings::ExtendableMessageEventBinding; use crate::dom::bindings::codegen::Bindings::ExtendableMessageEventBinding::ExtendableMessageEventMethods; use crate::dom::bindings::error::Fallible; use crate::dom::bindings::inheritance::Castable; use crate::dom::bindings::reflector::reflect_dom_object; use crate::dom::bindings::root::{Dom, DomRoot}; use crate::dom::bindings::str::DOMString; use crate::dom::bindings::trace::RootedTraceableBox; use crate::dom::bindings::utils::to_frozen_array; use crate::dom::event::Event; use crate::dom::eventtarget::EventTarget; use crate::dom::extendableevent::ExtendableEvent; use crate::dom::globalscope::GlobalScope; use crate::dom::messageport::MessagePort; use crate::dom::serviceworkerglobalscope::ServiceWorkerGlobalScope; use crate::script_runtime::JSContext; use dom_struct::dom_struct; use js::jsapi::Heap; use js::jsval::JSVal; use js::rust::HandleValue; use servo_atoms::Atom; #[dom_struct] #[allow(non_snake_case)] pub struct ExtendableMessageEvent { event: ExtendableEvent, #[ignore_malloc_size_of = "mozjs"] data: Heap, origin: DOMString, lastEventId: DOMString, ports: Vec>, #[ignore_malloc_size_of = "mozjs"] frozen_ports: DomRefCell>>, } #[allow(non_snake_case)] impl ExtendableMessageEvent { pub fn new_inherited( origin: DOMString, lastEventId: DOMString, ports: Vec>, ) -> ExtendableMessageEvent { ExtendableMessageEvent { event: ExtendableEvent::new_inherited(), data: Heap::default(), origin: origin, lastEventId: lastEventId, ports: ports .into_iter() .map(|port| Dom::from_ref(&*port)) .collect(), frozen_ports: DomRefCell::new(None), } } pub fn new( global: &GlobalScope, type_: Atom, bubbles: bool, cancelable: bool, data: HandleValue, origin: DOMString, lastEventId: DOMString, ports: Vec>, ) -> DomRoot { let ev = Box::new(ExtendableMessageEvent::new_inherited( origin, lastEventId, ports, )); let ev = reflect_dom_object(ev, global); { let event = ev.upcast::(); event.init_event(type_, bubbles, cancelable); } ev.data.set(data.get()); ev } pub fn Constructor( worker: &ServiceWorkerGlobalScope, type_: DOMString, init: RootedTraceableBox, ) -> Fallible> { let global = worker.upcast::(); let ev = ExtendableMessageEvent::new( global, Atom::from(type_), init.parent.parent.bubbles, init.parent.parent.cancelable, init.data.handle(), init.origin.clone().unwrap(), init.lastEventId.clone().unwrap(), vec![], ); Ok(ev) } } #[allow(non_snake_case)] impl ExtendableMessageEvent { pub fn dispatch_jsval( target: &EventTarget, scope: &GlobalScope, message: HandleValue, ports: Vec>, ) { let Extendablemessageevent = ExtendableMessageEvent::new( scope, atom!("message"), false, false, message, DOMString::new(), DOMString::new(), ports, ); Extendablemessageevent.upcast::().fire(target); } } impl ExtendableMessageEventMethods for ExtendableMessageEvent { // https://w3c.github.io/ServiceWorker/#extendablemessage-event-data-attribute fn Data(&self, _cx: JSContext) -> JSVal { self.data.get() } // https://w3c.github.io/ServiceWorker/#extendablemessage-event-origin-attribute fn Origin(&self) -> DOMString { self.origin.clone() } // https://w3c.github.io/ServiceWorker/#extendablemessage-event-lasteventid-attribute fn LastEventId(&self) -> DOMString { self.lastEventId.clone() } // https://dom.spec.whatwg.org/#dom-event-istrusted fn IsTrusted(&self) -> bool { self.event.IsTrusted() } /// https://w3c.github.io/ServiceWorker/#extendablemessage-event-ports fn Ports(&self, cx: JSContext) -> JSVal { if let Some(ports) = &*self.frozen_ports.borrow() { return ports.get(); } let ports: Vec> = self .ports .iter() .map(|port| DomRoot::from_ref(&**port)) .collect(); let frozen_ports = to_frozen_array(ports.as_slice(), cx); // Safety: need to create the Heap value in its final memory location before setting it. *self.frozen_ports.borrow_mut() = Some(Heap::default()); self.frozen_ports .borrow() .as_ref() .unwrap() .set(frozen_ports); frozen_ports } }