Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Merged by Bors] - reflect: add insert and remove methods to List #7063

Closed
wants to merge 7 commits into from
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 16 additions & 0 deletions crates/bevy_reflect/src/impls/smallvec.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,22 @@ impl<T: smallvec::Array + Send + Sync + 'static> List for SmallVec<T>
where
T::Item: FromReflect,
{
fn insert(&mut self, index: usize, value: Box<dyn Reflect>) {
let value = value.take::<T::Item>().unwrap_or_else(|value| {
<T as smallvec::Array>::Item::from_reflect(&*value).unwrap_or_else(|| {
panic!(
"Attempted to insert invalid value of type {}.",
value.type_name()
)
})
});
SmallVec::insert(self, index, value);
}

fn remove(&mut self, index: usize) -> Box<dyn Reflect> {
Box::new(self.remove(index)) as Box<dyn Reflect>
soqb marked this conversation as resolved.
Show resolved Hide resolved
}

fn push(&mut self, value: Box<dyn Reflect>) {
let value = value.take::<T::Item>().unwrap_or_else(|value| {
<T as smallvec::Array>::Item::from_reflect(&*value).unwrap_or_else(|| {
Expand Down
22 changes: 20 additions & 2 deletions crates/bevy_reflect/src/impls/std.rs
Original file line number Diff line number Diff line change
Expand Up @@ -179,7 +179,7 @@ impl_from_reflect_value!(NonZeroU8);
impl_from_reflect_value!(NonZeroI8);

macro_rules! impl_reflect_for_veclike {
($ty:ty, $push:expr, $pop:expr, $sub:ty) => {
($ty:ty, $insert:expr, $remove:expr, $push:expr, $pop:expr, $sub:ty) => {
impl<T: FromReflect> Array for $ty {
#[inline]
fn get(&self, index: usize) -> Option<&dyn Reflect> {
Expand Down Expand Up @@ -213,6 +213,22 @@ macro_rules! impl_reflect_for_veclike {
}

impl<T: FromReflect> List for $ty {
fn insert(&mut self, index: usize, value: Box<dyn Reflect>) {
let value = value.take::<T>().unwrap_or_else(|value| {
T::from_reflect(&*value).unwrap_or_else(|| {
panic!(
"Attempted to insert invalid value of type {}.",
value.type_name()
)
})
});
$insert(self, index, value);
}

fn remove(&mut self, index: usize) -> Box<dyn Reflect> {
Box::new($remove(self, index)) as Box<dyn Reflect>
soqb marked this conversation as resolved.
Show resolved Hide resolved
}

fn push(&mut self, value: Box<dyn Reflect>) {
let value = value.take::<T>().unwrap_or_else(|value| {
T::from_reflect(&*value).unwrap_or_else(|| {
Expand Down Expand Up @@ -328,9 +344,11 @@ macro_rules! impl_reflect_for_veclike {
};
}

impl_reflect_for_veclike!(Vec<T>, Vec::push, Vec::pop, [T]);
impl_reflect_for_veclike!(Vec<T>, Vec::insert, Vec::remove, Vec::push, Vec::pop, [T]);
impl_reflect_for_veclike!(
VecDeque<T>,
VecDeque::insert,
VecDeque::remove,
VecDeque::push_back,
VecDeque::pop_back,
VecDeque::<T>
Expand Down
39 changes: 35 additions & 4 deletions crates/bevy_reflect/src/list.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,18 +9,41 @@ use crate::{

/// An ordered, mutable list of [Reflect] items. This corresponds to types like [`std::vec::Vec`].
///
/// This is a sub-trait of [`Array`] as it implements a [`push`](List::push) function, allowing
/// it's internal size to grow.
/// This is a sub-trait of [`Array`] as it implements [insertion](List::insert) and [removal](List::remove),
/// allowing it's internal size to grow.
soqb marked this conversation as resolved.
Show resolved Hide resolved
///
/// This trait expects index 0 to contain the _front_ element.
/// The _back_ element must refer to the element with the largest index.
/// These two rules above should be upheld by manual implementors.
///
/// [`push`](List::push) and [`pop`](List::pop) have default implementations,
/// however it may be faster to implement them manually.
pub trait List: Reflect + Array {
/// Inserts an element at position `index` within the list.
soqb marked this conversation as resolved.
Show resolved Hide resolved
///
/// # Panics
/// Panics if `index > len`.
fn insert(&mut self, index: usize, element: Box<dyn Reflect>);

/// Removes and returns the element at position `index` within the list.
soqb marked this conversation as resolved.
Show resolved Hide resolved
///
/// # Panics
/// Panics if `index` is out of bounds.
fn remove(&mut self, index: usize) -> Box<dyn Reflect>;

/// Appends an element to the _back_ of the list.
fn push(&mut self, value: Box<dyn Reflect>);
fn push(&mut self, value: Box<dyn Reflect>) {
self.insert(self.len(), value);
}

/// Removes the _back_ element from the list and returns it, or [`None`] if it is empty.
fn pop(&mut self) -> Option<Box<dyn Reflect>>;
fn pop(&mut self) -> Option<Box<dyn Reflect>> {
if self.is_empty() {
None
} else {
Some(self.remove(self.len() - 1))
}
}

/// Clones the list, producing a [`DynamicList`].
fn clone_dynamic(&self) -> DynamicList {
Expand Down Expand Up @@ -174,6 +197,14 @@ impl Array for DynamicList {
}

impl List for DynamicList {
fn insert(&mut self, index: usize, element: Box<dyn Reflect>) {
self.values.insert(index, element);
}

fn remove(&mut self, index: usize) -> Box<dyn Reflect> {
self.values.remove(index)
}

fn push(&mut self, value: Box<dyn Reflect>) {
DynamicList::push_box(self, value);
}
Expand Down