use std::sync::Arc;
use std::sync::atomic::{AtomicBool, Ordering};
use crossbeam_channel::{Receiver, SendError, Sender};
use log::warn;
use servo_config::prefs::{PrefValue, PreferencesObserver};
use servo_constellation_traits::EmbedderToConstellationMessage;
#[derive(Clone)]
pub(crate) struct ConstellationProxy {
sender: Sender<EmbedderToConstellationMessage>,
disconnected: Arc<AtomicBool>,
}
impl ConstellationProxy {
pub fn new() -> (Self, Receiver<EmbedderToConstellationMessage>) {
let (sender, receiver) = crossbeam_channel::unbounded();
(
Self {
sender,
disconnected: Arc::default(),
},
receiver,
)
}
pub fn disconnected(&self) -> bool {
self.disconnected.load(Ordering::SeqCst)
}
pub fn send(&self, msg: EmbedderToConstellationMessage) {
if self.try_send(msg).is_err() {
warn!("Lost connection to Constellation. Will report to embedder.")
}
}
#[expect(clippy::result_large_err)]
fn try_send(
&self,
msg: EmbedderToConstellationMessage,
) -> Result<(), SendError<EmbedderToConstellationMessage>> {
if self.disconnected() {
return Err(SendError(msg));
}
if let Err(error) = self.sender.send(msg) {
self.disconnected.store(true, Ordering::SeqCst);
return Err(error);
}
Ok(())
}
pub fn sender(&self) -> Sender<EmbedderToConstellationMessage> {
self.sender.clone()
}
}
impl PreferencesObserver for ConstellationProxy {
fn prefs_changed(&self, changes: &[(&'static str, PrefValue)]) {
self.send(EmbedderToConstellationMessage::PreferencesUpdated(
changes.to_owned(),
));
}
}