Skip to main content

bevy_post_process/effect_stack/
chromatic_aberration.rs

1use bevy_asset::Handle;
2use bevy_camera::Camera;
3use bevy_ecs::{
4    component::Component,
5    query::{QueryItem, With},
6    reflect::ReflectComponent,
7    resource::Resource,
8    system::lifetimeless::Read,
9};
10use bevy_image::Image;
11use bevy_reflect::{std_traits::ReflectDefault, Reflect};
12use bevy_render::{
13    extract_component::ExtractComponent, render_resource::ShaderType, sync_component::SyncComponent,
14};
15
16/// The raw RGBA data for the default chromatic aberration gradient.
17///
18/// This consists of one red pixel, one green pixel, and one blue pixel, in that
19/// order.
20pub(super) static DEFAULT_CHROMATIC_ABERRATION_LUT_DATA: [u8; 12] =
21    [255, 0, 0, 255, 0, 255, 0, 255, 0, 0, 255, 255];
22
23#[derive(impl bevy_ecs::resource::Resource for DefaultChromaticAberrationLut where
    Self: ::core::marker::Send + ::core::marker::Sync + 'static {}Resource)]
24pub(crate) struct DefaultChromaticAberrationLut(pub(crate) Handle<Image>);
25
26/// Adds colored fringes to the edges of objects in the scene.
27///
28/// [Chromatic aberration] simulates the effect when lenses fail to focus all
29/// colors of light toward a single point. It causes rainbow-colored streaks to
30/// appear, which are especially apparent on the edges of objects. Chromatic
31/// aberration is commonly used for collision effects, especially in horror
32/// games.
33///
34/// Bevy's implementation is based on that of *Inside* ([Gjøl & Svendsen 2016]).
35/// It's based on a customizable lookup texture, which allows for changing the
36/// color pattern. By default, the color pattern is simply a 3×1 pixel texture
37/// consisting of red, green, and blue, in that order, but you can change it to
38/// any image in order to achieve different effects.
39///
40/// [Chromatic aberration]: https://en.wikipedia.org/wiki/Chromatic_aberration
41///
42/// [Gjøl & Svendsen 2016]: https://github.com/playdeadgames/publications/blob/master/INSIDE/rendering_inside_gdc2016.pdf
43#[derive(const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for ChromaticAberration where
            {
            fn get_type_registration() -> bevy_reflect::TypeRegistration {
                let mut registration =
                    bevy_reflect::TypeRegistration::of::<Self>();
                registration.insert::<bevy_reflect::ReflectFromPtr>(bevy_reflect::FromType::<Self>::from_type());
                registration.insert::<bevy_reflect::ReflectFromReflect>(bevy_reflect::FromType::<Self>::from_type());
                registration.register_type_data::<ReflectComponent, Self>();
                registration.register_type_data::<ReflectDefault, Self>();
                registration
            }
            #[inline(never)]
            fn register_type_dependencies(registry:
                    &mut bevy_reflect::TypeRegistry) {
                <Option<Handle<Image>> as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
                <f32 as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
                <u32 as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
            }
        }
        impl bevy_reflect::Typed for ChromaticAberration where  {
            #[inline]
            fn type_info() -> &'static bevy_reflect::TypeInfo {
                static CELL: bevy_reflect::utility::NonGenericTypeInfoCell =
                    bevy_reflect::utility::NonGenericTypeInfoCell::new();
                CELL.get_or_set(||
                        {
                            bevy_reflect::TypeInfo::Struct(bevy_reflect::structs::StructInfo::new::<Self>(&[bevy_reflect::NamedField::new::<Option<Handle<Image>>>("color_lut"),
                                                bevy_reflect::NamedField::new::<f32>("intensity"),
                                                bevy_reflect::NamedField::new::<u32>("max_samples")]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for ChromaticAberration where  {
            fn type_path() -> &'static str {
                "bevy_post_process::effect_stack::chromatic_aberration::ChromaticAberration"
            }
            fn short_type_path() -> &'static str { "ChromaticAberration" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("ChromaticAberration")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_post_process::effect_stack::chromatic_aberration".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_post_process::effect_stack::chromatic_aberration")
            }
        }
        impl bevy_reflect::Reflect for ChromaticAberration where  {
            #[inline]
            fn into_any(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                ->
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn ::core::any::Any> {
                self
            }
            #[inline]
            fn as_any(&self) -> &dyn ::core::any::Any { self }
            #[inline]
            fn as_any_mut(&mut self) -> &mut dyn ::core::any::Any { self }
            #[inline]
            fn into_reflect(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                ->
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect> {
                self
            }
            #[inline]
            fn as_reflect(&self) -> &dyn bevy_reflect::Reflect { self }
            #[inline]
            fn as_reflect_mut(&mut self) -> &mut dyn bevy_reflect::Reflect {
                self
            }
            #[inline]
            fn set(&mut self,
                value:
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>)
                ->
                    ::core::result::Result<(),
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>> {
                *self = <dyn bevy_reflect::Reflect>::take(value)?;
                ::core::result::Result::Ok(())
            }
        }
        #[allow(non_upper_case_globals)]
        const _: () =
            {
                static __INVENTORY: ::inventory::Node =
                    ::inventory::Node {
                        value: &{
                                bevy_reflect::__macro_exports::auto_register::AutomaticReflectRegistrations(<ChromaticAberration
                                        as
                                        bevy_reflect::__macro_exports::auto_register::RegisterForReflection>::__register)
                            },
                        next: ::inventory::__private::UnsafeCell::new(::inventory::__private::Option::None),
                    };
                #[link_section = ".text.startup"]
                unsafe extern "C" fn __ctor() {
                    unsafe {
                        ::inventory::ErasedNode::submit(__INVENTORY.value,
                            &__INVENTORY)
                    }
                }
                #[used]
                #[link_section = ".init_array"]
                static __CTOR: unsafe extern "C" fn() = __ctor;
            };
        impl bevy_reflect::structs::Struct for ChromaticAberration where  {
            fn field(&self, name: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match name {
                    "color_lut" =>
                        ::core::option::Option::Some(&self.color_lut),
                    "intensity" =>
                        ::core::option::Option::Some(&self.intensity),
                    "max_samples" =>
                        ::core::option::Option::Some(&self.max_samples),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_mut(&mut self, name: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match name {
                    "color_lut" =>
                        ::core::option::Option::Some(&mut self.color_lut),
                    "intensity" =>
                        ::core::option::Option::Some(&mut self.intensity),
                    "max_samples" =>
                        ::core::option::Option::Some(&mut self.max_samples),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_at(&self, index: usize)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match index {
                    0usize => ::core::option::Option::Some(&self.color_lut),
                    1usize => ::core::option::Option::Some(&self.intensity),
                    2usize => ::core::option::Option::Some(&self.max_samples),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_at_mut(&mut self, index: usize)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match index {
                    0usize => ::core::option::Option::Some(&mut self.color_lut),
                    1usize => ::core::option::Option::Some(&mut self.intensity),
                    2usize =>
                        ::core::option::Option::Some(&mut self.max_samples),
                    _ => ::core::option::Option::None,
                }
            }
            fn name_at(&self, index: usize) -> ::core::option::Option<&str> {
                match index {
                    0usize => ::core::option::Option::Some("color_lut"),
                    1usize => ::core::option::Option::Some("intensity"),
                    2usize => ::core::option::Option::Some("max_samples"),
                    _ => ::core::option::Option::None,
                }
            }
            fn index_of_name(&self, name: &str)
                -> ::core::option::Option<usize> {
                match name {
                    "color_lut" => ::core::option::Option::Some(0usize),
                    "intensity" => ::core::option::Option::Some(1usize),
                    "max_samples" => ::core::option::Option::Some(2usize),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_len(&self) -> usize { 3usize }
            fn iter_fields(&self) -> bevy_reflect::structs::FieldIter {
                bevy_reflect::structs::FieldIter::new(self)
            }
            fn to_dynamic_struct(&self)
                -> bevy_reflect::structs::DynamicStruct {
                let mut dynamic: bevy_reflect::structs::DynamicStruct =
                    ::core::default::Default::default();
                dynamic.set_represented_type(bevy_reflect::PartialReflect::get_represented_type_info(self));
                dynamic.insert_boxed("color_lut",
                    bevy_reflect::PartialReflect::to_dynamic(&self.color_lut));
                dynamic.insert_boxed("intensity",
                    bevy_reflect::PartialReflect::to_dynamic(&self.intensity));
                dynamic.insert_boxed("max_samples",
                    bevy_reflect::PartialReflect::to_dynamic(&self.max_samples));
                dynamic
            }
        }
        impl bevy_reflect::PartialReflect for ChromaticAberration where  {
            #[inline]
            fn get_represented_type_info(&self)
                -> ::core::option::Option<&'static bevy_reflect::TypeInfo> {
                ::core::option::Option::Some(<Self as
                            bevy_reflect::Typed>::type_info())
            }
            #[inline]
            fn try_apply(&mut self, value: &dyn bevy_reflect::PartialReflect)
                -> ::core::result::Result<(), bevy_reflect::ApplyError> {
                if let bevy_reflect::ReflectRef::Struct(struct_value) =
                        bevy_reflect::PartialReflect::reflect_ref(value) {
                    for (name, value) in
                        bevy_reflect::structs::Struct::iter_fields(struct_value) {
                        if let ::core::option::Option::Some(v) =
                                bevy_reflect::structs::Struct::field_mut(self, name) {
                            bevy_reflect::PartialReflect::try_apply(v, value)?;
                        }
                    }
                } else {
                    return ::core::result::Result::Err(bevy_reflect::ApplyError::MismatchedKinds {
                                from_kind: bevy_reflect::PartialReflect::reflect_kind(value),
                                to_kind: bevy_reflect::ReflectKind::Struct,
                            });
                }
                ::core::result::Result::Ok(())
            }
            #[inline]
            fn reflect_kind(&self) -> bevy_reflect::ReflectKind {
                bevy_reflect::ReflectKind::Struct
            }
            #[inline]
            fn reflect_ref(&self) -> bevy_reflect::ReflectRef {
                bevy_reflect::ReflectRef::Struct(self)
            }
            #[inline]
            fn reflect_mut(&mut self) -> bevy_reflect::ReflectMut {
                bevy_reflect::ReflectMut::Struct(self)
            }
            #[inline]
            fn reflect_owned(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                -> bevy_reflect::ReflectOwned {
                bevy_reflect::ReflectOwned::Struct(self)
            }
            #[inline]
            fn try_into_reflect(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                ->
                    ::core::result::Result<bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>,
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::PartialReflect>> {
                ::core::result::Result::Ok(self)
            }
            #[inline]
            fn try_as_reflect(&self)
                -> ::core::option::Option<&dyn bevy_reflect::Reflect> {
                ::core::option::Option::Some(self)
            }
            #[inline]
            fn try_as_reflect_mut(&mut self)
                -> ::core::option::Option<&mut dyn bevy_reflect::Reflect> {
                ::core::option::Option::Some(self)
            }
            #[inline]
            fn into_partial_reflect(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                ->
                    bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::PartialReflect> {
                self
            }
            #[inline]
            fn as_partial_reflect(&self)
                -> &dyn bevy_reflect::PartialReflect {
                self
            }
            #[inline]
            fn as_partial_reflect_mut(&mut self)
                -> &mut dyn bevy_reflect::PartialReflect {
                self
            }
            fn reflect_partial_eq(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<bool> {
                (bevy_reflect::structs::struct_partial_eq)(self, value)
            }
            fn reflect_partial_cmp(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<::core::cmp::Ordering> {
                (bevy_reflect::structs::struct_partial_cmp)(self, value)
            }
            #[inline]
            fn reflect_clone(&self)
                ->
                    ::core::result::Result<bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>,
                    bevy_reflect::ReflectCloneError> {
                ::core::result::Result::Ok(bevy_reflect::__macro_exports::alloc_utils::Box::new(::core::clone::Clone::clone(self)))
            }
        }
        impl bevy_reflect::FromReflect for ChromaticAberration where  {
            fn from_reflect(reflect: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<Self> {
                if let bevy_reflect::ReflectRef::Struct(__ref_struct) =
                        bevy_reflect::PartialReflect::reflect_ref(reflect) {
                    let mut __this =
                        <Self as ::core::default::Default>::default();
                    if let ::core::option::Option::Some(__field) =
                            (||
                                        <Option<Handle<Image>> as
                                                bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                                    "color_lut")?))() {
                        __this.color_lut = __field;
                    }
                    if let ::core::option::Option::Some(__field) =
                            (||
                                        <f32 as
                                                bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                                    "intensity")?))() {
                        __this.intensity = __field;
                    }
                    if let ::core::option::Option::Some(__field) =
                            (||
                                        <u32 as
                                                bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                                    "max_samples")?))() {
                        __this.max_samples = __field;
                    }
                    ::core::option::Option::Some(__this)
                } else { ::core::option::Option::None }
            }
        }
    };Reflect, impl bevy_ecs::component::Component for ChromaticAberration where
    Self: ::core::marker::Send + ::core::marker::Sync + 'static {
    const STORAGE_TYPE: bevy_ecs::component::StorageType =
        bevy_ecs::component::StorageType::Table;
    type Mutability = bevy_ecs::component::Mutable;
    fn register_required_components(_requiree:
            bevy_ecs::component::ComponentId,
        required_components:
            &mut bevy_ecs::component::RequiredComponentsRegistrator) {}
    fn clone_behavior() -> bevy_ecs::component::ComponentCloneBehavior {
        use bevy_ecs::component::{
            DefaultCloneBehaviorBase, DefaultCloneBehaviorViaClone,
        };
        (&&&bevy_ecs::component::DefaultCloneBehaviorSpecialization::<Self>::default()).default_clone_behavior()
    }
    fn relationship_accessor()
        ->
            ::core::option::Option<bevy_ecs::relationship::ComponentRelationshipAccessor<Self>> {
        ::core::option::Option::None
    }
}Component, #[automatically_derived]
impl ::core::clone::Clone for ChromaticAberration {
    #[inline]
    fn clone(&self) -> ChromaticAberration {
        ChromaticAberration {
            color_lut: ::core::clone::Clone::clone(&self.color_lut),
            intensity: ::core::clone::Clone::clone(&self.intensity),
            max_samples: ::core::clone::Clone::clone(&self.max_samples),
        }
    }
}Clone)]
44#[reflect(Component, Default, Clone)]
45pub struct ChromaticAberration {
46    /// The lookup texture that determines the color gradient.
47    ///
48    /// By default (if None), this is a 3×1 texel texture consisting of one red
49    /// pixel, one green pixel, and one blue texel, in that order. This
50    /// recreates the most typical chromatic aberration pattern. However, you
51    /// can change it to achieve different artistic effects.
52    ///
53    /// The texture is always sampled in its vertical center, so it should
54    /// ordinarily have a height of 1 texel.
55    pub color_lut: Option<Handle<Image>>,
56
57    /// The size of the streaks around the edges of objects, as a fraction of
58    /// the window size.
59    ///
60    /// The default value is 0.02.
61    pub intensity: f32,
62
63    /// A cap on the number of texture samples that will be performed.
64    ///
65    /// Higher values result in smoother-looking streaks but are slower.
66    ///
67    /// The default value is 8.
68    pub max_samples: u32,
69}
70
71impl Default for ChromaticAberration {
72    fn default() -> Self {
73        Self {
74            color_lut: None,
75            intensity: 0.02,
76            max_samples: 8,
77        }
78    }
79}
80
81impl SyncComponent for ChromaticAberration {
82    type Target = Self;
83}
84
85impl ExtractComponent for ChromaticAberration {
86    type QueryData = Read<ChromaticAberration>;
87    type QueryFilter = With<Camera>;
88    type Out = Self;
89
90    fn extract_component(
91        chromatic_aberration: QueryItem<'_, '_, Self::QueryData>,
92    ) -> Option<Self::Out> {
93        // Skip the postprocessing phase entirely if the intensity is negligible.
94        if chromatic_aberration.intensity > 1e-4 {
95            Some(chromatic_aberration.clone())
96        } else {
97            None
98        }
99    }
100}
101
102/// The on-GPU version of the [`ChromaticAberration`] settings.
103///
104/// See the documentation for [`ChromaticAberration`] for more information on
105/// each of these fields.
106#[derive(impl bevy_render::render_resource::encase::private::ShaderSize for
    ChromaticAberrationUniform where
    f32: bevy_render::render_resource::encase::private::ShaderSize,
    u32: bevy_render::render_resource::encase::private::ShaderSize,
    u32: bevy_render::render_resource::encase::private::ShaderSize,
    u32: bevy_render::render_resource::encase::private::ShaderSize {}ShaderType, #[automatically_derived]
impl ::core::default::Default for ChromaticAberrationUniform {
    #[inline]
    fn default() -> ChromaticAberrationUniform {
        ChromaticAberrationUniform {
            intensity: ::core::default::Default::default(),
            max_samples: ::core::default::Default::default(),
            unused_1: ::core::default::Default::default(),
            unused_2: ::core::default::Default::default(),
        }
    }
}Default)]
107pub struct ChromaticAberrationUniform {
108    pub(super) intensity: f32,
109    pub(super) max_samples: u32,
110    pub(super) unused_1: u32,
111    pub(super) unused_2: u32,
112}