aboutsummaryrefslogtreecommitdiff
path: root/azalea-core/src/data_registry.rs
blob: 6e2c29ffcc6c535109cdb2d060e07af749f503cf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
use std::{io::Cursor, str::FromStr};

use azalea_registry::DataRegistry;
use simdnbt::owned::NbtCompound;

use crate::{registry_holder::RegistryHolder, resource_location::ResourceLocation};

pub trait ResolvableDataRegistry: DataRegistry {
    fn resolve_name(&self, registries: &RegistryHolder) -> Option<ResourceLocation> {
        self.resolve(registries).map(|(name, _)| name.clone())
    }
    fn resolve<'a>(
        &self,
        registries: &'a RegistryHolder,
    ) -> Option<(&'a ResourceLocation, &'a NbtCompound)> {
        let name_resourcelocation = ResourceLocation::from_str(Self::NAME).unwrap_or_else(|_| {
            panic!(
                "Name for registry should be a valid ResourceLocation: {}",
                Self::NAME
            )
        });
        let registry_values = registries.map.get(&name_resourcelocation)?;
        let resolved = registry_values.get_index(self.protocol_id() as usize)?;
        Some(resolved)
    }

    fn resolve_and_deserialize<T: simdnbt::Deserialize>(
        &self,
        registries: &RegistryHolder,
    ) -> Option<Result<(ResourceLocation, T), simdnbt::DeserializeError>> {
        let (name, value) = self.resolve(registries)?;

        let mut nbt_bytes = Vec::new();
        value.write(&mut nbt_bytes);
        let nbt_borrow_compound =
            simdnbt::borrow::read_compound(&mut Cursor::new(&nbt_bytes)).ok()?;
        let value = match T::from_compound((&nbt_borrow_compound).into()) {
            Ok(value) => value,
            Err(err) => {
                return Some(Err(err));
            }
        };

        Some(Ok((name.clone(), value)))
    }
}
impl<T: DataRegistry> ResolvableDataRegistry for T {}