use crate::layout::{Layers, Viewports};
use dioxus_native_core::prelude::NodeImmutableDioxusExt;
use dioxus_native_core::real_dom::NodeImmutable;
use dioxus_native_core::NodeId;
use freya_dom::prelude::FreyaDOM;
use rustc_hash::FxHashMap;
pub use crate::events::{DomEvent, ElementsState, FreyaEvent};
use crate::types::{EventEmitter, EventsQueue, NodesEvents};
pub fn process_events(
dom: &FreyaDOM,
layers: &Layers,
events: &mut EventsQueue,
event_emitter: &EventEmitter,
elements_state: &mut ElementsState,
viewports: &Viewports,
scale_factor: f64,
) {
let global_events = measure_global_events(events);
let potential_events = measure_potential_event_listeners(layers, events, viewports, dom);
let emitted_events = measure_dom_events(potential_events, dom, scale_factor);
let potential_colateral_events =
elements_state.process_events(emitted_events, events, event_emitter);
let emitted_colateral_events =
measure_dom_events(potential_colateral_events, dom, scale_factor);
for event in emitted_colateral_events {
event_emitter.send(event).unwrap();
}
emit_global_events_listeners(global_events, dom, event_emitter, scale_factor);
events.clear();
}
pub fn measure_global_events(events: &EventsQueue) -> Vec<FreyaEvent> {
let mut global_events = Vec::default();
for event in events {
let event_name = match event.get_name() {
"click" => Some("globalclick"),
"mousedown" => Some("globalmousedown"),
"mouseover" => Some("globalmouseover"),
_ => None,
};
if let Some(event_name) = event_name {
let mut global_event = event.clone();
global_event.set_name(event_name.to_string());
global_events.push(global_event);
}
}
global_events
}
pub fn measure_potential_event_listeners(
layers: &Layers,
events: &EventsQueue,
viewports: &Viewports,
fdom: &FreyaDOM,
) -> NodesEvents {
let mut potential_events = FxHashMap::default();
let layout = fdom.layout();
for (_, layer) in layers.layers() {
for node_id in layer {
let areas = layout.get(*node_id);
if let Some(areas) = areas {
'events: for event in events.iter() {
if let FreyaEvent::Keyboard { name, .. } = event {
let event_data = (*node_id, event.clone());
potential_events
.entry(name.clone())
.or_insert_with(|| vec![event_data.clone()])
.push(event_data);
} else {
let data = match event {
FreyaEvent::Mouse { name, cursor, .. } => Some((name, cursor)),
FreyaEvent::Wheel { name, cursor, .. } => Some((name, cursor)),
FreyaEvent::Touch { name, location, .. } => Some((name, location)),
_ => None,
};
if let Some((name, cursor)) = data {
let cursor_is_inside = areas.area.contains(cursor.to_f32());
if cursor_is_inside {
let node_viewports = viewports.get(node_id);
if let Some((_, node_viewports)) = node_viewports {
for viewport_id in node_viewports {
let viewport = viewports.get(viewport_id).unwrap().0;
if let Some(viewport) = viewport {
if !viewport.contains(cursor.to_f32()) {
continue 'events;
}
}
}
}
let event_data = (*node_id, event.clone());
potential_events
.entry(name.clone())
.or_insert_with(Vec::new)
.push(event_data);
}
}
}
}
}
}
}
potential_events
}
fn get_derivated_events(event_name: &str) -> Vec<&str> {
match event_name {
"mouseover" => {
vec![event_name, "mouseenter", "pointerenter", "pointerover"]
}
"mousedown" | "touchdown" => {
vec![event_name, "pointerdown"]
}
"click" | "ontouchend" => {
vec![event_name, "pointerup"]
}
"mouseleave" => {
vec![event_name, "pointerleave"]
}
_ => vec![event_name],
}
}
fn measure_dom_events(
potential_events: NodesEvents,
fdom: &FreyaDOM,
scale_factor: f64,
) -> Vec<DomEvent> {
let mut new_events = Vec::new();
let rdom = fdom.rdom();
for (event_name, event_nodes) in potential_events {
let derivated_events = get_derivated_events(event_name.as_str());
let mut found_events: Vec<(NodeId, FreyaEvent)> = Vec::new();
'events: for derivated_event_name in derivated_events {
let listeners = rdom.get_listening_sorted(derivated_event_name);
for (node_id, request) in event_nodes.iter() {
for listener in &listeners {
if listener.id() == *node_id {
let mut request = request.clone();
request.set_name(derivated_event_name.to_string());
found_events.push((*node_id, request));
break 'events;
}
}
}
}
for (node_id, request_event) in found_events {
let areas = fdom.layout().get(node_id).cloned();
if let Some(areas) = areas {
let node_ref = fdom.rdom().get(node_id).unwrap();
let element_id = node_ref.mounted_id().unwrap();
let event = DomEvent::new(
node_id,
element_id,
&request_event,
Some(areas.visible_area()),
scale_factor,
);
new_events.push(event);
}
}
}
new_events
}
fn emit_global_events_listeners(
global_events: Vec<FreyaEvent>,
fdom: &FreyaDOM,
event_emitter: &EventEmitter,
scale_factor: f64,
) {
for global_event in global_events {
let event_name = global_event.get_name();
let listeners = fdom.rdom().get_listening_sorted(event_name);
for listener in listeners {
let element_id = listener.mounted_id().unwrap();
let event = DomEvent::new(listener.id(), element_id, &global_event, None, scale_factor);
event_emitter.send(event).unwrap();
}
}
}