use accesskit::Role;
use bevy_a11y::AccessibilityNode;
use bevy_app::{App, Plugin};
use bevy_ecs::event::EntityEvent;
use bevy_ecs::query::{Has, With, Without};
use bevy_ecs::system::ResMut;
use bevy_ecs::{
component::Component,
observer::On,
reflect::{ReflectComponent, ReflectEvent},
system::{Commands, Query},
};
use bevy_input::keyboard::{KeyCode, KeyboardInput};
use bevy_input::ButtonState;
use bevy_input_focus::{FocusCause, FocusedInput, InputFocus, InputFocusVisible};
use bevy_picking::events::{Cancel, Click, DragEnd, Pointer, Press, Release};
use bevy_reflect::Reflect;
use bevy_ui::{Checkable, Checked, InteractionDisabled, Pressed};
use crate::{ActivateOnPress, ValueChange};
use bevy_ecs::entity::Entity;
#[derive(Component, Debug, Default, Clone)]
#[require(AccessibilityNode(accesskit::Node::new(Role::CheckBox)), Checkable)]
#[derive(Reflect)]
#[reflect(Component)]
pub struct Checkbox;
fn checkbox_on_key_input(
mut ev: On<FocusedInput<KeyboardInput>>,
q_checkbox: Query<Has<Checked>, (With<Checkbox>, Without<InteractionDisabled>)>,
mut commands: Commands,
) {
if let Ok(is_checked) = q_checkbox.get(ev.focused_entity) {
let event = &ev.event().input;
if event.state == ButtonState::Pressed
&& !event.repeat
&& (event.key_code == KeyCode::Enter || event.key_code == KeyCode::Space)
{
ev.propagate(false);
commands.trigger(ValueChange {
source: ev.focused_entity,
value: !is_checked,
is_final: true,
});
}
}
}
fn checkbox_on_pointer_click(
mut click: On<Pointer<Click>>,
q_checkbox: Query<
(Has<Checked>, Has<InteractionDisabled>),
(With<Checkbox>, Without<ActivateOnPress>),
>,
mut commands: Commands,
) {
if let Ok((is_checked, disabled)) = q_checkbox.get(click.entity) {
click.propagate(false);
if !disabled {
commands.trigger(ValueChange {
source: click.entity,
value: !is_checked,
is_final: true,
});
}
}
}
fn checkbox_on_pointer_down(
mut press: On<Pointer<Press>>,
mut q_checkbox: Query<
(
Entity,
Has<InteractionDisabled>,
Has<Checked>,
Has<Pressed>,
Has<ActivateOnPress>,
),
With<Checkbox>,
>,
focus: Option<ResMut<InputFocus>>,
focus_visible: Option<ResMut<InputFocusVisible>>,
mut commands: Commands,
) {
if let Ok((checkbox, disabled, checked, pressed, activate_on_press)) =
q_checkbox.get_mut(press.entity)
{
if let Some(mut focus) = focus {
focus.set(press.entity, FocusCause::Pressed);
}
if let Some(mut focus_visible) = focus_visible {
focus_visible.0 = false;
}
press.propagate(false);
if !disabled && !pressed {
commands.entity(checkbox).insert(Pressed);
if activate_on_press {
commands.trigger(ValueChange {
source: press.entity,
value: !checked,
is_final: true,
});
}
}
}
}
fn checkbox_on_pointer_up(
mut release: On<Pointer<Release>>,
mut q_checkbox: Query<(Entity, Has<InteractionDisabled>, Has<Pressed>), With<Checkbox>>,
mut commands: Commands,
) {
if let Ok((checkbox, disabled, pressed)) = q_checkbox.get_mut(release.entity) {
release.propagate(false);
if !disabled && pressed {
commands.entity(checkbox).remove::<Pressed>();
}
}
}
fn checkbox_on_pointer_drag_end(
mut drag_end: On<Pointer<DragEnd>>,
mut q_checkbox: Query<(Entity, Has<InteractionDisabled>, Has<Pressed>), With<Checkbox>>,
mut commands: Commands,
) {
if let Ok((checkbox, disabled, pressed)) = q_checkbox.get_mut(drag_end.entity) {
drag_end.propagate(false);
if !disabled && pressed {
commands.entity(checkbox).remove::<Pressed>();
}
}
}
fn checkbox_on_pointer_cancel(
mut cancel: On<Pointer<Cancel>>,
mut q_checkbox: Query<(Entity, Has<InteractionDisabled>, Has<Pressed>), With<Checkbox>>,
mut commands: Commands,
) {
if let Ok((checkbox, disabled, pressed)) = q_checkbox.get_mut(cancel.entity) {
cancel.propagate(false);
if !disabled && pressed {
commands.entity(checkbox).remove::<Pressed>();
}
}
}
#[derive(EntityEvent, Reflect)]
#[reflect(Event)]
pub struct SetChecked {
pub entity: Entity,
pub checked: bool,
}
#[derive(EntityEvent, Reflect)]
#[reflect(Event)]
pub struct ToggleChecked {
pub entity: Entity,
}
fn checkbox_on_set_checked(
set_checked: On<SetChecked>,
q_checkbox: Query<(Has<Checked>, Has<InteractionDisabled>), With<Checkbox>>,
mut commands: Commands,
) {
if let Ok((is_checked, disabled)) = q_checkbox.get(set_checked.entity) {
if disabled {
return;
}
let will_be_checked = set_checked.checked;
if will_be_checked != is_checked {
commands.trigger(ValueChange {
source: set_checked.entity,
value: will_be_checked,
is_final: true,
});
}
}
}
fn checkbox_on_toggle_checked(
toggle_checked: On<ToggleChecked>,
q_checkbox: Query<(Has<Checked>, Has<InteractionDisabled>), With<Checkbox>>,
mut commands: Commands,
) {
if let Ok((is_checked, disabled)) = q_checkbox.get(toggle_checked.entity) {
if disabled {
return;
}
commands.trigger(ValueChange {
source: toggle_checked.entity,
value: !is_checked,
is_final: true,
});
}
}
pub struct CheckboxPlugin;
impl Plugin for CheckboxPlugin {
fn build(&self, app: &mut App) {
app.add_observer(checkbox_on_key_input)
.add_observer(checkbox_on_pointer_click)
.add_observer(checkbox_on_pointer_down)
.add_observer(checkbox_on_pointer_up)
.add_observer(checkbox_on_pointer_drag_end)
.add_observer(checkbox_on_pointer_cancel)
.add_observer(checkbox_on_set_checked)
.add_observer(checkbox_on_toggle_checked);
}
}
pub fn checkbox_self_update(value_change: On<ValueChange<bool>>, mut commands: Commands) {
if value_change.value {
commands.entity(value_change.source).insert(Checked);
} else {
commands.entity(value_change.source).remove::<Checked>();
}
}