use bevy_ecs::reflect::ReflectComponent; use bevy_reflect::Reflect; use bevy_utils::AHasher; use std::{ borrow::Cow, hash::{Hash, Hasher}, ops::Deref, }; /// Component used to identify an entity. Stores a hash for faster comparisons #[derive(Debug, Clone, Reflect)] #[reflect(Component)] pub struct Name { hash: u64, // TODO: Shouldn't be serialized name: Cow<'static, str>, } impl Default for Name { fn default() -> Self { Name::new("") } } impl Name { pub fn new(name: impl Into>) -> Self { let name = name.into(); let mut name = Name { name, hash: 0 }; name.update_hash(); name } #[inline(always)] pub fn set(&mut self, name: impl Into>) { *self = Name::new(name); } #[inline(always)] pub fn mutate(&mut self, f: F) { f(self.name.to_mut()); self.update_hash(); } #[inline(always)] pub fn as_str(&self) -> &str { &self.name } fn update_hash(&mut self) { let mut hasher = AHasher::default(); self.name.hash(&mut hasher); self.hash = hasher.finish(); } } impl From<&str> for Name { #[inline(always)] fn from(name: &str) -> Self { Name::new(name.to_owned()) } } impl Hash for Name { fn hash(&self, state: &mut H) { self.name.hash(state); } } impl PartialEq for Name { fn eq(&self, other: &Self) -> bool { if self.hash != other.hash { // Makes the common case of two strings not been equal very fast return false; } self.name.eq(&other.name) } } impl Eq for Name {} impl PartialOrd for Name { fn partial_cmp(&self, other: &Self) -> Option { self.name.partial_cmp(&other.name) } } impl Ord for Name { fn cmp(&self, other: &Self) -> std::cmp::Ordering { self.name.cmp(&other.name) } } impl Deref for Name { type Target = Cow<'static, str>; fn deref(&self) -> &Self::Target { &self.name } }