Compare commits
No commits in common. "6031eeab5c019fc67f1c5f923974b33e9bb34b3c" and "c7fd8a57d7fe7ae684de804347391b87ea27f947" have entirely different histories.
6031eeab5c
...
c7fd8a57d7
2 changed files with 86 additions and 95 deletions
161
src/bike.rs
161
src/bike.rs
|
@ -13,9 +13,7 @@ fn spawn_bike(
|
||||||
mut meshes: ResMut<Assets<Mesh>>,
|
mut meshes: ResMut<Assets<Mesh>>,
|
||||||
mut materials: ResMut<Assets<StandardMaterial>>,
|
mut materials: ResMut<Assets<StandardMaterial>>,
|
||||||
) {
|
) {
|
||||||
let pos = Vec3::new(0.0, 4.0, 0.0);
|
let xform = Transform::from_xyz(4.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));
|
||||||
|
|
||||||
|
@ -68,73 +66,6 @@ pub struct Rearing;
|
||||||
#[derive(Component)]
|
#[derive(Component)]
|
||||||
pub struct RearHub;
|
pub struct RearHub;
|
||||||
|
|
||||||
fn helper(
|
|
||||||
commands: &mut Commands,
|
|
||||||
meshes: &mut ResMut<Assets<Mesh>>,
|
|
||||||
materials: &mut ResMut<Assets<StandardMaterial>>,
|
|
||||||
position: Vec3,
|
|
||||||
) -> Entity {
|
|
||||||
let mut tire_mesh: Mesh = Torus::new(0.30, 0.40).into();
|
|
||||||
let tire_verts = tire_mesh
|
|
||||||
.attribute(Mesh::ATTRIBUTE_POSITION)
|
|
||||||
.unwrap()
|
|
||||||
.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 {
|
|
||||||
base_color: Color::srgb(0.01, 0.01, 0.01),
|
|
||||||
alpha_mode: AlphaMode::Opaque,
|
|
||||||
perceptual_roughness: 0.5,
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
let collider = Collider::convex_decomposition_from_mesh(&tire_mesh).unwrap();
|
|
||||||
let mut tire = None;
|
|
||||||
let tref = &mut tire;
|
|
||||||
|
|
||||||
let xform = Transform::from_translation(position);
|
|
||||||
|
|
||||||
let hub = commands
|
|
||||||
.spawn((
|
|
||||||
SpatialBundle::from_transform(xform),
|
|
||||||
RigidBody::Dynamic,
|
|
||||||
MassPropertiesBundle::new_computed(&Collider::sphere(0.1), 0.01),
|
|
||||||
))
|
|
||||||
.with_children(move |parent| {
|
|
||||||
let mesh: Mesh = Sphere::new(0.1).into();
|
|
||||||
parent.spawn(PbrBundle {
|
|
||||||
mesh: meshes.add(mesh),
|
|
||||||
material: materials.add(wheel_material.clone()),
|
|
||||||
..Default::default()
|
|
||||||
});
|
|
||||||
let tire_id = parent
|
|
||||||
.spawn((
|
|
||||||
CyberWheel,
|
|
||||||
RigidBody::Dynamic,
|
|
||||||
collider.clone(),
|
|
||||||
ColliderDensity(0.05),
|
|
||||||
CollisionLayers::from_bits(2, 2),
|
|
||||||
ExternalTorque::ZERO.with_persistence(false),
|
|
||||||
))
|
|
||||||
.id();
|
|
||||||
let _ = tref.insert(tire_id);
|
|
||||||
})
|
|
||||||
.id();
|
|
||||||
|
|
||||||
// connect hubs and tires to make wheels
|
|
||||||
commands.spawn(RevoluteJoint::new(hub, tire.unwrap()).with_aligned_axis(Vec3::X));
|
|
||||||
hub
|
|
||||||
}
|
|
||||||
|
|
||||||
fn spawn_wheels(
|
fn spawn_wheels(
|
||||||
mut commands: Commands,
|
mut commands: Commands,
|
||||||
mut meshes: ResMut<Assets<Mesh>>,
|
mut meshes: ResMut<Assets<Mesh>>,
|
||||||
|
@ -142,23 +73,97 @@ fn spawn_wheels(
|
||||||
xform: Transform,
|
xform: Transform,
|
||||||
body: Entity,
|
body: Entity,
|
||||||
) {
|
) {
|
||||||
let rake_vec = Vec3::new(0.0, -1.0, -0.57).normalize(); // about 30 degrees
|
let rake_vec = Vec3::new(0.0, -1.0, 0.57).normalize(); // about 30 degrees
|
||||||
let front_pos = xform.translation + rake_vec;
|
let front_pos = xform.translation + *xform.forward() + rake_vec;
|
||||||
|
|
||||||
let front_hub = helper(&mut commands, &mut meshes, &mut materials, front_pos);
|
let wheel_mesh: Mesh = Torus::new(0.30, 0.40).into();
|
||||||
commands.entity(front_hub).insert(FrontHub);
|
let wheel_material = StandardMaterial {
|
||||||
|
base_color: Color::srgb(0.01, 0.01, 0.01),
|
||||||
|
alpha_mode: AlphaMode::Opaque,
|
||||||
|
perceptual_roughness: 0.5,
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
let collider = Collider::convex_decomposition_from_mesh(&wheel_mesh).unwrap();
|
||||||
|
|
||||||
|
let front_hub = commands
|
||||||
|
.spawn((
|
||||||
|
FrontHub,
|
||||||
|
SpatialBundle::from_transform(Transform::from_translation(front_pos)),
|
||||||
|
RigidBody::Dynamic,
|
||||||
|
MassPropertiesBundle::new_computed(&Collider::sphere(0.01), 1.0),
|
||||||
|
))
|
||||||
|
.id();
|
||||||
|
|
||||||
|
let mut tire_rot = Transform::default();
|
||||||
|
tire_rot.rotate_z(FRAC_PI_2);
|
||||||
|
|
||||||
|
let front_tire = commands
|
||||||
|
.spawn((
|
||||||
|
SpatialBundle::from_transform(tire_rot),
|
||||||
|
CyberWheel,
|
||||||
|
RigidBody::Dynamic,
|
||||||
|
collider.clone(),
|
||||||
|
ColliderDensity(0.05),
|
||||||
|
CollisionLayers::from_bits(2, 2),
|
||||||
|
ExternalTorque::ZERO.with_persistence(false),
|
||||||
|
))
|
||||||
|
.with_children(|b| {
|
||||||
|
b.spawn(PbrBundle {
|
||||||
|
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();
|
||||||
|
|
||||||
|
// connect hubs and tires to make wheels
|
||||||
|
commands.spawn(RevoluteJoint::new(front_hub, front_tire).with_aligned_axis(Vec3::X));
|
||||||
|
|
||||||
|
// suspension joints connect the hubs and the body
|
||||||
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_hub = helper(&mut commands, &mut meshes, &mut materials, rear_pos);
|
let rear_pos = *xform.back() + xform.translation + rake_vec.y;
|
||||||
commands.entity(rear_hub).insert(RearHub);
|
let rear_hub = commands
|
||||||
|
.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() + rake_vec.y),
|
FixedJoint::new(rear_hub, body).with_local_anchor_2(*xform.back() + xform.translation),
|
||||||
));
|
));
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
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_normal() {
|
if tan_theta.is_finite() && !tan_theta.is_subnormal() {
|
||||||
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,24 +121,10 @@ 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 = Vec3::Y; //xform.translation.normalize();
|
let world_up = 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