~starkingdoms/starkingdoms

ad2517ff63342f5d076ebc7b72f83a4b68b9b19f — ghostly_zsh a month ago e2c74ff master
feat: crafting ui start
M crates/unified/assets/config/parts/hearty.part.toml => crates/unified/assets/config/parts/hearty.part.toml +4 -1
@@ 47,4 47,7 @@ snap = { translation = [ 0.0, -25.0, 0.0 ], rotation = 0.0 }
[[joint]]
id = "Left"
target = { translation = [ -55.0, 0.0, 0.0 ], rotation = -270.0 }
snap = { translation = [ -25.0, 0.0, 0.0 ], rotation = 0.0 }
\ No newline at end of file
snap = { translation = [ -25.0, 0.0, 0.0 ], rotation = 0.0 }

[crafting]
can_craft = true

A crates/unified/src/client/crafting/mod.rs => crates/unified/src/client/crafting/mod.rs +1 -0
@@ 0,0 1,1 @@
pub mod ui;

A crates/unified/src/client/crafting/ui.rs => crates/unified/src/client/crafting/ui.rs +70 -0
@@ 0,0 1,70 @@
use bevy::{input_focus::{AutoFocus, InputFocus}, ui::RelativeCursorPosition};

use crate::{attachment::PartInShip, client::colors, ecs::{CanCraft, CraftingUi, MainCamera, Me}, prelude::*};

pub fn crafting_ui_plugin(app: &mut App) {
    app.add_systems(Update, close_ui);
}

pub fn open_crafting_ui(
    ev: On<Pointer<Press>>,
    crafting_parts: Query<(Entity, &Transform), (With<PartInShip>, With<CanCraft>)>,
    hearty: Query<(Entity, &Transform), (With<Me>, With<CanCraft>)>,
    camera: Single<(Entity, &Camera, &GlobalTransform), (With<MainCamera>, Without<PartInShip>)>,
    commands: Commands,
) {
    if matches!(ev.button, PointerButton::Secondary) {
        let (entity, transform) = if let Ok(part) = crafting_parts.get(ev.entity) {
            part
        } else if let Ok(part) = hearty.get(ev.entity) {
            part
        } else {
            return
        };
        // we have our crafting entity!
        // now make the ui
        setup_ui(transform, commands, camera);
    }
}

fn setup_ui(
    parent_transform: &Transform,
    mut commands: Commands,
    camera: Single<(Entity, &Camera, &GlobalTransform), (With<MainCamera>, Without<PartInShip>)>,
) {
    let parent_pos = camera.1.world_to_viewport(camera.2, parent_transform.translation).unwrap();
    let entity = commands.spawn((
        UiTargetCamera(camera.0),
        Node {
            position_type: PositionType::Absolute,
            left: Val::Px(parent_pos.x),
            top: Val::Px(parent_pos.y),
            width: Val::Px(100.0),
            height: Val::Px(100.0),
            ..default()
        },
        AutoFocus,
        CraftingUi,
        BackgroundColor(colors::MANTLE),
        RelativeCursorPosition::default(),
    ));
}

pub fn close_ui(
    buttons: Res<ButtonInput<MouseButton>>,
    crafting_ui: Query<(Entity, &RelativeCursorPosition), With<CraftingUi>>,
    mut commands: Commands,
) {
    if crafting_ui.is_empty() { return }
    if !buttons.any_just_released([MouseButton::Left, MouseButton::Right]) { return }

    for (_, cursor_pos) in &crafting_ui {
        if cursor_pos.cursor_over {
            return;
        }
    }

    for (entity, _) in &crafting_ui {
        commands.entity(entity).despawn();
    }
}

M crates/unified/src/client/mod.rs => crates/unified/src/client/mod.rs +3 -0
@@ 1,3 1,4 @@
use crate::client::crafting::ui::crafting_ui_plugin;
use crate::client::key_input::key_input_plugin;
use crate::client::parts::parts_plugin;
use crate::client::planet::indicators::indicators_plugin;


@@ 28,6 29,7 @@ pub mod ship;
pub mod rendering;
pub mod input;
pub mod starguide;
pub mod crafting;

pub struct ClientPlugin {
    pub server: Option<String>,


@@ 61,6 63,7 @@ impl Plugin for ClientPlugin {
            .add_plugins(starguide_init_plugin)
            .add_plugins(starguide_input_plugin)
            .add_plugins(starguide_orbit_plugin)
            .add_plugins(crafting_ui_plugin)
            .insert_state(GameplayState::Main)
            .insert_resource(DebugPickingMode::Disabled);


M crates/unified/src/client/parts.rs => crates/unified/src/client/parts.rs +3 -1
@@ 1,6 1,7 @@
use std::f32::consts::PI;

use crate::attachment::{Joint, JointOf, Joints, PartInShip, Peer, SnapOf, SnapOfJoint};
use crate::client::crafting::ui::open_crafting_ui;
use crate::ecs::Me;
use crate::client::colors::GREEN;
use crate::ecs::{DragRequestEvent, Part, MAIN_LAYER};


@@ 45,7 46,8 @@ fn handle_incoming_parts(
            .insert(MAIN_LAYER)
            .insert(sprite)
            .insert(Pickable::default())
            .observe(on_part_click);
            .observe(on_part_click)
            .observe(open_crafting_ui);
    }
}
fn handle_updated_parts(

M crates/unified/src/client/rendering/mod.rs => crates/unified/src/client/rendering/mod.rs +3 -1
@@ 2,6 2,7 @@ use bevy::anti_alias::fxaa::Fxaa;
use bevy::app::{App, Startup};
use bevy::core_pipeline::tonemapping::DebandDither;
use bevy::post_process::bloom::Bloom;
use crate::client::crafting::ui::close_ui;
use crate::ecs::{GameplayState, MAIN_LAYER, MainCamera, Me, STARGUIDE_LAYER, StarguideGizmos};
use crate::prelude::*;



@@ 23,7 24,8 @@ pub fn setup_graphics(mut config_store: ResMut<GizmoConfigStore>, mut commands: 
        .insert(Bloom::default())
        .insert(DebandDither::Enabled)
        .insert(Fxaa::default())
        .insert(MainCamera);
        .insert(MainCamera)
        .insert(SpritePickingCamera);
    let (gizmo_config, _) = config_store.config_mut::<StarguideGizmos>();
    gizmo_config.render_layers = STARGUIDE_LAYER;
    gizmo_config.depth_bias = 1.0;

M crates/unified/src/client_plugins.rs => crates/unified/src/client_plugins.rs +3 -1
@@ 4,6 4,7 @@ use aeronet_websocket::client::WebSocketClientPlugin;
use bevy::app::{PluginGroup, PluginGroupBuilder};
use bevy::dev_tools::picking_debug::DebugPickingPlugin;
use bevy::ecs::schedule::ScheduleLabel;
use bevy::input_focus::InputDispatchPlugin;
use crate::prelude::*;
use bevy::ui::UiPlugin;
use leafwing_input_manager::plugin::InputManagerPlugin;


@@ 22,9 23,10 @@ impl PluginGroup for ClientPluginGroup {
                server: self.server,
            })
            .add(UiPlugin)
            .add(InputDispatchPlugin)
            .add(InputManagerPlugin::<crate::client::input::ClientAction>::default())
    }
}

