/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ use dom::bindings::callback::ReportExceptions; use dom::bindings::codegen::Bindings::EventBinding::EventMethods; use dom::bindings::codegen::InheritTypes::{EventTargetCast, NodeCast, NodeDerived}; use dom::bindings::js::{JS, JSRef, OptionalSettable, OptionalRootable, Root}; use dom::eventtarget::{Capturing, Bubbling, EventTarget}; use dom::event::{Event, PhaseAtTarget, PhaseNone, PhaseBubbling, PhaseCapturing}; use dom::node::{Node, NodeHelpers}; use dom::virtualmethods::vtable_for; // See http://dom.spec.whatwg.org/#concept-event-dispatch for the full dispatch algorithm pub fn dispatch_event<'a, 'b>(target: JSRef<'a, EventTarget>, pseudo_target: Option>, event: JSRef) -> bool { assert!(!event.dispatching.get()); event.target.assign(Some(match pseudo_target { Some(pseudo_target) => pseudo_target, None => target.clone(), })); event.dispatching.set(true); let type_ = event.Type(); //TODO: no chain if not participating in a tree let mut chain: Vec> = if target.is_node() { let target_node: JSRef = NodeCast::to_ref(target).unwrap(); target_node.ancestors().map(|ancestor| { let ancestor_target: JSRef = EventTargetCast::from_ref(ancestor); JS::from_rooted(ancestor_target).root() }).collect() } else { vec!() }; event.phase.set(PhaseCapturing); //FIXME: The "callback this value" should be currentTarget /* capturing */ for cur_target in chain.as_slice().iter().rev() { let stopped = match cur_target.get_listeners_for(type_.as_slice(), Capturing) { Some(listeners) => { event.current_target.assign(Some(cur_target.deref().clone())); for listener in listeners.iter() { // Explicitly drop any exception on the floor. let _ = listener.HandleEvent_(**cur_target, event, ReportExceptions); if event.stop_immediate.get() { break; } } event.stop_propagation.get() } None => false }; if stopped { break; } } /* at target */ if !event.stop_propagation.get() { event.phase.set(PhaseAtTarget); event.current_target.assign(Some(target.clone())); let opt_listeners = target.get_listeners(type_.as_slice()); for listeners in opt_listeners.iter() { for listener in listeners.iter() { // Explicitly drop any exception on the floor. let _ = listener.HandleEvent_(target, event, ReportExceptions); if event.stop_immediate.get() { break; } } } } /* bubbling */ if event.bubbles.get() && !event.stop_propagation.get() { event.phase.set(PhaseBubbling); for cur_target in chain.iter() { let stopped = match cur_target.get_listeners_for(type_.as_slice(), Bubbling) { Some(listeners) => { event.current_target.assign(Some(cur_target.deref().clone())); for listener in listeners.iter() { // Explicitly drop any exception on the floor. let _ = listener.HandleEvent_(**cur_target, event, ReportExceptions); if event.stop_immediate.get() { break; } } event.stop_propagation.get() } None => false }; if stopped { break; } } } /* default action */ let target = event.GetTarget().root(); match target { Some(target) => { let node: Option> = NodeCast::to_ref(*target); match node { Some(node) => { let vtable = vtable_for(&node); vtable.handle_event(event); } None => {} } } None => {} } // Root ordering restrictions mean we need to unroot the chain entries // in the same order they were rooted. while chain.len() > 0 { let _ = chain.pop(); } event.dispatching.set(false); event.phase.set(PhaseNone); event.current_target.clear(); !event.DefaultPrevented() }