Skip to main content

bevy_feathers/controls/
toggle_switch.rs

1use accesskit::Role;
2use bevy_a11y::AccessibilityNode;
3use bevy_app::{Plugin, PreUpdate};
4use bevy_ecs::{
5    bundle::Bundle,
6    children,
7    component::Component,
8    entity::Entity,
9    hierarchy::Children,
10    lifecycle::RemovedComponents,
11    query::{Added, Changed, Has, Or, With},
12    reflect::ReflectComponent,
13    schedule::IntoScheduleConfigs,
14    system::{Commands, Query},
15    world::Mut,
16};
17use bevy_input_focus::tab_navigation::TabIndex;
18use bevy_picking::{hover::Hovered, PickingSystems};
19use bevy_reflect::{prelude::ReflectDefault, Reflect};
20use bevy_scene::prelude::*;
21use bevy_ui::{
22    percent, px, BorderRadius, Checked, InteractionDisabled, Node, PositionType, Pressed, UiRect,
23};
24use bevy_ui_widgets::{ActivateOnPress, Checkbox};
25
26use crate::{
27    constants::size,
28    cursor::EntityCursor,
29    focus::FocusIndicator,
30    theme::{ThemeBackgroundColor, ThemeBorderColor},
31    tokens,
32};
33
34/// A toggle switch widget.
35///
36/// This is spawnable by inheriting it as a "scene component".
37///
38/// # Emitted events
39/// * [`bevy_ui_widgets::ValueChange<bool>`] with the new value when the toggle switch changes state.
40///
41/// These events can be disabled by adding an [`bevy_ui::InteractionDisabled`] component to the bundle
42#[derive(impl bevy_scene::SceneComponent for FeathersToggleSwitch where
    Self: ::core::marker::Send + ::core::marker::Sync + 'static {
    type Props = ();
    fn scene(props: Self::Props) -> impl bevy_scene::Scene {
        (Self::scene(),
            <bevy_scene::InitTemplate<<FeathersToggleSwitch as
                    bevy_ecs::template::FromTemplate>::Template> as
                    ::core::default::Default>::default(),
            bevy_scene::template_value(bevy_scene::SceneComponentInfo::new::<FeathersToggleSwitch>(true)))
    }
}SceneComponent, #[automatically_derived]
impl ::core::default::Default for FeathersToggleSwitch {
    #[inline]
    fn default() -> FeathersToggleSwitch { FeathersToggleSwitch {} }
}Default, #[automatically_derived]
impl ::core::clone::Clone for FeathersToggleSwitch {
    #[inline]
    fn clone(&self) -> FeathersToggleSwitch { FeathersToggleSwitch }
}Clone, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for FeathersToggleSwitch 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) {}
        }
        impl bevy_reflect::Typed for FeathersToggleSwitch 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>(&[]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for FeathersToggleSwitch where  {
            fn type_path() -> &'static str {
                "bevy_feathers::controls::toggle_switch::FeathersToggleSwitch"
            }
            fn short_type_path() -> &'static str { "FeathersToggleSwitch" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("FeathersToggleSwitch")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_feathers::controls::toggle_switch".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_feathers::controls::toggle_switch")
            }
        }
        impl bevy_reflect::Reflect for FeathersToggleSwitch 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(<FeathersToggleSwitch
                                        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 FeathersToggleSwitch where  {
            fn field(&self, name: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_mut(&mut self, name: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_at(&self, index: usize)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match index { _ => ::core::option::Option::None, }
            }
            fn field_at_mut(&mut self, index: usize)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match index { _ => ::core::option::Option::None, }
            }
            fn name_at(&self, index: usize) -> ::core::option::Option<&str> {
                match index { _ => ::core::option::Option::None, }
            }
            fn index_of_name(&self, name: &str)
                -> ::core::option::Option<usize> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_len(&self) -> usize { 0usize }
            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
            }
        }
        impl bevy_reflect::PartialReflect for FeathersToggleSwitch 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 FeathersToggleSwitch 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();
                    ::core::option::Option::Some(__this)
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
43#[reflect(Component, Clone, Default)]
44pub struct FeathersToggleSwitch;
45
46impl FeathersToggleSwitch {
47    fn scene() -> impl Scene {
48        bevy_scene::SceneScope({
        let _expr0 = { size::TOGGLE_WIDTH }.into();
        let _expr1 = { size::TOGGLE_HEIGHT }.into();
        let _expr2 = { px(2) }.into();
        let _expr3 = { px(5) }.into();
        let _expr4 = { tokens::SWITCH_BG }.into();
        let _expr5 = { tokens::SWITCH_BORDER }.into();
        let _expr6 = { accesskit::Node::new(Role::Switch) }.into();
        let _expr7 = { percent(0) }.into();
        let _expr8 = { px(0) }.into();
        let _expr9 = { px(0) }.into();
        let _expr10 = { percent(50) }.into();
        let _expr11 = { px(2) }.into();
        let _expr12 = { px(3) }.into();
        let _expr13 = { tokens::SWITCH_SLIDE_BG }.into();
        let _expr14 = { tokens::SWITCH_SLIDE_BORDER }.into();
        let _res =
            (bevy_scene::SceneFunction(move |_context, _scene|
                        {
                            let __value =
                                _scene.get_or_insert_template::<<Node as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            __value.width = _expr0;
                            __value.height = _expr1;
                            __value.border = _expr2;
                            __value.border_radius = _expr3;
                            let _ =
                                _scene.get_or_insert_template::<<Checkbox as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            let _ =
                                _scene.get_or_insert_template::<<FeathersToggleSwitch as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            let __value =
                                _scene.get_or_insert_template::<<ThemeBackgroundColor as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            __value.0 = _expr4;
                            let __value =
                                _scene.get_or_insert_template::<<ThemeBorderColor as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            __value.0 = _expr5;
                            let __value =
                                _scene.get_or_insert_template::<<AccessibilityNode as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            __value.0 = _expr6;
                            let _ =
                                _scene.get_or_insert_template::<<Hovered as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            let __value =
                                _scene.get_or_insert_template::<<EntityCursor as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            {
                                let _node = __value;
                                if !#[allow(non_exhaustive_omitted_patterns)] match _node {
                                            bevy_scene::macro_utils::PathResolveHelper::<<EntityCursor
                                                as bevy_ecs::template::FromTemplate>::Template>::System(..)
                                                => true,
                                            _ => false,
                                        } {
                                    *_node =
                                        bevy_scene::macro_utils::PathResolveHelper::<<EntityCursor
                                                as
                                                bevy_ecs::template::FromTemplate>::Template>::default_system();
                                }
                                if let bevy_scene::macro_utils::PathResolveHelper::<<EntityCursor
                                        as
                                        bevy_ecs::template::FromTemplate>::Template>::System(t0, ..)
                                        = _node {
                                    *t0 = (bevy_window::SystemCursorIcon::Pointer {}).into();
                                }
                            }
                            let __value =
                                _scene.get_or_insert_template::<<TabIndex as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                            __value.0 = 0;
                            let _ =
                                _scene.get_or_insert_template::<<FocusIndicator as
                                    bevy_ecs::template::FromTemplate>::Template>(_context);
                        }),
                bevy_scene::RelatedScenes::<<Children as
                        bevy_ecs::relationship::RelationshipTarget>::Relationship,
                        _>::new(bevy_scene::EntityScene(bevy_scene::SceneFunction(move
                                |_context, _scene|
                                {
                                    let __value =
                                        _scene.get_or_insert_template::<<Node as
                                            bevy_ecs::template::FromTemplate>::Template>(_context);
                                    __value.position_type = (PositionType::Absolute {}).into();
                                    __value.left = _expr7;
                                    __value.top = _expr8;
                                    __value.bottom = _expr9;
                                    __value.width = _expr10;
                                    __value.border = _expr11;
                                    __value.border_radius = _expr12;
                                    let _ =
                                        _scene.get_or_insert_template::<<ToggleSwitchSlide as
                                            bevy_ecs::template::FromTemplate>::Template>(_context);
                                    let __value =
                                        _scene.get_or_insert_template::<<ThemeBackgroundColor as
                                            bevy_ecs::template::FromTemplate>::Template>(_context);
                                    __value.0 = _expr13;
                                    let __value =
                                        _scene.get_or_insert_template::<<ThemeBorderColor as
                                            bevy_ecs::template::FromTemplate>::Template>(_context);
                                    __value.0 = _expr14;
                                }))));
        _res
    });bsn! {
49            Node {
50                width: size::TOGGLE_WIDTH,
51                height: size::TOGGLE_HEIGHT,
52                border: px(2),
53                border_radius: px(5),
54            }
55            Checkbox
56            FeathersToggleSwitch
57            ThemeBackgroundColor(tokens::SWITCH_BG)
58            ThemeBorderColor(tokens::SWITCH_BORDER)
59            AccessibilityNode(accesskit::Node::new(Role::Switch))
60            Hovered
61            EntityCursor::System(bevy_window::SystemCursorIcon::Pointer)
62            TabIndex(0)
63            FocusIndicator
64            Children [(
65                Node {
66                    position_type: PositionType::Absolute,
67                    left: percent(0),
68                    top: px(0),
69                    bottom: px(0),
70                    width: percent(50),
71                    border: px(2),
72                    border_radius: px(3),
73                }
74                ToggleSwitchSlide
75                ThemeBackgroundColor(tokens::SWITCH_SLIDE_BG)
76                ThemeBorderColor(tokens::SWITCH_SLIDE_BORDER)
77            )]
78        }
79    }
80}
81
82/// Marker for the toggle switch slide
83#[derive(impl bevy_ecs::component::Component for ToggleSwitchSlide 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::default::Default for ToggleSwitchSlide {
    #[inline]
    fn default() -> ToggleSwitchSlide { ToggleSwitchSlide {} }
}Default, #[automatically_derived]
impl ::core::clone::Clone for ToggleSwitchSlide {
    #[inline]
    fn clone(&self) -> ToggleSwitchSlide { ToggleSwitchSlide }
}Clone, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for ToggleSwitchSlide 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) {}
        }
        impl bevy_reflect::Typed for ToggleSwitchSlide 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>(&[]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for ToggleSwitchSlide where  {
            fn type_path() -> &'static str {
                "bevy_feathers::controls::toggle_switch::ToggleSwitchSlide"
            }
            fn short_type_path() -> &'static str { "ToggleSwitchSlide" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("ToggleSwitchSlide")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_feathers::controls::toggle_switch".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_feathers::controls::toggle_switch")
            }
        }
        impl bevy_reflect::Reflect for ToggleSwitchSlide 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(<ToggleSwitchSlide
                                        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 ToggleSwitchSlide where  {
            fn field(&self, name: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_mut(&mut self, name: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_at(&self, index: usize)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match index { _ => ::core::option::Option::None, }
            }
            fn field_at_mut(&mut self, index: usize)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match index { _ => ::core::option::Option::None, }
            }
            fn name_at(&self, index: usize) -> ::core::option::Option<&str> {
                match index { _ => ::core::option::Option::None, }
            }
            fn index_of_name(&self, name: &str)
                -> ::core::option::Option<usize> {
                match name { _ => ::core::option::Option::None, }
            }
            fn field_len(&self) -> usize { 0usize }
            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
            }
        }
        impl bevy_reflect::PartialReflect for ToggleSwitchSlide 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 ToggleSwitchSlide 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();
                    ::core::option::Option::Some(__this)
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
84#[reflect(Component, Clone, Default)]
85struct ToggleSwitchSlide;
86
87/// Template function to spawn a toggle switch.
88///
89/// # Arguments
90/// * `props` - construction properties for the toggle switch.
91/// * `overrides` - a bundle of components that are merged in with the normal toggle switch components.
92///
93/// # Emitted events
94/// * [`bevy_ui_widgets::ValueChange<bool>`] with the new value when the toggle switch changes state.
95///
96/// These events can be disabled by adding an [`bevy_ui::InteractionDisabled`] component to the bundle
97#[deprecated(since = "0.19.0", note = "Use the toggle_switch() BSN function")]
98pub fn toggle_switch_bundle<B: Bundle>(overrides: B) -> impl Bundle {
99    (
100        Node {
101            width: size::TOGGLE_WIDTH,
102            height: size::TOGGLE_HEIGHT,
103            border: UiRect::all(px(2)),
104            border_radius: BorderRadius::all(px(5)),
105            ..Default::default()
106        },
107        Checkbox,
108        FeathersToggleSwitch,
109        ThemeBackgroundColor(tokens::SWITCH_BG),
110        ThemeBorderColor(tokens::SWITCH_BORDER),
111        AccessibilityNode(accesskit::Node::new(Role::Switch)),
112        Hovered::default(),
113        EntityCursor::System(bevy_window::SystemCursorIcon::Pointer),
114        TabIndex(0),
115        FocusIndicator,
116        overrides,
117        <::bevy_ecs::hierarchy::Children as
        ::bevy_ecs::spawn::SpawnRelated>::spawn(::bevy_ecs::spawn::Spawn((Node {
                position_type: PositionType::Absolute,
                left: percent(0),
                top: px(0),
                bottom: px(0),
                width: percent(50),
                border: UiRect::all(px(2)),
                border_radius: BorderRadius::all(px(3)),
                ..Default::default()
            }, ToggleSwitchSlide,
            ThemeBackgroundColor(tokens::SWITCH_SLIDE_BG),
            ThemeBorderColor(tokens::SWITCH_SLIDE_BORDER))))children![(
118            Node {
119                position_type: PositionType::Absolute,
120                left: percent(0),
121                top: px(0),
122                bottom: px(0),
123                width: percent(50),
124                border: UiRect::all(px(2)),
125                border_radius: BorderRadius::all(px(3)),
126                ..Default::default()
127            },
128            ToggleSwitchSlide,
129            ThemeBackgroundColor(tokens::SWITCH_SLIDE_BG),
130            ThemeBorderColor(tokens::SWITCH_SLIDE_BORDER)
131        )],
132    )
133}
134
135fn update_switch_styles(
136    q_switches: Query<
137        (
138            Entity,
139            Has<InteractionDisabled>,
140            Has<Checked>,
141            Has<Pressed>,
142            Has<ActivateOnPress>,
143            &Hovered,
144            &ThemeBackgroundColor,
145            &ThemeBorderColor,
146        ),
147        (
148            With<FeathersToggleSwitch>,
149            Or<(
150                Changed<Hovered>,
151                Added<Checked>,
152                Added<Pressed>,
153                Added<InteractionDisabled>,
154            )>,
155        ),
156    >,
157    q_children: Query<&Children>,
158    mut q_slide: Query<
159        (&mut Node, &ThemeBackgroundColor, &ThemeBorderColor),
160        With<ToggleSwitchSlide>,
161    >,
162    mut commands: Commands,
163) {
164    for (
165        switch_ent,
166        disabled,
167        checked,
168        pressed,
169        activate_on_press,
170        hovered,
171        outline_bg,
172        outline_border,
173    ) in q_switches.iter()
174    {
175        let Some(slide_ent) = q_children
176            .iter_descendants(switch_ent)
177            .find(|en| q_slide.contains(*en))
178        else {
179            continue;
180        };
181        // Safety: since we just checked the query, should always work.
182        let (ref mut slide_style, slide_bg_color, slide_border_color) =
183            q_slide.get_mut(slide_ent).unwrap();
184        set_switch_styles(
185            switch_ent,
186            slide_ent,
187            disabled,
188            checked,
189            pressed,
190            hovered.0,
191            activate_on_press,
192            outline_bg,
193            outline_border,
194            slide_style,
195            slide_bg_color,
196            slide_border_color,
197            &mut commands,
198        );
199    }
200}
201
202fn update_switch_styles_remove(
203    q_switches: Query<
204        (
205            Entity,
206            Has<InteractionDisabled>,
207            Has<Checked>,
208            Has<Pressed>,
209            Has<ActivateOnPress>,
210            &Hovered,
211            &ThemeBackgroundColor,
212            &ThemeBorderColor,
213        ),
214        With<FeathersToggleSwitch>,
215    >,
216    q_children: Query<&Children>,
217    mut q_slide: Query<
218        (&mut Node, &ThemeBackgroundColor, &ThemeBorderColor),
219        With<ToggleSwitchSlide>,
220    >,
221    mut removed_disabled: RemovedComponents<InteractionDisabled>,
222    mut removed_checked: RemovedComponents<Checked>,
223    mut remove_pressed: RemovedComponents<Pressed>,
224    mut remove_activate_on_press: RemovedComponents<ActivateOnPress>,
225    mut commands: Commands,
226) {
227    removed_disabled
228        .read()
229        .chain(removed_checked.read())
230        .chain(remove_pressed.read())
231        .chain(remove_activate_on_press.read())
232        .for_each(|ent| {
233            if let Ok((
234                switch_ent,
235                disabled,
236                checked,
237                pressed,
238                activate_on_press,
239                hovered,
240                outline_bg,
241                outline_border,
242            )) = q_switches.get(ent)
243            {
244                let Some(slide_ent) = q_children
245                    .iter_descendants(switch_ent)
246                    .find(|en| q_slide.contains(*en))
247                else {
248                    return;
249                };
250                // Safety: since we just checked the query, should always work.
251                let (ref mut slide_style, slide_bg_color, slide_border_color) =
252                    q_slide.get_mut(slide_ent).unwrap();
253                set_switch_styles(
254                    switch_ent,
255                    slide_ent,
256                    disabled,
257                    checked,
258                    pressed,
259                    hovered.0,
260                    activate_on_press,
261                    outline_bg,
262                    outline_border,
263                    slide_style,
264                    slide_bg_color,
265                    slide_border_color,
266                    &mut commands,
267                );
268            }
269        });
270}
271
272fn set_switch_styles(
273    switch_ent: Entity,
274    slide_ent: Entity,
275    disabled: bool,
276    checked: bool,
277    pressed: bool,
278    hovered: bool,
279    activate_on_press: bool,
280    outline_bg: &ThemeBackgroundColor,
281    outline_border: &ThemeBorderColor,
282    slide_style: &mut Mut<Node>,
283    slide_bg_color: &ThemeBackgroundColor,
284    slide_border_color: &ThemeBorderColor,
285    commands: &mut Commands,
286) {
287    let outline_border_token = if checked {
288        if disabled {
289            tokens::SWITCH_BORDER_CHECKED_DISABLED
290        } else if pressed && !activate_on_press {
291            tokens::SWITCH_BORDER_CHECKED_PRESSED
292        } else if hovered {
293            tokens::SWITCH_BORDER_CHECKED_HOVER
294        } else {
295            tokens::SWITCH_BORDER_CHECKED
296        }
297    } else {
298        if disabled {
299            tokens::SWITCH_BORDER_DISABLED
300        } else if pressed && !activate_on_press {
301            tokens::SWITCH_BORDER_PRESSED
302        } else if hovered {
303            tokens::SWITCH_BORDER_HOVER
304        } else {
305            tokens::SWITCH_BORDER
306        }
307    };
308
309    let outline_bg_token = if checked {
310        if disabled {
311            tokens::SWITCH_BG_CHECKED_DISABLED
312        } else if pressed && !activate_on_press {
313            tokens::SWITCH_BG_CHECKED_PRESSED
314        } else if hovered {
315            tokens::SWITCH_BG_CHECKED_HOVER
316        } else {
317            tokens::SWITCH_BG_CHECKED
318        }
319    } else {
320        if disabled {
321            tokens::SWITCH_BG_DISABLED
322        } else if pressed && !activate_on_press {
323            tokens::SWITCH_BG_PRESSED
324        } else if hovered {
325            tokens::SWITCH_BG_HOVER
326        } else {
327            tokens::SWITCH_BG
328        }
329    };
330
331    let slide_border_token = if checked {
332        if disabled {
333            tokens::SWITCH_SLIDE_BORDER_CHECKED_DISABLED
334        } else if pressed && !activate_on_press {
335            tokens::SWITCH_SLIDE_BORDER_CHECKED_PRESSED
336        } else if hovered {
337            tokens::SWITCH_SLIDE_BORDER_CHECKED_HOVER
338        } else {
339            tokens::SWITCH_SLIDE_BORDER_CHECKED
340        }
341    } else {
342        if disabled {
343            tokens::SWITCH_SLIDE_BORDER_DISABLED
344        } else if pressed && !activate_on_press {
345            tokens::SWITCH_SLIDE_BORDER_PRESSED
346        } else if hovered {
347            tokens::SWITCH_SLIDE_BORDER_HOVER
348        } else {
349            tokens::SWITCH_SLIDE_BORDER
350        }
351    };
352
353    let slide_bg_token = if checked {
354        if disabled {
355            tokens::SWITCH_SLIDE_BG_CHECKED_DISABLED
356        } else if pressed && !activate_on_press {
357            tokens::SWITCH_SLIDE_BG_CHECKED_PRESSED
358        } else if hovered {
359            tokens::SWITCH_SLIDE_BG_CHECKED_HOVER
360        } else {
361            tokens::SWITCH_SLIDE_BG_CHECKED
362        }
363    } else {
364        if disabled {
365            tokens::SWITCH_SLIDE_BG_DISABLED
366        } else if pressed && !activate_on_press {
367            tokens::SWITCH_SLIDE_BG_PRESSED
368        } else if hovered {
369            tokens::SWITCH_SLIDE_BG_HOVER
370        } else {
371            tokens::SWITCH_SLIDE_BG
372        }
373    };
374
375    let slide_pos = match checked {
376        true => percent(50),
377        false => percent(0),
378    };
379
380    let cursor_shape = match disabled {
381        true => bevy_window::SystemCursorIcon::NotAllowed,
382        false => bevy_window::SystemCursorIcon::Pointer,
383    };
384
385    // Change outline background
386    if outline_bg.0 != outline_bg_token {
387        commands
388            .entity(switch_ent)
389            .insert(ThemeBackgroundColor(outline_bg_token));
390    }
391
392    // Change outline border
393    if outline_border.0 != outline_border_token {
394        commands
395            .entity(switch_ent)
396            .insert(ThemeBorderColor(outline_border_token));
397    }
398
399    // Change slide background color
400    if slide_bg_color.0 != slide_bg_token {
401        commands
402            .entity(slide_ent)
403            .insert(ThemeBackgroundColor(slide_bg_token));
404    }
405
406    // Change slide border color
407    if slide_border_color.0 != slide_border_token {
408        commands
409            .entity(slide_ent)
410            .insert(ThemeBorderColor(slide_border_token));
411    }
412
413    // Change slide position
414    if slide_pos != slide_style.left {
415        slide_style.left = slide_pos;
416    }
417
418    // Change cursor shape
419    commands
420        .entity(switch_ent)
421        .insert(EntityCursor::System(cursor_shape));
422}
423
424/// Plugin which registers the systems for updating the toggle switch styles.
425pub struct ToggleSwitchPlugin;
426
427impl Plugin for ToggleSwitchPlugin {
428    fn build(&self, app: &mut bevy_app::App) {
429        app.add_systems(
430            PreUpdate,
431            (update_switch_styles, update_switch_styles_remove).in_set(PickingSystems::Last),
432        );
433    }
434}