2 * Copyright 2008-2009 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
30 #include "mshtml_private.h"
31 #include "htmlevent.h"
32 #include "htmlscript.h"
34 #include "wine/debug.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(mshtml
);
39 IDispatch
*handler_prop
;
41 IDispatch
*handlers
[0];
44 struct event_target_t
{
45 handler_vector_t
*event_table
[EVENTID_LAST
];
48 static const WCHAR abortW
[] = {'a','b','o','r','t',0};
49 static const WCHAR onabortW
[] = {'o','n','a','b','o','r','t',0};
51 static const WCHAR beforeunloadW
[] = {'b','e','f','o','r','e','u','n','l','o','a','d',0};
52 static const WCHAR onbeforeunloadW
[] = {'o','n','b','e','f','o','r','e','u','n','l','o','a','d',0};
54 static const WCHAR blurW
[] = {'b','l','u','r',0};
55 static const WCHAR onblurW
[] = {'o','n','b','l','u','r',0};
57 static const WCHAR changeW
[] = {'c','h','a','n','g','e',0};
58 static const WCHAR onchangeW
[] = {'o','n','c','h','a','n','g','e',0};
60 static const WCHAR clickW
[] = {'c','l','i','c','k',0};
61 static const WCHAR onclickW
[] = {'o','n','c','l','i','c','k',0};
63 static const WCHAR contextmenuW
[] = {'c','o','n','t','e','x','t','m','e','n','u',0};
64 static const WCHAR oncontextmenuW
[] = {'o','n','c','o','n','t','e','x','t','m','e','n','u',0};
66 static const WCHAR dataavailableW
[] = {'d','a','t','a','a','v','a','i','l','a','b','l','e',0};
67 static const WCHAR ondataavailableW
[] = {'o','n','d','a','t','a','a','v','a','i','l','a','b','l','e',0};
69 static const WCHAR dblclickW
[] = {'d','b','l','c','l','i','c','k',0};
70 static const WCHAR ondblclickW
[] = {'o','n','d','b','l','c','l','i','c','k',0};
72 static const WCHAR dragW
[] = {'d','r','a','g',0};
73 static const WCHAR ondragW
[] = {'o','n','d','r','a','g',0};
75 static const WCHAR dragstartW
[] = {'d','r','a','g','s','t','a','r','t',0};
76 static const WCHAR ondragstartW
[] = {'o','n','d','r','a','g','s','t','a','r','t',0};
78 static const WCHAR errorW
[] = {'e','r','r','o','r',0};
79 static const WCHAR onerrorW
[] = {'o','n','e','r','r','o','r',0};
81 static const WCHAR focusW
[] = {'f','o','c','u','s',0};
82 static const WCHAR onfocusW
[] = {'o','n','f','o','c','u','s',0};
84 static const WCHAR focusinW
[] = {'f','o','c','u','s','i','n',0};
85 static const WCHAR onfocusinW
[] = {'o','n','f','o','c','u','s','i','n',0};
87 static const WCHAR focusoutW
[] = {'f','o','c','u','s','o','u','t',0};
88 static const WCHAR onfocusoutW
[] = {'o','n','f','o','c','u','s','o','u','t',0};
90 static const WCHAR helpW
[] = {'h','e','l','p',0};
91 static const WCHAR onhelpW
[] = {'o','n','h','e','l','p',0};
93 static const WCHAR keydownW
[] = {'k','e','y','d','o','w','n',0};
94 static const WCHAR onkeydownW
[] = {'o','n','k','e','y','d','o','w','n',0};
96 static const WCHAR keypressW
[] = {'k','e','y','p','r','e','s','s',0};
97 static const WCHAR onkeypressW
[] = {'o','n','k','e','y','p','r','e','s','s',0};
99 static const WCHAR keyupW
[] = {'k','e','y','u','p',0};
100 static const WCHAR onkeyupW
[] = {'o','n','k','e','y','u','p',0};
102 static const WCHAR loadW
[] = {'l','o','a','d',0};
103 static const WCHAR onloadW
[] = {'o','n','l','o','a','d',0};
105 static const WCHAR messageW
[] = {'m','e','s','s','a','g','e',0};
106 static const WCHAR onmessageW
[] = {'o','n','m','e','s','s','a','g','e',0};
108 static const WCHAR mousedownW
[] = {'m','o','u','s','e','d','o','w','n',0};
109 static const WCHAR onmousedownW
[] = {'o','n','m','o','u','s','e','d','o','w','n',0};
111 static const WCHAR mousemoveW
[] = {'m','o','u','s','e','m','o','v','e',0};
112 static const WCHAR onmousemoveW
[] = {'o','n','m','o','u','s','e','m','o','v','e',0};
114 static const WCHAR mouseoutW
[] = {'m','o','u','s','e','o','u','t',0};
115 static const WCHAR onmouseoutW
[] = {'o','n','m','o','u','s','e','o','u','t',0};
117 static const WCHAR mouseoverW
[] = {'m','o','u','s','e','o','v','e','r',0};
118 static const WCHAR onmouseoverW
[] = {'o','n','m','o','u','s','e','o','v','e','r',0};
120 static const WCHAR mouseupW
[] = {'m','o','u','s','e','u','p',0};
121 static const WCHAR onmouseupW
[] = {'o','n','m','o','u','s','e','u','p',0};
123 static const WCHAR mousewheelW
[] = {'m','o','u','s','e','w','h','e','e','l',0};
124 static const WCHAR onmousewheelW
[] = {'o','n','m','o','u','s','e','w','h','e','e','l',0};
126 static const WCHAR pasteW
[] = {'p','a','s','t','e',0};
127 static const WCHAR onpasteW
[] = {'o','n','p','a','s','t','e',0};
129 static const WCHAR readystatechangeW
[] = {'r','e','a','d','y','s','t','a','t','e','c','h','a','n','g','e',0};
130 static const WCHAR onreadystatechangeW
[] = {'o','n','r','e','a','d','y','s','t','a','t','e','c','h','a','n','g','e',0};
132 static const WCHAR resizeW
[] = {'r','e','s','i','z','e',0};
133 static const WCHAR onresizeW
[] = {'o','n','r','e','s','i','z','e',0};
135 static const WCHAR scrollW
[] = {'s','c','r','o','l','l',0};
136 static const WCHAR onscrollW
[] = {'o','n','s','c','r','o','l','l',0};
138 static const WCHAR selectstartW
[] = {'s','e','l','e','c','t','s','t','a','r','t',0};
139 static const WCHAR onselectstartW
[] = {'o','n','s','e','l','e','c','t','s','t','a','r','t',0};
141 static const WCHAR submitW
[] = {'s','u','b','m','i','t',0};
142 static const WCHAR onsubmitW
[] = {'o','n','s','u','b','m','i','t',0};
144 static const WCHAR unloadW
[] = {'u','n','l','o','a','d',0};
145 static const WCHAR onunloadW
[] = {'o','n','u','n','l','o','a','d',0};
147 static const WCHAR HTMLEventsW
[] = {'H','T','M','L','E','v','e','n','t','s',0};
148 static const WCHAR KeyboardEventW
[] = {'K','e','y','b','o','a','r','d','E','v','e','n','t',0};
149 static const WCHAR MouseEventW
[] = {'M','o','u','s','e','E','v','e','n','t',0};
158 static const WCHAR
*event_types
[] = {
173 #define EVENT_DEFAULTLISTENER 0x0001
174 #define EVENT_BUBBLE 0x0002
175 #define EVENT_FORWARDBODY 0x0004
176 #define EVENT_BIND_TO_BODY 0x0008
177 #define EVENT_CANCELABLE 0x0010
178 #define EVENT_HASDEFAULTHANDLERS 0x0020
179 #define EVENT_FIXME 0x0040
181 static const event_info_t event_info
[] = {
182 {abortW
, onabortW
, EVENTT_NONE
, DISPID_EVMETH_ONABORT
,
184 {beforeunloadW
, onbeforeunloadW
, EVENTT_NONE
, DISPID_EVMETH_ONBEFOREUNLOAD
,
185 EVENT_DEFAULTLISTENER
|EVENT_FORWARDBODY
},
186 {blurW
, onblurW
, EVENTT_HTML
, DISPID_EVMETH_ONBLUR
,
187 EVENT_DEFAULTLISTENER
},
188 {changeW
, onchangeW
, EVENTT_HTML
, DISPID_EVMETH_ONCHANGE
,
189 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
190 {clickW
, onclickW
, EVENTT_MOUSE
, DISPID_EVMETH_ONCLICK
,
191 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
|EVENT_CANCELABLE
|EVENT_HASDEFAULTHANDLERS
},
192 {contextmenuW
, oncontextmenuW
, EVENTT_MOUSE
, DISPID_EVMETH_ONCONTEXTMENU
,
193 EVENT_BUBBLE
|EVENT_CANCELABLE
},
194 {dataavailableW
, ondataavailableW
, EVENTT_NONE
, DISPID_EVMETH_ONDATAAVAILABLE
,
196 {dblclickW
, ondblclickW
, EVENTT_MOUSE
, DISPID_EVMETH_ONDBLCLICK
,
197 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
|EVENT_CANCELABLE
},
198 {dragW
, ondragW
, EVENTT_MOUSE
, DISPID_EVMETH_ONDRAG
,
199 EVENT_FIXME
|EVENT_CANCELABLE
},
200 {dragstartW
, ondragstartW
, EVENTT_MOUSE
, DISPID_EVMETH_ONDRAGSTART
,
201 EVENT_FIXME
|EVENT_CANCELABLE
},
202 {errorW
, onerrorW
, EVENTT_NONE
, DISPID_EVMETH_ONERROR
,
204 {focusW
, onfocusW
, EVENTT_HTML
, DISPID_EVMETH_ONFOCUS
,
205 EVENT_DEFAULTLISTENER
},
206 {focusinW
, onfocusinW
, EVENTT_HTML
, DISPID_EVMETH_ONFOCUSIN
,
208 {focusoutW
, onfocusoutW
, EVENTT_HTML
, DISPID_EVMETH_ONFOCUSOUT
,
210 {helpW
, onhelpW
, EVENTT_KEY
, DISPID_EVMETH_ONHELP
,
211 EVENT_BUBBLE
|EVENT_CANCELABLE
},
212 {keydownW
, onkeydownW
, EVENTT_KEY
, DISPID_EVMETH_ONKEYDOWN
,
213 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
|EVENT_HASDEFAULTHANDLERS
},
214 {keypressW
, onkeypressW
, EVENTT_KEY
, DISPID_EVMETH_ONKEYPRESS
,
215 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
216 {keyupW
, onkeyupW
, EVENTT_KEY
, DISPID_EVMETH_ONKEYUP
,
217 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
218 {loadW
, onloadW
, EVENTT_HTML
, DISPID_EVMETH_ONLOAD
,
220 {messageW
, onmessageW
, EVENTT_NONE
, DISPID_EVMETH_ONMESSAGE
,
221 EVENT_FORWARDBODY
/* FIXME: remove when we get the target right */ },
222 {mousedownW
, onmousedownW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEDOWN
,
223 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
|EVENT_CANCELABLE
},
224 {mousemoveW
, onmousemoveW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEMOVE
,
225 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
226 {mouseoutW
, onmouseoutW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEOUT
,
227 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
228 {mouseoverW
, onmouseoverW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEOVER
,
229 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
230 {mouseupW
, onmouseupW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEUP
,
231 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
232 {mousewheelW
, onmousewheelW
, EVENTT_MOUSE
, DISPID_EVMETH_ONMOUSEWHEEL
,
234 {pasteW
, onpasteW
, EVENTT_NONE
, DISPID_EVMETH_ONPASTE
,
236 {readystatechangeW
, onreadystatechangeW
, EVENTT_NONE
, DISPID_EVMETH_ONREADYSTATECHANGE
,
238 {resizeW
, onresizeW
, EVENTT_NONE
, DISPID_EVMETH_ONRESIZE
,
239 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
240 {scrollW
, onscrollW
, EVENTT_HTML
, DISPID_EVMETH_ONSCROLL
,
241 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
},
242 {selectstartW
, onselectstartW
, EVENTT_MOUSE
, DISPID_EVMETH_ONSELECTSTART
,
244 {submitW
, onsubmitW
, EVENTT_HTML
, DISPID_EVMETH_ONSUBMIT
,
245 EVENT_DEFAULTLISTENER
|EVENT_BUBBLE
|EVENT_CANCELABLE
|EVENT_HASDEFAULTHANDLERS
},
246 {unloadW
, onunloadW
, EVENTT_HTML
, DISPID_EVMETH_ONUNLOAD
,
250 eventid_t
str_to_eid(LPCWSTR str
)
254 for(i
=0; i
< sizeof(event_info
)/sizeof(event_info
[0]); i
++) {
255 if(!strcmpW(event_info
[i
].name
, str
))
259 ERR("unknown type %s\n", debugstr_w(str
));
263 static eventid_t
attr_to_eid(LPCWSTR str
)
267 for(i
=0; i
< sizeof(event_info
)/sizeof(event_info
[0]); i
++) {
268 if(!strcmpW(event_info
[i
].attr_name
, str
))
275 struct HTMLEventObj
{
277 IHTMLEventObj IHTMLEventObj_iface
;
282 const event_info_t
*type
;
283 nsIDOMEvent
*nsevent
;
284 VARIANT return_value
;
285 BOOL prevent_default
;
289 static inline HTMLEventObj
*impl_from_IHTMLEventObj(IHTMLEventObj
*iface
)
291 return CONTAINING_RECORD(iface
, HTMLEventObj
, IHTMLEventObj_iface
);
294 static HRESULT WINAPI
HTMLEventObj_QueryInterface(IHTMLEventObj
*iface
, REFIID riid
, void **ppv
)
296 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
298 TRACE("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
300 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
301 *ppv
= &This
->IHTMLEventObj_iface
;
302 }else if(IsEqualGUID(&IID_IHTMLEventObj
, riid
)) {
303 *ppv
= &This
->IHTMLEventObj_iface
;
304 }else if(dispex_query_interface(&This
->dispex
, riid
, ppv
)) {
305 return *ppv
? S_OK
: E_NOINTERFACE
;
308 WARN("(%p)->(%s %p)\n", This
, debugstr_mshtml_guid(riid
), ppv
);
309 return E_NOINTERFACE
;
312 IUnknown_AddRef((IUnknown
*)*ppv
);
316 static ULONG WINAPI
HTMLEventObj_AddRef(IHTMLEventObj
*iface
)
318 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
319 LONG ref
= InterlockedIncrement(&This
->ref
);
321 TRACE("(%p) ref=%d\n", This
, ref
);
326 static ULONG WINAPI
HTMLEventObj_Release(IHTMLEventObj
*iface
)
328 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
329 LONG ref
= InterlockedDecrement(&This
->ref
);
331 TRACE("(%p) ref=%d\n", This
, ref
);
335 IHTMLDOMNode_Release(&This
->target
->IHTMLDOMNode_iface
);
337 nsIDOMEvent_Release(This
->nsevent
);
338 release_dispex(&This
->dispex
);
345 static HRESULT WINAPI
HTMLEventObj_GetTypeInfoCount(IHTMLEventObj
*iface
, UINT
*pctinfo
)
347 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
348 return IDispatchEx_GetTypeInfoCount(&This
->dispex
.IDispatchEx_iface
, pctinfo
);
351 static HRESULT WINAPI
HTMLEventObj_GetTypeInfo(IHTMLEventObj
*iface
, UINT iTInfo
,
352 LCID lcid
, ITypeInfo
**ppTInfo
)
354 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
355 return IDispatchEx_GetTypeInfo(&This
->dispex
.IDispatchEx_iface
, iTInfo
, lcid
, ppTInfo
);
358 static HRESULT WINAPI
HTMLEventObj_GetIDsOfNames(IHTMLEventObj
*iface
, REFIID riid
,
359 LPOLESTR
*rgszNames
, UINT cNames
,
360 LCID lcid
, DISPID
*rgDispId
)
362 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
363 return IDispatchEx_GetIDsOfNames(&This
->dispex
.IDispatchEx_iface
, riid
, rgszNames
, cNames
,
367 static HRESULT WINAPI
HTMLEventObj_Invoke(IHTMLEventObj
*iface
, DISPID dispIdMember
,
368 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
369 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
371 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
372 return IDispatchEx_Invoke(&This
->dispex
.IDispatchEx_iface
, dispIdMember
, riid
, lcid
,
373 wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
376 static HRESULT WINAPI
HTMLEventObj_get_srcElement(IHTMLEventObj
*iface
, IHTMLElement
**p
)
378 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
380 TRACE("(%p)->(%p)\n", This
, p
);
384 IHTMLDOMNode_QueryInterface(&This
->target
->IHTMLDOMNode_iface
, &IID_IHTMLElement
, (void**)p
);
388 static HRESULT WINAPI
HTMLEventObj_get_altKey(IHTMLEventObj
*iface
, VARIANT_BOOL
*p
)
390 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
391 cpp_bool ret
= FALSE
;
393 TRACE("(%p)->(%p)\n", This
, p
);
396 nsIDOMKeyEvent
*key_event
;
399 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMKeyEvent
, (void**)&key_event
);
400 if(NS_SUCCEEDED(nsres
)) {
401 nsIDOMKeyEvent_GetAltKey(key_event
, &ret
);
402 nsIDOMKeyEvent_Release(key_event
);
404 nsIDOMMouseEvent
*mouse_event
;
406 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
407 if(NS_SUCCEEDED(nsres
)) {
408 nsIDOMMouseEvent_GetAltKey(mouse_event
, &ret
);
409 nsIDOMMouseEvent_Release(mouse_event
);
414 *p
= ret
? VARIANT_TRUE
: VARIANT_FALSE
;
418 static HRESULT WINAPI
HTMLEventObj_get_ctrlKey(IHTMLEventObj
*iface
, VARIANT_BOOL
*p
)
420 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
421 cpp_bool ret
= FALSE
;
423 TRACE("(%p)->(%p)\n", This
, p
);
426 nsIDOMKeyEvent
*key_event
;
429 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMKeyEvent
, (void**)&key_event
);
430 if(NS_SUCCEEDED(nsres
)) {
431 nsIDOMKeyEvent_GetCtrlKey(key_event
, &ret
);
432 nsIDOMKeyEvent_Release(key_event
);
434 nsIDOMMouseEvent
*mouse_event
;
436 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
437 if(NS_SUCCEEDED(nsres
)) {
438 nsIDOMMouseEvent_GetCtrlKey(mouse_event
, &ret
);
439 nsIDOMMouseEvent_Release(mouse_event
);
444 *p
= ret
? VARIANT_TRUE
: VARIANT_FALSE
;
448 static HRESULT WINAPI
HTMLEventObj_get_shiftKey(IHTMLEventObj
*iface
, VARIANT_BOOL
*p
)
450 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
451 cpp_bool ret
= FALSE
;
453 TRACE("(%p)->(%p)\n", This
, p
);
456 nsIDOMKeyEvent
*key_event
;
459 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMKeyEvent
, (void**)&key_event
);
460 if(NS_SUCCEEDED(nsres
)) {
461 nsIDOMKeyEvent_GetShiftKey(key_event
, &ret
);
462 nsIDOMKeyEvent_Release(key_event
);
464 nsIDOMMouseEvent
*mouse_event
;
466 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
467 if(NS_SUCCEEDED(nsres
)) {
468 nsIDOMMouseEvent_GetShiftKey(mouse_event
, &ret
);
469 nsIDOMMouseEvent_Release(mouse_event
);
474 *p
= ret
? VARIANT_TRUE
: VARIANT_FALSE
;
478 static HRESULT WINAPI
HTMLEventObj_put_returnValue(IHTMLEventObj
*iface
, VARIANT v
)
480 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
482 TRACE("(%p)->(%s)\n", This
, debugstr_variant(&v
));
484 if(V_VT(&v
) != VT_BOOL
) {
485 FIXME("unsupported value %s\n", debugstr_variant(&v
));
486 return DISP_E_BADVARTYPE
;
489 This
->return_value
= v
;
491 This
->prevent_default
= TRUE
;
495 static HRESULT WINAPI
HTMLEventObj_get_returnValue(IHTMLEventObj
*iface
, VARIANT
*p
)
497 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
499 TRACE("(%p)->(%p)\n", This
, p
);
502 return VariantCopy(p
, &This
->return_value
);
505 static HRESULT WINAPI
HTMLEventObj_put_cancelBubble(IHTMLEventObj
*iface
, VARIANT_BOOL v
)
507 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
509 TRACE("(%p)->(%x)\n", This
, v
);
511 This
->cancel_bubble
= !!v
;
515 static HRESULT WINAPI
HTMLEventObj_get_cancelBubble(IHTMLEventObj
*iface
, VARIANT_BOOL
*p
)
517 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
519 TRACE("(%p)->(%p)\n", This
, p
);
521 *p
= This
->cancel_bubble
? VARIANT_TRUE
: VARIANT_FALSE
;
525 static HRESULT WINAPI
HTMLEventObj_get_fromElement(IHTMLEventObj
*iface
, IHTMLElement
**p
)
527 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
529 FIXME("(%p)->(%p)\n", This
, p
);
535 static HRESULT WINAPI
HTMLEventObj_get_toElement(IHTMLEventObj
*iface
, IHTMLElement
**p
)
537 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
539 FIXME("(%p)->(%p)\n", This
, p
);
545 static HRESULT WINAPI
HTMLEventObj_put_keyCode(IHTMLEventObj
*iface
, LONG v
)
547 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
548 FIXME("(%p)->(%d)\n", This
, v
);
552 static HRESULT WINAPI
HTMLEventObj_get_keyCode(IHTMLEventObj
*iface
, LONG
*p
)
554 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
557 TRACE("(%p)->(%p)\n", This
, p
);
560 nsIDOMKeyEvent
*key_event
;
563 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMKeyEvent
, (void**)&key_event
);
564 if(NS_SUCCEEDED(nsres
)) {
565 nsIDOMKeyEvent_GetKeyCode(key_event
, &key_code
);
566 nsIDOMKeyEvent_Release(key_event
);
574 static HRESULT WINAPI
HTMLEventObj_get_button(IHTMLEventObj
*iface
, LONG
*p
)
576 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
579 TRACE("(%p)->(%p)\n", This
, p
);
582 nsIDOMMouseEvent
*mouse_event
;
585 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
586 if(NS_SUCCEEDED(nsres
)) {
587 nsIDOMMouseEvent_GetButton(mouse_event
, &button
);
588 nsIDOMMouseEvent_Release(mouse_event
);
596 static HRESULT WINAPI
HTMLEventObj_get_type(IHTMLEventObj
*iface
, BSTR
*p
)
598 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
600 TRACE("(%p)->(%p)\n", This
, p
);
607 *p
= SysAllocString(This
->type
->name
);
608 return *p
? S_OK
: E_OUTOFMEMORY
;
611 static HRESULT WINAPI
HTMLEventObj_get_qualifier(IHTMLEventObj
*iface
, BSTR
*p
)
613 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
615 FIXME("(%p)->(%p)\n", This
, p
);
621 static HRESULT WINAPI
HTMLEventObj_get_reason(IHTMLEventObj
*iface
, LONG
*p
)
623 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
625 FIXME("(%p)->(%p)\n", This
, p
);
631 static HRESULT WINAPI
HTMLEventObj_get_x(IHTMLEventObj
*iface
, LONG
*p
)
633 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
636 TRACE("(%p)->(%p)\n", This
, p
);
639 nsIDOMUIEvent
*ui_event
;
642 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMUIEvent
, (void**)&ui_event
);
643 if(NS_SUCCEEDED(nsres
)) {
644 /* NOTE: pageX is not exactly right here. */
645 nsres
= nsIDOMUIEvent_GetPageX(ui_event
, &x
);
646 assert(nsres
== NS_OK
);
647 nsIDOMUIEvent_Release(ui_event
);
655 static HRESULT WINAPI
HTMLEventObj_get_y(IHTMLEventObj
*iface
, LONG
*p
)
657 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
660 TRACE("(%p)->(%p)\n", This
, p
);
663 nsIDOMUIEvent
*ui_event
;
666 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMUIEvent
, (void**)&ui_event
);
667 if(NS_SUCCEEDED(nsres
)) {
668 /* NOTE: pageY is not exactly right here. */
669 nsres
= nsIDOMUIEvent_GetPageY(ui_event
, &y
);
670 assert(nsres
== NS_OK
);
671 nsIDOMUIEvent_Release(ui_event
);
679 static HRESULT WINAPI
HTMLEventObj_get_clientX(IHTMLEventObj
*iface
, LONG
*p
)
681 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
684 TRACE("(%p)->(%p)\n", This
, p
);
687 nsIDOMMouseEvent
*mouse_event
;
690 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
691 if(NS_SUCCEEDED(nsres
)) {
692 nsIDOMMouseEvent_GetClientX(mouse_event
, &x
);
693 nsIDOMMouseEvent_Release(mouse_event
);
701 static HRESULT WINAPI
HTMLEventObj_get_clientY(IHTMLEventObj
*iface
, LONG
*p
)
703 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
706 TRACE("(%p)->(%p)\n", This
, p
);
709 nsIDOMMouseEvent
*mouse_event
;
712 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
713 if(NS_SUCCEEDED(nsres
)) {
714 nsIDOMMouseEvent_GetClientY(mouse_event
, &y
);
715 nsIDOMMouseEvent_Release(mouse_event
);
723 static HRESULT WINAPI
HTMLEventObj_get_offsetX(IHTMLEventObj
*iface
, LONG
*p
)
725 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
727 FIXME("(%p)->(%p)\n", This
, p
);
733 static HRESULT WINAPI
HTMLEventObj_get_offsetY(IHTMLEventObj
*iface
, LONG
*p
)
735 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
737 FIXME("(%p)->(%p)\n", This
, p
);
743 static HRESULT WINAPI
HTMLEventObj_get_screenX(IHTMLEventObj
*iface
, LONG
*p
)
745 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
748 TRACE("(%p)->(%p)\n", This
, p
);
751 nsIDOMMouseEvent
*mouse_event
;
754 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
755 if(NS_SUCCEEDED(nsres
)) {
756 nsIDOMMouseEvent_GetScreenX(mouse_event
, &x
);
757 nsIDOMMouseEvent_Release(mouse_event
);
765 static HRESULT WINAPI
HTMLEventObj_get_screenY(IHTMLEventObj
*iface
, LONG
*p
)
767 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
770 TRACE("(%p)->(%p)\n", This
, p
);
773 nsIDOMMouseEvent
*mouse_event
;
776 nsres
= nsIDOMEvent_QueryInterface(This
->nsevent
, &IID_nsIDOMMouseEvent
, (void**)&mouse_event
);
777 if(NS_SUCCEEDED(nsres
)) {
778 nsIDOMMouseEvent_GetScreenY(mouse_event
, &y
);
779 nsIDOMMouseEvent_Release(mouse_event
);
787 static HRESULT WINAPI
HTMLEventObj_get_srcFilter(IHTMLEventObj
*iface
, IDispatch
**p
)
789 HTMLEventObj
*This
= impl_from_IHTMLEventObj(iface
);
791 FIXME("(%p)->(%p)\n", This
, p
);
797 static const IHTMLEventObjVtbl HTMLEventObjVtbl
= {
798 HTMLEventObj_QueryInterface
,
800 HTMLEventObj_Release
,
801 HTMLEventObj_GetTypeInfoCount
,
802 HTMLEventObj_GetTypeInfo
,
803 HTMLEventObj_GetIDsOfNames
,
805 HTMLEventObj_get_srcElement
,
806 HTMLEventObj_get_altKey
,
807 HTMLEventObj_get_ctrlKey
,
808 HTMLEventObj_get_shiftKey
,
809 HTMLEventObj_put_returnValue
,
810 HTMLEventObj_get_returnValue
,
811 HTMLEventObj_put_cancelBubble
,
812 HTMLEventObj_get_cancelBubble
,
813 HTMLEventObj_get_fromElement
,
814 HTMLEventObj_get_toElement
,
815 HTMLEventObj_put_keyCode
,
816 HTMLEventObj_get_keyCode
,
817 HTMLEventObj_get_button
,
818 HTMLEventObj_get_type
,
819 HTMLEventObj_get_qualifier
,
820 HTMLEventObj_get_reason
,
823 HTMLEventObj_get_clientX
,
824 HTMLEventObj_get_clientY
,
825 HTMLEventObj_get_offsetX
,
826 HTMLEventObj_get_offsetY
,
827 HTMLEventObj_get_screenX
,
828 HTMLEventObj_get_screenY
,
829 HTMLEventObj_get_srcFilter
832 static inline HTMLEventObj
*unsafe_impl_from_IHTMLEventObj(IHTMLEventObj
*iface
)
834 return iface
->lpVtbl
== &HTMLEventObjVtbl
? impl_from_IHTMLEventObj(iface
) : NULL
;
837 static const tid_t HTMLEventObj_iface_tids
[] = {
842 static dispex_static_data_t HTMLEventObj_dispex
= {
845 HTMLEventObj_iface_tids
848 static HTMLEventObj
*create_event(void)
852 ret
= heap_alloc_zero(sizeof(*ret
));
856 ret
->IHTMLEventObj_iface
.lpVtbl
= &HTMLEventObjVtbl
;
859 init_dispex(&ret
->dispex
, (IUnknown
*)&ret
->IHTMLEventObj_iface
, &HTMLEventObj_dispex
);
864 static HRESULT
set_event_info(HTMLEventObj
*event
, HTMLDOMNode
*target
, eventid_t eid
, nsIDOMEvent
*nsevent
)
866 event
->type
= event_info
+eid
;
867 event
->nsevent
= nsevent
;
870 nsIDOMEvent_AddRef(nsevent
);
871 }else if(event_types
[event_info
[eid
].type
]) {
875 nsAString_InitDepend(&type_str
, event_types
[event_info
[eid
].type
]);
876 nsres
= nsIDOMHTMLDocument_CreateEvent(target
->doc
->nsdoc
, &type_str
, &event
->nsevent
);
877 nsAString_Finish(&type_str
);
878 if(NS_FAILED(nsres
)) {
879 ERR("Could not create event: %08x\n", nsres
);
884 event
->target
= target
;
886 IHTMLDOMNode_AddRef(&target
->IHTMLDOMNode_iface
);
890 HRESULT
create_event_obj(IHTMLEventObj
**ret
)
894 event
= create_event();
896 return E_OUTOFMEMORY
;
898 *ret
= &event
->IHTMLEventObj_iface
;
902 static inline event_target_t
*get_event_target_data(EventTarget
*event_target
, BOOL alloc
)
904 const dispex_static_data_vtbl_t
*vtbl
= dispex_get_vtbl(&event_target
->dispex
);
905 event_target_t
**ptr
;
907 ptr
= vtbl
&& vtbl
->get_event_target_ptr
908 ? vtbl
->get_event_target_ptr(&event_target
->dispex
)
909 : &event_target
->ptr
;
913 return *ptr
= heap_alloc_zero(sizeof(event_target_t
));
916 static HRESULT
call_disp_func(IDispatch
*disp
, DISPPARAMS
*dp
, VARIANT
*retv
)
922 memset(&ei
, 0, sizeof(ei
));
924 hres
= IDispatch_QueryInterface(disp
, &IID_IDispatchEx
, (void**)&dispex
);
925 if(SUCCEEDED(hres
)) {
926 hres
= IDispatchEx_InvokeEx(dispex
, 0, GetUserDefaultLCID(), DISPATCH_METHOD
, dp
, retv
, &ei
, NULL
);
927 IDispatchEx_Release(dispex
);
929 TRACE("Could not get IDispatchEx interface: %08x\n", hres
);
930 hres
= IDispatch_Invoke(disp
, 0, &IID_NULL
, GetUserDefaultLCID(), DISPATCH_METHOD
,
931 dp
, retv
, &ei
, NULL
);
937 static HRESULT
call_cp_func(IDispatch
*disp
, DISPID dispid
, HTMLEventObj
*event_obj
, VARIANT
*retv
)
939 DISPPARAMS dp
= {NULL
,NULL
,0,0};
945 V_VT(&event_arg
) = VT_DISPATCH
;
946 V_DISPATCH(&event_arg
) = (IDispatch
*)&event_obj
->IHTMLEventObj_iface
;
947 dp
.rgvarg
= &event_arg
;
951 memset(&ei
, 0, sizeof(ei
));
952 return IDispatch_Invoke(disp
, dispid
, &IID_NULL
, 0, DISPATCH_METHOD
, &dp
, retv
, &ei
, &argerr
);
955 static BOOL
is_cp_event(cp_static_data_t
*data
, DISPID dispid
)
964 hres
= get_dispids(data
->tid
, &data
->id_cnt
, &data
->ids
);
970 max
= data
->id_cnt
-1;
973 if(data
->ids
[i
] == dispid
)
976 if(data
->ids
[i
] < dispid
)
985 void call_event_handlers(HTMLDocumentNode
*doc
, HTMLEventObj
*event_obj
, EventTarget
*event_target
,
986 ConnectionPointContainer
*cp_container
, eventid_t eid
, IDispatch
*this_obj
)
988 event_target_t
*data
= get_event_target_data(event_target
, FALSE
);
989 const BOOL cancelable
= event_info
[eid
].flags
& EVENT_CANCELABLE
;
993 if(data
&& data
->event_table
[eid
] && data
->event_table
[eid
]->handler_prop
) {
994 DISPID named_arg
= DISPID_THIS
;
996 DISPPARAMS dp
= {&arg
, &named_arg
, 1, 1};
998 V_VT(&arg
) = VT_DISPATCH
;
999 V_DISPATCH(&arg
) = this_obj
;
1000 V_VT(&v
) = VT_EMPTY
;
1002 TRACE("%s >>>\n", debugstr_w(event_info
[eid
].name
));
1003 hres
= call_disp_func(data
->event_table
[eid
]->handler_prop
, &dp
, &v
);
1005 TRACE("%s <<< %s\n", debugstr_w(event_info
[eid
].name
), debugstr_variant(&v
));
1008 if(V_VT(&v
) == VT_BOOL
) {
1010 event_obj
->prevent_default
= TRUE
;
1011 }else if(V_VT(&v
) != VT_EMPTY
) {
1012 FIXME("unhandled result %s\n", debugstr_variant(&v
));
1017 WARN("%s <<< %08x\n", debugstr_w(event_info
[eid
].name
), hres
);
1021 if(data
&& data
->event_table
[eid
] && data
->event_table
[eid
]->handler_cnt
) {
1023 DISPPARAMS dp
= {&arg
, NULL
, 1, 0};
1026 V_VT(&arg
) = VT_DISPATCH
;
1027 V_DISPATCH(&arg
) = (IDispatch
*)&event_obj
->dispex
.IDispatchEx_iface
;
1029 i
= data
->event_table
[eid
]->handler_cnt
;
1031 if(data
->event_table
[eid
]->handlers
[i
]) {
1032 V_VT(&v
) = VT_EMPTY
;
1034 TRACE("%s [%d] >>>\n", debugstr_w(event_info
[eid
].name
), i
);
1035 hres
= call_disp_func(data
->event_table
[eid
]->handlers
[i
], &dp
, &v
);
1037 TRACE("%s [%d] <<<\n", debugstr_w(event_info
[eid
].name
), i
);
1040 if(V_VT(&v
) == VT_BOOL
) {
1042 event_obj
->prevent_default
= TRUE
;
1043 }else if(V_VT(&v
) != VT_EMPTY
) {
1044 FIXME("unhandled result %s\n", debugstr_variant(&v
));
1049 WARN("%s [%d] <<< %08x\n", debugstr_w(event_info
[eid
].name
), i
, hres
);
1056 * NOTE: CP events may require doc_obj reference, which we don't own. We make sure that
1057 * it's safe to call event handler by checking nsevent_listener, which is NULL for
1058 * detached documents.
1060 if(cp_container
&& cp_container
->forward_container
)
1061 cp_container
= cp_container
->forward_container
;
1062 if(cp_container
&& cp_container
->cps
&& doc
->nsevent_listener
) {
1063 ConnectionPoint
*cp
;
1066 for(j
=0; cp_container
->cp_entries
[j
].riid
; j
++) {
1067 cp
= cp_container
->cps
+ j
;
1068 if(!cp
->sinks_size
|| !is_cp_event(cp
->data
, event_info
[eid
].dispid
))
1071 for(i
=0; doc
->nsevent_listener
&& i
< cp
->sinks_size
; i
++) {
1072 if(!cp
->sinks
[i
].disp
)
1075 V_VT(&v
) = VT_EMPTY
;
1077 TRACE("cp %s [%u] >>>\n", debugstr_w(event_info
[eid
].name
), i
);
1078 hres
= call_cp_func(cp
->sinks
[i
].disp
, event_info
[eid
].dispid
,
1079 cp
->data
->pass_event_arg
? event_obj
: NULL
, &v
);
1081 TRACE("cp %s [%u] <<<\n", debugstr_w(event_info
[eid
].name
), i
);
1084 if(V_VT(&v
) == VT_BOOL
) {
1086 event_obj
->prevent_default
= TRUE
;
1087 }else if(V_VT(&v
) != VT_EMPTY
) {
1088 FIXME("unhandled result %s\n", debugstr_variant(&v
));
1093 WARN("cp %s [%u] <<< %08x\n", debugstr_w(event_info
[eid
].name
), i
, hres
);
1097 if(!doc
->nsevent_listener
)
1103 static void fire_event_obj(HTMLDocumentNode
*doc
, eventid_t eid
, HTMLEventObj
*event_obj
,
1104 HTMLDOMNode
*target
, IDispatch
*script_this
)
1106 IHTMLEventObj
*prev_event
;
1107 nsIDOMNode
*parent
, *nsnode
;
1108 BOOL prevent_default
= FALSE
;
1109 HTMLInnerWindow
*window
;
1115 TRACE("(%p) %s\n", doc
, debugstr_w(event_info
[eid
].name
));
1117 window
= doc
->window
;
1119 WARN("NULL window\n");
1123 htmldoc_addref(&doc
->basedoc
);
1125 prev_event
= window
->event
;
1126 window
->event
= event_obj
? &event_obj
->IHTMLEventObj_iface
: NULL
;
1128 nsIDOMNode_GetNodeType(target
->nsnode
, &node_type
);
1129 nsnode
= target
->nsnode
;
1130 nsIDOMNode_AddRef(nsnode
);
1135 hres
= get_node(doc
, nsnode
, FALSE
, &node
);
1136 if(SUCCEEDED(hres
) && node
) {
1137 call_event_handlers(doc
, event_obj
, &node
->event_target
, node
->cp_container
, eid
,
1138 script_this
? script_this
: (IDispatch
*)&node
->IHTMLDOMNode_iface
);
1142 if(!(event_info
[eid
].flags
& EVENT_BUBBLE
) || (event_obj
&& event_obj
->cancel_bubble
))
1145 nsIDOMNode_GetParentNode(nsnode
, &parent
);
1146 nsIDOMNode_Release(nsnode
);
1151 nsIDOMNode_GetNodeType(nsnode
, &node_type
);
1152 }while(node_type
== ELEMENT_NODE
);
1154 if(!(event_info
[eid
].flags
& EVENT_BUBBLE
) || (event_obj
&& event_obj
->cancel_bubble
))
1158 if(event_info
[eid
].flags
& EVENT_FORWARDBODY
) {
1159 nsIDOMHTMLElement
*nsbody
;
1162 nsres
= nsIDOMHTMLDocument_GetBody(doc
->nsdoc
, &nsbody
);
1163 if(NS_SUCCEEDED(nsres
) && nsbody
) {
1164 hres
= get_node(doc
, (nsIDOMNode
*)nsbody
, FALSE
, &node
);
1165 if(SUCCEEDED(hres
) && node
) {
1166 call_event_handlers(doc
, event_obj
, &node
->event_target
, node
->cp_container
, eid
,
1167 script_this
? script_this
: (IDispatch
*)&node
->IHTMLDOMNode_iface
);
1170 nsIDOMHTMLElement_Release(nsbody
);
1172 ERR("Could not get body: %08x\n", nsres
);
1176 call_event_handlers(doc
, event_obj
, &doc
->node
.event_target
, &doc
->basedoc
.cp_container
, eid
,
1177 script_this
? script_this
: (IDispatch
*)&doc
->basedoc
.IHTMLDocument2_iface
);
1181 FIXME("unimplemented node type %d\n", node_type
);
1185 nsIDOMNode_Release(nsnode
);
1187 if(event_obj
&& event_obj
->prevent_default
)
1188 prevent_default
= TRUE
;
1189 window
->event
= prev_event
;
1191 if(!prevent_default
&& (event_info
[eid
].flags
& EVENT_HASDEFAULTHANDLERS
)) {
1192 nsnode
= target
->nsnode
;
1193 nsIDOMNode_AddRef(nsnode
);
1196 hres
= get_node(doc
, nsnode
, TRUE
, &node
);
1201 if(node
->vtbl
->handle_event
)
1202 hres
= node
->vtbl
->handle_event(node
, eid
, event_obj
? event_obj
->nsevent
: NULL
, &prevent_default
);
1204 if(FAILED(hres
) || prevent_default
|| (event_obj
&& event_obj
->cancel_bubble
))
1208 nsres
= nsIDOMNode_GetParentNode(nsnode
, &parent
);
1209 if(NS_FAILED(nsres
))
1212 nsIDOMNode_Release(nsnode
);
1217 nsIDOMNode_Release(nsnode
);
1220 if(prevent_default
&& event_obj
&& event_obj
->nsevent
) {
1221 TRACE("calling PreventDefault\n");
1222 nsIDOMEvent_PreventDefault(event_obj
->nsevent
);
1225 htmldoc_release(&doc
->basedoc
);
1228 void fire_event(HTMLDocumentNode
*doc
, eventid_t eid
, BOOL set_event
, HTMLDOMNode
*target
, nsIDOMEvent
*nsevent
,
1229 IDispatch
*script_this
)
1231 HTMLEventObj
*event_obj
= NULL
;
1235 event_obj
= create_event();
1239 hres
= set_event_info(event_obj
, target
, eid
, nsevent
);
1241 IHTMLEventObj_Release(&event_obj
->IHTMLEventObj_iface
);
1246 fire_event_obj(doc
, eid
, event_obj
, target
, script_this
);
1249 IHTMLEventObj_Release(&event_obj
->IHTMLEventObj_iface
);
1252 HRESULT
dispatch_event(HTMLDOMNode
*node
, const WCHAR
*event_name
, VARIANT
*event_var
, VARIANT_BOOL
*cancelled
)
1254 HTMLEventObj
*event_obj
= NULL
;
1258 eid
= attr_to_eid(event_name
);
1259 if(eid
== EVENTID_LAST
) {
1260 WARN("unknown event %s\n", debugstr_w(event_name
));
1261 return E_INVALIDARG
;
1264 if(event_var
&& V_VT(event_var
) != VT_EMPTY
&& V_VT(event_var
) != VT_ERROR
) {
1265 if(V_VT(event_var
) != VT_DISPATCH
) {
1266 FIXME("event_var %s not supported\n", debugstr_variant(event_var
));
1270 if(V_DISPATCH(event_var
)) {
1271 IHTMLEventObj
*event_iface
;
1273 hres
= IDispatch_QueryInterface(V_DISPATCH(event_var
), &IID_IHTMLEventObj
, (void**)&event_iface
);
1275 FIXME("No IHTMLEventObj iface\n");
1279 event_obj
= unsafe_impl_from_IHTMLEventObj(event_iface
);
1281 ERR("Not our IHTMLEventObj?\n");
1282 IHTMLEventObj_Release(event_iface
);
1289 hres
= set_event_info(event_obj
, node
, eid
, NULL
);
1291 fire_event_obj(node
->doc
, eid
, event_obj
, node
, NULL
);
1293 IHTMLEventObj_Release(&event_obj
->IHTMLEventObj_iface
);
1297 fire_event(node
->doc
, eid
, TRUE
, node
, NULL
, NULL
);
1300 *cancelled
= VARIANT_TRUE
; /* FIXME */
1304 HRESULT
call_fire_event(HTMLDOMNode
*node
, eventid_t eid
)
1308 if(node
->vtbl
->fire_event
) {
1309 BOOL handled
= FALSE
;
1311 hres
= node
->vtbl
->fire_event(node
, eid
, &handled
);
1316 fire_event(node
->doc
, eid
, TRUE
, node
, NULL
, NULL
);
1320 static BOOL
alloc_handler_vector(event_target_t
*event_target
, eventid_t eid
, int cnt
)
1322 handler_vector_t
*new_vector
, *handler_vector
= event_target
->event_table
[eid
];
1324 if(handler_vector
) {
1325 if(cnt
<= handler_vector
->handler_cnt
)
1328 new_vector
= heap_realloc_zero(handler_vector
, sizeof(handler_vector_t
) + sizeof(IDispatch
*)*cnt
);
1330 new_vector
= heap_alloc_zero(sizeof(handler_vector_t
) + sizeof(IDispatch
*)*cnt
);
1336 new_vector
->handler_cnt
= cnt
;
1337 event_target
->event_table
[eid
] = new_vector
;
1341 HRESULT
ensure_doc_nsevent_handler(HTMLDocumentNode
*doc
, eventid_t eid
)
1343 nsIDOMNode
*nsnode
= NULL
;
1345 TRACE("%s\n", debugstr_w(event_info
[eid
].name
));
1351 case EVENTID_FOCUSIN
:
1352 doc
->event_vector
[eid
] = TRUE
;
1353 eid
= EVENTID_FOCUS
;
1355 case EVENTID_FOCUSOUT
:
1356 doc
->event_vector
[eid
] = TRUE
;
1363 if(doc
->event_vector
[eid
] || !(event_info
[eid
].flags
& (EVENT_DEFAULTLISTENER
|EVENT_BIND_TO_BODY
)))
1366 if(event_info
[eid
].flags
& EVENT_BIND_TO_BODY
) {
1367 nsnode
= doc
->node
.nsnode
;
1368 nsIDOMNode_AddRef(nsnode
);
1371 doc
->event_vector
[eid
] = TRUE
;
1372 add_nsevent_listener(doc
, nsnode
, event_info
[eid
].name
);
1375 nsIDOMNode_Release(nsnode
);
1379 void detach_events(HTMLDocumentNode
*doc
)
1381 if(doc
->event_vector
) {
1384 for(i
=0; i
< EVENTID_LAST
; i
++) {
1385 if(doc
->event_vector
[i
]) {
1386 detach_nsevent(doc
, event_info
[i
].name
);
1387 doc
->event_vector
[i
] = FALSE
;
1392 release_nsevents(doc
);
1395 /* Caller should ensure that it's called only once for given event in the target. */
1396 static void bind_event(EventTarget
*event_target
, eventid_t eid
)
1398 const dispex_static_data_vtbl_t
*vtbl
= dispex_get_vtbl(&event_target
->dispex
);
1399 if(vtbl
->bind_event
)
1400 vtbl
->bind_event(&event_target
->dispex
, eid
);
1402 FIXME("Unsupported event binding on target %p\n", event_target
);
1405 static void remove_event_handler(EventTarget
*event_target
, eventid_t eid
)
1407 event_target_t
*data
;
1411 hres
= dispex_get_dprop_ref(&event_target
->dispex
, event_info
[eid
].attr_name
, FALSE
, &store
);
1413 VariantClear(store
);
1415 data
= get_event_target_data(event_target
, FALSE
);
1416 if(data
&& data
->event_table
[eid
] && data
->event_table
[eid
]->handler_prop
) {
1417 IDispatch_Release(data
->event_table
[eid
]->handler_prop
);
1418 data
->event_table
[eid
]->handler_prop
= NULL
;
1422 static HRESULT
set_event_handler_disp(EventTarget
*event_target
, eventid_t eid
, IDispatch
*disp
)
1424 event_target_t
*data
;
1426 if(event_info
[eid
].flags
& EVENT_FIXME
)
1427 FIXME("unimplemented event %s\n", debugstr_w(event_info
[eid
].name
));
1429 remove_event_handler(event_target
, eid
);
1433 data
= get_event_target_data(event_target
, TRUE
);
1435 return E_OUTOFMEMORY
;
1437 if(!data
->event_table
[eid
]) {
1438 if(!alloc_handler_vector(data
, eid
, 0))
1439 return E_OUTOFMEMORY
;
1441 bind_event(event_target
, eid
);
1442 }else if(data
->event_table
[eid
]->handler_prop
) {
1443 IDispatch_Release(data
->event_table
[eid
]->handler_prop
);
1446 data
->event_table
[eid
]->handler_prop
= disp
;
1447 IDispatch_AddRef(disp
);
1451 HRESULT
set_event_handler(EventTarget
*event_target
, eventid_t eid
, VARIANT
*var
)
1455 remove_event_handler(event_target
, eid
);
1459 return set_event_handler_disp(event_target
, eid
, V_DISPATCH(var
));
1466 * Setting event handler to string is a rare case and we don't want to
1467 * complicate nor increase memory of event_target_t for that. Instead,
1468 * we store the value in DispatchEx, which can already handle custom
1471 remove_event_handler(event_target
, eid
);
1473 hres
= dispex_get_dprop_ref(&event_target
->dispex
, event_info
[eid
].attr_name
, TRUE
, &v
);
1477 V_BSTR(v
) = SysAllocString(V_BSTR(var
));
1479 return E_OUTOFMEMORY
;
1485 FIXME("not handler %s\n", debugstr_variant(var
));
1494 HRESULT
get_event_handler(EventTarget
*event_target
, eventid_t eid
, VARIANT
*var
)
1496 event_target_t
*data
;
1500 hres
= dispex_get_dprop_ref(&event_target
->dispex
, event_info
[eid
].attr_name
, FALSE
, &v
);
1501 if(SUCCEEDED(hres
) && V_VT(v
) != VT_EMPTY
) {
1502 V_VT(var
) = VT_EMPTY
;
1503 return VariantCopy(var
, v
);
1506 data
= get_event_target_data(event_target
, FALSE
);
1507 if(data
&& data
->event_table
[eid
] && data
->event_table
[eid
]->handler_prop
) {
1508 V_VT(var
) = VT_DISPATCH
;
1509 V_DISPATCH(var
) = data
->event_table
[eid
]->handler_prop
;
1510 IDispatch_AddRef(V_DISPATCH(var
));
1512 V_VT(var
) = VT_NULL
;
1518 HRESULT
attach_event(EventTarget
*event_target
, BSTR name
, IDispatch
*disp
, VARIANT_BOOL
*res
)
1520 event_target_t
*data
;
1524 eid
= attr_to_eid(name
);
1525 if(eid
== EVENTID_LAST
) {
1526 WARN("Unknown event\n");
1527 *res
= VARIANT_TRUE
;
1531 if(event_info
[eid
].flags
& EVENT_FIXME
)
1532 FIXME("unimplemented event %s\n", debugstr_w(event_info
[eid
].name
));
1534 data
= get_event_target_data(event_target
, TRUE
);
1536 return E_OUTOFMEMORY
;
1538 if(data
->event_table
[eid
]) {
1539 while(i
< data
->event_table
[eid
]->handler_cnt
&& data
->event_table
[eid
]->handlers
[i
])
1541 if(i
== data
->event_table
[eid
]->handler_cnt
&& !alloc_handler_vector(data
, eid
, i
+1))
1542 return E_OUTOFMEMORY
;
1543 }else if(alloc_handler_vector(data
, eid
, i
+1)) {
1544 bind_event(event_target
, eid
);
1546 return E_OUTOFMEMORY
;
1549 IDispatch_AddRef(disp
);
1550 data
->event_table
[eid
]->handlers
[i
] = disp
;
1552 *res
= VARIANT_TRUE
;
1556 HRESULT
detach_event(EventTarget
*event_target
, BSTR name
, IDispatch
*disp
)
1558 event_target_t
*data
;
1562 eid
= attr_to_eid(name
);
1563 if(eid
== EVENTID_LAST
) {
1564 WARN("Unknown event\n");
1568 data
= get_event_target_data(event_target
, FALSE
);
1572 if(!data
->event_table
[eid
])
1575 while(i
< data
->event_table
[eid
]->handler_cnt
) {
1576 if(data
->event_table
[eid
]->handlers
[i
] == disp
) {
1577 IDispatch_Release(data
->event_table
[eid
]->handlers
[i
]);
1578 data
->event_table
[eid
]->handlers
[i
] = NULL
;
1586 void bind_target_event(HTMLDocumentNode
*doc
, EventTarget
*event_target
, const WCHAR
*event
, IDispatch
*disp
)
1590 TRACE("(%p %p %s %p)\n", doc
, event_target
, debugstr_w(event
), disp
);
1592 eid
= attr_to_eid(event
);
1593 if(eid
== EVENTID_LAST
) {
1594 WARN("Unsupported event %s\n", debugstr_w(event
));
1598 set_event_handler_disp(event_target
, eid
, disp
);
1601 void update_doc_cp_events(HTMLDocumentNode
*doc
, cp_static_data_t
*cp
)
1605 for(i
=0; i
< EVENTID_LAST
; i
++) {
1606 if((event_info
[i
].flags
& EVENT_DEFAULTLISTENER
) && is_cp_event(cp
, event_info
[i
].dispid
))
1607 ensure_doc_nsevent_handler(doc
, i
);
1611 void check_event_attr(HTMLDocumentNode
*doc
, nsIDOMHTMLElement
*nselem
)
1613 const PRUnichar
*attr_value
;
1614 nsAString attr_value_str
;
1621 for(i
=0; i
< EVENTID_LAST
; i
++) {
1622 nsres
= get_elem_attr_value(nselem
, event_info
[i
].attr_name
, &attr_value_str
, &attr_value
);
1623 if(NS_SUCCEEDED(nsres
)) {
1627 TRACE("%p.%s = %s\n", nselem
, debugstr_w(event_info
[i
].attr_name
), debugstr_w(attr_value
));
1629 disp
= script_parse_event(doc
->window
, attr_value
);
1631 hres
= get_node(doc
, (nsIDOMNode
*)nselem
, TRUE
, &node
);
1632 if(SUCCEEDED(hres
)) {
1633 set_event_handler_disp(&node
->event_target
, i
, disp
);
1636 IDispatch_Release(disp
);
1638 nsAString_Finish(&attr_value_str
);
1643 HRESULT
doc_init_events(HTMLDocumentNode
*doc
)
1648 doc
->event_vector
= heap_alloc_zero(EVENTID_LAST
*sizeof(BOOL
));
1649 if(!doc
->event_vector
)
1650 return E_OUTOFMEMORY
;
1654 for(i
=0; i
< EVENTID_LAST
; i
++) {
1655 if(event_info
[i
].flags
& EVENT_HASDEFAULTHANDLERS
) {
1656 hres
= ensure_doc_nsevent_handler(doc
, i
);
1665 void release_event_target(event_target_t
*event_target
)
1670 for(i
=0; i
< EVENTID_LAST
; i
++) {
1671 if(event_target
->event_table
[i
]) {
1672 if(event_target
->event_table
[i
]->handler_prop
)
1673 IDispatch_Release(event_target
->event_table
[i
]->handler_prop
);
1674 for(j
=0; j
< event_target
->event_table
[i
]->handler_cnt
; j
++)
1675 if(event_target
->event_table
[i
]->handlers
[j
])
1676 IDispatch_Release(event_target
->event_table
[i
]->handlers
[j
]);
1680 heap_free(event_target
);