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

feat: Inventories #151

Open
wants to merge 43 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
4d5527a
Simple Inventory Implementation
Outspending Dec 29, 2024
06899a8
better slot system + sized inventories
Outspending Dec 29, 2024
5cfaa0a
Simple Viewer System
Outspending Dec 30, 2024
7dbdab0
QOL Changes
Outspending Dec 30, 2024
e542d66
tests
Outspending Dec 30, 2024
0acaa1b
Added all related container packets
Outspending Dec 30, 2024
bd973c4
container content
Outspending Dec 30, 2024
d47862b
InventoryData + viewer support
Outspending Dec 30, 2024
802c8b8
updated viewers
Outspending Dec 30, 2024
b9831f9
clippy fix
Outspending Dec 30, 2024
09bbcc4
clippy + fmt check fixed
Outspending Dec 30, 2024
cb9ee29
Merge branch 'feat/inventories' of https://github.com/Outspending/fer…
Outspending Dec 30, 2024
767f8fa
Update login_process.rs
Outspending Dec 30, 2024
26e4794
inventory events
Outspending Dec 30, 2024
cd492a7
Simple ECS Implementation
Outspending Dec 30, 2024
a765aac
Merge branch 'master' into feat/inventories
Outspending Dec 30, 2024
7f61449
Fixed Inventories + More Packets
Outspending Dec 30, 2024
1d127c4
ECS Compatibility
Outspending Dec 30, 2024
cd09d0f
better error handling
Outspending Dec 30, 2024
816391b
Inventory Clicking
Outspending Dec 30, 2024
63ad32e
better inventory packets
Outspending Dec 31, 2024
91ba5f3
Merge remote-tracking branch 'upstream/master' into feat/inventories
Outspending Dec 31, 2024
f2acd8d
content packet
Outspending Dec 31, 2024
0eac9fe
Merge pull request #145 from Outspending/feat/inventories
Sweattypalms Dec 31, 2024
7d404b2
Fixed Container Content Packet
Outspending Dec 31, 2024
dd1b152
inventory syncing
Outspending Dec 31, 2024
1e0dbab
inventory sync check
Outspending Dec 31, 2024
5619a63
Player / Creative Inventory Support
Outspending Jan 4, 2025
e39b090
Inventory Clicking + Updating
Outspending Jan 4, 2025
7628ada
InventoryBuilder + type fixes
Outspending Jan 4, 2025
6dd2513
clippy fix + fmt
Outspending Jan 4, 2025
7a9b518
Inventory Types + Macros
Outspending Jan 5, 2025
8d90ee5
Merge remote-tracking branch 'upstream/master' into feature/inventories
Outspending Jan 5, 2025
2ff5a83
fixed merge conflicts
Outspending Jan 5, 2025
4a59780
Inventory Types
Outspending Jan 5, 2025
14788fb
format fix
Outspending Jan 5, 2025
3871f59
Update login_process.rs
Outspending Jan 5, 2025
23d624e
Fixed merge issues
Outspending Jan 5, 2025
70f9cde
Stuff
Outspending Jan 5, 2025
6c3a8ba
All Inventory Types + Better Macro Support
Outspending Jan 5, 2025
c44ee95
Update mod.rs
Outspending Jan 5, 2025
20ceb40
clippy fix
Outspending Jan 5, 2025
2db4348
Components (ish) [Wont Compile]
Outspending Jan 5, 2025
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
2 changes: 2 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ members = [
"src/lib/derive_macros",
"src/lib/ecs",
"src/lib/events",
"src/lib/inventory",
"src/lib/net",
"src/lib/net/crates/codec",
"src/lib/net/crates/encryption",
Expand Down Expand Up @@ -94,6 +95,7 @@ ferrumc-logging = { path = "src/lib/utils/logging" }
ferrumc-macros = { path = "src/lib/derive_macros" }
ferrumc-nbt = { path = "src/lib/adapters/nbt" }
ferrumc-net = { path = "src/lib/net" }
ferrumc-inventory = { path = "src/lib/inventory" }
ferrumc-net-codec = { path = "src/lib/net/crates/codec" }
ferrumc-net-encryption = { path = "src/lib/net/crates/encryption" }
ferrumc-plugins = { path = "src/lib/plugins" }
Expand Down
1 change: 1 addition & 0 deletions src/bin/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ ferrumc-plugins = { workspace = true }
ferrumc-storage = { workspace = true }
ferrumc-utils = { workspace = true }
ferrumc-config = { workspace = true }
ferrumc-inventory = { workspace = true }
ferrumc-profiling = { workspace = true }
ferrumc-logging = { workspace = true }
ferrumc-world = { workspace = true }
Expand Down
111 changes: 111 additions & 0 deletions src/bin/src/packet_handlers/containers.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
use ferrumc_inventory::inventory::{Inventory, InventorySyncType};
use ferrumc_inventory::slot::Slot;
use ferrumc_inventory::types::player::PlayerInventory;
use ferrumc_macros::event_handler;
use ferrumc_net::errors::NetError;
use ferrumc_net::packets::incoming::click_container::InventoryClickEvent;
use ferrumc_net::packets::incoming::close_container::InventoryCloseEvent;
use ferrumc_net::packets::incoming::set_creative_mode_slot::SetCreativeModeSlotEvent;
use ferrumc_net::packets::incoming::set_held_item::ChangeSlotEvent;
use ferrumc_net::packets::outgoing::set_equipment::{Equipment, EquipmentSlot, SetEquipmentPacket};
use ferrumc_net::utils::broadcast::{broadcast, BroadcastOptions};
use ferrumc_state::GlobalState;

#[event_handler]
async fn container_close(
container_close_event: InventoryCloseEvent,
state: GlobalState,
) -> Result<InventoryCloseEvent, NetError> {
let conn_id = container_close_event.conn_id;

let window_id = container_close_event.window_id;
if window_id != 0 {
state.universe.remove_component::<Inventory>(conn_id)?;
}

Ok(container_close_event)
}

#[event_handler]
async fn handle_container_click(
inventory_click_event: InventoryClickEvent,
state: GlobalState,
) -> Result<InventoryClickEvent, NetError> {
if inventory_click_event.is_canceled {
return Err(NetError::Other(String::default()));
}

let conn_id = inventory_click_event.conn_id;
let packet = &inventory_click_event.packet;
let mut inventory = state.universe.get_mut::<Inventory>(conn_id)?;

if inventory.is_synced {
match packet.changed_slots.data.as_slice() {
[changed_slot] => {
let slot_num = changed_slot.slot_number;

inventory
.set_slot(slot_num, Slot::from_network_slot(changed_slot.slot))
.sync_inventory(conn_id, &InventorySyncType::Single(slot_num), state)
.await
.map_err(|err| NetError::Other(err.to_string()))?;
}
changed_slots => {
changed_slots.iter().for_each(|changed_slot| {
inventory.set_slot(
changed_slot.slot_number,
Slot::from_network_slot(changed_slot.slot),
);
});

inventory
.sync_inventory(conn_id, &InventorySyncType::All, state)
.await
.map_err(|err| NetError::Other(err.to_string()))?;
}
}
}

Ok(inventory_click_event)
}

#[event_handler]
async fn set_creative_mode_slot(
creative_mode_slot: SetCreativeModeSlotEvent,
state: GlobalState,
) -> Result<SetCreativeModeSlotEvent, NetError> {
let conn_id = creative_mode_slot.conn_id;

let mut inventory = state.universe.get_mut::<PlayerInventory>(conn_id)?;
inventory.set_slot(
creative_mode_slot.slot,
Slot::from_network_slot(creative_mode_slot.clicked_item),
);

Ok(creative_mode_slot)
}

#[event_handler]
async fn handle_carried_item(
change_slot_event: ChangeSlotEvent,
state: GlobalState,
) -> Result<ChangeSlotEvent, NetError> {
let conn_id = change_slot_event.conn_id;

let mut inventory = state.universe.get_mut::<PlayerInventory>(conn_id)?;
inventory.set_carried_item(change_slot_event.slot);

let equipment = Equipment::new(
EquipmentSlot::MainHand,
Slot::with_item(change_slot_event.slot as i32).to_network_slot(),
);

broadcast(
&SetEquipmentPacket::new(conn_id, vec![equipment]),
&state,
BroadcastOptions::default().except([conn_id]),
)
.await?;

Ok(change_slot_event)
}
6 changes: 4 additions & 2 deletions src/bin/src/packet_handlers/login_process.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ use ferrumc_core::transform::position::Position;
use ferrumc_core::transform::rotation::Rotation;
use ferrumc_ecs::components::storage::ComponentRefMut;
use ferrumc_ecs::entities::Entity;
use ferrumc_inventory::types::player::PlayerInventory;
use ferrumc_macros::event_handler;
use ferrumc_net::connection::{ConnectionState, StreamWriter};
use ferrumc_net::errors::NetError;
Expand Down Expand Up @@ -53,7 +54,7 @@ async fn handle_login_start(
login_start_event.conn_id,
PlayerIdentity::new(username.to_string(), uuid),
)?
/*.add_component::<ChunkReceiver>(login_start_event.conn_id, ChunkReceiver::default())?*/;
/*.add_component::<ChunkReceiver>(login_start_event.conn_id, ChunkReceiver::default())?*/;

//Send a Login Success Response to further the login sequence
let mut writer = state
Expand Down Expand Up @@ -165,7 +166,8 @@ async fn handle_ack_finish_configuration(
.add_component::<Position>(entity_id, Position::default())?
.add_component::<Rotation>(entity_id, Rotation::default())?
.add_component::<OnGround>(entity_id, OnGround::default())?
.add_component::<ChunkReceiver>(entity_id, ChunkReceiver::default())?;
.add_component::<ChunkReceiver>(entity_id, ChunkReceiver::default())?
.add_component::<PlayerInventory>(entity_id, PlayerInventory::default())?;

let mut writer = state.universe.get_mut::<StreamWriter>(entity_id)?;

Expand Down
1 change: 1 addition & 0 deletions src/bin/src/packet_handlers/mod.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
mod animations;
mod containers;
mod handshake;
mod login_process;
mod player;
Expand Down
3 changes: 2 additions & 1 deletion src/lib/derive_macros/src/events/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,8 @@ pub(crate) fn derive(input: TokenStream) -> TokenStream {
quote! {crate}
}
FoundCrate::Name(name) => {
quote! {::#name}
let name = syn::Ident::new(&name, proc_macro2::Span::call_site());
quote! {#name}
}
};

Expand Down
185 changes: 185 additions & 0 deletions src/lib/derive_macros/src/inventory/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
use proc_macro::TokenStream;
use proc_macro_crate::{crate_name, FoundCrate};
use quote::quote;
use syn::{Data, Visibility};

pub fn create(input: TokenStream) -> TokenStream {
let input = syn::parse_macro_input!(input as syn::DeriveInput);
let name = &input.ident;

if let Data::Struct(data_struct) = &input.data {
let mut default_statements = Vec::new();
let mut field_statements = Vec::new();

let found_crate = crate_name("ferrumc-inventory").unwrap();

let net_crate = match found_crate {
FoundCrate::Itself => {
quote! { crate }
}
FoundCrate::Name(name) => {
let name = syn::Ident::new(&name, proc_macro2::Span::call_site());
quote! { #name }
}
};

let mut inventory_type_creator = quote! {};

// Extract inventory_type attribute
for attr in &input.attrs {
if attr.path().is_ident("inventory") {
attr.parse_nested_meta(|meta| {
if let Some(ident) = meta.path.get_ident() {
let ident_str = ident.to_string();
let ident_str_ident = syn::Ident::new(&ident_str, proc_macro2::Span::call_site());
match ident_str.as_str() {
"inventory_type" => {
if let Some(ident) = handle_meta("inventory_type", &meta, true) {
inventory_type_creator = quote! {
#inventory_type_creator
.#ident_str_ident(#net_crate::inventory::InventoryType::#ident)
};
}
},
"is_synced" => {
if let Some(expr) = handle_meta("is_synced", &meta, false) {
inventory_type_creator = quote! {
#inventory_type_creator
.#ident_str_ident(#expr)
};
}
},
"title" => {
if let Some(expr) = handle_meta("title", &meta, false) {
inventory_type_creator = quote! {
#inventory_type_creator
.#ident_str_ident(ferrumc_text::TextComponent::new(#expr).build())
};
}
}
_ => {},
}
}
Ok(())
})
.unwrap();
}
}

// Process fields
for field in &data_struct.fields {
let field_name = &field.ident.clone().expect("Missing field");

let mut id_expr = None;
let mut value_expr = None;

if let Visibility::Public(_) = &field.vis {
for attr in &field.attrs {
if attr.path().is_ident("slot") {
attr.parse_nested_meta(|meta| {
if let Some(ident) = meta.path.get_ident() {
let ident_str = ident.to_string();
match ident_str.as_str() {
"id" => {
id_expr = handle_meta("id", &meta, true);
}
"default_value" => {
value_expr = handle_meta("default_value", &meta, true);
}
_ => {}
}
}
Ok(())
})
.unwrap();
}
}
}

// Generate default initialization and setter methods
if let (Some(id), Some(value)) = (id_expr, value_expr) {
default_statements.push(quote! {
#field_name: #value,
});

let setter_name =
syn::Ident::new(&format!("set_{}", field_name), field_name.span());
field_statements.push(quote! {
pub fn #setter_name<S: Into<Slot> + Copy>(&mut self, #field_name: S) {
self.#field_name = #field_name.into();
self.set_slot(#id, #field_name);
}
});
}
}

// Generate the `new` method
let new_method = quote! {
pub fn new(id: u8) -> Self {
Self {
inventory: #net_crate::builder::InventoryBuilder::new(id)
#inventory_type_creator
.build(),
#(#default_statements)*
}
}
};

// Generate the complete implementation block
// Wacky ass code because rust is retarded
let output = quote! {
impl #name {
#new_method

#(#field_statements)*
}

impl std::ops::Deref for #name {
type Target = Inventory;

fn deref(&self) -> &Self::Target {
&self.inventory
}
}

impl std::ops::DerefMut for #name {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.inventory
}
}
};

output.into()
} else {
TokenStream::new()
}
}

