Skip to main content

bevy_ui_widgets/
popover.rs

1//! Framework for positioning of popups, tooltips, and other popover UI elements.
2
3use bevy_app::{App, Plugin, PostUpdate};
4use bevy_ecs::{
5    component::Component,
6    entity::Entity,
7    hierarchy::{ChildOf, Children},
8    query::Without,
9    reflect::ReflectComponent,
10    schedule::IntoScheduleConfigs,
11    system::{ParamSet, Query},
12};
13use bevy_math::{Affine2, Rect, Vec2};
14use bevy_reflect::Reflect;
15use bevy_ui::{
16    ui_layout_system, ComputedNode, ComputedUiRenderTargetInfo, Node, PositionType,
17    UiGlobalTransform, UiSystems, UiTransform, Val2,
18};
19
20use crate::update_scrollbar_thumb;
21
22/// Which side of the parent element the popover element should be placed.
23#[derive(#[automatically_derived]
impl ::core::fmt::Debug for PopoverSide {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                PopoverSide::Top => "Top",
                PopoverSide::Bottom => "Bottom",
                PopoverSide::Left => "Left",
                PopoverSide::Right => "Right",
            })
    }
}Debug, #[automatically_derived]
impl ::core::default::Default for PopoverSide {
    #[inline]
    fn default() -> PopoverSide { Self::Bottom }
}Default, #[automatically_derived]
impl ::core::clone::Clone for PopoverSide {
    #[inline]
    fn clone(&self) -> PopoverSide { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for PopoverSide { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for PopoverSide {
    #[inline]
    fn eq(&self, other: &PopoverSide) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for PopoverSide 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
            }
            #[inline(never)]
            fn register_type_dependencies(registry:
                    &mut bevy_reflect::TypeRegistry) {}
        }
        impl bevy_reflect::Typed for PopoverSide 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::Enum(bevy_reflect::enums::EnumInfo::new::<Self>(&[bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Top")),
                                                bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Bottom")),
                                                bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Left")),
                                                bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Right"))]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for PopoverSide where  {
            fn type_path() -> &'static str {
                "bevy_ui_widgets::popover::PopoverSide"
            }
            fn short_type_path() -> &'static str { "PopoverSide" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("PopoverSide")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover")
            }
        }
        impl bevy_reflect::Reflect for PopoverSide 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(<PopoverSide
                                        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::enums::Enum for PopoverSide where  {
            fn field(&self, __name_param: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_at(&self, __index_param: usize)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_mut(&mut self, __name_param: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_at_mut(&mut self, __index_param: usize)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn index_of(&self, __name_param: &str)
                -> ::core::option::Option<usize> {
                match self { _ => ::core::option::Option::None, }
            }
            fn name_at(&self, __index_param: usize)
                -> ::core::option::Option<&str> {
                match self { _ => ::core::option::Option::None, }
            }
            fn iter_fields(&self) -> bevy_reflect::enums::VariantFieldIter {
                bevy_reflect::enums::VariantFieldIter::new(self)
            }
            #[inline]
            fn field_len(&self) -> usize {
                match self {
                    PopoverSide::Top { .. } => 0usize,
                    PopoverSide::Bottom { .. } => 0usize,
                    PopoverSide::Left { .. } => 0usize,
                    PopoverSide::Right { .. } => 0usize,
                    _ => 0,
                }
            }
            #[inline]
            fn variant_name(&self) -> &str {
                match self {
                    PopoverSide::Top { .. } => "Top",
                    PopoverSide::Bottom { .. } => "Bottom",
                    PopoverSide::Left { .. } => "Left",
                    PopoverSide::Right { .. } => "Right",
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            #[inline]
            fn variant_index(&self) -> usize {
                match self {
                    PopoverSide::Top { .. } => 0usize,
                    PopoverSide::Bottom { .. } => 1usize,
                    PopoverSide::Left { .. } => 2usize,
                    PopoverSide::Right { .. } => 3usize,
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            #[inline]
            fn variant_type(&self) -> bevy_reflect::enums::VariantType {
                match self {
                    PopoverSide::Top { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    PopoverSide::Bottom { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    PopoverSide::Left { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    PopoverSide::Right { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            fn to_dynamic_enum(&self) -> bevy_reflect::enums::DynamicEnum {
                bevy_reflect::enums::DynamicEnum::from_ref::<Self>(self)
            }
        }
        impl bevy_reflect::PartialReflect for PopoverSide 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_param: &dyn bevy_reflect::PartialReflect)
                -> ::core::result::Result<(), bevy_reflect::ApplyError> {
                if let bevy_reflect::ReflectRef::Enum(__value_param) =
                        bevy_reflect::PartialReflect::reflect_ref(__value_param) {
                    if bevy_reflect::enums::Enum::variant_name(self) ==
                            bevy_reflect::enums::Enum::variant_name(__value_param) {
                        match bevy_reflect::enums::Enum::variant_type(__value_param)
                            {
                            bevy_reflect::enums::VariantType::Struct => {
                                for field in
                                    bevy_reflect::enums::Enum::iter_fields(__value_param) {
                                    let name = field.name().unwrap();
                                    if let ::core::option::Option::Some(v) =
                                            bevy_reflect::enums::Enum::field_mut(self, name) {
                                        bevy_reflect::PartialReflect::try_apply(v, field.value())?;
                                    }
                                }
                            }
                            bevy_reflect::enums::VariantType::Tuple => {
                                for (index, field) in
                                    ::core::iter::Iterator::enumerate(bevy_reflect::enums::Enum::iter_fields(__value_param))
                                    {
                                    if let ::core::option::Option::Some(v) =
                                            bevy_reflect::enums::Enum::field_at_mut(self, index) {
                                        bevy_reflect::PartialReflect::try_apply(v, field.value())?;
                                    }
                                }
                            }
                            _ => {}
                        }
                    } else {
                        match bevy_reflect::enums::Enum::variant_name(__value_param)
                            {
                            "Top" => { *self = PopoverSide::Top {} }
                            "Bottom" => { *self = PopoverSide::Bottom {} }
                            "Left" => { *self = PopoverSide::Left {} }
                            "Right" => { *self = PopoverSide::Right {} }
                            name => {
                                return ::core::result::Result::Err(bevy_reflect::ApplyError::UnknownVariant {
                                            enum_name: ::core::convert::Into::into(bevy_reflect::DynamicTypePath::reflect_type_path(self)),
                                            variant_name: ::core::convert::Into::into(name),
                                        });
                            }
                        }
                    }
                } else {
                    return ::core::result::Result::Err(bevy_reflect::ApplyError::MismatchedKinds {
                                from_kind: bevy_reflect::PartialReflect::reflect_kind(__value_param),
                                to_kind: bevy_reflect::ReflectKind::Enum,
                            });
                }
                ::core::result::Result::Ok(())
            }
            fn reflect_kind(&self) -> bevy_reflect::ReflectKind {
                bevy_reflect::ReflectKind::Enum
            }
            fn reflect_ref(&self) -> bevy_reflect::ReflectRef {
                bevy_reflect::ReflectRef::Enum(self)
            }
            fn reflect_mut(&mut self) -> bevy_reflect::ReflectMut {
                bevy_reflect::ReflectMut::Enum(self)
            }
            fn reflect_owned(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                -> bevy_reflect::ReflectOwned {
                bevy_reflect::ReflectOwned::Enum(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_hash(&self) -> ::core::option::Option<u64> {
                (bevy_reflect::enums::enum_hash)(self)
            }
            fn reflect_partial_eq(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<bool> {
                (bevy_reflect::enums::enum_partial_eq)(self, value)
            }
            fn reflect_partial_cmp(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<::core::cmp::Ordering> {
                (bevy_reflect::enums::enum_partial_cmp)(self, value)
            }
            #[inline]
            #[allow(unreachable_code, reason =
            "Ignored fields without a `clone` attribute will early-return with an error")]
            fn reflect_clone(&self)
                ->
                    ::core::result::Result<bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>,
                    bevy_reflect::ReflectCloneError> {
                let this = self;
                ::core::result::Result::Ok(bevy_reflect::__macro_exports::alloc_utils::Box::new(match this
                            {
                            PopoverSide::Top {} => PopoverSide::Top {},
                            PopoverSide::Bottom {} => PopoverSide::Bottom {},
                            PopoverSide::Left {} => PopoverSide::Left {},
                            PopoverSide::Right {} => PopoverSide::Right {},
                        }))
            }
        }
        impl bevy_reflect::FromReflect for PopoverSide where  {
            fn from_reflect(__param0: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<Self> {
                if let bevy_reflect::ReflectRef::Enum(__param0) =
                        bevy_reflect::PartialReflect::reflect_ref(__param0) {
                    match bevy_reflect::enums::Enum::variant_name(__param0) {
                        "Top" => ::core::option::Option::Some(PopoverSide::Top {}),
                        "Bottom" =>
                            ::core::option::Option::Some(PopoverSide::Bottom {}),
                        "Left" =>
                            ::core::option::Option::Some(PopoverSide::Left {}),
                        "Right" =>
                            ::core::option::Option::Some(PopoverSide::Right {}),
                        name => ::core::option::Option::None,
                    }
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
24pub enum PopoverSide {
25    /// The popover element should be placed above the parent.
26    Top,
27    /// The popover element should be placed below the parent.
28    #[default]
29    Bottom,
30    /// The popover element should be placed to the left of the parent.
31    Left,
32    /// The popover element should be placed to the right of the parent.
33    Right,
34}
35
36impl PopoverSide {
37    /// Returns the side that is the mirror image of this side.
38    pub fn mirror(&self) -> Self {
39        match self {
40            PopoverSide::Top => PopoverSide::Bottom,
41            PopoverSide::Bottom => PopoverSide::Top,
42            PopoverSide::Left => PopoverSide::Right,
43            PopoverSide::Right => PopoverSide::Left,
44        }
45    }
46}
47
48/// How the popover element should be aligned to the parent element. The alignment will be along an
49/// axis that is perpendicular to the direction of the popover side. So for example, if the popup is
50/// positioned below the parent, then the [`PopoverAlign`] variant controls the horizontal alignment
51/// of the popup.
52#[derive(#[automatically_derived]
impl ::core::fmt::Debug for PopoverAlign {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                PopoverAlign::Start => "Start",
                PopoverAlign::Center => "Center",
                PopoverAlign::End => "End",
            })
    }
}Debug, #[automatically_derived]
impl ::core::default::Default for PopoverAlign {
    #[inline]
    fn default() -> PopoverAlign { Self::Start }
}Default, #[automatically_derived]
impl ::core::clone::Clone for PopoverAlign {
    #[inline]
    fn clone(&self) -> PopoverAlign { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for PopoverAlign { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for PopoverAlign {
    #[inline]
    fn eq(&self, other: &PopoverAlign) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for PopoverAlign 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
            }
            #[inline(never)]
            fn register_type_dependencies(registry:
                    &mut bevy_reflect::TypeRegistry) {}
        }
        impl bevy_reflect::Typed for PopoverAlign 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::Enum(bevy_reflect::enums::EnumInfo::new::<Self>(&[bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Start")),
                                                bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("Center")),
                                                bevy_reflect::enums::VariantInfo::Unit(bevy_reflect::enums::UnitVariantInfo::new("End"))]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for PopoverAlign where  {
            fn type_path() -> &'static str {
                "bevy_ui_widgets::popover::PopoverAlign"
            }
            fn short_type_path() -> &'static str { "PopoverAlign" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("PopoverAlign")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover")
            }
        }
        impl bevy_reflect::Reflect for PopoverAlign 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(<PopoverAlign
                                        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::enums::Enum for PopoverAlign where  {
            fn field(&self, __name_param: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_at(&self, __index_param: usize)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_mut(&mut self, __name_param: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn field_at_mut(&mut self, __index_param: usize)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match self { _ => ::core::option::Option::None, }
            }
            fn index_of(&self, __name_param: &str)
                -> ::core::option::Option<usize> {
                match self { _ => ::core::option::Option::None, }
            }
            fn name_at(&self, __index_param: usize)
                -> ::core::option::Option<&str> {
                match self { _ => ::core::option::Option::None, }
            }
            fn iter_fields(&self) -> bevy_reflect::enums::VariantFieldIter {
                bevy_reflect::enums::VariantFieldIter::new(self)
            }
            #[inline]
            fn field_len(&self) -> usize {
                match self {
                    PopoverAlign::Start { .. } => 0usize,
                    PopoverAlign::Center { .. } => 0usize,
                    PopoverAlign::End { .. } => 0usize,
                    _ => 0,
                }
            }
            #[inline]
            fn variant_name(&self) -> &str {
                match self {
                    PopoverAlign::Start { .. } => "Start",
                    PopoverAlign::Center { .. } => "Center",
                    PopoverAlign::End { .. } => "End",
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            #[inline]
            fn variant_index(&self) -> usize {
                match self {
                    PopoverAlign::Start { .. } => 0usize,
                    PopoverAlign::Center { .. } => 1usize,
                    PopoverAlign::End { .. } => 2usize,
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            #[inline]
            fn variant_type(&self) -> bevy_reflect::enums::VariantType {
                match self {
                    PopoverAlign::Start { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    PopoverAlign::Center { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    PopoverAlign::End { .. } =>
                        bevy_reflect::enums::VariantType::Unit,
                    _ =>
                        ::core::panicking::panic("internal error: entered unreachable code"),
                }
            }
            fn to_dynamic_enum(&self) -> bevy_reflect::enums::DynamicEnum {
                bevy_reflect::enums::DynamicEnum::from_ref::<Self>(self)
            }
        }
        impl bevy_reflect::PartialReflect for PopoverAlign 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_param: &dyn bevy_reflect::PartialReflect)
                -> ::core::result::Result<(), bevy_reflect::ApplyError> {
                if let bevy_reflect::ReflectRef::Enum(__value_param) =
                        bevy_reflect::PartialReflect::reflect_ref(__value_param) {
                    if bevy_reflect::enums::Enum::variant_name(self) ==
                            bevy_reflect::enums::Enum::variant_name(__value_param) {
                        match bevy_reflect::enums::Enum::variant_type(__value_param)
                            {
                            bevy_reflect::enums::VariantType::Struct => {
                                for field in
                                    bevy_reflect::enums::Enum::iter_fields(__value_param) {
                                    let name = field.name().unwrap();
                                    if let ::core::option::Option::Some(v) =
                                            bevy_reflect::enums::Enum::field_mut(self, name) {
                                        bevy_reflect::PartialReflect::try_apply(v, field.value())?;
                                    }
                                }
                            }
                            bevy_reflect::enums::VariantType::Tuple => {
                                for (index, field) in
                                    ::core::iter::Iterator::enumerate(bevy_reflect::enums::Enum::iter_fields(__value_param))
                                    {
                                    if let ::core::option::Option::Some(v) =
                                            bevy_reflect::enums::Enum::field_at_mut(self, index) {
                                        bevy_reflect::PartialReflect::try_apply(v, field.value())?;
                                    }
                                }
                            }
                            _ => {}
                        }
                    } else {
                        match bevy_reflect::enums::Enum::variant_name(__value_param)
                            {
                            "Start" => { *self = PopoverAlign::Start {} }
                            "Center" => { *self = PopoverAlign::Center {} }
                            "End" => { *self = PopoverAlign::End {} }
                            name => {
                                return ::core::result::Result::Err(bevy_reflect::ApplyError::UnknownVariant {
                                            enum_name: ::core::convert::Into::into(bevy_reflect::DynamicTypePath::reflect_type_path(self)),
                                            variant_name: ::core::convert::Into::into(name),
                                        });
                            }
                        }
                    }
                } else {
                    return ::core::result::Result::Err(bevy_reflect::ApplyError::MismatchedKinds {
                                from_kind: bevy_reflect::PartialReflect::reflect_kind(__value_param),
                                to_kind: bevy_reflect::ReflectKind::Enum,
                            });
                }
                ::core::result::Result::Ok(())
            }
            fn reflect_kind(&self) -> bevy_reflect::ReflectKind {
                bevy_reflect::ReflectKind::Enum
            }
            fn reflect_ref(&self) -> bevy_reflect::ReflectRef {
                bevy_reflect::ReflectRef::Enum(self)
            }
            fn reflect_mut(&mut self) -> bevy_reflect::ReflectMut {
                bevy_reflect::ReflectMut::Enum(self)
            }
            fn reflect_owned(self:
                    bevy_reflect::__macro_exports::alloc_utils::Box<Self>)
                -> bevy_reflect::ReflectOwned {
                bevy_reflect::ReflectOwned::Enum(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_hash(&self) -> ::core::option::Option<u64> {
                (bevy_reflect::enums::enum_hash)(self)
            }
            fn reflect_partial_eq(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<bool> {
                (bevy_reflect::enums::enum_partial_eq)(self, value)
            }
            fn reflect_partial_cmp(&self,
                value: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<::core::cmp::Ordering> {
                (bevy_reflect::enums::enum_partial_cmp)(self, value)
            }
            #[inline]
            #[allow(unreachable_code, reason =
            "Ignored fields without a `clone` attribute will early-return with an error")]
            fn reflect_clone(&self)
                ->
                    ::core::result::Result<bevy_reflect::__macro_exports::alloc_utils::Box<dyn bevy_reflect::Reflect>,
                    bevy_reflect::ReflectCloneError> {
                let this = self;
                ::core::result::Result::Ok(bevy_reflect::__macro_exports::alloc_utils::Box::new(match this
                            {
                            PopoverAlign::Start {} => PopoverAlign::Start {},
                            PopoverAlign::Center {} => PopoverAlign::Center {},
                            PopoverAlign::End {} => PopoverAlign::End {},
                        }))
            }
        }
        impl bevy_reflect::FromReflect for PopoverAlign where  {
            fn from_reflect(__param0: &dyn bevy_reflect::PartialReflect)
                -> ::core::option::Option<Self> {
                if let bevy_reflect::ReflectRef::Enum(__param0) =
                        bevy_reflect::PartialReflect::reflect_ref(__param0) {
                    match bevy_reflect::enums::Enum::variant_name(__param0) {
                        "Start" =>
                            ::core::option::Option::Some(PopoverAlign::Start {}),
                        "Center" =>
                            ::core::option::Option::Some(PopoverAlign::Center {}),
                        "End" => ::core::option::Option::Some(PopoverAlign::End {}),
                        name => ::core::option::Option::None,
                    }
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
53pub enum PopoverAlign {
54    /// The starting edge of the popover element should be aligned to the starting edge of the
55    /// parent.
56    #[default]
57    Start,
58    /// The center of the popover element should be aligned to the center of the parent.
59    Center,
60    /// The ending edge of the popover element should be aligned to the ending edge of the parent.
61    End,
62}
63
64/// Indicates a possible position of a popover element relative to it's parent. You can
65/// specify multiple possible positions; the positioning code will check to see if there is
66/// sufficient space to display the popup without being clipped by the window edge. If any position
67/// has sufficient room, it will pick the first one; if there are none, then it will pick the least
68/// bad one.
69#[derive(#[automatically_derived]
impl ::core::fmt::Debug for PopoverPlacement {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f,
            "PopoverPlacement", "side", &self.side, "align", &self.align,
            "gap", &&self.gap)
    }
}Debug, #[automatically_derived]
impl ::core::default::Default for PopoverPlacement {
    #[inline]
    fn default() -> PopoverPlacement {
        PopoverPlacement {
            side: ::core::default::Default::default(),
            align: ::core::default::Default::default(),
            gap: ::core::default::Default::default(),
        }
    }
}Default, #[automatically_derived]
impl ::core::clone::Clone for PopoverPlacement {
    #[inline]
    fn clone(&self) -> PopoverPlacement {
        let _: ::core::clone::AssertParamIsClone<PopoverSide>;
        let _: ::core::clone::AssertParamIsClone<PopoverAlign>;
        let _: ::core::clone::AssertParamIsClone<f32>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for PopoverPlacement { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for PopoverPlacement {
    #[inline]
    fn eq(&self, other: &PopoverPlacement) -> bool {
        self.gap == other.gap && self.side == other.side &&
            self.align == other.align
    }
}PartialEq, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for PopoverPlacement 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
            }
            #[inline(never)]
            fn register_type_dependencies(registry:
                    &mut bevy_reflect::TypeRegistry) {
                <PopoverSide as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
                <PopoverAlign as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
                <f32 as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
            }
        }
        impl bevy_reflect::Typed for PopoverPlacement 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::<PopoverSide>("side"),
                                                bevy_reflect::NamedField::new::<PopoverAlign>("align"),
                                                bevy_reflect::NamedField::new::<f32>("gap")]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for PopoverPlacement where  {
            fn type_path() -> &'static str {
                "bevy_ui_widgets::popover::PopoverPlacement"
            }
            fn short_type_path() -> &'static str { "PopoverPlacement" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("PopoverPlacement")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover")
            }
        }
        impl bevy_reflect::Reflect for PopoverPlacement 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(<PopoverPlacement
                                        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 PopoverPlacement where  {
            fn field(&self, name: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match name {
                    "side" => ::core::option::Option::Some(&self.side),
                    "align" => ::core::option::Option::Some(&self.align),
                    "gap" => ::core::option::Option::Some(&self.gap),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_mut(&mut self, name: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match name {
                    "side" => ::core::option::Option::Some(&mut self.side),
                    "align" => ::core::option::Option::Some(&mut self.align),
                    "gap" => ::core::option::Option::Some(&mut self.gap),
                    _ => ::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.side),
                    1usize => ::core::option::Option::Some(&self.align),
                    2usize => ::core::option::Option::Some(&self.gap),
                    _ => ::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.side),
                    1usize => ::core::option::Option::Some(&mut self.align),
                    2usize => ::core::option::Option::Some(&mut self.gap),
                    _ => ::core::option::Option::None,
                }
            }
            fn name_at(&self, index: usize) -> ::core::option::Option<&str> {
                match index {
                    0usize => ::core::option::Option::Some("side"),
                    1usize => ::core::option::Option::Some("align"),
                    2usize => ::core::option::Option::Some("gap"),
                    _ => ::core::option::Option::None,
                }
            }
            fn index_of_name(&self, name: &str)
                -> ::core::option::Option<usize> {
                match name {
                    "side" => ::core::option::Option::Some(0usize),
                    "align" => ::core::option::Option::Some(1usize),
                    "gap" => ::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("side",
                    bevy_reflect::PartialReflect::to_dynamic(&self.side));
                dynamic.insert_boxed("align",
                    bevy_reflect::PartialReflect::to_dynamic(&self.align));
                dynamic.insert_boxed("gap",
                    bevy_reflect::PartialReflect::to_dynamic(&self.gap));
                dynamic
            }
        }
        impl bevy_reflect::PartialReflect for PopoverPlacement 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]
            #[allow(unreachable_code, reason =
            "Ignored fields without a `clone` attribute will early-return with an error")]
            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(Self {
                            side: <PopoverSide as
                                        bevy_reflect::PartialReflect>::reflect_clone_and_take(&self.side)?,
                            align: <PopoverAlign as
                                        bevy_reflect::PartialReflect>::reflect_clone_and_take(&self.align)?,
                            gap: <f32 as
                                        bevy_reflect::PartialReflect>::reflect_clone_and_take(&self.gap)?,
                        }))
            }
        }
        impl bevy_reflect::FromReflect for PopoverPlacement 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 __this =
                        Self {
                            side: <PopoverSide as
                                        bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                            "side")?)?,
                            align: <PopoverAlign as
                                        bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                            "align")?)?,
                            gap: <f32 as
                                        bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                            "gap")?)?,
                        };
                    ::core::option::Option::Some(__this)
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
70pub struct PopoverPlacement {
71    /// The side of the parent entity where the popover element should be placed.
72    pub side: PopoverSide,
73
74    /// How the popover element should be aligned to the parent entity.
75    pub align: PopoverAlign,
76
77    /// The size of the gap between the parent and the popover element, in logical pixels. This will
78    /// offset the popover along the direction of `side`.
79    pub gap: f32,
80}
81
82/// Component which is inserted into a popover element to make it dynamically position relative to
83/// an parent element.
84#[derive(impl bevy_ecs::component::Component for Popover 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::cmp::PartialEq for Popover {
    #[inline]
    fn eq(&self, other: &Popover) -> bool {
        self.window_margin == other.window_margin &&
            self.positions == other.positions
    }
}PartialEq, #[automatically_derived]
impl ::core::default::Default for Popover {
    #[inline]
    fn default() -> Popover {
        Popover {
            positions: ::core::default::Default::default(),
            window_margin: ::core::default::Default::default(),
        }
    }
}Default, const _: () =
    {
        impl bevy_reflect::GetTypeRegistration for Popover 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
            }
            #[inline(never)]
            fn register_type_dependencies(registry:
                    &mut bevy_reflect::TypeRegistry) {
                <Vec<PopoverPlacement> as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
                <f32 as
                        bevy_reflect::__macro_exports::RegisterForReflection>::__register(registry);
            }
        }
        impl bevy_reflect::Typed for Popover 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::<Vec<PopoverPlacement>>("positions"),
                                                bevy_reflect::NamedField::new::<f32>("window_margin")]))
                        })
            }
        }
        #[allow(deprecated, reason =
        "derives on a deprecated type shouldn't be considered a usage")]
        impl bevy_reflect::TypePath for Popover where  {
            fn type_path() -> &'static str {
                "bevy_ui_widgets::popover::Popover"
            }
            fn short_type_path() -> &'static str { "Popover" }
            fn type_ident() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("Popover")
            }
            fn crate_name() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover".split(':').next().unwrap())
            }
            fn module_path() -> ::core::option::Option<&'static str> {
                ::core::option::Option::Some("bevy_ui_widgets::popover")
            }
        }
        impl bevy_reflect::Reflect for Popover 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(<Popover
                                        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 Popover where  {
            fn field(&self, name: &str)
                -> ::core::option::Option<&dyn bevy_reflect::PartialReflect> {
                match name {
                    "positions" =>
                        ::core::option::Option::Some(&self.positions),
                    "window_margin" =>
                        ::core::option::Option::Some(&self.window_margin),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_mut(&mut self, name: &str)
                ->
                    ::core::option::Option<&mut dyn bevy_reflect::PartialReflect> {
                match name {
                    "positions" =>
                        ::core::option::Option::Some(&mut self.positions),
                    "window_margin" =>
                        ::core::option::Option::Some(&mut self.window_margin),
                    _ => ::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.positions),
                    1usize => ::core::option::Option::Some(&self.window_margin),
                    _ => ::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.positions),
                    1usize =>
                        ::core::option::Option::Some(&mut self.window_margin),
                    _ => ::core::option::Option::None,
                }
            }
            fn name_at(&self, index: usize) -> ::core::option::Option<&str> {
                match index {
                    0usize => ::core::option::Option::Some("positions"),
                    1usize => ::core::option::Option::Some("window_margin"),
                    _ => ::core::option::Option::None,
                }
            }
            fn index_of_name(&self, name: &str)
                -> ::core::option::Option<usize> {
                match name {
                    "positions" => ::core::option::Option::Some(0usize),
                    "window_margin" => ::core::option::Option::Some(1usize),
                    _ => ::core::option::Option::None,
                }
            }
            fn field_len(&self) -> usize { 2usize }
            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("positions",
                    bevy_reflect::PartialReflect::to_dynamic(&self.positions));
                dynamic.insert_boxed("window_margin",
                    bevy_reflect::PartialReflect::to_dynamic(&self.window_margin));
                dynamic
            }
        }
        impl bevy_reflect::PartialReflect for Popover 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]
            #[allow(unreachable_code, reason =
            "Ignored fields without a `clone` attribute will early-return with an error")]
            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(Self {
                            positions: <Vec<PopoverPlacement> as
                                        bevy_reflect::PartialReflect>::reflect_clone_and_take(&self.positions)?,
                            window_margin: <f32 as
                                        bevy_reflect::PartialReflect>::reflect_clone_and_take(&self.window_margin)?,
                        }))
            }
        }
        impl bevy_reflect::FromReflect for Popover 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 __this =
                        Self {
                            positions: <Vec<PopoverPlacement> as
                                        bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                            "positions")?)?,
                            window_margin: <f32 as
                                        bevy_reflect::FromReflect>::from_reflect(bevy_reflect::structs::Struct::field(__ref_struct,
                                            "window_margin")?)?,
                        };
                    ::core::option::Option::Some(__this)
                } else { ::core::option::Option::None }
            }
        }
    };Reflect)]
