valence/build/entity.rs

524 lines
17 KiB
Rust
Raw Permalink Normal View History

2022-07-28 00:10:35 +10:00
use std::collections::BTreeMap;
2022-04-27 12:48:35 +10:00
2022-07-28 00:10:35 +10:00
use heck::ToPascalCase;
use proc_macro2::{Ident, TokenStream};
2022-04-29 17:48:41 +10:00
use quote::quote;
use serde::Deserialize;
2022-07-28 00:10:35 +10:00
use crate::ident;
2022-04-27 12:48:35 +10:00
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Clone, Debug)]
struct Entity {
#[serde(rename = "type")]
typ: Option<String>,
translation_key: Option<String>,
fields: Vec<Field>,
parent: Option<String>,
2022-07-06 11:08:40 +10:00
}
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Clone, Debug)]
struct EntityData {
types: BTreeMap<String, i32>,
2022-04-27 12:48:35 +10:00
}
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Clone, Debug)]
2022-04-27 12:48:35 +10:00
struct Field {
2022-07-28 00:10:35 +10:00
name: String,
index: u8,
#[serde(flatten)]
default_value: Value,
bits: Vec<Bit>,
2022-04-27 12:48:35 +10:00
}
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Clone, Debug)]
#[serde(tag = "type", content = "default_value", rename_all = "snake_case")]
enum Value {
Byte(u8),
Integer(i32),
Float(f32),
String(String),
TextComponent(String),
OptionalTextComponent(Option<String>),
ItemStack(String),
Boolean(bool),
Rotation {
pitch: f32,
yaw: f32,
roll: f32,
},
BlockPos(BlockPos),
OptionalBlockPos(Option<BlockPos>),
Facing(String),
OptionalUuid(Option<String>),
OptionalBlockState(Option<String>),
NbtCompound(String),
Particle(String),
VillagerData {
#[serde(rename = "type")]
typ: String,
profession: String,
level: i32,
},
OptionalInt(Option<i32>),
EntityPose(String),
CatVariant(String),
FrogVariant(String),
OptionalGlobalPos(Option<()>), // TODO
PaintingVariant(String),
2022-07-06 11:08:40 +10:00
}
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Debug, Clone, Copy)]
struct BlockPos {
x: i32,
y: i32,
z: i32,
2022-07-06 11:08:40 +10:00
}
2022-07-28 00:10:35 +10:00
#[derive(Deserialize, Clone, Debug)]
struct Bit {
name: String,
index: u8,
2022-04-27 12:48:35 +10:00
}
2022-07-28 00:10:35 +10:00
impl Value {
pub fn type_id(&self) -> i32 {
match self {
2022-07-28 00:10:35 +10:00
Value::Byte(_) => 0,
Value::Integer(_) => 1,
Value::Float(_) => 2,
Value::String(_) => 3,
Value::TextComponent(_) => 4,
Value::OptionalTextComponent(_) => 5,
Value::ItemStack(_) => 6,
Value::Boolean(_) => 7,
Value::Rotation { .. } => 8,
Value::BlockPos(_) => 9,
Value::OptionalBlockPos(_) => 10,
Value::Facing(_) => 11,
Value::OptionalUuid(_) => 12,
Value::OptionalBlockState(_) => 13,
Value::NbtCompound(_) => 14,
Value::Particle(_) => 15,
Value::VillagerData { .. } => 16,
Value::OptionalInt(_) => 17,
Value::EntityPose(_) => 18,
Value::CatVariant(_) => 19,
Value::FrogVariant(_) => 20,
Value::OptionalGlobalPos(_) => 21,
Value::PaintingVariant(_) => 22,
}
}
2022-07-28 00:10:35 +10:00
pub fn field_type(&self) -> TokenStream {
match self {
2022-07-28 00:10:35 +10:00
Value::Byte(_) => quote!(u8),
Value::Integer(_) => quote!(i32),
Value::Float(_) => quote!(f32),
Value::String(_) => quote!(Box<str>),
Value::TextComponent(_) => quote!(Text),
Value::OptionalTextComponent(_) => quote!(Option<Text>),
Value::ItemStack(_) => quote!(()), // TODO
Value::Boolean(_) => quote!(bool),
Value::Rotation { .. } => quote!(EulerAngle),
Value::BlockPos(_) => quote!(BlockPos),
Value::OptionalBlockPos(_) => quote!(Option<BlockPos>),
Value::Facing(_) => quote!(Facing),
Value::OptionalUuid(_) => quote!(Option<Uuid>),
Value::OptionalBlockState(_) => quote!(BlockState),
2022-08-30 12:28:19 +10:00
Value::NbtCompound(_) => quote!(crate::nbt::Compound),
2022-07-28 00:10:35 +10:00
Value::Particle(_) => quote!(Particle),
Value::VillagerData { .. } => quote!(VillagerData),
Value::OptionalInt(_) => quote!(OptionalInt),
Value::EntityPose(_) => quote!(Pose),
Value::CatVariant(_) => quote!(CatKind),
Value::FrogVariant(_) => quote!(FrogKind),
Value::OptionalGlobalPos(_) => quote!(()), // TODO
Value::PaintingVariant(_) => quote!(PaintingKind),
}
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub fn getter_return_type(&self) -> TokenStream {
match self {
Value::String(_) => quote!(&str),
Value::TextComponent(_) => quote!(&Text),
Value::OptionalTextComponent(_) => quote!(Option<&Text>),
2022-08-30 12:28:19 +10:00
Value::NbtCompound(_) => quote!(&crate::nbt::Compound),
2022-07-28 00:10:35 +10:00
_ => self.field_type(),
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub fn getter_return_expr(&self, field_name: &Ident) -> TokenStream {
match self {
Value::String(_) | Value::TextComponent(_) | Value::NbtCompound(_) => {
quote!(&self.#field_name)
}
Value::OptionalTextComponent(_) => quote!(self.#field_name.as_ref()),
_ => quote!(self.#field_name),
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub fn default_expr(&self) -> TokenStream {
match self {
Value::Byte(b) => quote!(#b),
Value::Integer(i) => quote!(#i),
Value::Float(f) => quote!(#f),
Value::String(s) => quote!(#s.to_owned().into_boxed_str()),
Value::TextComponent(_) => quote!(Text::default()), // TODO
Value::OptionalTextComponent(t) => {
assert!(t.is_none());
quote!(None)
}
Value::ItemStack(_) => quote!(()), // TODO
Value::Boolean(b) => quote!(#b),
Value::Rotation { pitch, yaw, roll } => quote! {
EulerAngle {
pitch: #pitch,
yaw: #yaw,
roll: #roll,
}
},
Value::BlockPos(BlockPos { x, y, z }) => {
quote!(BlockPos { x: #x, y: #y, z: #z })
}
Value::OptionalBlockPos(_) => quote!(None), // TODO
Value::Facing(f) => {
let variant = ident(f.to_pascal_case());
quote!(Facing::#variant)
}
Value::OptionalUuid(_) => quote!(None), // TODO
Value::OptionalBlockState(_) => quote!(BlockState::default()), // TODO
2022-08-30 12:28:19 +10:00
Value::NbtCompound(_) => quote!(crate::nbt::Compound::default()), // TODO
2022-07-28 00:10:35 +10:00
Value::Particle(p) => {
let variant = ident(p.to_pascal_case());
quote!(Particle::#variant)
}
Value::VillagerData {
typ,
profession,
level,
} => {
let typ = ident(typ.to_pascal_case());
let profession = ident(profession.to_pascal_case());
quote!(VillagerData::new(VillagerKind::#typ, VillagerProfession::#profession, #level))
}
Value::OptionalInt(i) => {
assert!(i.is_none());
quote!(OptionalInt::default())
}
Value::EntityPose(p) => {
let variant = ident(p.to_pascal_case());
quote!(Pose::#variant)
}
Value::CatVariant(c) => {
let variant = ident(c.to_pascal_case());
quote!(CatKind::#variant)
}
Value::FrogVariant(f) => {
let variant = ident(f.to_pascal_case());
quote!(FrogKind::#variant)
}
Value::OptionalGlobalPos(_) => quote!(()),
Value::PaintingVariant(p) => {
let variant = ident(p.to_pascal_case());
quote!(PaintingKind::#variant)
2022-04-29 17:48:41 +10:00
}
}
}
2022-07-28 00:10:35 +10:00
pub fn encodable_expr(&self, self_lvalue: TokenStream) -> TokenStream {
match self {
Value::Integer(_) => quote!(VarInt(#self_lvalue)),
_ => self_lvalue,
}
}
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
type Entities = BTreeMap<String, Entity>;
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub fn build() -> anyhow::Result<TokenStream> {
let entities = serde_json::from_str::<Entities>(include_str!("../extracted/entities.json"))?
.into_iter()
.map(|(k, mut v)| {
let strip = |s: String| {
if let Some(stripped) = s.strip_suffix("Entity") {
if !stripped.is_empty() {
return stripped.to_owned();
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
}
s
};
v.parent = v.parent.map(strip);
(strip(k), v)
})
.collect::<Entities>();
let entity_types =
serde_json::from_str::<EntityData>(include_str!("../extracted/entity_data.json"))?.types;
let concrete_entities = entities
.clone()
.into_iter()
.filter(|(_, v)| v.typ.is_some())
.collect::<Entities>();
let entity_kind_variants = concrete_entities.iter().map(|(name, e)| {
let name = ident(name);
let id = entity_types[e.typ.as_ref().unwrap()] as isize;
quote! {
#name = #id,
}
});
2022-07-28 00:10:35 +10:00
let concrete_entity_names = concrete_entities
.iter()
.map(|(k, _)| ident(k))
.collect::<Vec<_>>();
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
let concrete_entity_structs = concrete_entities.iter().map(|(struct_name, _)| {
let fields = collect_all_fields(struct_name, &entities);
let struct_name = ident(struct_name);
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
let modified_flags_type =
ident("u".to_owned() + &fields.len().next_power_of_two().max(8).to_string());
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
let struct_fields = fields.iter().map(|&field| {
let name = ident(&field.name);
let typ = field.default_value.field_type();
quote! {
#name: #typ,
}
});
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
let field_initializers = fields.iter().map(|&field| {
let field_name = ident(&field.name);
let init = field.default_value.default_expr();
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
quote! {
#field_name: #init,
}
});
let getter_setters = fields.iter().map(|&field| {
let field_name = ident(&field.name);
let field_type = field.default_value.field_type();
let field_index = field.index;
if !field.bits.is_empty() {
field
.bits
.iter()
.map(|bit| {
let bit_name = ident(&bit.name);
let bit_index = bit.index;
let getter_name = ident(format!("get_{}", &bit.name));
let setter_name = ident(format!("set_{}", &bit.name));
quote! {
pub fn #getter_name(&self) -> bool {
self.#field_name >> #bit_index as #field_type & 1 == 1
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
pub fn #setter_name(&mut self, #bit_name: bool) {
if self.#getter_name() != #bit_name {
self.#field_name =
(self.#field_name & !(1 << #bit_index as #field_type))
| ((#bit_name as #field_type) << #bit_index);
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
self.__modified_flags |= 1 << #field_index
}
2022-04-29 17:48:41 +10:00
}
}
2022-07-28 00:10:35 +10:00
})
.collect::<TokenStream>()
} else {
let getter_name = ident(format!("get_{}", &field.name));
let setter_name = ident(format!("set_{}", &field.name));
let getter_return_type = field.default_value.getter_return_type();
let getter_return_expr = field.default_value.getter_return_expr(&field_name);
quote! {
pub fn #getter_name(&self) -> #getter_return_type {
#getter_return_expr
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub fn #setter_name(&mut self, #field_name: impl Into<#field_type>) {
let #field_name = #field_name.into();
if self.#field_name != #field_name {
self.__modified_flags |= 1 << #field_index as #modified_flags_type;
self.#field_name = #field_name;
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
}
2022-07-06 11:08:40 +10:00
}
}
2022-07-28 00:10:35 +10:00
});
let initial_tracked_data_stmts = fields.iter().map(|&field| {
let field_name = ident(&field.name);
let field_index = field.index;
let default_expr = field.default_value.default_expr();
let type_id = field.default_value.type_id();
2022-08-03 12:02:05 +10:00
let encodable = field.default_value.encodable_expr(quote!(self.#field_name));
2022-07-06 11:08:40 +10:00
2022-06-30 06:00:41 +10:00
quote! {
2022-07-28 00:10:35 +10:00
if self.#field_name != (#default_expr) {
data.push(#field_index);
2022-06-30 06:00:41 +10:00
VarInt(#type_id).encode(data).unwrap();
2022-08-03 12:02:05 +10:00
#encodable.encode(data).unwrap();
2022-06-30 06:00:41 +10:00
}
}
2022-07-28 00:10:35 +10:00
});
let updated_tracked_data_stmts = fields.iter().map(|&field| {
let field_name = ident(&field.name);
let field_index = field.index;
let type_id = field.default_value.type_id();
2022-08-03 12:02:05 +10:00
let encodable = field.default_value.encodable_expr(quote!(self.#field_name));
2022-06-30 06:00:41 +10:00
quote! {
2022-07-28 00:10:35 +10:00
if (self.__modified_flags >> #field_index as #modified_flags_type) & 1 == 1 {
data.push(#field_index);
2022-06-30 06:00:41 +10:00
VarInt(#type_id).encode(data).unwrap();
2022-08-03 12:02:05 +10:00
#encodable.encode(data).unwrap();
2022-06-30 06:00:41 +10:00
}
}
2022-07-28 00:10:35 +10:00
});
2022-06-30 06:00:41 +10:00
2022-04-29 17:48:41 +10:00
quote! {
2022-07-28 00:10:35 +10:00
pub struct #struct_name {
/// Contains a set bit for every modified field.
__modified_flags: #modified_flags_type,
2022-04-29 17:48:41 +10:00
#(#struct_fields)*
}
2022-07-28 00:10:35 +10:00
impl #struct_name {
2022-06-30 06:00:41 +10:00
pub(crate) fn new() -> Self {
2022-04-29 17:48:41 +10:00
Self {
2022-07-28 00:10:35 +10:00
__modified_flags: 0,
#(#field_initializers)*
2022-04-29 17:48:41 +10:00
}
}
2022-07-28 00:10:35 +10:00
pub(crate) fn initial_tracked_data(&self, data: &mut Vec<u8>) {
#(#initial_tracked_data_stmts)*
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
pub(crate) fn updated_tracked_data(&self, data: &mut Vec<u8>) {
if self.__modified_flags != 0 {
#(#updated_tracked_data_stmts)*
2022-06-30 06:00:41 +10:00
}
2022-07-06 11:08:40 +10:00
}
2022-07-28 00:10:35 +10:00
// TODO: remove this
#[allow(unused)]
2022-06-30 06:00:41 +10:00
pub(crate) fn clear_modifications(&mut self) {
2022-07-28 00:10:35 +10:00
self.__modified_flags = 0;
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
#(#getter_setters)*
2022-04-29 17:48:41 +10:00
}
}
});
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
let translation_key_arms = concrete_entities.iter().map(|(k, v)| {
let name = ident(k);
let key = v
.translation_key
.as_ref()
.expect("translation key should be present for concrete entity");
quote! {
Self::#name => #key,
}
});
Ok(quote! {
/// Contains a variant for each concrete entity type.
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
pub enum EntityKind {
2022-07-28 00:10:35 +10:00
#(#entity_kind_variants)*
2022-04-29 17:48:41 +10:00
}
2022-07-28 00:10:35 +10:00
impl EntityKind {
pub fn translation_key(self) -> &'static str {
match self {
#(#translation_key_arms)*
}
2022-04-29 17:48:41 +10:00
}
}
2022-08-03 12:02:05 +10:00
pub enum TrackedData {
2022-07-28 00:10:35 +10:00
#(#concrete_entity_names(#concrete_entity_names),)*
}
2022-08-03 12:02:05 +10:00
impl TrackedData {
pub(super) fn new(kind: EntityKind) -> Self {
match kind {
2022-07-28 00:10:35 +10:00
#(EntityKind::#concrete_entity_names => Self::#concrete_entity_names(#concrete_entity_names::new()),)*
}
}
2022-07-28 00:10:35 +10:00
pub fn kind(&self) -> EntityKind {
match self {
2022-07-28 00:10:35 +10:00
#(Self::#concrete_entity_names(_) => EntityKind::#concrete_entity_names,)*
}
}
2022-07-28 00:10:35 +10:00
pub(super) fn initial_tracked_data(&self) -> Option<Vec<u8>> {
let mut data = Vec::new();
match self {
2022-07-28 00:10:35 +10:00
#(Self::#concrete_entity_names(e) => e.initial_tracked_data(&mut data),)*
}
if data.is_empty() {
None
} else {
data.push(0xff);
Some(data)
}
}
2022-07-28 00:10:35 +10:00
pub(super) fn updated_tracked_data(&self) -> Option<Vec<u8>> {
let mut data = Vec::new();
match self {
2022-07-28 00:10:35 +10:00
#(Self::#concrete_entity_names(e) => e.updated_tracked_data(&mut data),)*
}
if data.is_empty() {
None
} else {
data.push(0xff);
Some(data)
}
}
pub(super) fn clear_modifications(&mut self) {
match self {
2022-07-28 00:10:35 +10:00
#(Self::#concrete_entity_names(e) => e.__modified_flags = 0,)*
}
}
}
2022-04-29 17:48:41 +10:00
2022-07-28 00:10:35 +10:00
#(#concrete_entity_structs)*
})
}
fn collect_all_fields<'a>(entity_name: &str, entities: &'a Entities) -> Vec<&'a Field> {
fn rec<'a>(entity_name: &str, entities: &'a Entities, fields: &mut Vec<&'a Field>) {
let e = &entities[entity_name];
fields.extend(&e.fields);
if let Some(parent) = &e.parent {
rec(parent, entities, fields);
}
}
let mut fields = Vec::new();
rec(entity_name, entities, &mut fields);
fields.sort_by_key(|f| f.index);
fields
2022-04-29 17:48:41 +10:00
}