aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJosh Matthews <josh@joshmatthews.net>2013-10-02 00:51:05 -0400
committerJosh Matthews <josh@joshmatthews.net>2013-11-05 12:58:28 -0500
commit88f5c2b1333d69feed4198b5e13b3314f17091e5 (patch)
tree8f99942a78fccd5c2bfe5342a184364bafff4ce1
parentbb97fd13f38090c460d79dad3322ab4b7e325a82 (diff)
downloadservo-88f5c2b1333d69feed4198b5e13b3314f17091e5.tar.gz
servo-88f5c2b1333d69feed4198b5e13b3314f17091e5.zip
Add basic event dispatch with bubbling, capturing, and propagation interruption.
-rw-r--r--src/components/script/dom/bindings/utils.rs3
-rw-r--r--src/components/script/dom/event.rs45
-rw-r--r--src/components/script/dom/eventdispatcher.rs111
-rw-r--r--src/components/script/dom/eventtarget.rs124
-rw-r--r--src/components/script/dom/node.rs11
-rw-r--r--src/components/script/dom/window.rs4
-rw-r--r--src/components/script/script.rc1
-rw-r--r--src/test/html/content/test_Event.html2
-rw-r--r--src/test/html/content/test_event_dispatch.html51
-rw-r--r--src/test/html/content/test_event_dispatch_dynamic.html21
-rw-r--r--src/test/html/content/test_event_dispatch_order.html42
-rw-r--r--src/test/html/content/test_event_listener.html35
12 files changed, 380 insertions, 70 deletions
diff --git a/src/components/script/dom/bindings/utils.rs b/src/components/script/dom/bindings/utils.rs
index 32639d706e7..7eb70ff145f 100644
--- a/src/components/script/dom/bindings/utils.rs
+++ b/src/components/script/dom/bindings/utils.rs
@@ -768,7 +768,8 @@ pub enum Error {
NotFound,
HierarchyRequest,
InvalidCharacter,
- NotSupported
+ NotSupported,
+ InvalidState
}
pub type Fallible<T> = Result<T, Error>;
diff --git a/src/components/script/dom/event.rs b/src/components/script/dom/event.rs
index b8ff54f5755..02acca10937 100644
--- a/src/components/script/dom/event.rs
+++ b/src/components/script/dom/event.rs
@@ -31,6 +31,13 @@ pub struct AbstractEvent {
event: *mut Box<Event>
}
+pub enum EventPhase {
+ Phase_None = 0,
+ Phase_Capturing,
+ Phase_At_Target,
+ Phase_Bubbling
+}
+
impl AbstractEvent {
pub fn from_box(box: *mut Box<Event>) -> AbstractEvent {
AbstractEvent {
@@ -95,6 +102,14 @@ impl AbstractEvent {
assert!(self.is_mouseevent());
self.transmute_mut()
}
+
+ pub fn propagation_stopped(&self) -> bool {
+ self.event().stop_propagation
+ }
+
+ pub fn bubbles(&self) -> bool {
+ self.event().bubbles
+ }
}
impl DerivedWrapper for AbstractEvent {
@@ -138,11 +153,18 @@ pub enum EventTypeId {
pub struct Event {
type_id: EventTypeId,
reflector_: Reflector,
+ current_target: Option<AbstractEventTarget>,
+ target: Option<AbstractEventTarget>,
type_: ~str,
+ phase: EventPhase,
default_prevented: bool,
+ stop_propagation: bool,
+ stop_immediate: bool,
cancelable: bool,
bubbles: bool,
trusted: bool,
+ dispatching: bool,
+ initialized: bool
}
impl Event {
@@ -150,11 +172,18 @@ impl Event {
Event {
type_id: type_id,
reflector_: Reflector::new(),
+ current_target: None,
+ target: None,
+ phase: Phase_None,
type_: ~"",
default_prevented: false,
cancelable: true,
bubbles: true,
- trusted: false
+ trusted: false,
+ dispatching: false,
+ stop_propagation: false,
+ stop_immediate: false,
+ initialized: false,
}
}
@@ -173,7 +202,7 @@ impl Event {
}
pub fn EventPhase(&self) -> u16 {
- 0
+ self.phase as u16
}
pub fn Type(&self) -> DOMString {
@@ -181,11 +210,11 @@ impl Event {
}
pub fn GetTarget(&self) -> Option<AbstractEventTarget> {
- None
+ self.target
}
pub fn GetCurrentTarget(&self) -> Option<AbstractEventTarget> {
- None
+ self.current_target
}
pub fn DefaultPrevented(&self) -> bool {
@@ -193,13 +222,18 @@ impl Event {
}
pub fn PreventDefault(&mut self) {
- self.default_prevented = true
+ if self.cancelable {
+ self.default_prevented = true
+ }
}
pub fn StopPropagation(&mut self) {
+ self.stop_propagation = true;
}
pub fn StopImmediatePropagation(&mut self) {
+ self.stop_immediate = true;
+ self.stop_propagation = true;
}
pub fn Bubbles(&self) -> bool {
@@ -221,6 +255,7 @@ impl Event {
self.type_ = null_str_as_word_null(type_);
self.cancelable = cancelable;
self.bubbles = bubbles;
+ self.initialized = true;
Ok(())
}
diff --git a/src/components/script/dom/eventdispatcher.rs b/src/components/script/dom/eventdispatcher.rs
new file mode 100644
index 00000000000..ba8e06795f5
--- /dev/null
+++ b/src/components/script/dom/eventdispatcher.rs
@@ -0,0 +1,111 @@
+/* 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::eReportExceptions;
+use dom::eventtarget::{AbstractEventTarget, Capturing, Bubbling};
+use dom::event::{AbstractEvent, Phase_At_Target, Phase_None, Phase_Bubbling, Phase_Capturing};
+use dom::node::AbstractNode;
+use servo_util::tree::{TreeNodeRef};
+
+// See http://dom.spec.whatwg.org/#concept-event-dispatch for the full dispatch algorithm
+pub fn dispatch_event(target: AbstractEventTarget, event: AbstractEvent) -> bool {
+ assert!(!event.event().dispatching);
+
+ {
+ let event = event.mut_event();
+ event.target = Some(target);
+ event.dispatching = true;
+ }
+
+ let type_ = event.event().type_.clone();
+ let mut chain = ~[];
+
+ //TODO: no chain if not participating in a tree
+ if target.is_node() {
+ for ancestor in AbstractNode::from_eventtarget(target).ancestors() {
+ chain.push(AbstractEventTarget::from_node(ancestor));
+ }
+ }
+
+ event.mut_event().phase = Phase_Capturing;
+
+ //FIXME: The "callback this value" should be currentTarget
+
+ /* capturing */
+ for &cur_target in chain.rev_iter() {
+ //XXX bad clone
+ let stopped = match cur_target.eventtarget().get_listeners_for(type_.clone(), Capturing) {
+ Some(listeners) => {
+ event.mut_event().current_target = Some(cur_target);
+ for listener in listeners.iter() {
+ listener.HandleEvent__(event, eReportExceptions);
+
+ if event.event().stop_immediate {
+ break;
+ }
+ }
+
+ event.propagation_stopped()
+ }
+ None => false
+ };
+
+ if stopped {
+ break;
+ }
+ }
+
+ /* at target */
+ if !event.propagation_stopped() {
+ {
+ let event = event.mut_event();
+ event.phase = Phase_At_Target;
+ event.current_target = Some(target);
+ }
+
+ let opt_listeners = target.eventtarget().get_listeners(type_.clone());
+ for listeners in opt_listeners.iter() {
+ for listener in listeners.iter() {
+ listener.HandleEvent__(event, eReportExceptions);
+ if event.event().stop_immediate {
+ break;
+ }
+ }
+ }
+ }
+
+ /* bubbling */
+ if event.bubbles() && !event.propagation_stopped() {
+ event.mut_event().phase = Phase_Bubbling;
+
+ for &cur_target in chain.iter() {
+ //XXX bad clone
+ let stopped = match cur_target.eventtarget().get_listeners_for(type_.clone(), Bubbling) {
+ Some(listeners) => {
+ event.mut_event().current_target = Some(cur_target);
+ for listener in listeners.iter() {
+ listener.HandleEvent__(event, eReportExceptions);
+
+ if event.event().stop_immediate {
+ break;
+ }
+ }
+
+ event.propagation_stopped()
+ }
+ None => false
+ };
+ if stopped {
+ break;
+ }
+ }
+ }
+
+ let event = event.mut_event();
+ event.dispatching = false;
+ event.phase = Phase_None;
+ event.current_target = None;
+
+ !event.DefaultPrevented()
+}
diff --git a/src/components/script/dom/eventtarget.rs b/src/components/script/dom/eventtarget.rs
index 4a33b1eba2d..d477d0f8f47 100644
--- a/src/components/script/dom/eventtarget.rs
+++ b/src/components/script/dom/eventtarget.rs
@@ -2,12 +2,12 @@
* 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, DOMString, Fallible, DerivedWrapper};
-use dom::bindings::utils::null_str_as_word_null;
+use dom::bindings::utils::{null_str_as_word_null, InvalidState};
use dom::bindings::codegen::EventListenerBinding::EventListener;
use dom::event::AbstractEvent;
+use dom::eventdispatcher::dispatch_event;
use dom::node::{AbstractNode, ScriptView};
use script_task::page_from_context;
@@ -18,10 +18,28 @@ use std::cast;
use std::hashmap::HashMap;
use std::unstable::raw::Box;
+#[deriving(Eq)]
+pub enum ListenerPhase {
+ Capturing,
+ Bubbling,
+}
+
+#[deriving(Eq)]
+pub enum EventTargetTypeId {
+ WindowTypeId,
+ NodeTypeId
+}
+
+#[deriving(Eq)]
+struct EventListenerEntry {
+ phase: ListenerPhase,
+ listener: EventListener
+}
+
pub struct EventTarget {
+ type_id: EventTargetTypeId,
reflector_: Reflector,
- capturing_handlers: HashMap<~str, ~[EventListener]>,
- bubbling_handlers: HashMap<~str, ~[EventListener]>
+ handlers: HashMap<~str, ~[EventListenerEntry]>,
}
pub struct AbstractEventTarget {
@@ -29,9 +47,9 @@ pub struct AbstractEventTarget {
}
impl AbstractEventTarget {
- pub fn from_box(box: *mut Box<EventTarget>) -> AbstractEventTarget {
+ pub fn from_box<T>(box: *mut Box<T>) -> AbstractEventTarget {
AbstractEventTarget {
- eventtarget: box
+ eventtarget: box as *mut Box<EventTarget>
}
}
@@ -41,6 +59,18 @@ impl AbstractEventTarget {
}
}
+ pub fn type_id(&self) -> EventTargetTypeId {
+ self.eventtarget().type_id
+ }
+
+ pub fn is_window(&self) -> bool {
+ self.type_id() == WindowTypeId
+ }
+
+ pub fn is_node(&self) -> bool {
+ self.type_id() == NodeTypeId
+ }
+
//
// Downcasting borrows
//
@@ -59,11 +89,11 @@ impl AbstractEventTarget {
}
}
- fn eventtarget<'a>(&'a self) -> &'a EventTarget {
+ pub fn eventtarget<'a>(&'a self) -> &'a EventTarget {
self.transmute()
}
- fn mut_eventtarget<'a>(&'a mut self) -> &'a mut EventTarget {
+ pub fn mut_eventtarget<'a>(&'a mut self) -> &'a mut EventTarget {
self.transmute_mut()
}
}
@@ -99,35 +129,41 @@ impl Reflectable for AbstractEventTarget {
}
impl EventTarget {
- pub fn new() -> EventTarget {
+ pub fn new_inherited(type_id: EventTargetTypeId) -> EventTarget {
EventTarget {
+ type_id: type_id,
reflector_: Reflector::new(),
- capturing_handlers: HashMap::new(),
- bubbling_handlers: HashMap::new(),
+ handlers: HashMap::new(),
+ }
+ }
+
+ pub fn get_listeners(&self, type_: ~str) -> Option<~[EventListener]> {
+ do self.handlers.find_equiv(&type_).map |listeners| {
+ listeners.iter().map(|entry| entry.listener).collect()
}
}
- pub fn init_wrapper(@mut self, cx: *JSContext, scope: *JSObject) {
- self.wrap_object_shared(cx, scope);
+ pub fn get_listeners_for(&self, type_: ~str, desired_phase: ListenerPhase)
+ -> Option<~[EventListener]> {
+ do self.handlers.find_equiv(&type_).map |listeners| {
+ let filtered = listeners.iter().filter(|entry| entry.phase == desired_phase);
+ filtered.map(|entry| entry.listener).collect()
+ }
}
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
+ for &listener in listener.iter() {
+ let entry = self.handlers.find_or_insert_with(null_str_as_word_null(ty), |_| ~[]);
+ let phase = if capture { Capturing } else { Bubbling };
+ let new_entry = EventListenerEntry {
+ phase: phase,
+ listener: listener
};
- let entry = handlers.find_or_insert_with(null_str_as_word_null(ty), |_| ~[]);
- if entry.position_elem(listener).is_none() {
- entry.push((*listener).clone());
+ if entry.position_elem(&new_entry).is_none() {
+ entry.push(new_entry);
}
}
}
@@ -136,15 +172,15 @@ impl EventTarget {
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(&null_str_as_word_null(ty));
+ for &listener in listener.iter() {
+ let mut entry = self.handlers.find_mut(&null_str_as_word_null(ty));
for entry in entry.mut_iter() {
- let position = entry.position_elem(listener);
+ let phase = if capture { Capturing } else { Bubbling };
+ let old_entry = EventListenerEntry {
+ phase: phase,
+ listener: listener
+ };
+ let position = entry.position_elem(&old_entry);
for &position in position.iter() {
entry.remove(position);
}
@@ -152,25 +188,11 @@ impl EventTarget {
}
}
- pub fn DispatchEvent(&self, _abstract_self: AbstractEventTarget, event: AbstractEvent) -> Fallible<bool> {
- //FIXME: get proper |this| object
-
- let type_ = event.event().type_.clone();
- let maybe_handlers = self.capturing_handlers.find(&type_);
- for handlers in maybe_handlers.iter() {
- for handler in handlers.iter() {
- handler.HandleEvent__(event, eReportExceptions);
- }
- }
- if event.event().bubbles {
- let maybe_handlers = self.bubbling_handlers.find(&type_);
- for handlers in maybe_handlers.iter() {
- for handler in handlers.iter() {
- handler.HandleEvent__(event, eReportExceptions);
- }
- }
+ pub fn DispatchEvent(&self, abstract_self: AbstractEventTarget, event: AbstractEvent) -> Fallible<bool> {
+ if event.event().dispatching || !event.event().initialized {
+ return Err(InvalidState);
}
- Ok(!event.event().DefaultPrevented())
+ Ok(dispatch_event(abstract_self, event))
}
}
diff --git a/src/components/script/dom/node.rs b/src/components/script/dom/node.rs
index 2e331cc82f9..ba74a580049 100644
--- a/src/components/script/dom/node.rs
+++ b/src/components/script/dom/node.rs
@@ -12,7 +12,7 @@ use dom::document::{AbstractDocument, DocumentTypeId};
use dom::documenttype::DocumentType;
use dom::element::{Element, ElementTypeId, HTMLImageElementTypeId, HTMLIframeElementTypeId};
use dom::element::{HTMLStyleElementTypeId};
-use dom::eventtarget::EventTarget;
+use dom::eventtarget::{AbstractEventTarget, EventTarget, NodeTypeId};
use dom::nodelist::{NodeList};
use dom::htmlimageelement::HTMLImageElement;
use dom::htmliframeelement::HTMLIFrameElement;
@@ -211,6 +211,13 @@ impl<'self, View> AbstractNode<View> {
}
}
+ pub fn from_eventtarget(target: AbstractEventTarget) -> AbstractNode<View> {
+ assert!(target.is_node());
+ unsafe {
+ cast::transmute(target)
+ }
+ }
+
// Convenience accessors
/// Returns the type ID of this node. Fails if this node is borrowed mutably.
@@ -522,7 +529,7 @@ impl Node<ScriptView> {
fn new_(type_id: NodeTypeId, doc: Option<AbstractDocument>) -> Node<ScriptView> {
Node {
- eventtarget: EventTarget::new(),
+ eventtarget: EventTarget::new_inherited(NodeTypeId),
type_id: type_id,
abstract: None,
diff --git a/src/components/script/dom/window.rs b/src/components/script/dom/window.rs
index 46565b7fb09..3701c22554d 100644
--- a/src/components/script/dom/window.rs
+++ b/src/components/script/dom/window.rs
@@ -6,7 +6,7 @@ use dom::bindings::codegen::WindowBinding;
use dom::bindings::utils::{Reflectable, Reflector};
use dom::bindings::utils::{DOMString, null_str_as_empty, Traceable};
use dom::document::AbstractDocument;
-use dom::eventtarget::EventTarget;
+use dom::eventtarget::{EventTarget, WindowTypeId};
use dom::node::{AbstractNode, ScriptView};
use dom::navigator::Navigator;
@@ -205,7 +205,7 @@ impl Window {
image_cache_task: ImageCacheTask)
-> @mut Window {
let win = @mut Window {
- eventtarget: EventTarget::new(),
+ eventtarget: EventTarget::new_inherited(WindowTypeId),
page: page,
script_chan: script_chan.clone(),
compositor: compositor,
diff --git a/src/components/script/script.rc b/src/components/script/script.rc
index 675796e2c47..0307e70b580 100644
--- a/src/components/script/script.rc
+++ b/src/components/script/script.rc
@@ -55,6 +55,7 @@ pub mod dom {
pub mod domparser;
pub mod element;
pub mod event;
+ pub mod eventdispatcher;
pub mod eventtarget;
pub mod formdata;
pub mod htmlanchorelement;
diff --git a/src/test/html/content/test_Event.html b/src/test/html/content/test_Event.html
index 7523c843206..0643df62e44 100644
--- a/src/test/html/content/test_Event.html
+++ b/src/test/html/content/test_Event.html
@@ -4,7 +4,7 @@
<script>
is_function(Event, "Event");
-let ev = new Event("foopy");
+let ev = new Event("foopy", {cancelable: true});
is_a(ev, Event);
is(ev.type, 'foopy');
diff --git a/src/test/html/content/test_event_dispatch.html b/src/test/html/content/test_event_dispatch.html
new file mode 100644
index 00000000000..cf9eb4ee675
--- /dev/null
+++ b/src/test/html/content/test_event_dispatch.html
@@ -0,0 +1,51 @@
+<html>
+<head>
+<script src="harness.js"></script>
+</head>
+<body>
+<span>Paragraph containing <div>event listener</div>.</span>
+<script>
+ var bodyTimes = 0;
+ function bodyListener(ev) {
+ bodyTimes++;
+ is(ev.currentTarget, document.getElementsByTagName('body')[0]);
+ is(ev.target, document.getElementsByTagName('div')[0]);
+ if (bodyTimes == 1) {
+ is(ev.eventPhase, ev.CAPTURING_PHASE);
+ } else if (bodyTimes == 2) {
+ is(ev.eventPhase, ev.BUBBLING_PHASE);
+ }
+ }
+
+ var spanTimes = 0;
+ function spanListener(ev) {
+ is(ev.currentTarget, document.getElementsByTagName('span')[0]);
+ is(ev.target, document.getElementsByTagName('div')[0]);
+ is(ev.eventPhase, ev.BUBBLING_PHASE);
+ spanTimes++;
+ }
+
+ var divTimes = 0;
+ function divListener(ev) {
+ var self = document.getElementsByTagName('div')[0];
+ is(ev.currentTarget, self);
+ is(ev.target, self);
+ is(ev.eventPhase, ev.AT_TARGET);
+ divTimes++;
+ }
+
+ document.getElementsByTagName('body')[0].addEventListener("foopy", bodyListener, true);
+ document.getElementsByTagName('body')[0].addEventListener("foopy", bodyListener, false);
+ document.getElementsByTagName('span')[0].addEventListener("foopy", spanListener, false);
+ document.getElementsByTagName('div')[0].addEventListener("foopy", divListener, false);
+ var ev = new Event('foopy', {bubbles: true});
+ is(ev.bubbles, true);
+ document.getElementsByTagName('div')[0].dispatchEvent(ev);
+ is(bodyTimes, 2, 'body listener should be called multiple times');
+ is(divTimes, 1, 'target listener should be called once');
+ is(spanTimes, 1, 'span listener should be called while bubbling');
+
+ finish();
+</script>
+</body>
+</html>
diff --git a/src/test/html/content/test_event_dispatch_dynamic.html b/src/test/html/content/test_event_dispatch_dynamic.html
new file mode 100644
index 00000000000..3b852fe497d
--- /dev/null
+++ b/src/test/html/content/test_event_dispatch_dynamic.html
@@ -0,0 +1,21 @@
+<!DOCTYPE html>
+<script src="harness.js"></script>
+<b><b><b></b></b></b>
+<script>
+var sawmiddle = -1;
+var sawouter = -1;
+var step = 0;
+var outerb = document.getElementsByTagName('b')[0];
+var middleb = outerb.firstChild;
+var innerb = middleb.firstChild;
+outerb.addEventListener("x", function() {
+ middleb.addEventListener("x", function() {
+ sawmiddle = step++;
+ }, true);
+ sawouter = step++;
+}, true);
+innerb.dispatchEvent(new Event("x"));
+is(sawmiddle, 1);
+is(sawouter, 0);
+finish();
+</script>
diff --git a/src/test/html/content/test_event_dispatch_order.html b/src/test/html/content/test_event_dispatch_order.html
new file mode 100644
index 00000000000..e1b381d0b77
--- /dev/null
+++ b/src/test/html/content/test_event_dispatch_order.html
@@ -0,0 +1,42 @@
+<html>
+<head>
+<script src="harness.js"></script>
+</head>
+<body>
+<div id="foo"></div>
+<script>
+ var sawBubble = false;
+ var sawCapture = false;
+ var sawBubbleTwice = false;
+ function handler(ev) {
+ is(ev.eventPhase, ev.AT_TARGET);
+ is(sawBubble, false);
+ is(sawCapture, false);
+ sawBubble = true;
+ }
+ function handler2(ev) {
+ is(ev.eventPhase, ev.AT_TARGET);
+ is(sawBubble, true);
+ is(sawCapture, false);
+ sawCapture = true;
+ }
+ function handler3(ev) {
+ is(ev.eventPhase, ev.AT_TARGET);
+ is(sawBubble, true);
+ is(sawCapture, true);
+ sawBubbleTwice = true;
+ }
+
+ var target = document.getElementById('foo');
+ target.addEventListener('foopy', handler, false);
+ target.addEventListener('foopy', handler2, true);
+ target.addEventListener('foopy', handler3, false);
+ var ev = new Event('foopy', {bubbles: true});
+ target.dispatchEvent(ev);
+ is(sawBubble, true);
+ is(sawCapture, true);
+ is(sawBubbleTwice, true);
+ finish();
+</script>
+</body>
+</html>
diff --git a/src/test/html/content/test_event_listener.html b/src/test/html/content/test_event_listener.html
index 414f8e8a156..5096d76349a 100644
--- a/src/test/html/content/test_event_listener.html
+++ b/src/test/html/content/test_event_listener.html
@@ -4,16 +4,35 @@
</head>
<body>
<script>
- var saw_event = false;
- function onFoopy() {
+ function onFoopy(ev) {
window.removeEventListener('foopy', onFoopy);
- saw_event = true;
+ is(ev instanceof expected, true);
+ is(ev.type, 'foopy');
}
- window.addEventListener('foopy', onFoopy);
- var ev = document.createEvent('HTMLEvents');
- ev.initEvent('foopy', true, true);
- window.dispatchEvent(ev);
- is(saw_event, true);
+
+ var expected;
+ var events = [['HTMLEvents', Event, function(ev) { ev.initEvent('foopy', true, true); }],
+ ['UIEvents', UIEvent, function(ev) { ev.initUIEvent('foopy', true, true, null, 0); }],
+ ['MouseEvents', MouseEvent,
+ function(ev) { ev.initMouseEvent('foopy', true, true, null, 0,
+ 0, 0, 0, 0, false, false,
+ false, false, 0, null); }]];
+ for (var i = 0; i < events.length; i++) {
+ addEventListener('foopy', onFoopy);
+ expected = events[i][1];
+ var ev = document.createEvent(events[i][0]);
+ events[i][2](ev);
+ window.dispatchEvent(ev);
+ }
+
+ var constructors = [Event, UIEvent, MouseEvent];
+ for (var i = 0; i < constructors.length; i++) {
+ addEventListener('foopy', onFoopy);
+ expected = constructors[i];
+ var ev = new constructors[i]('foopy', {cancelable: true, bubbles: true});
+ window.dispatchEvent(ev);
+ }
+
finish();
</script>
</body>