Generate code for handling callbacks. Implement add/removeEventListener and hacky dispatchEvent proof-of-concept.

This commit is contained in:
Josh Matthews 2013-11-01 10:19:45 -04:00
parent 06b1db8818
commit 7ecf5abbbd
18 changed files with 491 additions and 331 deletions

View file

@ -2,26 +2,96 @@
* 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::eReportExceptions;
use dom::bindings::codegen::EventTargetBinding;
use dom::bindings::utils::{Reflectable, Reflector};
use dom::bindings::utils::{Reflectable, Reflector, DOMString, Fallible};
use dom::bindings::codegen::EventListenerBinding::EventListener;
use dom::event::Event;
use script_task::page_from_context;
use js::jsapi::{JSObject, JSContext};
use std::hashmap::HashMap;
pub struct EventTarget {
reflector_: Reflector
reflector_: Reflector,
capturing_handlers: HashMap<~str, ~[EventListener]>,
bubbling_handlers: HashMap<~str, ~[EventListener]>
}
impl EventTarget {
pub fn new() -> ~EventTarget {
~EventTarget {
reflector_: Reflector::new()
pub fn new() -> EventTarget {
EventTarget {
reflector_: Reflector::new(),
capturing_handlers: HashMap::new(),
bubbling_handlers: HashMap::new(),
}
}
pub fn init_wrapper(@mut self, cx: *JSContext, scope: *JSObject) {
self.wrap_object_shared(cx, scope);
}
pub fn AddEventListener(&mut self,
ty: &DOMString,
listener: Option<EventListener>,
capture: bool) {
// TODO: Handle adding a listener during event dispatch: should not be invoked during
// current phase.
// (https://developer.mozilla.org/en-US/docs/Web/API/EventTarget.addEventListener#Adding_a_listener_during_event_dispatch)
for listener in listener.iter() {
let handlers = if capture {
&mut self.capturing_handlers
} else {
&mut self.bubbling_handlers
};
let entry = handlers.find_or_insert_with(ty.to_str(), |_| ~[]);
if entry.position_elem(listener).is_none() {
entry.push((*listener).clone());
}
}
}
pub fn RemoveEventListener(&mut self,
ty: &DOMString,
listener: Option<EventListener>,
capture: bool) {
for listener in listener.iter() {
let handlers = if capture {
&mut self.capturing_handlers
} else {
&mut self.bubbling_handlers
};
let mut entry = handlers.find_mut(&ty.to_str());
for entry in entry.mut_iter() {
let position = entry.position_elem(listener);
for &position in position.iter() {
entry.remove(position);
}
}
}
}
pub fn DispatchEvent(&self, event: @mut Event) -> Fallible<bool> {
//FIXME: get proper |this| object
let maybe_handlers = self.capturing_handlers.find(&event.type_);
for handlers in maybe_handlers.iter() {
for handler in handlers.iter() {
handler.HandleEvent__(event, eReportExceptions);
}
}
if event.bubbles {
let maybe_handlers = self.bubbling_handlers.find(&event.type_);
for handlers in maybe_handlers.iter() {
for handler in handlers.iter() {
handler.HandleEvent__(event, eReportExceptions);
}
}
}
Ok(!event.DefaultPrevented())
}
}
impl Reflectable for EventTarget {