#[derive(ScheduleLabel, Clone, Eq, Debug, Hash, PartialEq)]
pub struct DontRunSchedule;
\ No newline at end of file
pub struct DontRunSchedule;

M crates/unified/src/config/part.rs => crates/unified/src/config/part.rs +5 -0
@@ 13,6 13,7 @@ pub struct PartConfig {
    #[serde(default)]
    #[serde(rename = "joint")]
    pub joints: Vec<JointConfig>,
    pub crafting: Option<CraftingConfig>,
}
#[derive(Deserialize, TypePath, Serialize, Clone, Debug, PartialEq)]
pub struct PartPartConfig {


@@ 53,3 54,7 @@ impl From<JointOffset> for Transform {
        }
    }
}
#[derive(Deserialize, TypePath, Serialize, Clone, Debug, PartialEq)]
pub struct CraftingConfig {
    pub can_craft: bool,
}

M crates/unified/src/ecs.rs => crates/unified/src/ecs.rs +5 -0
@@ 108,3 108,8 @@ pub struct Hi {
    #[entities]
    pub you_are: Entity
}

#[derive(Component, Serialize, Deserialize, Debug)]
pub struct CanCraft;
#[derive(Component, Serialize, Deserialize, Debug)]
pub struct CraftingUi;

M crates/unified/src/server/part.rs => crates/unified/src/server/part.rs +11 -2
@@ 1,6 1,6 @@
use crate::attachment::{Joint, JointId, JointOf, Joints, Peer, SnapOf, SnapOfJoint};
use crate::config::part::{JointConfig, PartConfig};
use crate::ecs::{Part, PartHandle};
use crate::config::part::{CraftingConfig, JointConfig, PartConfig};
use crate::ecs::{CanCraft, Part, PartHandle};
use crate::prelude::*;
use bevy_replicon::prelude::Replicated;
use crate::ecs::thruster::{PartThrusters, Thruster, ThrusterBundle, ThrusterId, ThrusterOfPart};


@@ 32,6 32,9 @@ fn handle_ready_parts(
                .entity(entity)
                .insert(calculate_bundle(strong_config, &loading_part.0))
                .remove::<SpawnPartRequest>();
            if let Some(ref crafting_config) = strong_config.crafting {
                crafting_bundle(&mut commands.entity(entity), &crafting_config);
            }
            spawn_joints(strong_config, entity, commands.reborrow());
            spawn_thrusters(strong_config, entity, commands.reborrow());
        }


@@ 129,6 132,11 @@ fn calculate_bundle(config: &PartConfig, handle: &Handle<PartConfig>) -> impl Bu
        mass,
    )
}
fn crafting_bundle(entity: &mut EntityCommands, config: &CraftingConfig) {
    if config.can_craft {
        entity.insert(CanCraft);
    }
}
fn spawn_joint_bundle(joint: &JointConfig, part: &PartConfig, parent: &Entity) -> impl Bundle {
    let j_comp = Joint {
        id: JointId::from_part_and_joint_id(part.part.name.clone(), joint.id.clone()),


@@ 169,3 177,4 @@ fn spawn_thrusters(config: &PartConfig, part: Entity, mut commands: Commands) {
            });
    }
}


M crates/unified/src/shared_plugins.rs => crates/unified/src/shared_plugins.rs +2 -1
@@ 1,6 1,6 @@
use crate::attachment::{Joint, JointOf, PartInShip, Peer, Ship, SnapOf, SnapOfJoint};
use crate::config::planet::{Planet, PlanetConfigCollection};
use crate::ecs::{DragRequestEvent, Hi, Part, Particles, Player, PlayerStorage};
use crate::ecs::{CanCraft, DragRequestEvent, Hi, Part, Particles, Player, PlayerStorage};
use bevy::app::{App, PluginGroup, PluginGroupBuilder};
use bevy_common_assets::toml::TomlAssetPlugin;
use crate::prelude::*;


@@ 52,6 52,7 @@ pub fn register_everything(app: &mut App) {
    app.replicate::<PlayerStorage>();
    app.replicate::<Thruster>();
    app.replicate::<ThrusterOfPart>();
    app.replicate::<CanCraft>();
}

fn physics_setup_plugin(app: &mut App) {