honeycomb_render/scene/
camera.rs1use bevy::input::mouse::{MouseMotion, MouseWheel};
2use bevy::math::vec2;
3use bevy::prelude::*;
4
5#[derive(Component)]
10pub struct PanOrbitCamera {
11 pub(crate) focus: Vec3,
12 pub(crate) radius: f32,
13 pub(crate) upside_down: bool,
14}
15
16impl Default for PanOrbitCamera {
17 fn default() -> Self {
18 Self {
19 focus: Vec3::ZERO,
20 radius: 5.0,
21 upside_down: false,
22 }
23 }
24}
25
26pub fn update_camera(
28 window_q: Query<&Window>,
29 mut ev_motion: EventReader<MouseMotion>,
30 mut ev_scroll: EventReader<MouseWheel>,
31 input_mouse: Res<ButtonInput<MouseButton>>,
32 mut query: Query<(&mut PanOrbitCamera, &mut Transform, &Projection)>,
33) {
34 let window = window_q.single();
35 let orbit_button = MouseButton::Right;
37 let pan_button = MouseButton::Left;
38
39 let mut pan = Vec2::ZERO;
40 let mut rotation_move = Vec2::ZERO;
41 let mut scroll = 0.0;
42 let mut orbit_button_changed = false;
43
44 if input_mouse.pressed(orbit_button) {
45 for ev in ev_motion.read() {
46 rotation_move += ev.delta;
47 }
48 } else if input_mouse.pressed(pan_button) {
49 for ev in ev_motion.read() {
51 pan += ev.delta * 2.;
52 }
53 }
54
55 for ev in ev_scroll.read() {
56 scroll += ev.y;
57
58 scroll /= if cfg!(target_arch = "wasm32") {
59 100.0
60 } else {
61 20.0
62 };
63 }
64 if input_mouse.just_released(orbit_button) || input_mouse.just_pressed(orbit_button) {
65 orbit_button_changed = true;
66 }
67
68 for (mut pan_orbit, mut transform, projection) in &mut query {
69 if orbit_button_changed {
70 let up = transform.rotation * Vec3::Y;
73 pan_orbit.upside_down = up.y <= 0.0;
74 }
75 let window = vec2(
76 window.physical_width() as f32,
77 window.physical_height() as f32,
78 );
79
80 let mut any = false;
81 if rotation_move.length_squared() > 0.0 {
82 any = true;
83 let delta_x = {
84 let delta = rotation_move.x / window.x * std::f32::consts::PI * 2.0;
85 if pan_orbit.upside_down { -delta } else { delta }
86 };
87 let delta_y = rotation_move.y / window.y * std::f32::consts::PI;
88 let yaw = Quat::from_rotation_y(-delta_x);
89 let pitch = Quat::from_rotation_x(-delta_y);
90 transform.rotation = yaw * transform.rotation; transform.rotation *= pitch; } else if pan.length_squared() > 0.0 {
93 any = true;
94
95 if let Projection::Perspective(projection) = projection {
96 pan *= Vec2::new(projection.fov * projection.aspect_ratio, projection.fov) / window;
97 }
98 let right = transform.rotation * Vec3::X * -pan.x;
100 let up = transform.rotation * Vec3::Y * pan.y;
101 let translation = (right + up) * pan_orbit.radius;
103 pan_orbit.focus += translation;
104 } else if scroll.abs() > 0.0 {
105 any = true;
106 pan_orbit.radius -= scroll * pan_orbit.radius * 0.2;
107 pan_orbit.radius = f32::max(pan_orbit.radius, 0.05);
109 }
110
111 if any {
112 let rot_matrix = Mat3::from_quat(transform.rotation);
113 transform.translation =
114 pan_orbit.focus + rot_matrix.mul_vec3(Vec3::new(0.0, 0.0, pan_orbit.radius));
115 }
116 }
117
118 ev_motion.clear();
119}