medea-jason 0.9.1

Client library for Medea media server.
Documentation
//! Rust side representation of a Dart side [`List`].
//!
//! [`List`]: https://api.dart.dev/stable/dart-core/List-class.html

use dart_sys::Dart_Handle;
use derive_more::with_trait::From;
use medea_macro::dart_bridge;

use crate::{
    api::DartValue,
    platform::dart::utils::{NonNullDartValueArgExt as _, handle::DartHandle},
};

#[dart_bridge("flutter/lib/src/native/ffi/list.g.dart")]
mod list {
    use std::ptr;

    use dart_sys::Dart_Handle;

    use crate::{
        api::{DartValue, DartValueArg},
        platform::{Error, dart::utils::handle::DartHandle},
    };

    extern "C" {
        /// Returns an element by the provided `index` from the provided
        /// [`List`].
        ///
        /// [`List`]: https://api.dart.dev/stable/dart-core/List-class.html
        pub fn get(
            list: Dart_Handle,
            index: u32,
        ) -> Result<ptr::NonNull<DartValueArg<Option<DartHandle>>>, Error>;

        /// Returns [`length`] of the provided [`List`].
        ///
        /// [`length`]: https://api.dart.dev/stable/dart-core/List/length.html
        /// [`List`]: https://api.dart.dev/stable/dart-core/List-class.html
        pub fn length(list: Dart_Handle) -> Result<u32, Error>;

        /// Initializes a new empty [`List`].
        ///
        /// [`List`]: https://api.dart.dev/stable/dart-core/List-class.html
        pub fn init() -> Result<Dart_Handle, Error>;

        /// Adds the provided [`DartValue`] to the provided [`List`].
        ///
        /// [`List`]: https://api.dart.dev/stable/dart-core/List-class.html
        pub fn add(map: Dart_Handle, value: DartValue) -> Result<(), Error>;
    }
}

/// Rust side representation of a Dart side [`List`].
///
/// [`List`]: https://api.dart.dev/stable/dart-core/List-class.html
#[derive(Debug, From)]
pub struct DartList(DartHandle);

impl DartList {
    /// Creates a new empty [`DartList`].
    #[must_use]
    pub fn new() -> Self {
        let map = unsafe { list::init() }.unwrap();

        Self(unsafe { DartHandle::new(map) })
    }

    /// Adds the provided [`DartValue`] to the end of this [`DartList`],
    /// extending the length by one.
    pub fn add(&mut self, value: DartValue) {
        unsafe { list::add(self.0.get(), value) }.unwrap();
    }

    /// Returns an element by the provided `index` from this [`DartList`].
    #[expect(clippy::unwrap_in_result, reason = "unrelated")]
    #[must_use]
    pub fn get(&self, index: usize) -> Option<DartHandle> {
        #[expect( // intended
            clippy::cast_possible_truncation,
            reason = "overflow is unexpected"
        )]
        let item_ptr =
            unsafe { list::get(self.0.get(), index as u32) }.unwrap();
        unsafe { item_ptr.unbox() }.try_into().unwrap()
    }

    /// Returns [`length`] of this [`DartList`].
    ///
    /// [`length`]: https://api.dart.dev/stable/dart-core/List/length.html
    #[must_use]
    pub fn length(&self) -> usize {
        unsafe { list::length(self.0.get()).unwrap() as usize }
    }

    /// Returns the underlying [`Dart_Handle`] of this [`DartList`].
    #[must_use]
    pub fn handle(&self) -> Dart_Handle {
        self.0.get()
    }
}

impl<T: From<DartHandle>> From<DartList> for Vec<T> {
    fn from(list: DartList) -> Self {
        let len = list.length();
        let mut out = Self::with_capacity(len);
        for i in 0..len {
            if let Some(v) = list.get(i) {
                out.push(v.into());
            }
        }
        out
    }
}

impl Default for DartList {
    fn default() -> Self {
        Self::new()
    }
}