dinput: Fix printing NULL strings.
[wine] / dlls / mshtml / tests / events.c
1 /*
2  * Copyright 2008-2011 Jacek Caban for CodeWeavers
3  *
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.
8  *
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.
13  *
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
17  */
18
19 #define COBJMACROS
20 #define CONST_VTABLE
21
22 #include <wine/test.h>
23 #include <stdarg.h>
24 #include <stdio.h>
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "ole2.h"
29 #include "mshtml.h"
30 #include "mshtmdid.h"
31 #include "docobj.h"
32 #include "hlink.h"
33 #include "dispex.h"
34
35 #define DEFINE_EXPECT(func) \
36     static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
37
38 #define SET_EXPECT(func) \
39     do { called_ ## func = FALSE; expect_ ## func = TRUE; } while(0)
40
41 #define CHECK_EXPECT2(func) \
42     do { \
43     trace(#func "\n"); \
44         ok(expect_ ##func, "unexpected call " #func "\n"); \
45         called_ ## func = TRUE; \
46     }while(0)
47
48 #define CHECK_EXPECT(func) \
49     do { \
50         CHECK_EXPECT2(func); \
51         expect_ ## func = FALSE; \
52     }while(0)
53
54 #define CHECK_CALLED(func) \
55     do { \
56         ok(called_ ## func, "expected " #func "\n"); \
57         expect_ ## func = called_ ## func = FALSE; \
58     }while(0)
59
60 DEFINE_EXPECT(document_onclick);
61 DEFINE_EXPECT(body_onclick);
62 DEFINE_EXPECT(doc_onclick_attached);
63 DEFINE_EXPECT(div_onclick);
64 DEFINE_EXPECT(div_onclick_attached);
65 DEFINE_EXPECT(timeout);
66 DEFINE_EXPECT(doccp_onclick);
67 DEFINE_EXPECT(doccp_onclick_cancel);
68 DEFINE_EXPECT(div_onclick_disp);
69 DEFINE_EXPECT(invoke_onclick);
70 DEFINE_EXPECT(iframe_onreadystatechange_loading);
71 DEFINE_EXPECT(iframe_onreadystatechange_interactive);
72 DEFINE_EXPECT(iframe_onreadystatechange_complete);
73 DEFINE_EXPECT(iframedoc_onreadystatechange);
74 DEFINE_EXPECT(img_onload);
75 DEFINE_EXPECT(input_onfocus);
76 DEFINE_EXPECT(form_onsubmit);
77 DEFINE_EXPECT(form_onclick);
78 DEFINE_EXPECT(submit_onclick);
79 DEFINE_EXPECT(submit_onclick_attached);
80 DEFINE_EXPECT(submit_onclick_setret);
81
82 static HWND container_hwnd = NULL;
83 static IHTMLWindow2 *window;
84 static IOleDocumentView *view;
85 static BOOL xy_todo;
86
87 typedef struct {
88     LONG x;
89     LONG y;
90     LONG offset_x;
91     LONG offset_y;
92 } xy_test_t;
93
94 static const xy_test_t no_xy = {-10,-10,-10,-10};
95 static const xy_test_t zero_xy = {0,0,0,0};
96
97 static const char empty_doc_str[] =
98     "<html></html>";
99
100 static const char click_doc_str[] =
101     "<html><body>"
102     "<div id=\"clickdiv\" style=\"text-align: center; background: red; font-size: 32\">click here</div>"
103     "</body></html>";
104
105 static const char readystate_doc_str[] =
106     "<html><body><iframe id=\"iframe\"></iframe></body></html>";
107
108 static const char img_doc_str[] =
109     "<html><body><img id=\"imgid\"></img></body></html>";
110
111 static const char input_doc_str[] =
112     "<html><body><input id=\"inputid\"></input></body></html>";    
113
114 static const char iframe_doc_str[] =
115     "<html><body><iframe id=\"ifr\">Testing</iframe></body></html>";
116
117 static const char form_doc_str[] =
118     "<html><body><form id=\"formid\" method=\"post\" action=\"about:blank\">"
119     "<input type=\"text\" value=\"test\" name=\"i\"/>"
120     "<input type=\"submit\" id=\"submitid\" />"
121     "</form></body></html>";
122
123 static const char *debugstr_guid(REFIID riid)
124 {
125     static char buf[50];
126
127     sprintf(buf, "{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
128             riid->Data1, riid->Data2, riid->Data3, riid->Data4[0],
129             riid->Data4[1], riid->Data4[2], riid->Data4[3], riid->Data4[4],
130             riid->Data4[5], riid->Data4[6], riid->Data4[7]);
131
132     return buf;
133 }
134
135 static int strcmp_wa(LPCWSTR strw, const char *stra)
136 {
137     CHAR buf[512];
138     WideCharToMultiByte(CP_ACP, 0, strw, -1, buf, sizeof(buf), NULL, NULL);
139     return lstrcmpA(stra, buf);
140 }
141
142 static BSTR a2bstr(const char *str)
143 {
144     BSTR ret;
145     int len;
146
147     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
148     ret = SysAllocStringLen(NULL, len-1);
149     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
150
151     return ret;
152 }
153
154 static BOOL iface_cmp(IUnknown *iface1, IUnknown *iface2)
155 {
156     IUnknown *unk1, *unk2;
157
158     if(iface1 == iface2)
159         return TRUE;
160
161     IUnknown_QueryInterface(iface1, &IID_IUnknown, (void**)&unk1);
162     IUnknown_Release(unk1);
163     IUnknown_QueryInterface(iface1, &IID_IUnknown, (void**)&unk2);
164     IUnknown_Release(unk2);
165
166     return unk1 == unk2;
167 }
168
169 #define test_disp(u,id) _test_disp(__LINE__,u,id)
170 static void _test_disp(unsigned line, IUnknown *unk, const IID *diid)
171 {
172     IDispatchEx *dispex;
173     ITypeInfo *typeinfo;
174     UINT ticnt;
175     HRESULT hres;
176
177     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
178     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatch: %08x\n", hres);
179     if(FAILED(hres))
180         return;
181
182     ticnt = 0xdeadbeef;
183     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
184     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
185     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
186
187     hres = IDispatchEx_GetTypeInfo(dispex, 0, 0, &typeinfo);
188     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfo failed: %08x\n", hres);
189
190     if(SUCCEEDED(hres)) {
191         TYPEATTR *type_attr;
192
193         hres = ITypeInfo_GetTypeAttr(typeinfo, &type_attr);
194         ok_(__FILE__,line) (hres == S_OK, "GetTypeAttr failed: %08x\n", hres);
195         ok_(__FILE__,line) (IsEqualGUID(&type_attr->guid, diid), "unexpected guid %s\n",
196                             debugstr_guid(&type_attr->guid));
197
198         ITypeInfo_ReleaseTypeAttr(typeinfo, type_attr);
199         ITypeInfo_Release(typeinfo);
200     }
201
202     IDispatchEx_Release(dispex);
203 }
204
205 #define get_doc3_iface(u) _get_doc3_iface(__LINE__,u)
206 static IHTMLDocument3 *_get_doc3_iface(unsigned line, IUnknown *unk)
207 {
208     IHTMLDocument3 *doc3;
209     HRESULT hres;
210
211     hres = IUnknown_QueryInterface(unk, &IID_IHTMLDocument3, (void**)&doc3);
212     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLDocument3 iface: %08x\n", hres);
213
214     return doc3;
215 }
216
217 #define get_elem_iface(u) _get_elem_iface(__LINE__,u)
218 static IHTMLElement *_get_elem_iface(unsigned line, IUnknown *unk)
219 {
220     IHTMLElement *elem;
221     HRESULT hres;
222
223     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement, (void**)&elem);
224     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement iface: %08x\n", hres);
225
226     return elem;
227 }
228
229 #define get_elem2_iface(u) _get_elem2_iface(__LINE__,u)
230 static IHTMLElement2 *_get_elem2_iface(unsigned line, IUnknown *unk)
231 {
232     IHTMLElement2 *elem2;
233     HRESULT hres;
234
235     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement2, (void**)&elem2);
236     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement2 iface: %08x\n", hres);
237
238     return elem2;
239 }
240
241 #define get_elem3_iface(u) _get_elem3_iface(__LINE__,u)
242 static IHTMLElement3 *_get_elem3_iface(unsigned line, IUnknown *unk)
243 {
244     IHTMLElement3 *elem3;
245     HRESULT hres;
246
247     hres = IUnknown_QueryInterface(unk, &IID_IHTMLElement3, (void**)&elem3);
248     ok_(__FILE__,line) (hres == S_OK, "Could not get IHTMLElement3 iface: %08x\n", hres);
249
250     return elem3;
251 }
252
253 #define doc_get_body(d) _doc_get_body(__LINE__,d)
254 static IHTMLElement *_doc_get_body(unsigned line, IHTMLDocument2 *doc)
255 {
256     IHTMLElement *body = NULL;
257     HRESULT hres;
258
259     hres = IHTMLDocument2_get_body(doc, &body);
260     ok_(__FILE__,line) (hres == S_OK, "get_body failed: %08x\n", hres);
261     ok_(__FILE__,line) (body != NULL, "body == NULL\n");
262
263     return body;
264 }
265
266 #define get_elem_id(d,i) _get_elem_id(__LINE__,d,i)
267 static IHTMLElement *_get_elem_id(unsigned line, IHTMLDocument2 *doc, const char *id)
268 {
269     IHTMLDocument3 *doc3 = _get_doc3_iface(line, (IUnknown*)doc);
270     IHTMLElement *elem;
271     BSTR str;
272     HRESULT hres;
273
274     str = a2bstr(id);
275     hres = IHTMLDocument3_getElementById(doc3, str, &elem);
276     SysFreeString(str);
277     IHTMLDocument3_Release(doc3);
278     ok_(__FILE__,line) (hres == S_OK, "getElementById failed: %08x\n", hres);
279
280     return elem;
281 }
282
283 #define test_elem_tag(u,n) _test_elem_tag(__LINE__,u,n)
284 static void _test_elem_tag(unsigned line, IUnknown *unk, const char *extag)
285 {
286     IHTMLElement *elem = _get_elem_iface(line, unk);
287     BSTR tag;
288     HRESULT hres;
289
290     hres = IHTMLElement_get_tagName(elem, &tag);
291     IHTMLElement_Release(elem);
292     ok_(__FILE__, line) (hres == S_OK, "get_tagName failed: %08x\n", hres);
293     ok_(__FILE__, line) (!strcmp_wa(tag, extag), "got tag: %s, expected %s\n", wine_dbgstr_w(tag), extag);
294
295     SysFreeString(tag);
296 }
297
298 #define get_event_obj() _get_event_obj(__LINE__)
299 static IHTMLEventObj *_get_event_obj(unsigned line)
300 {
301     IHTMLEventObj *event = NULL;
302     HRESULT hres;
303
304     hres = IHTMLWindow2_get_event(window, &event);
305     ok_(__FILE__,line) (hres == S_OK, "get_event failed: %08x\n", hres);
306     ok_(__FILE__,line) (event != NULL, "event = NULL\n");
307     _test_disp(line, (IUnknown*)event, &DIID_DispCEventObj);
308
309     return event;
310 }
311
312 #define elem_fire_event(a,b,c) _elem_fire_event(__LINE__,a,b,c)
313 static void _elem_fire_event(unsigned line, IUnknown *unk, const char *event, VARIANT *evobj)
314 {
315     IHTMLElement3 *elem3 = _get_elem3_iface(line, unk);
316     VARIANT_BOOL b;
317     BSTR str;
318     HRESULT hres;
319
320     b = 100;
321     str = a2bstr(event);
322     hres = IHTMLElement3_fireEvent(elem3, str, evobj, &b);
323     SysFreeString(str);
324     ok_(__FILE__,line)(hres == S_OK, "fireEvent failed: %08x\n", hres);
325     ok_(__FILE__,line)(b == VARIANT_TRUE, "fireEvent returned %x\n", b);
326 }
327
328 #define test_event_args(a,b,c,d,e,f,g) _test_event_args(__LINE__,a,b,c,d,e,f,g)
329 static void _test_event_args(unsigned line, const IID *dispiid, DISPID id, WORD wFlags, DISPPARAMS *pdp,
330         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
331 {
332     ok_(__FILE__,line) (id == DISPID_VALUE, "id = %d\n", id);
333     ok_(__FILE__,line) (wFlags == DISPATCH_METHOD, "wFlags = %x\n", wFlags);
334     ok_(__FILE__,line) (pdp != NULL, "pdp == NULL\n");
335     ok_(__FILE__,line) (pdp->cArgs == 1, "pdp->cArgs = %d\n", pdp->cArgs);
336     ok_(__FILE__,line) (pdp->cNamedArgs == 1, "pdp->cNamedArgs = %d\n", pdp->cNamedArgs);
337     ok_(__FILE__,line) (pdp->rgdispidNamedArgs[0] == DISPID_THIS, "pdp->rgdispidNamedArgs[0] = %d\n",
338                         pdp->rgdispidNamedArgs[0]);
339     ok_(__FILE__,line) (V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(rgvarg) = %d\n", V_VT(pdp->rgvarg));
340     ok_(__FILE__,line) (pvarRes != NULL, "pvarRes == NULL\n");
341     ok_(__FILE__,line) (pei != NULL, "pei == NULL");
342     ok_(__FILE__,line) (!pspCaller, "pspCaller != NULL\n");
343
344     if(dispiid)
345         _test_disp(line, (IUnknown*)V_DISPATCH(pdp->rgvarg), dispiid);
346 }
347
348 #define test_attached_event_args(a,b,c,d,e) _test_attached_event_args(__LINE__,a,b,c,d,e)
349 static void _test_attached_event_args(unsigned line, DISPID id, WORD wFlags, DISPPARAMS *pdp,
350         VARIANT *pvarRes, EXCEPINFO *pei)
351 {
352     IHTMLEventObj *event;
353
354     ok(id == DISPID_VALUE, "id = %d\n", id);
355     ok(wFlags == DISPATCH_METHOD, "wFlags = %x\n", wFlags);
356     ok(pdp != NULL, "pDispParams == NULL\n");
357     ok(pdp->cArgs == 1, "pdp->cArgs = %d\n", pdp->cArgs);
358     ok(!pdp->cNamedArgs, "pdp->cNamedArgs = %d\n", pdp->cNamedArgs);
359     ok(!pdp->rgdispidNamedArgs, "pdp->rgdispidNamedArgs = %p\n", pdp->rgdispidNamedArgs);
360     ok(pdp->rgvarg != NULL, "rgvarg = NULL\n");
361     ok(pvarRes != NULL, "pvarRes = NULL\n");
362     ok(pei != NULL, "pei = NULL\n");
363     ok(V_VT(pvarRes) == VT_EMPTY, "V_VT(pVarRes) = %d\n", V_VT(pvarRes));
364     ok(V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(pdp->rgvarg) = %d\n", V_VT(pdp->rgvarg));
365     ok(V_DISPATCH(pdp->rgvarg) != NULL, "V_DISPATCH(pdp->rgvarg) = %p\n", V_DISPATCH(pdp->rgvarg));
366
367     event = _get_event_obj(line);
368     ok(iface_cmp((IUnknown*)event, (IUnknown*)V_DISPATCH(pdp->rgvarg)), "event != arg0\n");
369     IHTMLEventObj_Release(event);
370 }
371
372 #define get_event_src() _get_event_src(__LINE__)
373 static IHTMLElement *_get_event_src(unsigned line)
374 {
375     IHTMLEventObj *event = _get_event_obj(line);
376     IHTMLElement *src_elem = NULL;
377     HRESULT hres;
378
379     hres = IHTMLEventObj_get_srcElement(event, &src_elem);
380     IHTMLEventObj_Release(event);
381     ok_(__FILE__,line) (hres == S_OK, "get_srcElement failed: %08x\n", hres);
382
383     return src_elem;
384 }
385
386 #define test_event_src(t) _test_event_src(__LINE__,t)
387 static void _test_event_src(unsigned line, const char *src_tag)
388 {
389     IHTMLElement *src_elem = _get_event_src(line);
390
391     if(src_tag) {
392         ok_(__FILE__,line) (src_elem != NULL, "src_elem = NULL\n");
393         _test_elem_tag(line, (IUnknown*)src_elem, src_tag);
394         IHTMLElement_Release(src_elem);
395     }else {
396         ok_(__FILE__,line) (!src_elem, "src_elem != NULL\n");
397     }
398 }
399
400 static void _test_event_altkey(unsigned line, IHTMLEventObj *event, VARIANT_BOOL exval)
401 {
402     VARIANT_BOOL b;
403     HRESULT hres;
404
405     hres = IHTMLEventObj_get_altKey(event, &b);
406     ok_(__FILE__,line)(hres == S_OK, "get_altKey failed: %08x\n", hres);
407     ok_(__FILE__,line)(b == exval, "altKey = %x, expected %x\n", b, exval);
408 }
409
410 static void _test_event_ctrlkey(unsigned line, IHTMLEventObj *event, VARIANT_BOOL exval)
411 {
412     VARIANT_BOOL b;
413     HRESULT hres;
414
415     hres = IHTMLEventObj_get_ctrlKey(event, &b);
416     ok_(__FILE__,line)(hres == S_OK, "get_ctrlKey failed: %08x\n", hres);
417     ok_(__FILE__,line)(b == exval, "ctrlKey = %x, expected %x\n", b, exval);
418 }
419
420 static void _test_event_shiftkey(unsigned line, IHTMLEventObj *event, VARIANT_BOOL exval)
421 {
422     VARIANT_BOOL b;
423     HRESULT hres;
424
425     hres = IHTMLEventObj_get_shiftKey(event, &b);
426     ok_(__FILE__,line)(hres == S_OK, "get_shiftKey failed: %08x\n", hres);
427     ok_(__FILE__,line)(b == exval, "shiftKey = %x, expected %x\n", b, exval);
428 }
429
430 static void _test_event_cancelbubble(unsigned line, IHTMLEventObj *event, VARIANT_BOOL exval)
431 {
432     VARIANT_BOOL b;
433     HRESULT hres;
434
435     hres = IHTMLEventObj_get_cancelBubble(event, &b);
436     ok_(__FILE__,line)(hres == S_OK, "get_cancelBubble failed: %08x\n", hres);
437     ok_(__FILE__,line)(b == exval, "cancelBubble = %x, expected %x\n", b, exval);
438 }
439
440 static void _test_event_fromelem(unsigned line, IHTMLEventObj *event, const char *from_tag)
441 {
442     IHTMLElement *elem;
443     HRESULT hres;
444
445     hres = IHTMLEventObj_get_fromElement(event, &elem);
446     ok_(__FILE__,line)(hres == S_OK, "get_fromElement failed: %08x\n", hres);
447     if(from_tag)
448         _test_elem_tag(line, (IUnknown*)elem, from_tag);
449     else
450         ok_(__FILE__,line)(elem == NULL, "fromElement != NULL\n");
451     if(elem)
452         IHTMLElement_Release(elem);
453 }
454
455 static void _test_event_toelem(unsigned line, IHTMLEventObj *event, const char *to_tag)
456 {
457     IHTMLElement *elem;
458     HRESULT hres;
459
460     hres = IHTMLEventObj_get_toElement(event, &elem);
461     ok_(__FILE__,line)(hres == S_OK, "get_toElement failed: %08x\n", hres);
462     if(to_tag)
463         _test_elem_tag(line, (IUnknown*)elem, to_tag);
464     else
465         ok_(__FILE__,line)(elem == NULL, "toElement != NULL\n");
466     if(elem)
467         IHTMLElement_Release(elem);
468 }
469
470 static void _test_event_keycode(unsigned line, IHTMLEventObj *event, LONG exl)
471 {
472     LONG l;
473     HRESULT hres;
474
475     hres = IHTMLEventObj_get_keyCode(event, &l);
476     ok_(__FILE__,line)(hres == S_OK, "get_keyCode failed: %08x\n", hres);
477     ok_(__FILE__,line)(l == exl, "keyCode = %x, expected %x\n", l, exl);
478 }
479
480 static void _test_event_button(unsigned line, IHTMLEventObj *event, LONG exl)
481 {
482     LONG l;
483     HRESULT hres;
484
485     hres = IHTMLEventObj_get_button(event, &l);
486     ok_(__FILE__,line)(hres == S_OK, "get_button failed: %08x\n", hres);
487     ok_(__FILE__,line)(l == exl, "button = %x, expected %x\n", l, exl);
488 }
489
490 static void _test_event_reason(unsigned line, IHTMLEventObj *event, LONG exl)
491 {
492     LONG l;
493     HRESULT hres;
494
495     hres = IHTMLEventObj_get_reason(event, &l);
496     ok_(__FILE__,line)(hres == S_OK, "get_reason failed: %08x\n", hres);
497     ok_(__FILE__,line)(l == exl, "reason = %x, expected %x\n", l, exl);
498 }
499
500 static void _test_event_x(unsigned line, IHTMLEventObj *event, LONG exl)
501 {
502     LONG l;
503     HRESULT hres;
504
505     hres = IHTMLEventObj_get_x(event, &l);
506     ok_(__FILE__,line)(hres == S_OK, "get_x failed: %08x\n", hres);
507     if(exl != -10) /* don't test the exact value */
508         ok_(__FILE__,line)(l == exl, "x = %d, expected %d\n", l, exl);
509 }
510
511 static void _test_event_y(unsigned line, IHTMLEventObj *event, LONG exl)
512 {
513     LONG l;
514     HRESULT hres;
515
516     hres = IHTMLEventObj_get_y(event, &l);
517     ok_(__FILE__,line)(hres == S_OK, "get_y failed: %08x\n", hres);
518     if(exl != -10) /* don't test the exact value */
519         ok_(__FILE__,line)(l == exl, "y = %d, expected %d\n", l, exl);
520 }
521
522 static void _test_event_clientx(unsigned line, IHTMLEventObj *event, LONG exl)
523 {
524     LONG l;
525     HRESULT hres;
526
527     hres = IHTMLEventObj_get_clientX(event, &l);
528     ok_(__FILE__,line)(hres == S_OK, "get_clientX failed: %08x\n", hres);
529     if(exl != -10) /* don't test the exact value */
530         ok_(__FILE__,line)(l == exl, "clientX = %d, expected %d\n", l, exl);
531 }
532
533 static void _test_event_clienty(unsigned line, IHTMLEventObj *event, LONG exl)
534 {
535     LONG l;
536     HRESULT hres;
537
538     hres = IHTMLEventObj_get_clientY(event, &l);
539     ok_(__FILE__,line)(hres == S_OK, "get_clientY failed: %08x\n", hres);
540     if(exl != -10) /* don't test the exact value */
541         ok_(__FILE__,line)(l == exl, "clientY = %d, expected %d\n", l, exl);
542 }
543
544 static void _test_event_offsetx(unsigned line, IHTMLEventObj *event, LONG exl)
545 {
546     LONG l;
547     HRESULT hres;
548
549     hres = IHTMLEventObj_get_offsetX(event, &l);
550     ok_(__FILE__,line)(hres == S_OK, "get_offsetX failed: %08x\n", hres);
551     if(exl != -10) /* don't test the exact value */
552         ok_(__FILE__,line)(l == exl, "offsetX = %d, expected %d\n", l, exl);
553 }
554
555 static void _test_event_offsety(unsigned line, IHTMLEventObj *event, LONG exl)
556 {
557     LONG l;
558     HRESULT hres;
559
560     hres = IHTMLEventObj_get_offsetY(event, &l);
561     ok_(__FILE__,line)(hres == S_OK, "get_offsetY failed: %08x\n", hres);
562     if(exl != -10) /* don't test the exact value */
563         ok_(__FILE__,line)(l == exl, "offsetY = %d, expected %d\n", l, exl);
564 }
565
566 static void _test_event_screenx(unsigned line, IHTMLEventObj *event, LONG exl)
567 {
568     LONG l;
569     HRESULT hres;
570
571     hres = IHTMLEventObj_get_screenX(event, &l);
572     ok_(__FILE__,line)(hres == S_OK, "get_screenX failed: %08x\n", hres);
573     if(exl != -10) /* don't test the exact value */
574         ok_(__FILE__,line)(l == exl, "screenX = %d, expected %d\n", l, exl);
575 }
576
577 static void _test_event_screeny(unsigned line, IHTMLEventObj *event, LONG exl)
578 {
579     LONG l;
580     HRESULT hres;
581
582     hres = IHTMLEventObj_get_screenY(event, &l);
583     ok_(__FILE__,line)(hres == S_OK, "get_screenY failed: %08x\n", hres);
584     if(exl != -10) /* don't test the exact value for -10 */
585         ok_(__FILE__,line)(l == exl, "screenY = %d, expected %d\n", l, exl);
586 }
587
588 static void _test_event_type(unsigned line, IHTMLEventObj *event, const char *exstr)
589 {
590     BSTR str;
591     HRESULT hres;
592
593     hres = IHTMLEventObj_get_type(event, &str);
594     ok_(__FILE__,line)(hres == S_OK, "get_type failed: %08x\n", hres);
595     ok_(__FILE__,line)(!strcmp_wa(str, exstr), "type = %s, expected %s\n", wine_dbgstr_w(str), exstr);
596 }
597
598 static void _test_event_qualifier(unsigned line, IHTMLEventObj *event, const char *exstr)
599 {
600     BSTR str;
601     HRESULT hres;
602
603     hres = IHTMLEventObj_get_qualifier(event, &str);
604     ok_(__FILE__,line)(hres == S_OK, "get_qualifier failed: %08x\n", hres);
605     if(exstr)
606         ok_(__FILE__,line)(!strcmp_wa(str, exstr), "qualifier = %s, expected %s\n", wine_dbgstr_w(str), exstr);
607     else
608         ok_(__FILE__,line)(!str, "qualifier != NULL\n");
609 }
610
611 static void _test_event_srcfilter(unsigned line, IHTMLEventObj *event)
612 {
613     IDispatch *disp;
614     HRESULT hres;
615
616     hres = IHTMLEventObj_get_srcFilter(event, &disp);
617     ok_(__FILE__,line)(hres == S_OK, "get_srcFilter failed: %08x\n", hres);
618     ok_(__FILE__,line)(!disp, "srcFilter != NULL\n");
619 }
620
621 #define test_event_obj(t,x) _test_event_obj(__LINE__,t,x)
622 static void _test_event_obj(unsigned line, const char *type, const xy_test_t *xy)
623 {
624     IHTMLEventObj *event = _get_event_obj(line);
625     VARIANT v;
626     HRESULT hres;
627
628     _test_event_altkey(line, event, VARIANT_FALSE);
629     _test_event_ctrlkey(line, event, VARIANT_FALSE);
630     _test_event_shiftkey(line, event, VARIANT_FALSE);
631     _test_event_cancelbubble(line, event, VARIANT_FALSE);
632     _test_event_fromelem(line, event, NULL);
633     _test_event_toelem(line, event, NULL);
634     _test_event_keycode(line, event, 0);
635     _test_event_button(line, event, 0);
636     _test_event_type(line, event, type);
637     _test_event_qualifier(line, event, NULL);
638     _test_event_reason(line, event, 0);
639     _test_event_srcfilter(line, event);
640     _test_event_x(line, event, xy->x);
641     _test_event_y(line, event, xy->y);
642     _test_event_clientx(line, event, -10);
643     _test_event_clienty(line, event, -10);
644     _test_event_offsetx(line, event, xy->offset_x);
645     _test_event_offsety(line, event, xy->offset_y);
646     _test_event_screenx(line, event, -10);
647     _test_event_screeny(line, event, -10);
648
649     hres = IHTMLEventObj_get_returnValue(event, &v);
650     ok_(__FILE__,line)(hres == S_OK, "get_returnValue failed: %08x\n", hres);
651     ok_(__FILE__,line)(V_VT(&v) == VT_EMPTY, "V_VT(returnValue) = %d\n", V_VT(&v));
652
653     IHTMLEventObj_Release(event);
654 }
655
656 #define elem_attach_event(a,b,c) _elem_attach_event(__LINE__,a,b,c)
657 static void _elem_attach_event(unsigned line, IUnknown *unk, const char *namea, IDispatch *disp)
658 {
659     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
660     VARIANT_BOOL res;
661     BSTR name;
662     HRESULT hres;
663
664     name = a2bstr(namea);
665     hres = IHTMLElement2_attachEvent(elem, name, disp, &res);
666     IHTMLElement2_Release(elem);
667     SysFreeString(name);
668     ok_(__FILE__,line)(hres == S_OK, "attachEvent failed: %08x\n", hres);
669     ok_(__FILE__,line)(res == VARIANT_TRUE, "attachEvent returned %x\n", res);
670 }
671
672 #define elem_detach_event(a,b,c) _elem_detach_event(__LINE__,a,b,c)
673 static void _elem_detach_event(unsigned line, IUnknown *unk, const char *namea, IDispatch *disp)
674 {
675     IHTMLElement2 *elem = _get_elem2_iface(line, unk);
676     BSTR name;
677     HRESULT hres;
678
679     name = a2bstr(namea);
680     hres = IHTMLElement2_detachEvent(elem, name, disp);
681     IHTMLElement2_Release(elem);
682     SysFreeString(name);
683     ok_(__FILE__,line)(hres == S_OK, "detachEvent failed: %08x\n", hres);
684 }
685
686 #define doc_attach_event(a,b,c) _doc_attach_event(__LINE__,a,b,c)
687 static void _doc_attach_event(unsigned line, IHTMLDocument2 *doc, const char *namea, IDispatch *disp)
688 {
689     IHTMLDocument3 *doc3 = _get_doc3_iface(line, (IUnknown*)doc);
690     VARIANT_BOOL res;
691     BSTR name;
692     HRESULT hres;
693
694     name = a2bstr(namea);
695     hres = IHTMLDocument3_attachEvent(doc3, name, disp, &res);
696     IHTMLDocument3_Release(doc3);
697     SysFreeString(name);
698     ok_(__FILE__,line)(hres == S_OK, "attachEvent failed: %08x\n", hres);
699     ok_(__FILE__,line)(res == VARIANT_TRUE, "attachEvent returned %x\n", res);
700 }
701
702 #define doc_detach_event(a,b,c) _doc_detach_event(__LINE__,a,b,c)
703 static void _doc_detach_event(unsigned line, IHTMLDocument2 *doc, const char *namea, IDispatch *disp)
704 {
705     IHTMLDocument3 *doc3 = _get_doc3_iface(line, (IUnknown*)doc);
706     BSTR name;
707     HRESULT hres;
708
709     name = a2bstr(namea);
710     hres = IHTMLDocument3_detachEvent(doc3, name, disp);
711     IHTMLDocument3_Release(doc3);
712     SysFreeString(name);
713     ok_(__FILE__,line)(hres == S_OK, "detachEvent failed: %08x\n", hres);
714 }
715
716 static HRESULT WINAPI DispatchEx_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
717 {
718     *ppv = NULL;
719
720     if(IsEqualGUID(riid, &IID_IUnknown)
721        || IsEqualGUID(riid, &IID_IDispatch)
722        || IsEqualGUID(riid, &IID_IDispatchEx))
723         *ppv = iface;
724     else {
725         ok(0, "unexpected riid %s\n", debugstr_guid(riid));
726         return E_NOINTERFACE;
727     }
728
729     return S_OK;
730 }
731
732 static HRESULT WINAPI Dispatch_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
733 {
734     *ppv = NULL;
735
736     if(IsEqualGUID(riid, &IID_IUnknown)
737        || IsEqualGUID(riid, &IID_IDispatch)) {
738         *ppv = iface;
739     }else if(IsEqualGUID(riid, &IID_IDispatchEx)) {
740         return E_NOINTERFACE;
741     }else {
742         ok(0, "unexpected riid %s\n", debugstr_guid(riid));
743         return E_NOINTERFACE;
744     }
745
746     return S_OK;
747 }
748
749 static ULONG WINAPI DispatchEx_AddRef(IDispatchEx *iface)
750 {
751     return 2;
752 }
753
754 static ULONG WINAPI DispatchEx_Release(IDispatchEx *iface)
755 {
756     return 1;
757 }
758
759 static HRESULT WINAPI DispatchEx_GetTypeInfoCount(IDispatchEx *iface, UINT *pctinfo)
760 {
761     ok(0, "unexpected call\n");
762     return E_NOTIMPL;
763 }
764
765 static HRESULT WINAPI DispatchEx_GetTypeInfo(IDispatchEx *iface, UINT iTInfo,
766                                               LCID lcid, ITypeInfo **ppTInfo)
767 {
768     ok(0, "unexpected call\n");
769     return E_NOTIMPL;
770 }
771
772 static HRESULT WINAPI DispatchEx_GetIDsOfNames(IDispatchEx *iface, REFIID riid,
773                                                 LPOLESTR *rgszNames, UINT cNames,
774                                                 LCID lcid, DISPID *rgDispId)
775 {
776     ok(0, "unexpected call\n");
777     return E_NOTIMPL;
778 }
779
780 static HRESULT WINAPI DispatchEx_Invoke(IDispatchEx *iface, DISPID dispIdMember,
781                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
782                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
783 {
784     ok(0, "unexpected call\n");
785     return E_NOTIMPL;
786 }
787
788 static HRESULT WINAPI DispatchEx_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
789 {
790     ok(0, "unexpected call\n");
791     return E_NOTIMPL;
792 }
793
794 static HRESULT WINAPI DispatchEx_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
795         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
796 {
797     ok(0, "unexpected call\n");
798     return E_NOTIMPL;
799 }
800
801 static HRESULT WINAPI DispatchEx_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
802 {
803     ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
804     return E_NOTIMPL;
805 }
806
807 static HRESULT WINAPI DispatchEx_DeleteMemberByDispID(IDispatchEx *iface, DISPID id)
808 {
809     ok(0, "unexpected call\n");
810     return E_NOTIMPL;
811 }
812
813 static HRESULT WINAPI DispatchEx_GetMemberProperties(IDispatchEx *iface, DISPID id, DWORD grfdexFetch, DWORD *pgrfdex)
814 {
815     ok(0, "unexpected call\n");
816     return E_NOTIMPL;
817 }
818
819 static HRESULT WINAPI DispatchEx_GetMemberName(IDispatchEx *iface, DISPID id, BSTR *pbstrName)
820 {
821     ok(0, "unexpected call\n");
822     return E_NOTIMPL;
823 }
824
825 static HRESULT WINAPI DispatchEx_GetNextDispID(IDispatchEx *iface, DWORD grfdex, DISPID id, DISPID *pid)
826 {
827     ok(0, "unexpected call\n");
828     return E_NOTIMPL;
829 }
830
831 static HRESULT WINAPI DispatchEx_GetNameSpaceParent(IDispatchEx *iface, IUnknown **ppunk)
832 {
833     ok(0, "unexpected call\n");
834     return E_NOTIMPL;
835 }
836
837 #define EVENT_HANDLER_FUNC_OBJ(event) \
838     static IDispatchExVtbl event ## FuncVtbl = { \
839         DispatchEx_QueryInterface, \
840         DispatchEx_AddRef, \
841         DispatchEx_Release, \
842         DispatchEx_GetTypeInfoCount, \
843         DispatchEx_GetTypeInfo, \
844         DispatchEx_GetIDsOfNames, \
845         DispatchEx_Invoke, \
846         DispatchEx_GetDispID, \
847         event, \
848         DispatchEx_DeleteMemberByName, \
849         DispatchEx_DeleteMemberByDispID, \
850         DispatchEx_GetMemberProperties, \
851         DispatchEx_GetMemberName, \
852         DispatchEx_GetNextDispID, \
853         DispatchEx_GetNameSpaceParent \
854     }; \
855     static IDispatchEx event ## _obj = { &event ## FuncVtbl };
856
857 static HRESULT WINAPI document_onclick(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
858         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
859 {
860     IHTMLDocument3 *doc3;
861     CHECK_EXPECT(document_onclick);
862     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
863     doc3 = get_doc3_iface((IUnknown*)V_DISPATCH(pdp->rgvarg));
864     IHTMLDocument3_Release(doc3);
865     test_event_src("DIV");
866     test_event_obj("click", &no_xy);
867     return S_OK;
868 }
869
870 EVENT_HANDLER_FUNC_OBJ(document_onclick);
871
872 static HRESULT WINAPI div_onclick(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
873         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
874 {
875     CHECK_EXPECT(div_onclick);
876     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
877     test_event_src("DIV");
878     return S_OK;
879 }
880
881 EVENT_HANDLER_FUNC_OBJ(div_onclick);
882
883 static HRESULT WINAPI div_onclick_attached(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
884         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
885 {
886     CHECK_EXPECT(div_onclick_attached);
887
888     test_attached_event_args(id, wFlags, pdp, pvarRes, pei);
889     test_event_src("DIV");
890     return S_OK;
891 }
892
893 EVENT_HANDLER_FUNC_OBJ(div_onclick_attached);
894
895 static HRESULT WINAPI doc_onclick_attached(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
896         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
897 {
898     CHECK_EXPECT(doc_onclick_attached);
899
900     test_attached_event_args(id, wFlags, pdp, pvarRes, pei);
901     test_event_src("DIV");
902     return S_OK;
903 }
904
905 EVENT_HANDLER_FUNC_OBJ(doc_onclick_attached);
906
907 static HRESULT WINAPI body_onclick(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
908         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
909 {
910     CHECK_EXPECT(body_onclick);
911     test_event_args(&DIID_DispHTMLBody, id, wFlags, pdp, pvarRes, pei, pspCaller);
912     test_event_src("DIV");
913     return S_OK;
914 }
915
916 EVENT_HANDLER_FUNC_OBJ(body_onclick);
917
918 static HRESULT WINAPI img_onload(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
919         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
920 {
921     CHECK_EXPECT(img_onload);
922     test_event_args(&DIID_DispHTMLImg, id, wFlags, pdp, pvarRes, pei, pspCaller);
923     test_event_src("IMG");
924     return S_OK;
925 }
926
927 EVENT_HANDLER_FUNC_OBJ(img_onload);
928
929 static HRESULT WINAPI input_onfocus(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
930         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
931 {
932     CHECK_EXPECT(input_onfocus);
933     test_event_args(&DIID_DispHTMLInputElement, id, wFlags, pdp, pvarRes, pei, pspCaller);
934     test_event_src("INPUT");
935     return S_OK;
936 }
937
938 EVENT_HANDLER_FUNC_OBJ(input_onfocus);
939
940 static HRESULT WINAPI form_onsubmit(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
941         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
942 {
943     CHECK_EXPECT(form_onsubmit);
944     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
945     test_event_src("FORM");
946
947     V_VT(pvarRes) = VT_BOOL;
948     V_BOOL(pvarRes) = VARIANT_FALSE;
949     return S_OK;
950 }
951
952 EVENT_HANDLER_FUNC_OBJ(form_onsubmit);
953
954 static HRESULT WINAPI form_onclick(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
955         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
956 {
957     CHECK_EXPECT(form_onclick);
958     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
959
960     return S_OK;
961 }
962
963 EVENT_HANDLER_FUNC_OBJ(form_onclick);
964
965 static HRESULT WINAPI submit_onclick(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
966         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
967 {
968     CHECK_EXPECT(submit_onclick);
969     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
970     test_event_src("INPUT");
971
972     V_VT(pvarRes) = VT_BOOL;
973     V_BOOL(pvarRes) = VARIANT_FALSE;
974     return S_OK;
975 }
976
977 EVENT_HANDLER_FUNC_OBJ(submit_onclick);
978
979 static HRESULT WINAPI submit_onclick_attached(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
980         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
981 {
982     CHECK_EXPECT(submit_onclick_attached);
983     test_attached_event_args(id, wFlags, pdp, pvarRes, pei);
984     test_event_src("INPUT");
985
986     V_VT(pvarRes) = VT_BOOL;
987     V_BOOL(pvarRes) = VARIANT_FALSE;
988     return S_OK;
989 }
990
991 EVENT_HANDLER_FUNC_OBJ(submit_onclick_attached);
992
993 static VARIANT onclick_retval, onclick_event_retval;
994
995 static HRESULT WINAPI submit_onclick_setret(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
996         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
997 {
998     IHTMLEventObj *event;
999     HRESULT hres;
1000
1001     CHECK_EXPECT(submit_onclick_setret);
1002     test_event_args(NULL, id, wFlags, pdp, pvarRes, pei, pspCaller);
1003     test_event_src("INPUT");
1004
1005     event = NULL;
1006     hres = IHTMLWindow2_get_event(window, &event);
1007     ok(hres == S_OK, "get_event failed: %08x\n", hres);
1008     ok(event != NULL, "event == NULL\n");
1009
1010     hres = IHTMLEventObj_put_returnValue(event, onclick_event_retval);
1011     ok(hres == S_OK, "put_returnValue failed: %08x\n", hres);
1012     IHTMLEventObj_Release(event);
1013
1014     *pvarRes = onclick_retval;
1015     return S_OK;
1016 }
1017
1018 EVENT_HANDLER_FUNC_OBJ(submit_onclick_setret);
1019
1020 static HRESULT WINAPI iframedoc_onreadystatechange(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1021         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
1022 {
1023     IHTMLEventObj *event = NULL;
1024     HRESULT hres;
1025
1026     CHECK_EXPECT2(iframedoc_onreadystatechange);
1027     test_event_args(&DIID_DispHTMLDocument, id, wFlags, pdp, pvarRes, pei, pspCaller);
1028
1029     event = (void*)0xdeadbeef;
1030     hres = IHTMLWindow2_get_event(window, &event);
1031     ok(hres == S_OK, "get_event failed: %08x\n", hres);
1032     ok(!event, "event = %p\n", event);
1033
1034     return S_OK;
1035 }
1036
1037 EVENT_HANDLER_FUNC_OBJ(iframedoc_onreadystatechange);
1038
1039 static HRESULT WINAPI iframe_onreadystatechange(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1040         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
1041 {
1042     IHTMLWindow2 *iframe_window;
1043     IHTMLDocument2 *iframe_doc;
1044     IHTMLFrameBase2 *iframe;
1045     IHTMLElement2 *elem2;
1046     IHTMLElement *elem;
1047     VARIANT v;
1048     BSTR str, str2;
1049     HRESULT hres;
1050
1051     test_event_args(&DIID_DispHTMLIFrame, id, wFlags, pdp, pvarRes, pei, pspCaller);
1052     test_event_src("IFRAME");
1053
1054     elem = get_event_src();
1055     elem2 = get_elem2_iface((IUnknown*)elem);
1056     IHTMLElement_Release(elem);
1057
1058     V_VT(&v) = VT_EMPTY;
1059     hres = IHTMLElement2_get_readyState(elem2, &v);
1060     ok(hres == S_OK, "get_readyState failed: %08x\n", hres);
1061     ok(V_VT(&v) == VT_BSTR, "V_VT(readyState) = %d\n", V_VT(&v));
1062
1063     hres = IHTMLElement2_QueryInterface(elem2, &IID_IHTMLFrameBase2, (void**)&iframe);
1064     IHTMLElement2_Release(elem2);
1065     ok(hres == S_OK, "Could not get IHTMLFrameBase2 iface: %08x\n", hres);
1066
1067     str = NULL;
1068     hres = IHTMLFrameBase2_get_readyState(iframe, &str);
1069     ok(hres == S_OK, "get_readyState failed: %08x\n", hres);
1070     ok(str != NULL, "readyState == NULL\n");
1071     ok(!lstrcmpW(str, V_BSTR(&v)), "ready states differ\n");
1072     VariantClear(&v);
1073
1074     hres = IHTMLFrameBase2_get_contentWindow(iframe, &iframe_window);
1075     ok(hres == S_OK, "get_contentDocument failed: %08x\n", hres);
1076
1077     hres = IHTMLWindow2_get_document(iframe_window, &iframe_doc);
1078     IHTMLWindow2_Release(iframe_window);
1079     ok(hres == S_OK, "get_document failed: %08x\n", hres);
1080
1081     hres = IHTMLDocument2_get_readyState(iframe_doc, &str2);
1082     ok(hres == S_OK, "get_document failed: %08x\n", hres);
1083     ok(!lstrcmpW(str, str2), "unexpected document readyState %s\n", wine_dbgstr_w(str2));
1084     SysFreeString(str2);
1085
1086     if(!strcmp_wa(str, "loading")) {
1087         CHECK_EXPECT(iframe_onreadystatechange_loading);
1088
1089         V_VT(&v) = VT_DISPATCH;
1090         V_DISPATCH(&v) = (IDispatch*)&iframedoc_onreadystatechange_obj;
1091         hres = IHTMLDocument2_put_onreadystatechange(iframe_doc, v);
1092         ok(hres == S_OK, "put_onreadystatechange: %08x\n", hres);
1093     }else if(!strcmp_wa(str, "interactive"))
1094         CHECK_EXPECT(iframe_onreadystatechange_interactive);
1095     else if(!strcmp_wa(str, "complete"))
1096         CHECK_EXPECT(iframe_onreadystatechange_complete);
1097     else
1098         ok(0, "unexpected state %s\n", wine_dbgstr_w(str));
1099
1100     SysFreeString(str);
1101     IHTMLDocument2_Release(iframe_doc);
1102     IHTMLFrameBase2_Release(iframe);
1103     return S_OK;
1104 }
1105
1106 EVENT_HANDLER_FUNC_OBJ(iframe_onreadystatechange);
1107
1108 static HRESULT WINAPI nocall(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1109         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
1110 {
1111     ok(0, "unexpected call\n");
1112     return S_OK;
1113 }
1114
1115 EVENT_HANDLER_FUNC_OBJ(nocall);
1116
1117 #define CONNECTION_POINT_OBJ(cpname, diid) \
1118     static HRESULT WINAPI cpname ## _QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv) \
1119     { \
1120         *ppv = NULL; \
1121         if(IsEqualGUID(riid, &IID_IUnknown) \
1122            || IsEqualGUID(riid, &IID_IDispatch) \
1123            || IsEqualGUID(riid, &diid)) \
1124             *ppv = iface; \
1125         else { \
1126             ok(0, "unexpected riid %s\n", debugstr_guid(riid)); \
1127             return E_NOINTERFACE; \
1128         } \
1129         return S_OK; \
1130     } \
1131     static IDispatchExVtbl cpname ## Vtbl = { \
1132         cpname ## _QueryInterface, \
1133         DispatchEx_AddRef,  \
1134         DispatchEx_Release, \
1135         DispatchEx_GetTypeInfoCount, \
1136         DispatchEx_GetTypeInfo, \
1137         DispatchEx_GetIDsOfNames, \
1138         cpname, \
1139         DispatchEx_GetDispID, \
1140         DispatchEx_InvokeEx, \
1141         DispatchEx_DeleteMemberByName, \
1142         DispatchEx_DeleteMemberByDispID, \
1143         DispatchEx_GetMemberProperties, \
1144         DispatchEx_GetMemberName, \
1145         DispatchEx_GetNextDispID, \
1146         DispatchEx_GetNameSpaceParent \
1147     }; \
1148     static IDispatchEx cpname ## _obj = { &cpname ## Vtbl }
1149
1150 #define test_cp_args(a,b,c,d,e,f) _test_cp_args(__LINE__,a,b,c,d,e,f)
1151 static void _test_cp_args(unsigned line, REFIID riid, WORD flags, DISPPARAMS *dp, VARIANT *vres, EXCEPINFO *ei, UINT *argerr)
1152 {
1153     ok_(__FILE__,line)(IsEqualGUID(&IID_NULL, riid), "riid = %s\n", debugstr_guid(riid));
1154     ok_(__FILE__,line)(flags == DISPATCH_METHOD, "flags = %x\n", flags);
1155     ok_(__FILE__,line)(dp != NULL, "dp == NULL\n");
1156     ok_(__FILE__,line)(!dp->cArgs, "dp->cArgs = %d\n", dp->cArgs);
1157     ok_(__FILE__,line)(!dp->rgvarg, "dp->rgvarg = %p\n", dp->rgvarg);
1158     ok_(__FILE__,line)(!dp->cNamedArgs, "dp->cNamedArgs = %d\n", dp->cNamedArgs);
1159     ok_(__FILE__,line)(!dp->rgdispidNamedArgs, "dp->rgdispidNamedArgs = %p\n", dp->rgdispidNamedArgs);
1160     ok_(__FILE__,line)(vres != NULL, "vres == NULL\n");
1161     ok_(__FILE__,line)(V_VT(vres) == VT_EMPTY, "V_VT(vres) = %d\n", V_VT(vres));
1162     ok_(__FILE__,line)(ei != NULL, "ei == NULL\n");
1163     ok_(__FILE__,line)(argerr != NULL, "argerr == NULL\n");
1164 }
1165
1166 static HRESULT WINAPI doccp(IDispatchEx *iface, DISPID dispIdMember,
1167                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1168                             VARIANT *pVarResult, EXCEPINFO *pei, UINT *puArgErr)
1169 {
1170     switch(dispIdMember) {
1171     case DISPID_HTMLDOCUMENTEVENTS_ONCLICK:
1172         CHECK_EXPECT(doccp_onclick);
1173         test_cp_args(riid, wFlags, pdp, pVarResult, pei, puArgErr);
1174         break;
1175     default:
1176         ok(0, "unexpected call %d\n", dispIdMember);
1177         return E_NOTIMPL;
1178     }
1179
1180     return S_OK;
1181 }
1182
1183 CONNECTION_POINT_OBJ(doccp, DIID_HTMLDocumentEvents);
1184
1185 static HRESULT WINAPI doccp_onclick_cancel(IDispatchEx *iface, DISPID dispIdMember,
1186         REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pdp, VARIANT *pVarResult, EXCEPINFO *pei, UINT *puArgErr)
1187 {
1188     switch(dispIdMember) {
1189     case DISPID_HTMLDOCUMENTEVENTS_ONCLICK:
1190         CHECK_EXPECT(doccp_onclick_cancel);
1191         test_cp_args(riid, wFlags, pdp, pVarResult, pei, puArgErr);
1192         V_VT(pVarResult) = VT_BOOL;
1193         V_BOOL(pVarResult) = VARIANT_FALSE;
1194         break;
1195     default:
1196         ok(0, "unexpected call %d\n", dispIdMember);
1197         return E_NOTIMPL;
1198     }
1199
1200     return S_OK;
1201 }
1202
1203 CONNECTION_POINT_OBJ(doccp_onclick_cancel, DIID_HTMLDocumentEvents);
1204
1205 static HRESULT WINAPI timeoutFunc_Invoke(IDispatchEx *iface, DISPID dispIdMember,
1206                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
1207                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
1208 {
1209     CHECK_EXPECT(timeout);
1210
1211     ok(dispIdMember == DISPID_VALUE, "dispIdMember = %d\n", dispIdMember);
1212     ok(IsEqualGUID(&IID_NULL, riid), "riid = %s\n", debugstr_guid(riid));
1213     ok(wFlags == DISPATCH_METHOD, "wFlags = %x\n", wFlags);
1214     ok(!lcid, "lcid = %x\n", lcid);
1215     ok(pDispParams != NULL, "pDispParams == NULL\n");
1216     ok(!pDispParams->cArgs, "pdp->cArgs = %d\n", pDispParams->cArgs);
1217     ok(!pDispParams->cNamedArgs, "pdp->cNamedArgs = %d\n", pDispParams->cNamedArgs);
1218     ok(!pDispParams->rgdispidNamedArgs, "pdp->rgdispidNamedArgs = %p\n", pDispParams->rgdispidNamedArgs);
1219     ok(!pDispParams->rgvarg, "rgvarg = %p\n", pDispParams->rgvarg);
1220     ok(pVarResult != NULL, "pVarResult = NULL\n");
1221     ok(pExcepInfo != NULL, "pExcepInfo = NULL\n");
1222     ok(!puArgErr, "puArgErr = %p\n", puArgErr);
1223     ok(V_VT(pVarResult) == VT_EMPTY, "V_VT(pVarResult) = %d\n", V_VT(pVarResult));
1224
1225     return S_OK;
1226 }
1227
1228 static const IDispatchExVtbl timeoutFuncVtbl = {
1229     DispatchEx_QueryInterface,
1230     DispatchEx_AddRef,
1231     DispatchEx_Release,
1232     DispatchEx_GetTypeInfoCount,
1233     DispatchEx_GetTypeInfo,
1234     DispatchEx_GetIDsOfNames,
1235     timeoutFunc_Invoke,
1236     DispatchEx_GetDispID,
1237     DispatchEx_InvokeEx,
1238     DispatchEx_DeleteMemberByName,
1239     DispatchEx_DeleteMemberByDispID,
1240     DispatchEx_GetMemberProperties,
1241     DispatchEx_GetMemberName,
1242     DispatchEx_GetNextDispID,
1243     DispatchEx_GetNameSpaceParent
1244 };
1245
1246 static IDispatchEx timeoutFunc = { &timeoutFuncVtbl };
1247
1248 static HRESULT WINAPI div_onclick_disp_Invoke(IDispatchEx *iface, DISPID id,
1249         REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1250         VARIANT *pvarRes, EXCEPINFO *pei, UINT *puArgErr)
1251 {
1252     CHECK_EXPECT(div_onclick_disp);
1253
1254     test_attached_event_args(id, wFlags, pdp, pvarRes, pei);
1255
1256     ok(IsEqualGUID(&IID_NULL, riid), "riid = %s\n", debugstr_guid(riid));
1257     ok(!puArgErr, "puArgErr = %p\n", puArgErr);
1258
1259     return S_OK;
1260 }
1261
1262 static const IDispatchExVtbl div_onclick_dispVtbl = {
1263     Dispatch_QueryInterface,
1264     DispatchEx_AddRef,
1265     DispatchEx_Release,
1266     DispatchEx_GetTypeInfoCount,
1267     DispatchEx_GetTypeInfo,
1268     DispatchEx_GetIDsOfNames,
1269     div_onclick_disp_Invoke,
1270 };
1271
1272 static IDispatchEx div_onclick_disp = { &div_onclick_dispVtbl };
1273
1274 static void pump_msgs(BOOL *b)
1275 {
1276     MSG msg;
1277
1278     if(b) {
1279         while(!*b && GetMessageW(&msg, NULL, 0, 0)) {
1280             TranslateMessage(&msg);
1281             DispatchMessage(&msg);
1282         }
1283     }else {
1284         while(!b && PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE)) {
1285             TranslateMessage(&msg);
1286             DispatchMessage(&msg);
1287         }
1288     }
1289 }
1290
1291 static IConnectionPoint *get_cp(IUnknown *unk, REFIID riid)
1292 {
1293     IConnectionPointContainer *cp_container;
1294     IConnectionPoint *cp;
1295     HRESULT hres;
1296
1297     hres = IUnknown_QueryInterface(unk, &IID_IConnectionPointContainer, (void**)&cp_container);
1298     ok(hres == S_OK, "Could not get IConnectionPointContainer: %08x\n", hres);
1299
1300     hres = IConnectionPointContainer_FindConnectionPoint(cp_container, riid, &cp);
1301     IConnectionPointContainer_Release(cp_container);
1302     ok(hres == S_OK, "FindConnectionPoint failed: %08x\n", hres);
1303
1304     return cp;
1305 }
1306
1307 static DWORD register_cp(IUnknown *unk, REFIID riid, IUnknown *sink)
1308 {
1309     IConnectionPoint *cp;
1310     DWORD cookie;
1311     HRESULT hres;
1312
1313     cp = get_cp(unk, riid);
1314     hres = IConnectionPoint_Advise(cp, sink, &cookie);
1315     IConnectionPoint_Release(cp);
1316     ok(hres == S_OK, "Advise failed: %08x\n", hres);
1317
1318     return cookie;
1319 }
1320
1321 static void unregister_cp(IUnknown *unk, REFIID riid, DWORD cookie)
1322 {
1323     IConnectionPoint *cp;
1324     HRESULT hres;
1325
1326     cp = get_cp(unk, riid);
1327     hres = IConnectionPoint_Unadvise(cp, cookie);
1328     IConnectionPoint_Release(cp);
1329     ok(hres == S_OK, "Unadvise failed: %08x\n", hres);
1330 }
1331
1332 static HRESULT WINAPI EventDispatch_QueryInterface(IDispatch *iface, REFIID riid, void **ppv)
1333 {
1334     if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IDispatch, riid)) {
1335         *ppv = iface;
1336         return S_OK;
1337     }
1338
1339     ok(0, "Unexpected call\n");
1340     return E_NOINTERFACE;
1341 }
1342
1343 static DWORD WINAPI EventDispatch_AddRef(IDispatch *iface)
1344 {
1345     return 2;
1346 }
1347
1348 static DWORD WINAPI EventDispatch_Release(IDispatch *iface)
1349 {
1350     return 1;
1351 }
1352
1353 static HRESULT WINAPI EventDispatch_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
1354 {
1355     ok(0, "Unexpected call\n");
1356     return E_NOTIMPL;
1357 }
1358
1359 static HRESULT WINAPI EventDispatch_GetTypeInfo(IDispatch *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo)
1360 {
1361     ok(0, "Unexpected call\n");
1362     return E_NOTIMPL;
1363 }
1364
1365 static HRESULT WINAPI EventDispatch_GetIDsOfNames(IDispatch *iface, REFIID riid, LPOLESTR *rgszNames,
1366         UINT cNames, LCID lcid, DISPID *rgDispId)
1367 {
1368     ok(0, "Unexpected call\n");
1369     return E_NOTIMPL;
1370 }
1371
1372 static HRESULT WINAPI EventDispatch_Invoke(IDispatch *iface, DISPID dispIdMember, REFIID riid,
1373         LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
1374         EXCEPINFO *pExcepInfo, UINT *puArgErr)
1375 {
1376     ok(IsEqualGUID(&IID_NULL, riid), "riid = %s\n", debugstr_guid(riid));
1377     ok(pDispParams != NULL, "pDispParams == NULL\n");
1378     ok(pExcepInfo != NULL, "pExcepInfo == NULL\n");
1379     ok(puArgErr != NULL, "puArgErr == NULL\n");
1380     ok(V_VT(pVarResult) == 0, "V_VT(pVarResult) = %d\n", V_VT(pVarResult));
1381     ok(wFlags == DISPATCH_METHOD, "wFlags = %d\n", wFlags);
1382
1383     switch(dispIdMember) {
1384     case DISPID_HTMLDOCUMENTEVENTS_ONCLICK:
1385         CHECK_EXPECT2(invoke_onclick);
1386         break;
1387     case DISPID_HTMLDOCUMENTEVENTS2_ONPROPERTYCHANGE:
1388     case DISPID_HTMLDOCUMENTEVENTS2_ONREADYSTATECHANGE:
1389     case 1027:
1390     case 1034:
1391     case 1037:
1392     case 1044:
1393     case 1045:
1394     case 1047:
1395     case 1048:
1396     case 1049:
1397         break; /* TODO */
1398     default:
1399         ok(0, "Unexpected call: %d\n", dispIdMember);
1400     }
1401
1402     return S_OK;
1403 }
1404
1405 static const IDispatchVtbl EventDispatchVtbl = {
1406     EventDispatch_QueryInterface,
1407     EventDispatch_AddRef,
1408     EventDispatch_Release,
1409     EventDispatch_GetTypeInfoCount,
1410     EventDispatch_GetTypeInfo,
1411     EventDispatch_GetIDsOfNames,
1412     EventDispatch_Invoke
1413 };
1414
1415 static IDispatch EventDispatch = { &EventDispatchVtbl };
1416
1417 static void test_onclick(IHTMLDocument2 *doc)
1418 {
1419     IHTMLElement *div, *body;
1420     DWORD cp_cookie;
1421     VARIANT v;
1422     HRESULT hres;
1423
1424     register_cp((IUnknown*)doc, &IID_IDispatch, (IUnknown*)&EventDispatch);
1425
1426     div = get_elem_id(doc, "clickdiv");
1427
1428     elem_attach_event((IUnknown*)div, "abcde", (IDispatch*)&nocall_obj);
1429     elem_attach_event((IUnknown*)div, "onclick", (IDispatch*)&div_onclick_attached_obj);
1430
1431     V_VT(&v) = VT_EMPTY;
1432     hres = IHTMLElement_get_onclick(div, &v);
1433     ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1434     ok(V_VT(&v) == VT_NULL, "V_VT(onclick) = %d\n", V_VT(&v));
1435
1436     V_VT(&v) = VT_EMPTY;
1437     hres = IHTMLElement_put_onclick(div, v);
1438     ok(hres == E_NOTIMPL, "put_onclick failed: %08x\n", hres);
1439
1440     V_VT(&v) = VT_BSTR;
1441     V_BSTR(&v) = a2bstr("function();");
1442     hres = IHTMLElement_put_onclick(div, v);
1443     todo_wine ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1444
1445     if(hres == S_OK) {
1446         V_VT(&v) = VT_EMPTY;
1447         hres = IHTMLElement_get_onclick(div, &v);
1448         ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1449         ok(V_VT(&v) == VT_BSTR, "V_VT(onclick) = %d\n", V_VT(&v));
1450         ok(!strcmp_wa(V_BSTR(&v), "function();"), "V_BSTR(onclick) = %s\n", wine_dbgstr_w(V_BSTR(&v)));
1451     }
1452     VariantClear(&v);
1453
1454     V_VT(&v) = VT_DISPATCH;
1455     V_DISPATCH(&v) = (IDispatch*)&div_onclick_obj;
1456     hres = IHTMLElement_put_onclick(div, v);
1457     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1458
1459     V_VT(&v) = VT_NULL;
1460     hres = IHTMLElement_put_ondblclick(div, v);
1461     ok(hres == S_OK, "put_ondblclick failed: %08x\n", hres);
1462
1463     V_VT(&v) = VT_EMPTY;
1464     hres = IHTMLElement_get_onclick(div, &v);
1465     ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1466     ok(V_VT(&v) == VT_DISPATCH, "V_VT(onclick) = %d\n", V_VT(&v));
1467     ok(V_DISPATCH(&v) == (IDispatch*)&div_onclick_obj, "V_DISPATCH(onclick) != onclickFunc\n");
1468     VariantClear(&v);
1469
1470     V_VT(&v) = VT_EMPTY;
1471     hres = IHTMLDocument2_get_onclick(doc, &v);
1472     ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1473     ok(V_VT(&v) == VT_NULL, "V_VT(onclick) = %d\n", V_VT(&v));
1474
1475     V_VT(&v) = VT_DISPATCH;
1476     V_DISPATCH(&v) = (IDispatch*)&document_onclick_obj;
1477     hres = IHTMLDocument2_put_onclick(doc, v);
1478     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1479
1480     V_VT(&v) = VT_EMPTY;
1481     hres = IHTMLDocument2_get_onclick(doc, &v);
1482     ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1483     ok(V_VT(&v) == VT_DISPATCH, "V_VT(onclick) = %d\n", V_VT(&v));
1484     ok(V_DISPATCH(&v) == (IDispatch*)&document_onclick_obj, "V_DISPATCH(onclick) != onclickFunc\n");
1485     VariantClear(&v);
1486
1487     body = doc_get_body(doc);
1488
1489     V_VT(&v) = VT_DISPATCH;
1490     V_DISPATCH(&v) = (IDispatch*)&body_onclick_obj;
1491     hres = IHTMLElement_put_onclick(body, v);
1492     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1493
1494     if(winetest_interactive) {
1495         SET_EXPECT(div_onclick);
1496         SET_EXPECT(div_onclick_attached);
1497         SET_EXPECT(body_onclick);
1498         SET_EXPECT(document_onclick);
1499         SET_EXPECT(invoke_onclick);
1500         pump_msgs(&called_document_onclick);
1501         CHECK_CALLED(div_onclick);
1502         CHECK_CALLED(div_onclick_attached);
1503         CHECK_CALLED(body_onclick);
1504         CHECK_CALLED(document_onclick);
1505         CHECK_CALLED(invoke_onclick);
1506     }
1507
1508     xy_todo = TRUE;
1509
1510     SET_EXPECT(div_onclick);
1511     SET_EXPECT(div_onclick_attached);
1512     SET_EXPECT(body_onclick);
1513     SET_EXPECT(document_onclick);
1514     SET_EXPECT(invoke_onclick);
1515
1516     hres = IHTMLElement_click(div);
1517     ok(hres == S_OK, "click failed: %08x\n", hres);
1518
1519     CHECK_CALLED(div_onclick);
1520     CHECK_CALLED(div_onclick_attached);
1521     CHECK_CALLED(body_onclick);
1522     CHECK_CALLED(document_onclick);
1523     CHECK_CALLED(invoke_onclick);
1524
1525     SET_EXPECT(div_onclick);
1526     SET_EXPECT(div_onclick_attached);
1527     SET_EXPECT(body_onclick);
1528     SET_EXPECT(document_onclick);
1529     SET_EXPECT(invoke_onclick);
1530
1531     V_VT(&v) = VT_EMPTY;
1532     elem_fire_event((IUnknown*)div, "onclick", &v);
1533
1534     CHECK_CALLED(div_onclick);
1535     CHECK_CALLED(div_onclick_attached);
1536     CHECK_CALLED(body_onclick);
1537     CHECK_CALLED(document_onclick);
1538     CHECK_CALLED(invoke_onclick);
1539
1540     cp_cookie = register_cp((IUnknown*)doc, &DIID_HTMLDocumentEvents, (IUnknown*)&doccp_obj);
1541     elem_attach_event((IUnknown*)div, "onclick", (IDispatch*)&div_onclick_disp);
1542     doc_attach_event(doc, "onclick", (IDispatch*)&doc_onclick_attached_obj);
1543
1544     SET_EXPECT(div_onclick);
1545     SET_EXPECT(div_onclick_disp);
1546     SET_EXPECT(div_onclick_attached);
1547     SET_EXPECT(body_onclick);
1548     SET_EXPECT(document_onclick);
1549     SET_EXPECT(doc_onclick_attached);
1550     SET_EXPECT(doccp_onclick);
1551     SET_EXPECT(invoke_onclick);
1552
1553     hres = IHTMLElement_click(div);
1554     ok(hres == S_OK, "click failed: %08x\n", hres);
1555
1556     CHECK_CALLED(div_onclick);
1557     CHECK_CALLED(div_onclick_disp);
1558     CHECK_CALLED(div_onclick_attached);
1559     CHECK_CALLED(body_onclick);
1560     CHECK_CALLED(document_onclick);
1561     CHECK_CALLED(doc_onclick_attached);
1562     CHECK_CALLED(doccp_onclick);
1563     CHECK_CALLED(invoke_onclick);
1564
1565     unregister_cp((IUnknown*)doc, &DIID_HTMLDocumentEvents, cp_cookie);
1566
1567     V_VT(&v) = VT_NULL;
1568     hres = IHTMLElement_put_onclick(div, v);
1569     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1570
1571     hres = IHTMLElement_get_onclick(div, &v);
1572     ok(hres == S_OK, "get_onclick failed: %08x\n", hres);
1573     ok(V_VT(&v) == VT_NULL, "get_onclick returned vt %d\n", V_VT(&v));
1574
1575     elem_detach_event((IUnknown*)div, "onclick", (IDispatch*)&div_onclick_disp);
1576     elem_detach_event((IUnknown*)div, "onclick", (IDispatch*)&div_onclick_disp);
1577     elem_detach_event((IUnknown*)div, "test", (IDispatch*)&div_onclick_disp);
1578     doc_detach_event(doc, "onclick", (IDispatch*)&doc_onclick_attached_obj);
1579
1580     SET_EXPECT(div_onclick_attached);
1581     SET_EXPECT(body_onclick);
1582     SET_EXPECT(document_onclick);
1583     SET_EXPECT(invoke_onclick);
1584
1585     hres = IHTMLElement_click(div);
1586     ok(hres == S_OK, "click failed: %08x\n", hres);
1587
1588     CHECK_CALLED(div_onclick_attached);
1589     CHECK_CALLED(body_onclick);
1590     CHECK_CALLED(document_onclick);
1591     CHECK_CALLED(invoke_onclick);
1592
1593     IHTMLElement_Release(div);
1594     IHTMLElement_Release(body);
1595 }
1596
1597 static void test_onreadystatechange(IHTMLDocument2 *doc)
1598 {
1599     IHTMLFrameBase *iframe;
1600     IHTMLElement2 *elem2;
1601     IHTMLElement *elem;
1602     VARIANT v;
1603     BSTR str;
1604     HRESULT hres;
1605
1606     elem = get_elem_id(doc, "iframe");
1607     elem2 = get_elem2_iface((IUnknown*)elem);
1608     IHTMLElement_Release(elem);
1609
1610     V_VT(&v) = VT_EMPTY;
1611     hres = IHTMLElement2_get_onreadystatechange(elem2, &v);
1612     ok(hres == S_OK, "get_onreadystatechange failed: %08x\n", hres);
1613     ok(V_VT(&v) == VT_NULL, "V_VT(onreadystatechange) = %d\n", V_VT(&v));
1614
1615     V_VT(&v) = VT_DISPATCH;
1616     V_DISPATCH(&v) = (IDispatch*)&iframe_onreadystatechange_obj;
1617     hres = IHTMLElement2_put_onreadystatechange(elem2, v);
1618     ok(hres == S_OK, "put_onreadystatechange failed: %08x\n", hres);
1619
1620     V_VT(&v) = VT_EMPTY;
1621     hres = IHTMLElement2_get_onreadystatechange(elem2, &v);
1622     ok(hres == S_OK, "get_onreadystatechange failed: %08x\n", hres);
1623     ok(V_VT(&v) == VT_DISPATCH, "V_VT(onreadystatechange) = %d\n", V_VT(&v));
1624     ok(V_DISPATCH(&v) == (IDispatch*)&iframe_onreadystatechange_obj, "unexpected onreadystatechange value\n");
1625
1626     hres = IHTMLElement2_QueryInterface(elem2, &IID_IHTMLFrameBase, (void**)&iframe);
1627     IHTMLElement2_Release(elem2);
1628     ok(hres == S_OK, "Could not get IHTMLFrameBase iface: %08x\n", hres);
1629
1630     hres = IHTMLFrameBase_put_src(iframe, (str = a2bstr("about:blank")));
1631     SysFreeString(str);
1632     ok(hres == S_OK, "put_src failed: %08x\n", hres);
1633
1634     SET_EXPECT(iframe_onreadystatechange_loading);
1635     SET_EXPECT(iframedoc_onreadystatechange);
1636     SET_EXPECT(iframe_onreadystatechange_interactive);
1637     SET_EXPECT(iframe_onreadystatechange_complete);
1638     pump_msgs(&called_iframe_onreadystatechange_complete);
1639     CHECK_CALLED(iframe_onreadystatechange_loading);
1640     CHECK_CALLED(iframedoc_onreadystatechange);
1641     CHECK_CALLED(iframe_onreadystatechange_interactive);
1642     CHECK_CALLED(iframe_onreadystatechange_complete);
1643
1644     IHTMLFrameBase_Release(iframe);
1645 }
1646
1647 static void test_imgload(IHTMLDocument2 *doc)
1648 {
1649     IHTMLImgElement *img;
1650     IHTMLElement *elem;
1651     VARIANT v;
1652     BSTR str;
1653     HRESULT hres;
1654
1655     elem = get_elem_id(doc, "imgid");
1656     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLImgElement, (void**)&img);
1657     IHTMLElement_Release(elem);
1658     ok(hres == S_OK, "Could not get IHTMLImgElement iface: %08x\n", hres);
1659
1660     V_VT(&v) = VT_EMPTY;
1661     hres = IHTMLImgElement_get_onload(img, &v);
1662     ok(hres == S_OK, "get_onload failed: %08x\n", hres);
1663     ok(V_VT(&v) == VT_NULL, "V_VT(onload) = %d\n", V_VT(&v));
1664
1665     V_VT(&v) = VT_DISPATCH;
1666     V_DISPATCH(&v) = (IDispatch*)&img_onload_obj;
1667     hres = IHTMLImgElement_put_onload(img, v);
1668     ok(hres == S_OK, "put_onload failed: %08x\n", hres);
1669
1670     V_VT(&v) = VT_EMPTY;
1671     hres = IHTMLImgElement_get_onload(img, &v);
1672     ok(hres == S_OK, "get_onload failed: %08x\n", hres);
1673     ok(V_VT(&v) == VT_DISPATCH, "V_VT(onload) = %d\n", V_VT(&v));
1674     ok(V_DISPATCH(&v) == (IDispatch*)&img_onload_obj, "V_DISPATCH(onload) != onloadkFunc\n");
1675     VariantClear(&v);
1676
1677     str = a2bstr("http://www.winehq.org/images/winehq_logo_text.png");
1678     hres = IHTMLImgElement_put_src(img, str);
1679     ok(hres == S_OK, "put_src failed: %08x\n", hres);
1680     SysFreeString(str);
1681
1682     SET_EXPECT(img_onload);
1683     pump_msgs(&called_img_onload);
1684     CHECK_CALLED(img_onload);
1685
1686     IHTMLImgElement_Release(img);
1687 }
1688
1689 static void test_focus(IHTMLDocument2 *doc)
1690 {
1691     IHTMLElement2 *elem2;
1692     IHTMLElement *elem;
1693     VARIANT v;
1694     HRESULT hres;
1695
1696     elem = get_elem_id(doc, "inputid");
1697     elem2 = get_elem2_iface((IUnknown*)elem);
1698     IHTMLElement_Release(elem);
1699
1700     V_VT(&v) = VT_EMPTY;
1701     hres = IHTMLElement2_get_onfocus(elem2, &v);
1702     ok(hres == S_OK, "get_onfocus failed: %08x\n", hres);
1703     ok(V_VT(&v) == VT_NULL, "V_VT(onfocus) = %d\n", V_VT(&v));
1704
1705     V_VT(&v) = VT_DISPATCH;
1706     V_DISPATCH(&v) = (IDispatch*)&input_onfocus_obj;
1707     hres = IHTMLElement2_put_onfocus(elem2, v);
1708     ok(hres == S_OK, "put_onfocus failed: %08x\n", hres);
1709
1710     V_VT(&v) = VT_EMPTY;
1711     hres = IHTMLElement2_get_onfocus(elem2, &v);
1712     ok(hres == S_OK, "get_onfocus failed: %08x\n", hres);
1713     ok(V_VT(&v) == VT_DISPATCH, "V_VT(onfocus) = %d\n", V_VT(&v));
1714     ok(V_DISPATCH(&v) == (IDispatch*)&input_onfocus_obj, "V_DISPATCH(onfocus) != onfocusFunc\n");
1715     VariantClear(&v);
1716
1717     if(!winetest_interactive)
1718         ShowWindow(container_hwnd, SW_SHOW);
1719
1720     SetFocus(NULL);
1721     ok(!IsChild(container_hwnd, GetFocus()), "focus belongs to document window\n");
1722
1723     hres = IHTMLWindow2_focus(window);
1724     ok(hres == S_OK, "focus failed: %08x\n", hres);
1725
1726     ok(IsChild(container_hwnd, GetFocus()), "focus does not belong to document window\n");
1727     pump_msgs(NULL);
1728
1729     SET_EXPECT(input_onfocus);
1730     hres = IHTMLElement2_focus(elem2);
1731     pump_msgs(NULL);
1732     CHECK_CALLED(input_onfocus);
1733     ok(hres == S_OK, "focus failed: %08x\n", hres);
1734
1735     if(!winetest_interactive)
1736         ShowWindow(container_hwnd, SW_HIDE);
1737
1738     IHTMLElement2_Release(elem2);
1739 }
1740
1741 static void test_submit(IHTMLDocument2 *doc)
1742 {
1743     IHTMLElement *elem, *submit;
1744     IHTMLFormElement *form;
1745     VARIANT v;
1746     DWORD cp_cookie;
1747     HRESULT hres;
1748
1749     elem = get_elem_id(doc, "formid");
1750
1751     V_VT(&v) = VT_DISPATCH;
1752     V_DISPATCH(&v) = (IDispatch*)&form_onclick_obj;
1753     hres = IHTMLElement_put_onclick(elem, v);
1754     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1755
1756     hres = IHTMLElement_QueryInterface(elem, &IID_IHTMLFormElement, (void**)&form);
1757     IHTMLElement_Release(elem);
1758     ok(hres == S_OK, "Could not get IHTMLFormElement iface: %08x\n", hres);
1759
1760     V_VT(&v) = VT_DISPATCH;
1761     V_DISPATCH(&v) = (IDispatch*)&form_onsubmit_obj;
1762     hres = IHTMLFormElement_put_onsubmit(form, v);
1763     ok(hres == S_OK, "put_onsubmit failed: %08x\n", hres);
1764
1765     IHTMLFormElement_Release(form);
1766
1767     submit = get_elem_id(doc, "submitid");
1768
1769     SET_EXPECT(form_onclick);
1770     SET_EXPECT(form_onsubmit);
1771     hres = IHTMLElement_click(submit);
1772     ok(hres == S_OK, "click failed: %08x\n", hres);
1773     CHECK_CALLED(form_onclick);
1774     CHECK_CALLED(form_onsubmit);
1775
1776     V_VT(&v) = VT_DISPATCH;
1777     V_DISPATCH(&v) = (IDispatch*)&submit_onclick_obj;
1778     hres = IHTMLElement_put_onclick(submit, v);
1779     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1780
1781     SET_EXPECT(form_onclick);
1782     SET_EXPECT(submit_onclick);
1783     hres = IHTMLElement_click(submit);
1784     ok(hres == S_OK, "click failed: %08x\n", hres);
1785     CHECK_CALLED(form_onclick);
1786     CHECK_CALLED(submit_onclick);
1787
1788     elem_attach_event((IUnknown*)submit, "onclick", (IDispatch*)&submit_onclick_attached_obj);
1789
1790     SET_EXPECT(form_onclick);
1791     SET_EXPECT(submit_onclick);
1792     SET_EXPECT(submit_onclick_attached);
1793     hres = IHTMLElement_click(submit);
1794     ok(hres == S_OK, "click failed: %08x\n", hres);
1795     CHECK_CALLED(form_onclick);
1796     CHECK_CALLED(submit_onclick);
1797     CHECK_CALLED(submit_onclick_attached);
1798
1799     V_VT(&v) = VT_NULL;
1800     hres = IHTMLElement_put_onclick(submit, v);
1801     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1802
1803     SET_EXPECT(form_onclick);
1804     SET_EXPECT(submit_onclick_attached);
1805     hres = IHTMLElement_click(submit);
1806     ok(hres == S_OK, "click failed: %08x\n", hres);
1807     CHECK_CALLED(form_onclick);
1808     CHECK_CALLED(submit_onclick_attached);
1809
1810     elem_detach_event((IUnknown*)submit, "onclick", (IDispatch*)&submit_onclick_attached_obj);
1811
1812     cp_cookie = register_cp((IUnknown*)doc, &DIID_HTMLDocumentEvents, (IUnknown*)&doccp_onclick_cancel_obj);
1813
1814     SET_EXPECT(form_onclick);
1815     SET_EXPECT(doccp_onclick_cancel);
1816     hres = IHTMLElement_click(submit);
1817     ok(hres == S_OK, "click failed: %08x\n", hres);
1818     CHECK_CALLED(form_onclick);
1819     CHECK_CALLED(doccp_onclick_cancel);
1820
1821     unregister_cp((IUnknown*)doc, &DIID_HTMLDocumentEvents, cp_cookie);
1822
1823     V_VT(&v) = VT_DISPATCH;
1824     V_DISPATCH(&v) = (IDispatch*)&submit_onclick_setret_obj;
1825     hres = IHTMLElement_put_onclick(submit, v);
1826     ok(hres == S_OK, "put_onclick failed: %08x\n", hres);
1827
1828     V_VT(&onclick_retval) = VT_BOOL;
1829     V_BOOL(&onclick_retval) = VARIANT_TRUE;
1830     V_VT(&onclick_event_retval) = VT_BOOL;
1831     V_BOOL(&onclick_event_retval) = VARIANT_TRUE;
1832
1833     SET_EXPECT(submit_onclick_setret);
1834     SET_EXPECT(form_onclick);
1835     SET_EXPECT(form_onsubmit);
1836     hres = IHTMLElement_click(submit);
1837     ok(hres == S_OK, "click failed: %08x\n", hres);
1838     CHECK_CALLED(submit_onclick_setret);
1839     CHECK_CALLED(form_onclick);
1840     CHECK_CALLED(form_onsubmit);
1841
1842     V_VT(&onclick_event_retval) = VT_BOOL;
1843     V_BOOL(&onclick_event_retval) = VARIANT_FALSE;
1844
1845     SET_EXPECT(submit_onclick_setret);
1846     SET_EXPECT(form_onclick);
1847     hres = IHTMLElement_click(submit);
1848     ok(hres == S_OK, "click failed: %08x\n", hres);
1849     CHECK_CALLED(submit_onclick_setret);
1850     CHECK_CALLED(form_onclick);
1851
1852     V_VT(&onclick_retval) = VT_BOOL;
1853     V_BOOL(&onclick_retval) = VARIANT_FALSE;
1854     V_VT(&onclick_event_retval) = VT_BOOL;
1855     V_BOOL(&onclick_event_retval) = VARIANT_TRUE;
1856
1857     SET_EXPECT(submit_onclick_setret);
1858     SET_EXPECT(form_onclick);
1859     hres = IHTMLElement_click(submit);
1860     ok(hres == S_OK, "click failed: %08x\n", hres);
1861     CHECK_CALLED(submit_onclick_setret);
1862     CHECK_CALLED(form_onclick);
1863
1864     V_VT(&onclick_event_retval) = VT_BOOL;
1865     V_BOOL(&onclick_event_retval) = VARIANT_FALSE;
1866
1867     SET_EXPECT(submit_onclick_setret);
1868     SET_EXPECT(form_onclick);
1869     hres = IHTMLElement_click(submit);
1870     ok(hres == S_OK, "click failed: %08x\n", hres);
1871     CHECK_CALLED(submit_onclick_setret);
1872     CHECK_CALLED(form_onclick);
1873
1874     IHTMLElement_Release(submit);
1875 }
1876
1877 static void test_timeout(IHTMLDocument2 *doc)
1878 {
1879     IHTMLWindow3 *win3;
1880     VARIANT expr, var;
1881     LONG id;
1882     HRESULT hres;
1883
1884     hres = IHTMLWindow2_QueryInterface(window, &IID_IHTMLWindow3, (void**)&win3);
1885     ok(hres == S_OK, "Could not get IHTMLWindow3 iface: %08x\n", hres);
1886
1887     V_VT(&expr) = VT_DISPATCH;
1888     V_DISPATCH(&expr) = (IDispatch*)&timeoutFunc;
1889     V_VT(&var) = VT_EMPTY;
1890     id = 0;
1891     hres = IHTMLWindow3_setTimeout(win3, &expr, 0, &var, &id);
1892     ok(hres == S_OK, "setTimeout failed: %08x\n", hres);
1893     ok(id, "id = 0\n");
1894
1895     SET_EXPECT(timeout);
1896     pump_msgs(&called_timeout);
1897     CHECK_CALLED(timeout);
1898
1899     V_VT(&expr) = VT_DISPATCH;
1900     V_DISPATCH(&expr) = (IDispatch*)&timeoutFunc;
1901     V_VT(&var) = VT_EMPTY;
1902     id = 0;
1903     hres = IHTMLWindow3_setTimeout(win3, &expr, 0, &var, &id);
1904     ok(hres == S_OK, "setTimeout failed: %08x\n", hres);
1905     ok(id, "id = 0\n");
1906
1907     hres = IHTMLWindow2_clearTimeout(window, id);
1908     ok(hres == S_OK, "clearTimeout failed: %08x\n", hres);
1909
1910     IHTMLWindow3_Release(win3);
1911 }
1912
1913 static IHTMLElement* find_element_by_id(IHTMLDocument2 *doc, const char *id)
1914 {
1915     HRESULT hres;
1916     IHTMLDocument3 *doc3;
1917     IHTMLElement *result;
1918     BSTR idW = a2bstr(id);
1919
1920     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument3, (void**)&doc3);
1921     ok(hres == S_OK, "QueryInterface(IID_IHTMLDocument3) failed: %08x\n", hres);
1922
1923     hres = IHTMLDocument3_getElementById(doc3, idW, &result);
1924     ok(hres == S_OK, "getElementById failed: %08x\n", hres);
1925     ok(result != NULL, "result == NULL\n");
1926     SysFreeString(idW);
1927
1928     IHTMLDocument3_Release(doc3);
1929     return result;
1930 }
1931
1932 static IHTMLDocument2* get_iframe_doc(IHTMLIFrameElement *iframe)
1933 {
1934     HRESULT hres;
1935     IHTMLFrameBase2 *base;
1936     IHTMLDocument2 *result = NULL;
1937
1938     hres = IHTMLIFrameElement_QueryInterface(iframe, &IID_IHTMLFrameBase2, (void**)&base);
1939     ok(hres == S_OK, "QueryInterface(IID_IHTMLFrameBase2) failed: %08x\n", hres);
1940     if(hres == S_OK) {
1941         IHTMLWindow2 *window;
1942
1943         hres = IHTMLFrameBase2_get_contentWindow(base, &window);
1944         ok(hres == S_OK, "get_contentWindow failed: %08x\n", hres);
1945         ok(window != NULL, "window == NULL\n");
1946         if(window) {
1947             hres = IHTMLWindow2_get_document(window, &result);
1948             ok(hres == S_OK, "get_document failed: %08x\n", hres);
1949             ok(result != NULL, "result == NULL\n");
1950             IHTMLWindow2_Release(window);
1951         }
1952     }
1953     if(base) IHTMLFrameBase2_Release(base);
1954
1955     return result;
1956 }
1957
1958 static void test_iframe_connections(IHTMLDocument2 *doc)
1959 {
1960     HRESULT hres;
1961     IHTMLIFrameElement *iframe;
1962     IHTMLDocument2 *iframes_doc;
1963     DWORD cookie;
1964     IConnectionPoint *cp;
1965     IHTMLElement *element = find_element_by_id(doc, "ifr");
1966
1967     hres = IHTMLElement_QueryInterface(element, &IID_IHTMLIFrameElement, (void**)&iframe);
1968     IHTMLElement_Release(element);
1969     ok(hres == S_OK, "QueryInterface(IID_IHTMLIFrameElement) failed: %08x\n", hres);
1970
1971     iframes_doc = get_iframe_doc(iframe);
1972     IHTMLIFrameElement_Release(iframe);
1973
1974     cookie = register_cp((IUnknown*)iframes_doc, &IID_IDispatch, (IUnknown*)&div_onclick_disp);
1975
1976     cp = get_cp((IUnknown*)doc, &IID_IDispatch);
1977     hres = IConnectionPoint_Unadvise(cp, cookie);
1978     IConnectionPoint_Release(cp);
1979     ok(hres == CONNECT_E_NOCONNECTION, "Unadvise returned %08x, expected CONNECT_E_NOCONNECTION\n", hres);
1980
1981     IHTMLDocument2_Release(iframes_doc);
1982 }
1983
1984 static HRESULT QueryInterface(REFIID,void**);
1985
1986 static HRESULT WINAPI InPlaceFrame_QueryInterface(IOleInPlaceFrame *iface, REFIID riid, void **ppv)
1987 {
1988     return E_NOINTERFACE;
1989 }
1990
1991 static ULONG WINAPI InPlaceFrame_AddRef(IOleInPlaceFrame *iface)
1992 {
1993     return 2;
1994 }
1995
1996 static ULONG WINAPI InPlaceFrame_Release(IOleInPlaceFrame *iface)
1997 {
1998     return 1;
1999 }
2000
2001 static HRESULT WINAPI InPlaceFrame_GetWindow(IOleInPlaceFrame *iface, HWND *phwnd)
2002 {
2003     return E_NOTIMPL;
2004 }
2005
2006 static HRESULT WINAPI InPlaceFrame_ContextSensitiveHelp(IOleInPlaceFrame *iface, BOOL fEnterMode)
2007 {
2008     return E_NOTIMPL;
2009 }
2010
2011 static HRESULT WINAPI InPlaceFrame_GetBorder(IOleInPlaceFrame *iface, LPRECT lprectBorder)
2012 {
2013     return E_NOTIMPL;
2014 }
2015
2016 static HRESULT WINAPI InPlaceFrame_RequestBorderSpace(IOleInPlaceFrame *iface,
2017         LPCBORDERWIDTHS pborderwidths)
2018 {
2019     return E_NOTIMPL;
2020 }
2021
2022 static HRESULT WINAPI InPlaceFrame_SetBorderSpace(IOleInPlaceFrame *iface,
2023         LPCBORDERWIDTHS pborderwidths)
2024 {
2025     return S_OK;
2026 }
2027
2028 static HRESULT WINAPI InPlaceUIWindow_SetActiveObject(IOleInPlaceFrame *iface,
2029         IOleInPlaceActiveObject *pActiveObject, LPCOLESTR pszObjName)
2030 {
2031     return S_OK;
2032 }
2033
2034 static HRESULT WINAPI InPlaceFrame_SetActiveObject(IOleInPlaceFrame *iface,
2035         IOleInPlaceActiveObject *pActiveObject, LPCOLESTR pszObjName)
2036 {
2037     return S_OK;
2038 }
2039
2040 static HRESULT WINAPI InPlaceFrame_InsertMenus(IOleInPlaceFrame *iface, HMENU hmenuShared,
2041         LPOLEMENUGROUPWIDTHS lpMenuWidths)
2042 {
2043     return E_NOTIMPL;
2044 }
2045
2046 static HRESULT WINAPI InPlaceFrame_SetMenu(IOleInPlaceFrame *iface, HMENU hmenuShared,
2047         HOLEMENU holemenu, HWND hwndActiveObject)
2048 {
2049     ok(0, "unexpected call\n");
2050     return E_NOTIMPL;
2051 }
2052
2053 static HRESULT WINAPI InPlaceFrame_RemoveMenus(IOleInPlaceFrame *iface, HMENU hmenuShared)
2054 {
2055     ok(0, "unexpected call\n");
2056     return E_NOTIMPL;
2057 }
2058
2059 static HRESULT WINAPI InPlaceFrame_SetStatusText(IOleInPlaceFrame *iface, LPCOLESTR pszStatusText)
2060 {
2061     return S_OK;
2062 }
2063
2064 static HRESULT WINAPI InPlaceFrame_EnableModeless(IOleInPlaceFrame *iface, BOOL fEnable)
2065 {
2066     return E_NOTIMPL;
2067 }
2068
2069 static HRESULT WINAPI InPlaceFrame_TranslateAccelerator(IOleInPlaceFrame *iface, LPMSG lpmsg, WORD wID)
2070 {
2071     ok(0, "unexpected call\n");
2072     return E_NOTIMPL;
2073 }
2074
2075 static const IOleInPlaceFrameVtbl InPlaceFrameVtbl = {
2076     InPlaceFrame_QueryInterface,
2077     InPlaceFrame_AddRef,
2078     InPlaceFrame_Release,
2079     InPlaceFrame_GetWindow,
2080     InPlaceFrame_ContextSensitiveHelp,
2081     InPlaceFrame_GetBorder,
2082     InPlaceFrame_RequestBorderSpace,
2083     InPlaceFrame_SetBorderSpace,
2084     InPlaceFrame_SetActiveObject,
2085     InPlaceFrame_InsertMenus,
2086     InPlaceFrame_SetMenu,
2087     InPlaceFrame_RemoveMenus,
2088     InPlaceFrame_SetStatusText,
2089     InPlaceFrame_EnableModeless,
2090     InPlaceFrame_TranslateAccelerator
2091 };
2092
2093 static IOleInPlaceFrame InPlaceFrame = { &InPlaceFrameVtbl };
2094
2095 static const IOleInPlaceFrameVtbl InPlaceUIWindowVtbl = {
2096     InPlaceFrame_QueryInterface,
2097     InPlaceFrame_AddRef,
2098     InPlaceFrame_Release,
2099     InPlaceFrame_GetWindow,
2100     InPlaceFrame_ContextSensitiveHelp,
2101     InPlaceFrame_GetBorder,
2102     InPlaceFrame_RequestBorderSpace,
2103     InPlaceFrame_SetBorderSpace,
2104     InPlaceUIWindow_SetActiveObject,
2105 };
2106
2107 static IOleInPlaceFrame InPlaceUIWindow = { &InPlaceUIWindowVtbl };
2108
2109 static HRESULT WINAPI InPlaceSite_QueryInterface(IOleInPlaceSite *iface, REFIID riid, void **ppv)
2110 {
2111     return QueryInterface(riid, ppv);
2112 }
2113
2114 static ULONG WINAPI InPlaceSite_AddRef(IOleInPlaceSite *iface)
2115 {
2116     return 2;
2117 }
2118
2119 static ULONG WINAPI InPlaceSite_Release(IOleInPlaceSite *iface)
2120 {
2121     return 1;
2122 }
2123
2124 static HRESULT WINAPI InPlaceSite_GetWindow(IOleInPlaceSite *iface, HWND *phwnd)
2125 {
2126     *phwnd = container_hwnd;
2127     return S_OK;
2128 }
2129
2130 static HRESULT WINAPI InPlaceSite_ContextSensitiveHelp(IOleInPlaceSite *iface, BOOL fEnterMode)
2131 {
2132     ok(0, "unexpected call\n");
2133     return E_NOTIMPL;
2134 }
2135
2136 static HRESULT WINAPI InPlaceSite_CanInPlaceActivate(IOleInPlaceSite *iface)
2137 {
2138     return S_OK;
2139 }
2140
2141 static HRESULT WINAPI InPlaceSite_OnInPlaceActivate(IOleInPlaceSite *iface)
2142 {
2143     return S_OK;
2144 }
2145
2146 static HRESULT WINAPI InPlaceSite_OnUIActivate(IOleInPlaceSite *iface)
2147 {
2148     return S_OK;
2149 }
2150
2151 static HRESULT WINAPI InPlaceSite_GetWindowContext(IOleInPlaceSite *iface,
2152         IOleInPlaceFrame **ppFrame, IOleInPlaceUIWindow **ppDoc, LPRECT lprcPosRect,
2153         LPRECT lprcClipRect, LPOLEINPLACEFRAMEINFO lpFrameInfo)
2154 {
2155     static const RECT rect = {0,0,300,300};
2156
2157     *ppFrame = &InPlaceFrame;
2158     *ppDoc = (IOleInPlaceUIWindow*)&InPlaceUIWindow;
2159     *lprcPosRect = rect;
2160     *lprcClipRect = rect;
2161
2162     ok(lpFrameInfo->cb == sizeof(*lpFrameInfo), "lpFrameInfo->cb = %u, expected %u\n", lpFrameInfo->cb, (unsigned)sizeof(*lpFrameInfo));
2163     lpFrameInfo->fMDIApp = FALSE;
2164     lpFrameInfo->hwndFrame = container_hwnd;
2165     lpFrameInfo->haccel = NULL;
2166     lpFrameInfo->cAccelEntries = 0;
2167
2168     return S_OK;
2169 }
2170
2171 static HRESULT WINAPI InPlaceSite_Scroll(IOleInPlaceSite *iface, SIZE scrollExtant)
2172 {
2173     return E_NOTIMPL;
2174 }
2175
2176 static HRESULT WINAPI InPlaceSite_OnUIDeactivate(IOleInPlaceSite *iface, BOOL fUndoable)
2177 {
2178     return S_OK;
2179 }
2180
2181 static HRESULT WINAPI InPlaceSite_OnInPlaceDeactivate(IOleInPlaceSite *iface)
2182 {
2183     return S_OK;
2184 }
2185
2186 static HRESULT WINAPI InPlaceSite_DiscardUndoState(IOleInPlaceSite *iface)
2187 {
2188     return E_NOTIMPL;
2189 }
2190
2191 static HRESULT WINAPI InPlaceSite_DeactivateAndUndo(IOleInPlaceSite *iface)
2192 {
2193     return E_NOTIMPL;
2194 }
2195
2196 static HRESULT WINAPI InPlaceSite_OnPosRectChange(IOleInPlaceSite *iface, LPCRECT lprcPosRect)
2197 {
2198     return E_NOTIMPL;
2199 }
2200
2201 static const IOleInPlaceSiteVtbl InPlaceSiteVtbl = {
2202     InPlaceSite_QueryInterface,
2203     InPlaceSite_AddRef,
2204     InPlaceSite_Release,
2205     InPlaceSite_GetWindow,
2206     InPlaceSite_ContextSensitiveHelp,
2207     InPlaceSite_CanInPlaceActivate,
2208     InPlaceSite_OnInPlaceActivate,
2209     InPlaceSite_OnUIActivate,
2210     InPlaceSite_GetWindowContext,
2211     InPlaceSite_Scroll,
2212     InPlaceSite_OnUIDeactivate,
2213     InPlaceSite_OnInPlaceDeactivate,
2214     InPlaceSite_DiscardUndoState,
2215     InPlaceSite_DeactivateAndUndo,
2216     InPlaceSite_OnPosRectChange,
2217 };
2218
2219 static IOleInPlaceSite InPlaceSite = { &InPlaceSiteVtbl };
2220
2221 static HRESULT WINAPI ClientSite_QueryInterface(IOleClientSite *iface, REFIID riid, void **ppv)
2222 {
2223     return QueryInterface(riid, ppv);
2224 }
2225
2226 static ULONG WINAPI ClientSite_AddRef(IOleClientSite *iface)
2227 {
2228     return 2;
2229 }
2230
2231 static ULONG WINAPI ClientSite_Release(IOleClientSite *iface)
2232 {
2233     return 1;
2234 }
2235
2236 static HRESULT WINAPI ClientSite_SaveObject(IOleClientSite *iface)
2237 {
2238     ok(0, "unexpected call\n");
2239     return E_NOTIMPL;
2240 }
2241
2242 static HRESULT WINAPI ClientSite_GetMoniker(IOleClientSite *iface, DWORD dwAssign, DWORD dwWhichMoniker,
2243         IMoniker **ppmon)
2244 {
2245     ok(0, "unexpected call\n");
2246     return E_NOTIMPL;
2247 }
2248
2249 static HRESULT WINAPI ClientSite_GetContainer(IOleClientSite *iface, IOleContainer **ppContainer)
2250 {
2251     return E_NOTIMPL;
2252 }
2253
2254 static HRESULT WINAPI ClientSite_ShowObject(IOleClientSite *iface)
2255 {
2256     ok(0, "unexpected call\n");
2257     return E_NOTIMPL;
2258 }
2259
2260 static HRESULT WINAPI ClientSite_OnShowWindow(IOleClientSite *iface, BOOL fShow)
2261 {
2262     ok(0, "unexpected call\n");
2263     return E_NOTIMPL;
2264 }
2265
2266 static HRESULT WINAPI ClientSite_RequestNewObjectLayout(IOleClientSite *iface)
2267 {
2268     ok(0, "unexpected call\n");
2269     return E_NOTIMPL;
2270 }
2271
2272 static const IOleClientSiteVtbl ClientSiteVtbl = {
2273     ClientSite_QueryInterface,
2274     ClientSite_AddRef,
2275     ClientSite_Release,
2276     ClientSite_SaveObject,
2277     ClientSite_GetMoniker,
2278     ClientSite_GetContainer,
2279     ClientSite_ShowObject,
2280     ClientSite_OnShowWindow,
2281     ClientSite_RequestNewObjectLayout
2282 };
2283
2284 static IOleClientSite ClientSite = { &ClientSiteVtbl };
2285
2286 static HRESULT WINAPI DocumentSite_QueryInterface(IOleDocumentSite *iface, REFIID riid, void **ppv)
2287 {
2288     return QueryInterface(riid, ppv);
2289 }
2290
2291 static ULONG WINAPI DocumentSite_AddRef(IOleDocumentSite *iface)
2292 {
2293     return 2;
2294 }
2295
2296 static ULONG WINAPI DocumentSite_Release(IOleDocumentSite *iface)
2297 {
2298     return 1;
2299 }
2300
2301 static HRESULT WINAPI DocumentSite_ActivateMe(IOleDocumentSite *iface, IOleDocumentView *pViewToActivate)
2302 {
2303     RECT rect = {0,0,300,300};
2304     IOleDocument *document;
2305     HRESULT hres;
2306
2307     hres = IOleDocumentView_QueryInterface(pViewToActivate, &IID_IOleDocument, (void**)&document);
2308     ok(hres == S_OK, "could not get IOleDocument: %08x\n", hres);
2309
2310     hres = IOleDocument_CreateView(document, &InPlaceSite, NULL, 0, &view);
2311     IOleDocument_Release(document);
2312     ok(hres == S_OK, "CreateView failed: %08x\n", hres);
2313
2314     hres = IOleDocumentView_SetInPlaceSite(view, &InPlaceSite);
2315     ok(hres == S_OK, "SetInPlaceSite failed: %08x\n", hres);
2316
2317     hres = IOleDocumentView_UIActivate(view, TRUE);
2318     ok(hres == S_OK, "UIActivate failed: %08x\n", hres);
2319
2320     hres = IOleDocumentView_SetRect(view, &rect);
2321     ok(hres == S_OK, "SetRect failed: %08x\n", hres);
2322
2323     hres = IOleDocumentView_Show(view, TRUE);
2324     ok(hres == S_OK, "Show failed: %08x\n", hres);
2325
2326     return S_OK;
2327 }
2328
2329 static const IOleDocumentSiteVtbl DocumentSiteVtbl = {
2330     DocumentSite_QueryInterface,
2331     DocumentSite_AddRef,
2332     DocumentSite_Release,
2333     DocumentSite_ActivateMe
2334 };
2335
2336 static IOleDocumentSite DocumentSite = { &DocumentSiteVtbl };
2337
2338 static HRESULT QueryInterface(REFIID riid, void **ppv)
2339 {
2340     *ppv = NULL;
2341
2342     if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IOleClientSite, riid))
2343         *ppv = &ClientSite;
2344     else if(IsEqualGUID(&IID_IOleDocumentSite, riid))
2345         *ppv = &DocumentSite;
2346     else if(IsEqualGUID(&IID_IOleWindow, riid) || IsEqualGUID(&IID_IOleInPlaceSite, riid))
2347         *ppv = &InPlaceSite;
2348
2349     return *ppv ? S_OK : E_NOINTERFACE;
2350 }
2351
2352 static IHTMLDocument2 *notif_doc;
2353 static BOOL doc_complete;
2354
2355 static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface,
2356         REFIID riid, void**ppv)
2357 {
2358     if(IsEqualGUID(&IID_IPropertyNotifySink, riid)) {
2359         *ppv = iface;
2360         return S_OK;
2361     }
2362
2363     ok(0, "unexpected call\n");
2364     return E_NOINTERFACE;
2365 }
2366
2367 static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface)
2368 {
2369     return 2;
2370 }
2371
2372 static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface)
2373 {
2374     return 1;
2375 }
2376
2377 static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
2378 {
2379     if(dispID == DISPID_READYSTATE){
2380         BSTR state;
2381         HRESULT hres;
2382
2383         hres = IHTMLDocument2_get_readyState(notif_doc, &state);
2384         ok(hres == S_OK, "get_readyState failed: %08x\n", hres);
2385
2386         if(!strcmp_wa(state, "complete"))
2387             doc_complete = TRUE;
2388
2389         SysFreeString(state);
2390     }
2391
2392     return S_OK;
2393 }
2394
2395 static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
2396 {
2397     ok(0, "unexpected call\n");
2398     return E_NOTIMPL;
2399 }
2400
2401 static const IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
2402     PropertyNotifySink_QueryInterface,
2403     PropertyNotifySink_AddRef,
2404     PropertyNotifySink_Release,
2405     PropertyNotifySink_OnChanged,
2406     PropertyNotifySink_OnRequestEdit
2407 };
2408
2409 static IPropertyNotifySink PropertyNotifySink = { &PropertyNotifySinkVtbl };
2410
2411 static void doc_load_string(IHTMLDocument2 *doc, const char *str)
2412 {
2413     IPersistStreamInit *init;
2414     IStream *stream;
2415     HGLOBAL mem;
2416     SIZE_T len;
2417
2418     notif_doc = doc;
2419
2420     doc_complete = FALSE;
2421     len = strlen(str);
2422     mem = GlobalAlloc(0, len);
2423     memcpy(mem, str, len);
2424     CreateStreamOnHGlobal(mem, TRUE, &stream);
2425
2426     IHTMLDocument2_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&init);
2427
2428     IPersistStreamInit_Load(init, stream);
2429     IPersistStreamInit_Release(init);
2430     IStream_Release(stream);
2431 }
2432
2433 static void do_advise(IUnknown *unk, REFIID riid, IUnknown *unk_advise)
2434 {
2435     IConnectionPointContainer *container;
2436     IConnectionPoint *cp;
2437     DWORD cookie;
2438     HRESULT hres;
2439
2440     hres = IUnknown_QueryInterface(unk, &IID_IConnectionPointContainer, (void**)&container);
2441     ok(hres == S_OK, "QueryInterface(IID_IConnectionPointContainer) failed: %08x\n", hres);
2442
2443     hres = IConnectionPointContainer_FindConnectionPoint(container, riid, &cp);
2444     IConnectionPointContainer_Release(container);
2445     ok(hres == S_OK, "FindConnectionPoint failed: %08x\n", hres);
2446
2447     hres = IConnectionPoint_Advise(cp, unk_advise, &cookie);
2448     IConnectionPoint_Release(cp);
2449     ok(hres == S_OK, "Advise failed: %08x\n", hres);
2450 }
2451
2452 static void set_client_site(IHTMLDocument2 *doc, BOOL set)
2453 {
2454     IOleObject *oleobj;
2455     HRESULT hres;
2456
2457     if(!set && view) {
2458         IOleDocumentView_Show(view, FALSE);
2459         IOleDocumentView_CloseView(view, 0);
2460         IOleDocumentView_SetInPlaceSite(view, NULL);
2461         IOleDocumentView_Release(view);
2462         view = NULL;
2463     }
2464
2465     hres = IHTMLDocument2_QueryInterface(doc, &IID_IOleObject, (void**)&oleobj);
2466     ok(hres == S_OK, "Could not et IOleObject: %08x\n", hres);
2467
2468     hres = IOleObject_SetClientSite(oleobj, set ? &ClientSite : NULL);
2469     ok(hres == S_OK, "SetClientSite failed: %08x\n", hres);
2470
2471     if(set) {
2472         IHlinkTarget *hlink;
2473
2474         hres = IOleObject_QueryInterface(oleobj, &IID_IHlinkTarget, (void**)&hlink);
2475         ok(hres == S_OK, "Could not get IHlinkTarget iface: %08x\n", hres);
2476
2477         hres = IHlinkTarget_Navigate(hlink, 0, NULL);
2478         ok(hres == S_OK, "Navgate failed: %08x\n", hres);
2479
2480         IHlinkTarget_Release(hlink);
2481     }
2482
2483     IOleObject_Release(oleobj);
2484 }
2485 static IHTMLDocument2 *create_document(void)
2486 {
2487     IHTMLDocument2 *doc;
2488     IHTMLDocument5 *doc5;
2489     HRESULT hres;
2490
2491     hres = CoCreateInstance(&CLSID_HTMLDocument, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
2492             &IID_IHTMLDocument2, (void**)&doc);
2493     ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
2494     if (FAILED(hres))
2495         return NULL;
2496
2497     hres = IHTMLDocument2_QueryInterface(doc, &IID_IHTMLDocument5, (void**)&doc5);
2498     if(FAILED(hres)) {
2499         win_skip("Could not get IHTMLDocument5 interface, probably too old IE\n");
2500         IHTMLDocument2_Release(doc);
2501         return NULL;
2502     }
2503
2504     IHTMLDocument5_Release(doc5);
2505     return doc;
2506 }
2507
2508
2509 typedef void (*testfunc_t)(IHTMLDocument2*);
2510
2511 static void run_test(const char *str, testfunc_t test)
2512 {
2513     IHTMLDocument2 *doc;
2514     IHTMLElement *body = NULL;
2515     MSG msg;
2516     HRESULT hres;
2517
2518     xy_todo = FALSE;
2519     doc = create_document();
2520     if (!doc)
2521         return;
2522     set_client_site(doc, TRUE);
2523     doc_load_string(doc, str);
2524     do_advise((IUnknown*)doc, &IID_IPropertyNotifySink, (IUnknown*)&PropertyNotifySink);
2525
2526     while(!doc_complete && GetMessage(&msg, NULL, 0, 0)) {
2527         TranslateMessage(&msg);
2528         DispatchMessage(&msg);
2529     }
2530
2531     hres = IHTMLDocument2_get_body(doc, &body);
2532     ok(hres == S_OK, "get_body failed: %08x\n", hres);
2533
2534     if(body) {
2535         IHTMLElement_Release(body);
2536
2537         hres = IHTMLDocument2_get_parentWindow(doc, &window);
2538         ok(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
2539         ok(window != NULL, "window == NULL\n");
2540
2541         test(doc);
2542
2543         IHTMLWindow2_Release(window);
2544         window = NULL;
2545     }else {
2546         skip("Could not get document body. Assuming no Gecko installed.\n");
2547     }
2548
2549     set_client_site(doc, FALSE);
2550     IHTMLDocument2_Release(doc);
2551 }
2552
2553 static LRESULT WINAPI wnd_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
2554 {
2555     return DefWindowProc(hwnd, msg, wParam, lParam);
2556 }
2557
2558 static HWND create_container_window(void)
2559 {
2560     static const CHAR szHTMLDocumentTest[] = "HTMLDocumentTest";
2561     static WNDCLASSEXA wndclass = {
2562         sizeof(WNDCLASSEXA),
2563         0,
2564         wnd_proc,
2565         0, 0, NULL, NULL, NULL, NULL, NULL,
2566         szHTMLDocumentTest,
2567         NULL
2568     };
2569
2570     RegisterClassExA(&wndclass);
2571     return CreateWindowA(szHTMLDocumentTest, szHTMLDocumentTest,
2572             WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT,
2573             300, 300, NULL, NULL, NULL, NULL);
2574 }
2575
2576 static void test_empty_document(void)
2577 {
2578     HRESULT hres;
2579     IHTMLWindow2 *window;
2580     IHTMLDocument2 *windows_doc, *doc;
2581     IConnectionPoint *cp;
2582     DWORD cookie;
2583
2584     doc = create_document();
2585     if(!doc)
2586         return;
2587
2588     hres = IHTMLDocument2_get_parentWindow(doc, &window);
2589     ok(hres == S_OK, "get_parentWindow failed: %08x\n", hres);
2590
2591     hres = IHTMLWindow2_get_document(window, &windows_doc);
2592     IHTMLWindow2_Release(window);
2593     ok(hres == S_OK, "get_document failed: %08x\n", hres);
2594
2595     cookie = register_cp((IUnknown*)windows_doc, &IID_IDispatch, (IUnknown*)&div_onclick_disp);
2596
2597     cp = get_cp((IUnknown*)doc, &IID_IDispatch);
2598     hres = IConnectionPoint_Unadvise(cp, cookie);
2599     IConnectionPoint_Release(cp);
2600     ok(hres == S_OK, "Unadvise failed: %08x\n", hres);
2601
2602     IHTMLDocument2_Release(windows_doc);
2603     IHTMLDocument2_Release(doc);
2604 }
2605
2606 START_TEST(events)
2607 {
2608     CoInitialize(NULL);
2609     container_hwnd = create_container_window();
2610
2611     if(winetest_interactive)
2612         ShowWindow(container_hwnd, SW_SHOW);
2613
2614     run_test(empty_doc_str, test_timeout);
2615     run_test(click_doc_str, test_onclick);
2616     run_test(readystate_doc_str, test_onreadystatechange);
2617     run_test(img_doc_str, test_imgload);
2618     run_test(input_doc_str, test_focus);
2619     run_test(form_doc_str, test_submit);
2620     run_test(iframe_doc_str, test_iframe_connections);
2621
2622     test_empty_document();
2623
2624     DestroyWindow(container_hwnd);
2625     CoUninitialize();
2626 }