fn handle_meta(
name: &str,
meta: &syn::meta::ParseNestedMeta,
is_required: bool,
) -> Option<syn::Expr> {
match meta.value() {
Ok(value) => {
if let Ok(value) = value.parse::<syn::Expr>() {
Some(value)
} else if is_required {
panic!("Failed to parse value for attribute '{}'", name);
} else {
None
}
}
Err(_) => {
if is_required {
panic!("Missing required attribute '{}'", name);
}

None
}
}
}

pub fn inventory_type(_args: TokenStream, input: TokenStream) -> TokenStream {
input
}
13 changes: 13 additions & 0 deletions src/lib/derive_macros/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ use proc_macro::TokenStream;

mod events;
mod helpers;
mod inventory;
mod nbt;
mod net;
mod profiling;
Expand Down Expand Up @@ -75,3 +76,15 @@ pub fn get_packet_entry(input: TokenStream) -> TokenStream {
pub fn get_registry_entry(input: TokenStream) -> TokenStream {
static_loading::registry::get(input)
}

// #=================== INVENTORY ===================#
#[proc_macro_derive(Inventory, attributes(slot))]
pub fn create_inventory(input: TokenStream) -> TokenStream {
inventory::create(input)
}

#[proc_macro_attribute]
pub fn inventory(args: TokenStream, input: TokenStream) -> TokenStream {
inventory::inventory_type(args, input)
}
// #=================== INVENTORY ===================#
Loading
Loading