medea-jason 0.13.1

Client library for Medea media server.
Documentation
//! Weak reference to a [`MediaManager`].
//!
//! [`MediaManager`]: media::MediaManager

use derive_more::with_trait::From;
use js_sys::Promise;
use wasm_bindgen::prelude::*;
use wasm_bindgen_futures::future_to_promise;

use super::Error;
use crate::{
    api::{LocalMediaTrack, MediaDeviceDetails, MediaStreamSettings},
    media,
};

/// [`MediaManagerHandle`] is a weak reference to a [`MediaManager`].
///
/// [`MediaManager`] performs all the media acquisition requests
/// ([getUserMedia()][1]/[getDisplayMedia()][2]) and stores all the received
/// tracks for further re-usage.
///
/// [`MediaManager`] stores weak references to [`LocalMediaTrack`]s, so if there
/// are no strong references to some track, then this track is stopped and
/// removed from [`MediaManager`].
///
/// Like all the handles it contains a weak reference to the object that is
/// managed by Rust, so its methods will fail if a weak reference could not be
/// upgraded.
///
/// [`MediaManager`]: media::MediaManager
/// [1]: https://w3.org/TR/mediacapture-streams#dom-mediadevices-getusermedia
/// [2]: https://w3.org/TR/screen-capture/#dom-mediadevices-getdisplaymedia
#[wasm_bindgen]
#[derive(Debug, From)]
pub struct MediaManagerHandle(media::MediaManagerHandleImpl);

#[wasm_bindgen]
impl MediaManagerHandle {
    /// Returns a list of [`MediaDeviceDetails`] objects representing available
    /// media input and output devices, such as microphones, cameras, and so
    /// forth.
    ///
    /// # Errors
    ///
    /// With a [`StateError`] if the underlying object has been disposed, e.g.
    /// `free` was called on this [`MediaManagerHandle`], or on a [`Jason`] that
    /// implicitly owns native object behind this [`MediaManagerHandle`].
    ///
    /// With a [`EnumerateDevicesException`][0] if a request of platform media
    /// devices access failed.
    ///
    /// [`Jason`]: crate::api::Jason
    /// [`StateError`]: crate::api::err::StateError
    /// [0]: crate::api::err::EnumerateDevicesException
    pub fn enumerate_devices(&self) -> Promise {
        let this = self.0.clone();

        future_to_promise(async move {
            this.enumerate_devices()
                .await
                .map(|devices| {
                    devices
                        .into_iter()
                        .fold(js_sys::Array::new(), |devices_info, info| {
                            _ = devices_info.push(&JsValue::from(
                                MediaDeviceDetails::from(info),
                            ));
                            devices_info
                        })
                        .into()
                })
                .map_err(Error::from)
                .map_err(Into::into)
        })
    }

    /// Returns [`LocalMediaTrack`]s objects, built from the provided
    /// [`MediaStreamSettings`].
    ///
    /// # Errors
    ///
    /// With a [`StateError`] if the underlying object has been disposed, e.g.
    /// `free` was called on this [`MediaManagerHandle`], or on a [`Jason`] that
    /// implicitly owns native object behind this [`MediaManagerHandle`].
    ///
    /// With a [`LocalMediaInitException`] if a request of platform media
    /// devices access failed.
    ///
    /// [`Jason`]: crate::api::Jason
    /// [`LocalMediaInitException`]: crate::api::err::LocalMediaInitException
    /// [`StateError`]: crate::api::err::StateError
    pub fn init_local_tracks(&self, caps: &MediaStreamSettings) -> Promise {
        let this = self.0.clone();
        let caps = caps.clone();

        future_to_promise(async move {
            this.init_local_tracks(caps.into())
                .await
                .map(|tracks| {
                    tracks
                        .into_iter()
                        .map(|t| JsValue::from(LocalMediaTrack::from(t)))
                        .collect::<js_sys::Array>()
                        .into()
                })
                .map_err(Error::from)
                .map_err(Into::into)
        })
    }

    /// Subscribes onto the [`MediaManagerHandle`]'s `devicechange` event.
    ///
    /// # Errors
    ///
    /// With a [`StateError`] if the underlying object has been disposed, e.g.
    /// `free` was called on this [`MediaManagerHandle`], or on a [`Jason`] that
    /// implicitly owns native object behind this [`MediaManagerHandle`].
    ///
    /// [`Jason`]: crate::api::Jason
    /// [`StateError`]: crate::api::err::StateError
    pub fn on_device_change(
        &self,
        cb: js_sys::Function,
    ) -> Result<(), JsValue> {
        let this = self.0.clone();
        this.on_device_change(cb.into())
            .map_err(Error::from)
            .map_err(Into::into)
    }
}