85#[reflect(Component)]
86pub struct Popover {
87    /// List of potential positions for the popover element relative to the parent.
88    pub positions: Vec<PopoverPlacement>,
89
90    /// Indicates how close to the window edge the popup is allowed to go.
91    pub window_margin: f32,
92}
93
94impl Clone for Popover {
95    fn clone(&self) -> Self {
96        Self {
97            positions: self.positions.clone(),
98            window_margin: self.window_margin,
99        }
100    }
101}
102
103pub(crate) fn position_popover(
104    mut q_popover: Query<(
105        Entity,
106        &mut Node,
107        &mut UiTransform,
108        &mut UiGlobalTransform,
109        &ComputedNode,
110        &ComputedUiRenderTargetInfo,
111        &Popover,
112        &ChildOf,
113    )>,
114    mut qs_transform: ParamSet<(
115        Query<(&ComputedNode, &UiGlobalTransform), Without<Popover>>,
116        Query<&mut UiGlobalTransform, Without<Popover>>,
117    )>,
118    q_children: Query<&Children>,
119) {
120    for (
121        popover_entity,
122        mut node,
123        mut transform,
124        mut ui_global_transform,
125        computed_node,
126        computed_target,
127        popover,
128        parent,
129    ) in q_popover.iter_mut()
130    {
131        // A rectangle which represents the area of the window.
132        let window_rect = Rect {
133            min: Vec2::ZERO,
134            max: computed_target.logical_size(),
135        }
136        .inflate(-popover.window_margin);
137
138        // Compute the parent rectangle.
139        let q_parent = qs_transform.p0();
140        let Ok((parent_node, parent_transform)) = q_parent.get(parent.parent()) else {
141            continue;
142        };
143
144        // Computed node size includes the border, but since absolute positioning doesn't include
145        // border we need to remove it from the calculations.
146        let parent_size =
147            parent_node.size() - parent_node.border.min_inset - parent_node.border.max_inset;
148        let parent_rect = scale_rect(
149            Rect::from_center_size(parent_transform.translation, parent_size),
150            parent_node.inverse_scale_factor,
151        );
152        let parent_matrix = parent_transform.affine().matrix2;
153
154        let mut best_occluded = f32::MAX;
155        let mut best_rect = Rect::default();
156
157        // Loop through all the potential positions and find a good one.
158        for position in &popover.positions {
159            let popover_size = computed_node.size() * computed_node.inverse_scale_factor;
160            let mut rect = Rect::default();
161
162            let target_width = popover_size.x;
163            let target_height = popover_size.y;
164
165            // Position along main axis.
166            match position.side {
167                PopoverSide::Top => {
168                    rect.max.y = parent_rect.min.y - position.gap;
169                    rect.min.y = rect.max.y - popover_size.y;
170                }
171
172                PopoverSide::Bottom => {
173                    rect.min.y = parent_rect.max.y + position.gap;
174                    rect.max.y = rect.min.y + popover_size.y;
175                }
176
177                PopoverSide::Left => {
178                    rect.max.x = parent_rect.min.x - position.gap;
179                    rect.min.x = rect.max.x - popover_size.x;
180                }
181
182                PopoverSide::Right => {
183                    rect.min.x = parent_rect.max.x + position.gap;
184                    rect.max.x = rect.min.x + popover_size.x;
185                }
186            }
187
188            // Position along secondary axis.
189            match position.align {
190                PopoverAlign::Start => match position.side {
191                    PopoverSide::Top | PopoverSide::Bottom => {
192                        rect.min.x = parent_rect.min.x;
193                        rect.max.x = rect.min.x + target_width;
194                    }
195
196                    PopoverSide::Left | PopoverSide::Right => {
197                        rect.min.y = parent_rect.min.y;
198                        rect.max.y = rect.min.y + target_height;
199                    }
200                },
201
202                PopoverAlign::End => match position.side {
203                    PopoverSide::Top | PopoverSide::Bottom => {
204                        rect.max.x = parent_rect.max.x;
205                        rect.min.x = rect.max.x - target_width;
206                    }
207
208                    PopoverSide::Left | PopoverSide::Right => {
209                        rect.max.y = parent_rect.max.y;
210                        rect.min.y = rect.max.y - target_height;
211                    }
212                },
213
214                PopoverAlign::Center => match position.side {
215                    PopoverSide::Top | PopoverSide::Bottom => {
216                        rect.min.x = parent_rect.min.x + (parent_rect.width() - target_width) * 0.5;
217                        rect.max.x = rect.min.x + target_width;
218                    }
219
220                    PopoverSide::Left | PopoverSide::Right => {
221                        rect.min.y =
222                            parent_rect.min.y + (parent_rect.height() - target_height) * 0.5;
223                        rect.max.y = rect.min.y + target_height;
224                    }
225                },
226            }
227
228            // Clip to window and see how much of the popover element is occluded. We can calculate
229            // how much was clipped by intersecting the rectangle against the window bounds, and
230            // then subtracting the area from the area of the unclipped rectangle.
231            let clipped_rect = rect.intersect(window_rect);
232            let occlusion = rect.area() - clipped_rect.area();
233
234            // Find the position that has the least occlusion.
235            if occlusion < best_occluded {
236                best_occluded = occlusion;
237                best_rect = rect;
238            }
239        }
240
241        // Update node properties, but only if they are different from before (to avoid setting
242        // change detection bit).
243        if best_occluded < f32::MAX {
244            let best_center = 0.5 * (best_rect.min + best_rect.max);
245            let current_center =
246                ui_global_transform.translation * computed_node.inverse_scale_factor;
247            let physical_translation =
248                (best_center - current_center) * computed_target.scale_factor();
249            if parent_matrix.determinant() == 0.0 {
250                continue;
251            }
252            let resolved_translation = transform.translation.resolve(
253                computed_target.scale_factor(),
254                computed_node.size(),
255                computed_target.physical_size().as_vec2(),
256            );
257            let logical_translation = (resolved_translation
258                + parent_matrix.inverse() * physical_translation)
259                / computed_target.scale_factor();
260            let ui_translation = Val2::px(logical_translation.x, logical_translation.y);
261            if transform.translation != ui_translation {
262                transform.translation = ui_translation;
263            }
264            if node.position_type != PositionType::Absolute {
265                node.position_type = PositionType::Absolute;
266            }
267
268            if physical_translation != Vec2::ZERO {
269                let mut affine = ui_global_transform.affine();
270                affine.translation += physical_translation;
271                *ui_global_transform = affine.into();
272
273                if let Ok(children) = q_children.get(popover_entity) {
274                    for child in children.iter() {
275                        translate_ui_children_recursive(
276                            *child,
277                            physical_translation,
278                            &q_children,
279                            &mut qs_transform.p1(),
280                        );
281                    }
282                }
283            }
284        }
285    }
286}
287
288fn translate_ui_children_recursive(
289    entity: Entity,
290    translation: Vec2,
291    q_children: &Query<&Children>,
292    q_transform: &mut Query<&mut UiGlobalTransform, Without<Popover>>,
293) {
294    let Ok(mut ui_global_transform) = q_transform.get_mut(entity) else {
295        return;
296    };
297
298    *ui_global_transform =
299        (ui_global_transform.affine() * Affine2::from_translation(translation)).into();
300
301    if let Ok(children) = q_children.get(entity) {
302        for child in children.iter() {
303            translate_ui_children_recursive(*child, translation, q_children, q_transform);
304        }
305    }
306}
307
308/// Plugin that adds systems for the [`Popover`] component.
309pub struct PopoverPlugin;
310
311impl Plugin for PopoverPlugin {
312    fn build(&self, app: &mut App) {
313        app.add_systems(
314            PostUpdate,
315            position_popover
316                .in_set(UiSystems::Layout)
317                .after(ui_layout_system)
318                .before(update_scrollbar_thumb),
319        );
320    }
321}
322
323#[inline]
324fn scale_rect(rect: Rect, factor: f32) -> Rect {
325    Rect {
326        min: rect.min * factor,
327        max: rect.max * factor,
328    }
329}