1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
//! Utility to adjust the aspect ratio of cameras automatically

use amethyst_assets::PrefabData;
use amethyst_core::{
    ecs::{
        Component, Entity, HashMapStorage, Join, ReadExpect, ReadStorage, System, SystemData,
        World, WriteStorage,
    },
    SystemDesc,
};
use amethyst_derive::{PrefabData, SystemDesc};
use amethyst_error::Error;
use amethyst_rendy::camera::Camera;
use amethyst_window::ScreenDimensions;

use serde::{Deserialize, Serialize};

#[cfg(feature = "profiler")]
use thread_profiler::profile_scope;

/// A component describing the behavior of the camera in accordance with the screen dimensions
#[derive(Clone, Debug, Deserialize, PrefabData, Serialize)]
#[prefab(Component)]
#[serde(default)]
pub struct AutoFov {
    /// The horizontal FOV value at the aspect ratio in the field `base_aspect_ratio`
    base_fovx: f32,

    /// The factor determining how sensitive the FOV change should be
    fovx_growth_rate: f32,

    /// If the FOV grow rate specified in the field `fovx_growth_rate` should be applied as-is
    fixed_growth_rate: bool,

    /// The aspect ratio when the camera's horizontal FOV is identical to `base_fovx`
    base_aspect_ratio: (usize, usize),

    /// The minimum value the horizontal FOV can have
    min_fovx: f32,

    /// The maximum value the horizontal FOV can have
    max_fovx: f32,
}

impl AutoFov {
    /// Creates a new instance with the default values for all fields
    pub fn new() -> Self {
        Default::default()
    }

    /// The horizontal FOV value at the aspect ratio in the field `base_aspect_ratio`
    ///
    /// This value should be between the `min_fovx` and `max_fovx` values. Value in radians.
    /// Defaults to `1.861684535`, which is the horizontal FOV value when the vertial FOV and aspect
    /// ratio for a camera is set to `1.0471975512`(60 deg) and `16/9`, respectively.
    pub fn base_fovx(&self) -> f32 {
        self.base_fovx
    }

    /// The factor determining how sensitive the FOV change should be
    ///
    /// Defaults to `1.0`.
    pub fn fovx_growth_rate(&self) -> f32 {
        self.fovx_growth_rate
    }

    /// If the FOV grow rate specified in the field `fovx_growth_rate` should be applied as-is
    ///
    /// Defaults to `false`. When `false`, the `fovx_growth_rate` field is multiplied with the
    /// camera's current vertical FOV value.
    pub fn fixed_growth_rate(&self) -> bool {
        self.fixed_growth_rate
    }

    /// The aspect ratio when the camera's horizontal FOV is identical to `base_fovx`
    ///
    /// Defaults to `(16, 9)`.
    pub fn base_aspect_ratio(&self) -> (usize, usize) {
        self.base_aspect_ratio
    }

    /// The minimum value the horizontal FOV can have
    ///
    /// This value should be larger than 0. Defaults to `0.1`.
    pub fn min_fovx(&self) -> f32 {
        self.min_fovx
    }

    /// The maximum value the horizontal FOV can have
    ///
    /// This value should be larger than 0. Defaults to `PI`. The rendered view will be stretched if
    /// the screen aspect ratio keeps growing after the point where the camera's horizontal FOV
    /// reaches this maximum value.
    pub fn max_fovx(&self) -> f32 {
        self.max_fovx
    }

    /// Sets `base_fovx` to the given value
    ///
    /// This function panics if the given value is not between `min_fovx` and `max_fovx`.
    pub fn set_base_fovx(&mut self, base_fovx: f32) {
        assert!(
            base_fovx >= self.min_fovx,
            format!(
                "`base_fovx` should be larger than `min_fovx` which is `{}`, but `{}` given",
                self.min_fovx, base_fovx
            )
        );
        assert!(
            base_fovx <= self.max_fovx,
            format!(
                "`base_fovx` should be smaller than `max_fovx` which is `{}`, but `{}` given",
                self.max_fovx, base_fovx
            )
        );

        self.base_fovx = base_fovx;
    }

    /// Sets `fovx_growth_rate` to the given value
    pub fn set_fovx_growth_rate(&mut self, fovx_growth_rate: f32) {
        self.fovx_growth_rate = fovx_growth_rate;
    }

    /// Sets `fixed_growth_rate` to the given value
    ///
    /// You can optionally give the new `fovx_growth_rate` value to be set.
    pub fn set_fixed_growth_rate(&mut self, fix: bool, new_growth_rate: Option<f32>) {
        self.fixed_growth_rate = fix;

        if let Some(fovx_growth_rate) = new_growth_rate {
            self.fovx_growth_rate = fovx_growth_rate;
        }
    }

