Compare commits
5 commits
c7fd8a57d7
...
6031eeab5c
Author | SHA1 | Date | |
---|---|---|---|
|
6031eeab5c | ||
|
52f8c91232 | ||
|
610f141318 | ||
|
061583648d | ||
|
bf2d9727f5 |
2 changed files with 95 additions and 86 deletions
131
src/bike.rs
131
src/bike.rs
|
@ -13,7 +13,9 @@ fn spawn_bike(
|
||||||
mut meshes: ResMut<Assets<Mesh>>,
|
mut meshes: ResMut<Assets<Mesh>>,
|
||||||
mut materials: ResMut<Assets<StandardMaterial>>,
|
mut materials: ResMut<Assets<StandardMaterial>>,
|
||||||
) {
|
) {
|
||||||
let xform = Transform::from_xyz(4.0, 4.0, 0.0);
|
let pos = Vec3::new(0.0, 4.0, 0.0);
|
||||||
|
let xform = Transform::from_translation(pos).with_rotation(Quat::from_rotation_z(0.0));
|
||||||
|
|
||||||
let body_collider =
|
let body_collider =
|
||||||
Collider::capsule_endpoints(0.5, Vec3::new(0.0, 0.0, -0.65), Vec3::new(0.0, 0.0, 0.8));
|
Collider::capsule_endpoints(0.5, Vec3::new(0.0, 0.0, -0.65), Vec3::new(0.0, 0.0, 0.8));
|
||||||
|
|
||||||
|
@ -66,40 +68,56 @@ pub struct Rearing;
|
||||||
#[derive(Component)]
|
#[derive(Component)]
|
||||||
pub struct RearHub;
|
pub struct RearHub;
|
||||||
|
|
||||||
fn spawn_wheels(
|
fn helper(
|
||||||
mut commands: Commands,
|
commands: &mut Commands,
|
||||||
mut meshes: ResMut<Assets<Mesh>>,
|
meshes: &mut ResMut<Assets<Mesh>>,
|
||||||
mut materials: ResMut<Assets<StandardMaterial>>,
|
materials: &mut ResMut<Assets<StandardMaterial>>,
|
||||||
xform: Transform,
|
position: Vec3,
|
||||||
body: Entity,
|
) -> Entity {
|
||||||
) {
|
let mut tire_mesh: Mesh = Torus::new(0.30, 0.40).into();
|
||||||
let rake_vec = Vec3::new(0.0, -1.0, 0.57).normalize(); // about 30 degrees
|
let tire_verts = tire_mesh
|
||||||
let front_pos = xform.translation + *xform.forward() + rake_vec;
|
.attribute(Mesh::ATTRIBUTE_POSITION)
|
||||||
|
.unwrap()
|
||||||
let wheel_mesh: Mesh = Torus::new(0.30, 0.40).into();
|
.as_float3()
|
||||||
|
.unwrap()
|
||||||
|
.iter()
|
||||||
|
.map(|v| {
|
||||||
|
//
|
||||||
|
let v = Vec3::from_array(*v);
|
||||||
|
let m = Mat3::from_rotation_z(90.0f32.to_radians());
|
||||||
|
let p = m.mul_vec3(v);
|
||||||
|
p.to_array()
|
||||||
|
})
|
||||||
|
.collect::<Vec<[f32; 3]>>();
|
||||||
|
tire_mesh.remove_attribute(Mesh::ATTRIBUTE_POSITION);
|
||||||
|
tire_mesh.insert_attribute(Mesh::ATTRIBUTE_POSITION, tire_verts);
|
||||||
let wheel_material = StandardMaterial {
|
let wheel_material = StandardMaterial {
|
||||||
base_color: Color::srgb(0.01, 0.01, 0.01),
|
base_color: Color::srgb(0.01, 0.01, 0.01),
|
||||||
alpha_mode: AlphaMode::Opaque,
|
alpha_mode: AlphaMode::Opaque,
|
||||||
perceptual_roughness: 0.5,
|
perceptual_roughness: 0.5,
|
||||||
..Default::default()
|
..Default::default()
|
||||||
};
|
};
|
||||||
let collider = Collider::convex_decomposition_from_mesh(&wheel_mesh).unwrap();
|
let collider = Collider::convex_decomposition_from_mesh(&tire_mesh).unwrap();
|
||||||
|
let mut tire = None;
|
||||||
|
let tref = &mut tire;
|
||||||
|
|
||||||
let front_hub = commands
|
let xform = Transform::from_translation(position);
|
||||||
|
|
||||||
|
let hub = commands
|
||||||
.spawn((
|
.spawn((
|
||||||
FrontHub,
|
SpatialBundle::from_transform(xform),
|
||||||
SpatialBundle::from_transform(Transform::from_translation(front_pos)),
|
|
||||||
RigidBody::Dynamic,
|
RigidBody::Dynamic,
|
||||||
MassPropertiesBundle::new_computed(&Collider::sphere(0.01), 1.0),
|
MassPropertiesBundle::new_computed(&Collider::sphere(0.1), 0.01),
|
||||||
))
|
))
|
||||||
.id();
|
.with_children(move |parent| {
|
||||||
|
let mesh: Mesh = Sphere::new(0.1).into();
|
||||||
let mut tire_rot = Transform::default();
|
parent.spawn(PbrBundle {
|
||||||
tire_rot.rotate_z(FRAC_PI_2);
|
mesh: meshes.add(mesh),
|
||||||
|
material: materials.add(wheel_material.clone()),
|
||||||
let front_tire = commands
|
..Default::default()
|
||||||
|
});
|
||||||
|
let tire_id = parent
|
||||||
.spawn((
|
.spawn((
|
||||||
SpatialBundle::from_transform(tire_rot),
|
|
||||||
CyberWheel,
|
CyberWheel,
|
||||||
RigidBody::Dynamic,
|
RigidBody::Dynamic,
|
||||||
collider.clone(),
|
collider.clone(),
|
||||||
|
@ -107,63 +125,40 @@ fn spawn_wheels(
|
||||||
CollisionLayers::from_bits(2, 2),
|
CollisionLayers::from_bits(2, 2),
|
||||||
ExternalTorque::ZERO.with_persistence(false),
|
ExternalTorque::ZERO.with_persistence(false),
|
||||||
))
|
))
|
||||||
.with_children(|b| {
|
.id();
|
||||||
b.spawn(PbrBundle {
|
let _ = tref.insert(tire_id);
|
||||||
mesh: meshes.add(wheel_mesh.clone()),
|
|
||||||
material: materials.add(wheel_material.clone()),
|
|
||||||
// transform: Transform::from_translation(front_pos)
|
|
||||||
// .with_rotation(Quat::from_rotation_z(FRAC_PI_2)),
|
|
||||||
..Default::default()
|
|
||||||
});
|
|
||||||
})
|
})
|
||||||
.id();
|
.id();
|
||||||
|
|
||||||
// connect hubs and tires to make wheels
|
// connect hubs and tires to make wheels
|
||||||
commands.spawn(RevoluteJoint::new(front_hub, front_tire).with_aligned_axis(Vec3::X));
|
commands.spawn(RevoluteJoint::new(hub, tire.unwrap()).with_aligned_axis(Vec3::X));
|
||||||
|
hub
|
||||||
|
}
|
||||||
|
|
||||||
// suspension joints connect the hubs and the body
|
fn spawn_wheels(
|
||||||
|
mut commands: Commands,
|
||||||
|
mut meshes: ResMut<Assets<Mesh>>,
|
||||||
|
mut materials: ResMut<Assets<StandardMaterial>>,
|
||||||
|
xform: Transform,
|
||||||
|
body: Entity,
|
||||||
|
) {
|
||||||
|
let rake_vec = Vec3::new(0.0, -1.0, -0.57).normalize(); // about 30 degrees
|
||||||
|
let front_pos = xform.translation + rake_vec;
|
||||||
|
|
||||||
|
let front_hub = helper(&mut commands, &mut meshes, &mut materials, front_pos);
|
||||||
|
commands.entity(front_hub).insert(FrontHub);
|
||||||
commands.spawn((
|
commands.spawn((
|
||||||
Steering,
|
Steering,
|
||||||
FixedJoint::new(front_hub, body).with_local_anchor_2(*xform.forward() + rake_vec),
|
FixedJoint::new(front_hub, body).with_local_anchor_2(*xform.forward() + rake_vec),
|
||||||
));
|
));
|
||||||
|
|
||||||
/*
|
let rear_pos = xform.translation + Vec3::new(0.0, -1.0, 0.57).normalize();
|
||||||
let rear_pos = *xform.back() + xform.translation + rake_vec.y;
|
let rear_hub = helper(&mut commands, &mut meshes, &mut materials, rear_pos);
|
||||||
let rear_hub = commands
|
commands.entity(rear_hub).insert(RearHub);
|
||||||
.spawn((
|
|
||||||
RearHub,
|
|
||||||
SpatialBundle::from_transform(Transform::from_translation(front_pos)),
|
|
||||||
RigidBody::Dynamic,
|
|
||||||
MassPropertiesBundle::new_computed(&Collider::sphere(0.01), 1.0),
|
|
||||||
))
|
|
||||||
.id();
|
|
||||||
|
|
||||||
let rear_tire = commands
|
|
||||||
.spawn((
|
|
||||||
CyberWheel,
|
|
||||||
RigidBody::Dynamic,
|
|
||||||
collider,
|
|
||||||
ColliderDensity(0.05),
|
|
||||||
CollisionLayers::from_bits(2, 2),
|
|
||||||
PbrBundle {
|
|
||||||
mesh: meshes.add(wheel_mesh),
|
|
||||||
material: materials.add(wheel_material),
|
|
||||||
transform: Transform::from_translation(rear_pos)
|
|
||||||
.with_rotation(Quat::from_rotation_z(FRAC_PI_2)),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
ExternalTorque::ZERO.with_persistence(false),
|
|
||||||
))
|
|
||||||
.id();
|
|
||||||
|
|
||||||
commands.spawn(RevoluteJoint::new(rear_hub,
|
|
||||||
rear_tire).with_aligned_axis(Vec3::X));
|
|
||||||
|
|
||||||
commands.spawn((
|
commands.spawn((
|
||||||
Rearing,
|
Rearing,
|
||||||
FixedJoint::new(rear_hub, body).with_local_anchor_2(*xform.back() + xform.translation),
|
FixedJoint::new(rear_hub, body).with_local_anchor_2(*xform.back() + rake_vec.y),
|
||||||
));
|
));
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct CyberBikePlugin;
|
pub struct CyberBikePlugin;
|
||||||
|
|
|
@ -106,8 +106,8 @@ mod systems {
|
||||||
let v2_r = v_squared / radius;
|
let v2_r = v_squared / radius;
|
||||||
let tan_theta = (v2_r / gravity).clamp(-FRAC_PI_3, FRAC_PI_3);
|
let tan_theta = (v2_r / gravity).clamp(-FRAC_PI_3, FRAC_PI_3);
|
||||||
|
|
||||||
if tan_theta.is_finite() && !tan_theta.is_subnormal() {
|
if tan_theta.is_normal() {
|
||||||
lean.lean = tan_theta.atan().clamp(-FRAC_PI_3, FRAC_PI_3);
|
lean.lean = -tan_theta.atan().clamp(-FRAC_PI_3, FRAC_PI_3);
|
||||||
} else {
|
} else {
|
||||||
lean.lean = 0.0;
|
lean.lean = 0.0;
|
||||||
}
|
}
|
||||||
|
@ -121,10 +121,24 @@ mod systems {
|
||||||
mut gizmos: Gizmos,
|
mut gizmos: Gizmos,
|
||||||
) {
|
) {
|
||||||
let (xform, mut torque, mut control_vars) = bike_query.single_mut();
|
let (xform, mut torque, mut control_vars) = bike_query.single_mut();
|
||||||
let world_up = xform.translation.normalize();
|
let world_up = Vec3::Y; //xform.translation.normalize();
|
||||||
let rot = Quat::from_axis_angle(*xform.back(), lean.lean);
|
let rot = Quat::from_axis_angle(*xform.back(), lean.lean);
|
||||||
let target_up = rotate_point(&world_up, &rot).normalize();
|
let target_up = rotate_point(&world_up, &rot).normalize();
|
||||||
|
|
||||||
|
// show which is up
|
||||||
|
gizmos.arrow(
|
||||||
|
xform.translation,
|
||||||
|
xform.translation + *xform.up(),
|
||||||
|
bevy::color::palettes::basic::YELLOW,
|
||||||
|
);
|
||||||
|
|
||||||
|
// show which is forward
|
||||||
|
gizmos.arrow(
|
||||||
|
*xform.forward() + xform.translation,
|
||||||
|
1.5 * *xform.forward() + xform.translation,
|
||||||
|
bevy::color::palettes::basic::PURPLE,
|
||||||
|
);
|
||||||
|
|
||||||
let bike_right = xform.right();
|
let bike_right = xform.right();
|
||||||
|
|
||||||
let roll_error = bike_right.dot(target_up);
|
let roll_error = bike_right.dot(target_up);
|
||||||
|
|
Loading…
Reference in a new issue