/* 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 http://mozilla.org/MPL/2.0/. */ use std::cell::Cell; use constellation_traits::BroadcastMsg; use dom_struct::dom_struct; use js::rust::{HandleObject, HandleValue}; use uuid::Uuid; use crate::dom::bindings::codegen::Bindings::BroadcastChannelBinding::BroadcastChannelMethods; use crate::dom::bindings::error::{Error, ErrorResult}; use crate::dom::bindings::reflector::{DomGlobal, reflect_dom_object_with_proto}; use crate::dom::bindings::root::DomRoot; use crate::dom::bindings::str::DOMString; use crate::dom::bindings::structuredclone; use crate::dom::eventtarget::EventTarget; use crate::dom::globalscope::GlobalScope; use crate::script_runtime::{CanGc, JSContext as SafeJSContext}; #[dom_struct] pub(crate) struct BroadcastChannel { eventtarget: EventTarget, name: DOMString, closed: Cell, #[no_trace] id: Uuid, } impl BroadcastChannel { fn new( global: &GlobalScope, proto: Option, name: DOMString, can_gc: CanGc, ) -> DomRoot { let channel = reflect_dom_object_with_proto( Box::new(BroadcastChannel::new_inherited(name)), global, proto, can_gc, ); global.track_broadcast_channel(&channel); channel } pub(crate) fn new_inherited(name: DOMString) -> BroadcastChannel { BroadcastChannel { eventtarget: EventTarget::new_inherited(), name, closed: Default::default(), id: Uuid::new_v4(), } } /// The unique Id of this channel. /// Used for filtering out the sender from the local broadcast. pub(crate) fn id(&self) -> &Uuid { &self.id } /// Is this channel closed? pub(crate) fn closed(&self) -> bool { self.closed.get() } } impl BroadcastChannelMethods for BroadcastChannel { /// fn Constructor( global: &GlobalScope, proto: Option, can_gc: CanGc, name: DOMString, ) -> DomRoot { BroadcastChannel::new(global, proto, name, can_gc) } /// fn PostMessage(&self, cx: SafeJSContext, message: HandleValue) -> ErrorResult { // Step 3, if closed. if self.closed.get() { return Err(Error::InvalidState); } // Step 6, StructuredSerialize(message). let data = structuredclone::write(cx, message, None)?; let global = self.global(); let msg = BroadcastMsg { origin: global.origin().immutable().clone(), channel_name: self.Name().to_string(), data, }; global.schedule_broadcast(msg, &self.id); Ok(()) } /// fn Name(&self) -> DOMString { self.name.clone() } /// fn Close(&self) { self.closed.set(true); } // event_handler!(messageerror, GetOnmessageerror, SetOnmessageerror); // event_handler!(message, GetOnmessage, SetOnmessage); }