    /// Sets `base_aspect_ratio` to the given value
    ///
    /// This function panics if the horizontal or vertical ratio value is zero.
    pub fn set_base_aspect_ratio(&mut self, horizontal: usize, vertical: usize) {
        assert!(
            horizontal != 0,
            "The horizontal value of aspect ratio should be larger than 0"
        );
        assert!(
            vertical != 0,
            "The vertical value of aspect ratio should be larger than 0"
        );
        self.base_aspect_ratio = (horizontal, vertical);
    }

    /// Sets `min_fovx` to the given value
    ///
    /// This function panics if the given `min_fovx` is not larger than zero or is larger than
    /// `max_fovx`.
    pub fn set_min(&mut self, min: f32) {
        let max = self.max_fovx;
        self.set_min_max(min, max);
    }

    /// Sets `max_fovx` to the given value
    ///
    /// This function panics if the given `max_fovx` is smaller than `min_fovx`.
    pub fn set_max(&mut self, max: f32) {
        let min = self.min_fovx;
        self.set_min_max(min, max);
    }

    /// Sets `min_fovx` and `max_fovx` to the given vaues
    ///
    /// This function panics if the given `min_fovx` is not larger than zero, or if the given
    /// `max_fovx` is smaller than the given `min_fovx`.
    pub fn set_min_max(&mut self, min: f32, max: f32) {
        assert!(
            min > 0.0,
            format!("`min_fovx` should be larger than 0, but `{}` given", min)
        );
        assert!(
            max >= min,
            format!(
                "`max_fovx` should be larger than or equal to `min_fovx` which is `{}`, but `{}` \
                 given",
                min, max,
            ),
        );
        self.min_fovx = min;
        self.max_fovx = max;
    }

    /// Computes the new horizontal FOV from the current screen aspect ratio and vertical FOV
    pub fn new_fovx(&self, current_aspect_ratio: f32, fovy: f32) -> f32 {
        let delta_aspect = current_aspect_ratio - self.base_aspect_value();

        if delta_aspect.abs() <= ::std::f32::EPSILON {
            return self.base_fovx;
        }

        let fovy = if self.fixed_growth_rate { 1.0 } else { fovy };
        let delta_fovx = self.fovx_growth_rate * fovy * delta_aspect;
        let new_fovx = self.base_fovx + delta_fovx;

        new_fovx.max(self.min_fovx).min(self.max_fovx)
    }

    #[inline]
    fn base_aspect_value(&self) -> f32 {
        self.base_aspect_ratio.0 as f32 / self.base_aspect_ratio.1 as f32
    }
}

impl Component for AutoFov {
    type Storage = HashMapStorage<Self>;
}

impl Default for AutoFov {
    fn default() -> Self {
        AutoFov {
            // This is actually 1.861_684_535, but float precision is lost beyond this
            base_fovx: 1.861_684_6,
            fovx_growth_rate: 1.0,
            fixed_growth_rate: false,
            base_aspect_ratio: (16, 9),
            min_fovx: 0.1,
            max_fovx: std::f32::consts::PI,
        }
    }
}

/// System that automatically adjusts the horizontal FOV based on the screen dimensions
///
/// For a camera component to be managed by this system, the entity with the camera component should
/// also have an `AutoFov` component attached to it.
///
/// If the camera is being loaded by a prefab, it is best to have the `PrefabLoaderSystem` loading
/// the camera as a dependency of this system. It enables the system to adjust the camera right
/// after it is created -- simply put, in the same frame.
#[derive(Debug, SystemDesc)]
pub struct AutoFovSystem {
    last_dimensions: ScreenDimensions,
}

impl AutoFovSystem {
    /// Sets up `SystemData` and returns a new `AutoFovSystem`.
    pub fn new() -> Self {
        Self {
            last_dimensions: ScreenDimensions::new(0, 0, 0.0),
        }
    }
}

impl<'a> System<'a> for AutoFovSystem {
    type SystemData = (
        ReadExpect<'a, ScreenDimensions>,
        ReadStorage<'a, AutoFov>,
        WriteStorage<'a, Camera>,
    );

    fn run(&mut self, (screen, auto_fovs, mut cameras): Self::SystemData) {
        #[cfg(feature = "profiler")]
        profile_scope!("auto_fov_system");

        if self.last_dimensions != *screen {
            for (camera, auto_fov) in (&mut cameras, &auto_fovs).join() {
                if let Some(perspective) = camera.projection_mut().as_perspective_mut() {
                    let fovy = perspective.fovy();
                    let fovx = auto_fov.new_fovx(screen.aspect_ratio(), fovy);
                    perspective.set_aspect(fovx / fovy);
                }
            }
            self.last_dimensions = screen.clone();
        }
    }
}

impl Default for AutoFovSystem {
    fn default() -> Self {
        Self {
            last_dimensions: ScreenDimensions::new(0, 0, 0.0),
        }
    }
}