diff --git a/common/src/cmd.rs b/common/src/cmd.rs index 21c2ee26b3..a1692dbabf 100644 --- a/common/src/cmd.rs +++ b/common/src/cmd.rs @@ -41,6 +41,7 @@ pub enum ChatCommand { Ban, Build, BuildAreaAdd, + BuildAreaList, BuildAreaRemove, Campfire, Debug, @@ -100,6 +101,7 @@ pub static CHAT_COMMANDS: &[ChatCommand] = &[ ChatCommand::Ban, ChatCommand::Build, ChatCommand::BuildAreaAdd, + ChatCommand::BuildAreaList, ChatCommand::BuildAreaRemove, ChatCommand::Campfire, ChatCommand::Debug, @@ -259,6 +261,7 @@ impl ChatCommand { "Adds a new build area", Admin, ), + ChatCommand::BuildAreaList => cmd(vec![], "List all build areas", Admin), ChatCommand::BuildAreaRemove => cmd( vec![Any("name", Required)], "Removes specified build area", @@ -397,7 +400,7 @@ impl ChatCommand { Admin, ), ChatCommand::PermitBuild => cmd( - vec![PlayerName(Required), Any("area_name", Required)], + vec![Any("area_name", Required)], "Grants player a bounded box they can build in", Admin, ), @@ -408,13 +411,13 @@ impl ChatCommand { Admin, ), ChatCommand::RevokeBuild => cmd( - vec![PlayerName(Required), Any("area_name", Required)], - "Revokes build area permission for given player", + vec![Any("area_name", Required)], + "Revokes build area permission for player", Admin, ), ChatCommand::RevokeBuildAll => cmd( - vec![PlayerName(Required)], - "Revokes all build area permissions for given player", + vec![], + "Revokes all build area permissions for player", Admin, ), ChatCommand::Region => cmd( @@ -504,6 +507,7 @@ impl ChatCommand { ChatCommand::Ban => "ban", ChatCommand::Build => "build", ChatCommand::BuildAreaAdd => "build_area_add", + ChatCommand::BuildAreaList => "build_area_list", ChatCommand::BuildAreaRemove => "build_area_remove", ChatCommand::Campfire => "campfire", ChatCommand::Debug => "debug", diff --git a/common/src/comp/inputs.rs b/common/src/comp/inputs.rs index 822f8d5c9e..fddbe7d4e2 100644 --- a/common/src/comp/inputs.rs +++ b/common/src/comp/inputs.rs @@ -2,12 +2,13 @@ use crate::depot::Id; use serde::{Deserialize, Serialize}; use specs::{Component, DerefFlaggedStorage}; use specs_idvs::IdvStorage; +use std::collections::HashSet; use vek::geom::Aabb; #[derive(Clone, Debug, Default, PartialEq, Serialize, Deserialize)] pub struct CanBuild { - pub building_is_on: bool, - pub build_areas: Vec>>, + pub enabled: bool, + pub build_areas: HashSet>>, } impl Component for CanBuild { type Storage = DerefFlaggedStorage>; diff --git a/server/src/cmd.rs b/server/src/cmd.rs index 3f56194afa..ed2e834ee4 100644 --- a/server/src/cmd.rs +++ b/server/src/cmd.rs @@ -34,6 +34,7 @@ use common_sys::state::BuildAreas; use rand::Rng; use specs::{Builder, Entity as EcsEntity, Join, WorldExt}; use std::{ + collections::HashSet, convert::TryFrom, ops::{Deref, DerefMut}, time::Duration, @@ -87,6 +88,7 @@ fn get_handler(cmd: &ChatCommand) -> CommandHandler { ChatCommand::Ban => handle_ban, ChatCommand::Build => handle_build, ChatCommand::BuildAreaAdd => handle_build_area_add, + ChatCommand::BuildAreaList => handle_build_area_list, ChatCommand::BuildAreaRemove => handle_build_area_remove, ChatCommand::Campfire => handle_spawn_campfire, ChatCommand::Debug => handle_debug, @@ -1129,60 +1131,42 @@ fn handle_safezone( fn handle_permit_build( server: &mut Server, client: EcsEntity, - _target: EcsEntity, + target: EcsEntity, args: String, action: &ChatCommand, ) { - if let (Some(target_alias), Some(area_name)) = - scan_fmt_some!(&args, &action.arg_fmt(), String, String) - { + if let Some(area_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let ecs = server.state.ecs(); - let target_player_opt = (&ecs.entities(), &ecs.read_storage::()) - .join() - .find(|(_, player)| player.alias == target_alias) - .map(|(entity, _)| entity); - - if let Some(target_player) = target_player_opt { - if server - .state - .read_storage::() - .get(target_player) - .is_none() + if server + .state + .read_storage::() + .get(target) + .is_none() + { + let _ = ecs + .write_storage::() + .insert(target, comp::CanBuild { + enabled: false, + build_areas: HashSet::new(), + }); + } + if let Some(bb_id) = ecs + .read_resource::() + .deref() + .area_names + .get(&area_name) + { + if let Some(mut comp_can_build) = ecs.write_storage::().get_mut(target) { - let _ = - ecs.write_storage::() - .insert(target_player, comp::CanBuild { - building_is_on: false, - build_areas: Vec::new(), - }); + comp_can_build.build_areas.insert(*bb_id); + server.notify_client( + client, + ServerGeneral::server_msg( + ChatType::CommandInfo, + format!("Permission to build in {} granted", area_name), + ), + ); } - if let Some(bb_id) = ecs - .read_resource::() - .deref() - .area_names - .get(&area_name) - { - if let Some(mut comp_can_build) = - ecs.write_storage::().get_mut(target_player) - { - comp_can_build.build_areas.push(*bb_id); - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandInfo, - format!("Gave {} permission to build in {}", target_alias, area_name), - ), - ); - } - } - } else { - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandError, - format!("Player '{}' not found!", target_alias), - ), - ); } } else { server.notify_client( @@ -1195,50 +1179,29 @@ fn handle_permit_build( fn handle_revoke_build( server: &mut Server, client: EcsEntity, - _target: EcsEntity, + target: EcsEntity, args: String, action: &ChatCommand, ) { - if let (Some(target_alias), Some(area_name)) = - scan_fmt_some!(&args, &action.arg_fmt(), String, String) - { + if let Some(area_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let ecs = server.state.ecs(); - let target_player_opt = (&ecs.entities(), &ecs.read_storage::()) - .join() - .find(|(_, player)| player.alias == target_alias) - .map(|(entity, _)| entity); - - if let Some(target_player) = target_player_opt { - if let Some(bb_id) = ecs - .read_resource::() - .deref() - .area_names - .get(&area_name) + if let Some(bb_id) = ecs + .read_resource::() + .deref() + .area_names + .get(&area_name) + { + if let Some(mut comp_can_build) = ecs.write_storage::().get_mut(target) { - if let Some(mut comp_can_build) = - ecs.write_storage::().get_mut(target_player) - { - comp_can_build.build_areas.retain(|&x| x != *bb_id); - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandInfo, - format!( - "Revoked {}'s permission to build in {}", - target_alias, area_name - ), - ), - ); - } + comp_can_build.build_areas.retain(|&x| x != *bb_id); + server.notify_client( + client, + ServerGeneral::server_msg( + ChatType::CommandInfo, + format!("Permission to build in {} revoked", area_name), + ), + ); } - } else { - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandError, - format!("Player '{}' not found!", target_alias), - ), - ); } } else { server.notify_client( @@ -1251,41 +1214,17 @@ fn handle_revoke_build( fn handle_revoke_build_all( server: &mut Server, client: EcsEntity, - _target: EcsEntity, - args: String, - action: &ChatCommand, + target: EcsEntity, + _args: String, + _action: &ChatCommand, ) { - if let Some(target_alias) = scan_fmt_some!(&args, &action.arg_fmt(), String) { - let ecs = server.state.ecs(); - let target_player_opt = (&ecs.entities(), &ecs.read_storage::()) - .join() - .find(|(_, player)| player.alias == target_alias) - .map(|(entity, _)| entity); + let ecs = server.state.ecs(); - if let Some(target_player) = target_player_opt { - ecs.write_storage::().remove(target_player); - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandInfo, - format!("Revoked {}'s permission to build everywhere", target_alias), - ), - ); - } else { - server.notify_client( - client, - ServerGeneral::server_msg( - ChatType::CommandError, - format!("Player '{}' not found!", target_alias), - ), - ); - } - } else { - server.notify_client( - client, - ServerGeneral::server_msg(ChatType::CommandError, action.help_string()), - ); - } + ecs.write_storage::().remove(target); + server.notify_client( + client, + ServerGeneral::server_msg(ChatType::CommandInfo, "All build permissions revoked"), + ); } fn handle_players( @@ -1328,14 +1267,14 @@ fn handle_build( .write_storage::() .get_mut(target) { - if can_build.building_is_on { - can_build.building_is_on = false; + if can_build.enabled { + can_build.enabled = false; server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Toggled off build mode!"), ); } else { - can_build.building_is_on = true; + can_build.enabled = true; server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Toggled on build mode!"), @@ -1371,13 +1310,28 @@ fn handle_build_area_add( i32 ) { let ecs = server.state.ecs(); + if ecs + .read_resource::() + .deref() + .area_names + .contains_key(&area_name) + { + server.notify_client( + client, + ServerGeneral::server_msg( + ChatType::CommandError, + format!("Build zone {} already exists!", area_name), + ), + ); + return; + } let bb_id = ecs .write_resource::() .deref_mut() .areas .insert(Aabb { - min: Vec3::new(xlo, ylo, zlo), - max: Vec3::new(xhi, yhi, zhi), + min: Vec3::new(xlo.min(xhi), ylo.min(yhi), zlo.min(zhi)), + max: Vec3::new(xhi.max(xlo), yhi.max(ylo), zhi.max(zlo)), }); ecs.write_resource::() .deref_mut() @@ -1393,6 +1347,34 @@ fn handle_build_area_add( } } +fn handle_build_area_list( + server: &mut Server, + client: EcsEntity, + _target: EcsEntity, + _args: String, + _action: &ChatCommand, +) { + let ecs = server.state.ecs(); + let build_areas = ecs.read_resource::(); + + server.notify_client( + client, + ServerGeneral::server_msg( + ChatType::CommandInfo, + build_areas.area_names.iter().fold( + "Build Areas:".to_string(), + |acc, (area_name, bb_id)| { + if let Some(aabb) = build_areas.areas.get(*bb_id) { + format!("{}\n{}: {} to {}", acc, area_name, aabb.min, aabb.max) + } else { + acc + } + }, + ), + ), + ); +} + fn handle_build_area_remove( server: &mut Server, client: EcsEntity, diff --git a/server/src/sys/msg/in_game.rs b/server/src/sys/msg/in_game.rs index 01135133a5..ff8fdc3b69 100644 --- a/server/src/sys/msg/in_game.rs +++ b/server/src/sys/msg/in_game.rs @@ -104,14 +104,17 @@ impl Sys { }, ClientGeneral::BreakBlock(pos) => { if let Some(comp_can_build) = can_build.get(entity) { - if comp_can_build.building_is_on { + if comp_can_build.enabled { for area in comp_can_build.build_areas.iter() { - if let Some(aabb) = build_areas.areas.get(*area) { - if aabb.contains_point(pos) { - if let Ok(block) = terrain.get(pos) { - block_changes.set(pos, block.into_vacant()); - } - } + if let Some(block) = build_areas + .areas + .get(*area) + // TODO: Make this an exclusive check on the upper bound of the AABB + // Vek defaults to inclusive which is not optimal + .filter(|aabb| aabb.contains_point(pos)) + .and_then(|_| terrain.get(pos).ok()) + { + block_changes.set(pos, block.into_vacant()); } } } @@ -119,12 +122,17 @@ impl Sys { }, ClientGeneral::PlaceBlock(pos, block) => { if let Some(comp_can_build) = can_build.get(entity) { - if comp_can_build.building_is_on { + if comp_can_build.enabled { for area in comp_can_build.build_areas.iter() { - if let Some(aabb) = build_areas.areas.get(*area) { - if aabb.contains_point(pos) { - block_changes.try_set(pos, block); - } + if build_areas + .areas + .get(*area) + // TODO: Make this an exclusive check on the upper bound of the AABB + // Vek defaults to inclusive which is not optimal + .filter(|aabb| aabb.contains_point(pos)) + .is_some() + { + block_changes.try_set(pos, block); } } } diff --git a/voxygen/src/session.rs b/voxygen/src/session.rs index 016219eb17..a7c04828b3 100644 --- a/voxygen/src/session.rs +++ b/voxygen/src/session.rs @@ -317,7 +317,7 @@ impl PlayState for SessionState { .state() .read_storage::() .get(player_entity) - .map_or_else(|| false, |cb| cb.building_is_on); + .map_or_else(|| false, |cb| cb.enabled); let is_mining = self .client