mshtml: Implement IHTMLWindow2::get_parent.
[wine] / dlls / mshtml / htmlwindow.c
1 /*
2  * Copyright 2006 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 #include <stdarg.h>
20
21 #define COBJMACROS
22
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winuser.h"
26 #include "ole2.h"
27
28 #include "wine/debug.h"
29 #include "wine/unicode.h"
30
31 #include "mshtml_private.h"
32 #include "htmlevent.h"
33 #include "resource.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
36
37 static struct list window_list = LIST_INIT(window_list);
38
39 static void window_set_docnode(HTMLWindow *window, HTMLDocumentNode *doc_node)
40 {
41     if(window->doc) {
42         abort_document_bindings(window->doc);
43         window->doc->basedoc.window = NULL;
44         htmldoc_release(&window->doc->basedoc);
45     }
46     window->doc = doc_node;
47     if(doc_node)
48         htmldoc_addref(&doc_node->basedoc);
49
50     if(window->doc_obj && window->doc_obj->basedoc.window == window) {
51         if(window->doc_obj->basedoc.doc_node)
52             htmldoc_release(&window->doc_obj->basedoc.doc_node->basedoc);
53         window->doc_obj->basedoc.doc_node = doc_node;
54         if(doc_node)
55             htmldoc_addref(&doc_node->basedoc);
56     }
57 }
58
59 nsIDOMWindow *get_nsdoc_window(nsIDOMDocument *nsdoc)
60 {
61     nsIDOMDocumentView *nsdocview;
62     nsIDOMAbstractView *nsview;
63     nsIDOMWindow *nswindow;
64     nsresult nsres;
65
66     nsres = nsIDOMDocument_QueryInterface(nsdoc, &IID_nsIDOMDocumentView, (void**)&nsdocview);
67     nsIDOMDocument_Release(nsdoc);
68     if(NS_FAILED(nsres)) {
69         ERR("Could not get nsIDOMDocumentView iface: %08x\n", nsres);
70         return NULL;
71     }
72
73     nsres = nsIDOMDocumentView_GetDefaultView(nsdocview, &nsview);
74     nsIDOMDocumentView_Release(nsview);
75     if(NS_FAILED(nsres)) {
76         ERR("GetDefaultView failed: %08x\n", nsres);
77         return NULL;
78     }
79
80     nsres = nsIDOMAbstractView_QueryInterface(nsview, &IID_nsIDOMWindow, (void**)&nswindow);
81     nsIDOMAbstractView_Release(nsview);
82     if(NS_FAILED(nsres)) {
83         ERR("Coult not get nsIDOMWindow iface: %08x\n", nsres);
84         return NULL;
85     }
86
87     return nswindow;
88 }
89
90 static void release_children(HTMLWindow *This)
91 {
92     HTMLWindow *child;
93
94     while(!list_empty(&This->children)) {
95         child = LIST_ENTRY(list_tail(&This->children), HTMLWindow, sibling_entry);
96
97         list_remove(&child->sibling_entry);
98         child->parent = NULL;
99         IHTMLWindow2_Release(HTMLWINDOW2(child));
100     }
101 }
102
103 #define HTMLWINDOW2_THIS(iface) DEFINE_THIS(HTMLWindow, HTMLWindow2, iface)
104
105 static HRESULT WINAPI HTMLWindow2_QueryInterface(IHTMLWindow2 *iface, REFIID riid, void **ppv)
106 {
107     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
108
109     *ppv = NULL;
110
111     if(IsEqualGUID(&IID_IUnknown, riid)) {
112         TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
113         *ppv = HTMLWINDOW2(This);
114     }else if(IsEqualGUID(&IID_IDispatch, riid)) {
115         TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
116         *ppv = HTMLWINDOW2(This);
117     }else if(IsEqualGUID(&IID_IDispatchEx, riid)) {
118         TRACE("(%p)->(IID_IDispatchEx %p)\n", This, ppv);
119         *ppv = DISPATCHEX(This);
120     }else if(IsEqualGUID(&IID_IHTMLFramesCollection2, riid)) {
121         TRACE("(%p)->(IID_IHTMLFramesCollection2 %p)\n", This, ppv);
122         *ppv = HTMLWINDOW2(This);
123     }else if(IsEqualGUID(&IID_IHTMLWindow2, riid)) {
124         TRACE("(%p)->(IID_IHTMLWindow2 %p)\n", This, ppv);
125         *ppv = HTMLWINDOW2(This);
126     }else if(IsEqualGUID(&IID_IHTMLWindow3, riid)) {
127         TRACE("(%p)->(IID_IHTMLWindow3 %p)\n", This, ppv);
128         *ppv = HTMLWINDOW3(This);
129     }else if(IsEqualGUID(&IID_IHTMLWindow4, riid)) {
130         TRACE("(%p)->(IID_IHTMLWindow4 %p)\n", This, ppv);
131         *ppv = HTMLWINDOW4(This);
132     }else if(dispex_query_interface(&This->dispex, riid, ppv)) {
133         return *ppv ? S_OK : E_NOINTERFACE;
134     }
135
136     if(*ppv) {
137         IUnknown_AddRef((IUnknown*)*ppv);
138         return S_OK;
139     }
140
141     WARN("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
142     return E_NOINTERFACE;
143 }
144
145 static ULONG WINAPI HTMLWindow2_AddRef(IHTMLWindow2 *iface)
146 {
147     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
148     LONG ref = InterlockedIncrement(&This->ref);
149
150     TRACE("(%p) ref=%d\n", This, ref);
151
152     return ref;
153 }
154
155 static ULONG WINAPI HTMLWindow2_Release(IHTMLWindow2 *iface)
156 {
157     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
158     LONG ref = InterlockedDecrement(&This->ref);
159
160     TRACE("(%p) ref=%d\n", This, ref);
161
162     if(!ref) {
163         DWORD i;
164
165         remove_target_tasks(This->task_magic);
166         set_window_bscallback(This, NULL);
167         set_current_mon(This, NULL);
168         window_set_docnode(This, NULL);
169         release_children(This);
170
171         if(This->option_factory) {
172             This->option_factory->window = NULL;
173             IHTMLOptionElementFactory_Release(HTMLOPTFACTORY(This->option_factory));
174         }
175
176         if(This->image_factory) {
177             This->image_factory->window = NULL;
178             IHTMLImageElementFactory_Release(HTMLIMGFACTORY(This->image_factory));
179         }
180
181         if(This->location) {
182             This->location->window = NULL;
183             IHTMLLocation_Release(HTMLLOCATION(This->location));
184         }
185
186         if(This->event_target)
187             release_event_target(This->event_target);
188         for(i=0; i < This->global_prop_cnt; i++)
189             heap_free(This->global_props[i].name);
190
191         This->window_ref->window = NULL;
192         windowref_release(This->window_ref);
193
194         heap_free(This->global_props);
195         release_script_hosts(This);
196         list_remove(&This->entry);
197         release_dispex(&This->dispex);
198         heap_free(This);
199     }
200
201     return ref;
202 }
203
204 static HRESULT WINAPI HTMLWindow2_GetTypeInfoCount(IHTMLWindow2 *iface, UINT *pctinfo)
205 {
206     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
207
208     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(This), pctinfo);
209 }
210
211 static HRESULT WINAPI HTMLWindow2_GetTypeInfo(IHTMLWindow2 *iface, UINT iTInfo,
212                                               LCID lcid, ITypeInfo **ppTInfo)
213 {
214     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
215
216     return IDispatchEx_GetTypeInfo(DISPATCHEX(This), iTInfo, lcid, ppTInfo);
217 }
218
219 static HRESULT WINAPI HTMLWindow2_GetIDsOfNames(IHTMLWindow2 *iface, REFIID riid,
220                                                 LPOLESTR *rgszNames, UINT cNames,
221                                                 LCID lcid, DISPID *rgDispId)
222 {
223     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
224
225     return IDispatchEx_GetIDsOfNames(DISPATCHEX(This), riid, rgszNames, cNames, lcid, rgDispId);
226 }
227
228 static HRESULT WINAPI HTMLWindow2_Invoke(IHTMLWindow2 *iface, DISPID dispIdMember,
229                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
230                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
231 {
232     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
233
234     return IDispatchEx_Invoke(DISPATCHEX(This), dispIdMember, riid, lcid, wFlags, pDispParams,
235             pVarResult, pExcepInfo, puArgErr);
236 }
237
238 static HRESULT get_frame_by_index(nsIDOMWindowCollection *nsFrames, PRUint32 index, HTMLWindow **ret)
239 {
240     PRUint32 length;
241     nsIDOMWindow *nsWindow;
242     nsresult nsres;
243
244     nsres = nsIDOMWindowCollection_GetLength(nsFrames, &length);
245     if(NS_FAILED(nsres)) {
246         FIXME("nsIDOMWindowCollection_GetLength failed: 0x%08x\n", nsres);
247         return E_FAIL;
248     }
249
250     if(index >= length)
251         return DISP_E_MEMBERNOTFOUND;
252
253     nsres = nsIDOMWindowCollection_Item(nsFrames, index, &nsWindow);
254     if(NS_FAILED(nsres)) {
255         FIXME("nsIDOMWindowCollection_Item failed: 0x%08x\n", nsres);
256         return E_FAIL;
257     }
258
259     *ret = nswindow_to_window(nsWindow);
260
261     nsIDOMWindow_Release(nsWindow);
262
263     return S_OK;
264 }
265
266 static HRESULT WINAPI HTMLWindow2_item(IHTMLWindow2 *iface, VARIANT *pvarIndex, VARIANT *pvarResult)
267 {
268     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
269     nsIDOMWindowCollection *nsFrames;
270     HTMLWindow *window;
271     HRESULT hres;
272     nsresult nsres;
273
274     TRACE("(%p)->(%p %p)\n", This, pvarIndex, pvarResult);
275
276     nsres = nsIDOMWindow_GetFrames(This->nswindow, &nsFrames);
277     if(NS_FAILED(nsres)) {
278         FIXME("nsIDOMWindow_GetFrames failed: 0x%08x\n", nsres);
279         return E_FAIL;
280     }
281
282     if(V_VT(pvarIndex) == VT_I4) {
283         int index = V_I4(pvarIndex);
284         TRACE("Getting index %d\n", index);
285         if(index < 0) {
286             hres = DISP_E_MEMBERNOTFOUND;
287             goto cleanup;
288         }
289         hres = get_frame_by_index(nsFrames, index, &window);
290         if(FAILED(hres))
291             goto cleanup;
292     }else if(V_VT(pvarIndex) == VT_UINT) {
293         unsigned int index = V_UINT(pvarIndex);
294         TRACE("Getting index %u\n", index);
295         hres = get_frame_by_index(nsFrames, index, &window);
296         if(FAILED(hres))
297             goto cleanup;
298     }else if(V_VT(pvarIndex) == VT_BSTR) {
299         BSTR str = V_BSTR(pvarIndex);
300         PRUint32 length, i;
301
302         TRACE("Getting name %s\n", wine_dbgstr_w(str));
303
304         nsres = nsIDOMWindowCollection_GetLength(nsFrames, &length);
305
306         window = NULL;
307         for(i = 0; i < length && !window; ++i) {
308             HTMLWindow *cur_window;
309             nsIDOMWindow *nsWindow;
310             BSTR id;
311
312             nsres = nsIDOMWindowCollection_Item(nsFrames, i, &nsWindow);
313             if(NS_FAILED(nsres)) {
314                 FIXME("nsIDOMWindowCollection_Item failed: 0x%08x\n", nsres);
315                 hres = E_FAIL;
316                 goto cleanup;
317             }
318
319             cur_window = nswindow_to_window(nsWindow);
320
321             nsIDOMWindow_Release(nsWindow);
322
323             hres = IHTMLElement_get_id(HTMLELEM(&cur_window->frame_element->element), &id);
324             if(FAILED(hres)) {
325                 FIXME("IHTMLElement_get_id failed: 0x%08x\n", hres);
326                 goto cleanup;
327             }
328
329             if(!strcmpW(id, str))
330                 window = cur_window;
331
332             SysFreeString(id);
333         }
334
335         if(!window) {
336             hres = DISP_E_MEMBERNOTFOUND;
337             goto cleanup;
338         }
339     }else {
340         hres = E_INVALIDARG;
341         goto cleanup;
342     }
343
344     IHTMLWindow2_AddRef(HTMLWINDOW2(window));
345     V_VT(pvarResult) = VT_DISPATCH;
346     V_DISPATCH(pvarResult) = (IDispatch*)window;
347
348     hres = S_OK;
349
350 cleanup:
351     nsIDOMWindowCollection_Release(nsFrames);
352
353     return hres;
354 }
355
356 static HRESULT WINAPI HTMLWindow2_get_length(IHTMLWindow2 *iface, LONG *p)
357 {
358     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
359     nsIDOMWindowCollection *nscollection;
360     PRUint32 length;
361     nsresult nsres;
362
363     TRACE("(%p)->(%p)\n", This, p);
364
365     nsres = nsIDOMWindow_GetFrames(This->nswindow, &nscollection);
366     if(NS_FAILED(nsres)) {
367         ERR("GetFrames failed: %08x\n", nsres);
368         return E_FAIL;
369     }
370
371     nsres = nsIDOMWindowCollection_GetLength(nscollection, &length);
372     nsIDOMWindowCollection_Release(nscollection);
373     if(NS_FAILED(nsres)) {
374         ERR("GetLength failed: %08x\n", nsres);
375         return E_FAIL;
376     }
377
378     *p = length;
379     return S_OK;
380 }
381
382 static HRESULT WINAPI HTMLWindow2_get_frames(IHTMLWindow2 *iface, IHTMLFramesCollection2 **p)
383 {
384     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
385     FIXME("(%p)->(%p): semi-stub\n", This, p);
386
387     /* FIXME: Should return a separate Window object */
388     *p = (IHTMLFramesCollection2*)HTMLWINDOW2(This);
389     HTMLWindow2_AddRef(iface);
390     return S_OK;
391 }
392
393 static HRESULT WINAPI HTMLWindow2_put_defaultStatus(IHTMLWindow2 *iface, BSTR v)
394 {
395     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
396     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
397     return E_NOTIMPL;
398 }
399
400 static HRESULT WINAPI HTMLWindow2_get_defaultStatus(IHTMLWindow2 *iface, BSTR *p)
401 {
402     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
403     FIXME("(%p)->(%p)\n", This, p);
404     return E_NOTIMPL;
405 }
406
407 static HRESULT WINAPI HTMLWindow2_put_status(IHTMLWindow2 *iface, BSTR v)
408 {
409     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
410     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
411     return E_NOTIMPL;
412 }
413
414 static HRESULT WINAPI HTMLWindow2_get_status(IHTMLWindow2 *iface, BSTR *p)
415 {
416     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
417     FIXME("(%p)->(%p)\n", This, p);
418     return E_NOTIMPL;
419 }
420
421 static HRESULT WINAPI HTMLWindow2_setTimeout(IHTMLWindow2 *iface, BSTR expression,
422         LONG msec, VARIANT *language, LONG *timerID)
423 {
424     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
425     VARIANT expr_var;
426
427     TRACE("(%p)->(%s %d %p %p)\n", This, debugstr_w(expression), msec, language, timerID);
428
429     V_VT(&expr_var) = VT_BSTR;
430     V_BSTR(&expr_var) = expression;
431
432     return IHTMLWindow3_setTimeout(HTMLWINDOW3(This), &expr_var, msec, language, timerID);
433 }
434
435 static HRESULT WINAPI HTMLWindow2_clearTimeout(IHTMLWindow2 *iface, LONG timerID)
436 {
437     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
438
439     TRACE("(%p)->(%d)\n", This, timerID);
440
441     return clear_task_timer(&This->doc->basedoc, FALSE, timerID);
442 }
443
444 static HRESULT WINAPI HTMLWindow2_alert(IHTMLWindow2 *iface, BSTR message)
445 {
446     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
447     WCHAR wszTitle[100];
448
449     TRACE("(%p)->(%s)\n", This, debugstr_w(message));
450
451     if(!LoadStringW(get_shdoclc(), IDS_MESSAGE_BOX_TITLE, wszTitle,
452                     sizeof(wszTitle)/sizeof(WCHAR))) {
453         WARN("Could not load message box title: %d\n", GetLastError());
454         return S_OK;
455     }
456
457     MessageBoxW(This->doc_obj->hwnd, message, wszTitle, MB_ICONWARNING);
458     return S_OK;
459 }
460
461 static HRESULT WINAPI HTMLWindow2_confirm(IHTMLWindow2 *iface, BSTR message,
462         VARIANT_BOOL *confirmed)
463 {
464     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
465     WCHAR wszTitle[100];
466
467     TRACE("(%p)->(%s %p)\n", This, debugstr_w(message), confirmed);
468
469     if(!confirmed) return E_INVALIDARG;
470
471     if(!LoadStringW(get_shdoclc(), IDS_MESSAGE_BOX_TITLE, wszTitle,
472                 sizeof(wszTitle)/sizeof(WCHAR))) {
473         WARN("Could not load message box title: %d\n", GetLastError());
474         *confirmed = VARIANT_TRUE;
475         return S_OK;
476     }
477
478     if(MessageBoxW(This->doc_obj->hwnd, message, wszTitle,
479                 MB_OKCANCEL|MB_ICONQUESTION)==IDOK)
480         *confirmed = VARIANT_TRUE;
481     else *confirmed = VARIANT_FALSE;
482
483     return S_OK;
484 }
485
486 typedef struct
487 {
488     BSTR message;
489     BSTR dststr;
490     VARIANT *textdata;
491 }prompt_arg;
492
493 static INT_PTR CALLBACK prompt_dlgproc(HWND hwnd, UINT msg,
494         WPARAM wparam, LPARAM lparam)
495 {
496     switch(msg)
497     {
498         case WM_INITDIALOG:
499         {
500             prompt_arg *arg = (prompt_arg*)lparam;
501             WCHAR wszTitle[100];
502
503             if(!LoadStringW(get_shdoclc(), IDS_MESSAGE_BOX_TITLE, wszTitle,
504                         sizeof(wszTitle)/sizeof(WCHAR))) {
505                 WARN("Could not load message box title: %d\n", GetLastError());
506                 EndDialog(hwnd, wparam);
507                 return FALSE;
508             }
509
510             SetWindowLongPtrW(hwnd, DWLP_USER, lparam);
511             SetWindowTextW(hwnd, wszTitle);
512             SetWindowTextW(GetDlgItem(hwnd, ID_PROMPT_PROMPT), arg->message);
513             SetWindowTextW(GetDlgItem(hwnd, ID_PROMPT_EDIT), arg->dststr);
514             return FALSE;
515         }
516         case WM_COMMAND:
517             switch(wparam)
518             {
519                 case MAKEWPARAM(IDCANCEL, BN_CLICKED):
520                     EndDialog(hwnd, wparam);
521                     return TRUE;
522                 case MAKEWPARAM(IDOK, BN_CLICKED):
523                 {
524                     prompt_arg *arg =
525                         (prompt_arg*)GetWindowLongPtrW(hwnd, DWLP_USER);
526                     HWND hwndPrompt = GetDlgItem(hwnd, ID_PROMPT_EDIT);
527                     INT len = GetWindowTextLengthW(hwndPrompt);
528
529                     if(!arg->textdata)
530                     {
531                         EndDialog(hwnd, wparam);
532                         return TRUE;
533                     }
534
535                     V_VT(arg->textdata) = VT_BSTR;
536                     if(!len && !arg->dststr)
537                         V_BSTR(arg->textdata) = NULL;
538                     else
539                     {
540                         V_BSTR(arg->textdata) = SysAllocStringLen(NULL, len);
541                         GetWindowTextW(hwndPrompt, V_BSTR(arg->textdata), len+1);
542                     }
543                     EndDialog(hwnd, wparam);
544                     return TRUE;
545                 }
546             }
547             return FALSE;
548         case WM_CLOSE:
549             EndDialog(hwnd, IDCANCEL);
550             return TRUE;
551         default:
552             return FALSE;
553     }
554 }
555
556 static HRESULT WINAPI HTMLWindow2_prompt(IHTMLWindow2 *iface, BSTR message,
557         BSTR dststr, VARIANT *textdata)
558 {
559     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
560     prompt_arg arg;
561
562     TRACE("(%p)->(%s %s %p)\n", This, debugstr_w(message), debugstr_w(dststr), textdata);
563
564     if(textdata) V_VT(textdata) = VT_NULL;
565
566     arg.message = message;
567     arg.dststr = dststr;
568     arg.textdata = textdata;
569
570     DialogBoxParamW(hInst, MAKEINTRESOURCEW(ID_PROMPT_DIALOG),
571             This->doc_obj->hwnd, prompt_dlgproc, (LPARAM)&arg);
572     return S_OK;
573 }
574
575 static HRESULT WINAPI HTMLWindow2_get_Image(IHTMLWindow2 *iface, IHTMLImageElementFactory **p)
576 {
577     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
578
579     TRACE("(%p)->(%p)\n", This, p);
580
581     if(!This->image_factory)
582         This->image_factory = HTMLImageElementFactory_Create(This);
583
584     *p = HTMLIMGFACTORY(This->image_factory);
585     IHTMLImageElementFactory_AddRef(*p);
586
587     return S_OK;
588 }
589
590 static HRESULT WINAPI HTMLWindow2_get_location(IHTMLWindow2 *iface, IHTMLLocation **p)
591 {
592     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
593
594     TRACE("(%p)->(%p)\n", This, p);
595
596     if(This->location) {
597         IHTMLLocation_AddRef(HTMLLOCATION(This->location));
598     }else {
599         HRESULT hres;
600
601         hres = HTMLLocation_Create(This, &This->location);
602         if(FAILED(hres))
603             return hres;
604     }
605
606     *p = HTMLLOCATION(This->location);
607     return S_OK;
608 }
609
610 static HRESULT WINAPI HTMLWindow2_get_history(IHTMLWindow2 *iface, IOmHistory **p)
611 {
612     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
613     FIXME("(%p)->(%p)\n", This, p);
614     return E_NOTIMPL;
615 }
616
617 static HRESULT WINAPI HTMLWindow2_close(IHTMLWindow2 *iface)
618 {
619     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
620     FIXME("(%p)->()\n", This);
621     return E_NOTIMPL;
622 }
623
624 static HRESULT WINAPI HTMLWindow2_put_opener(IHTMLWindow2 *iface, VARIANT v)
625 {
626     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
627     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
628     return E_NOTIMPL;
629 }
630
631 static HRESULT WINAPI HTMLWindow2_get_opener(IHTMLWindow2 *iface, VARIANT *p)
632 {
633     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
634     FIXME("(%p)->(%p)\n", This, p);
635     return E_NOTIMPL;
636 }
637
638 static HRESULT WINAPI HTMLWindow2_get_navigator(IHTMLWindow2 *iface, IOmNavigator **p)
639 {
640     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
641
642     TRACE("(%p)->(%p)\n", This, p);
643
644     *p = OmNavigator_Create();
645     return S_OK;
646 }
647
648 static HRESULT WINAPI HTMLWindow2_put_name(IHTMLWindow2 *iface, BSTR v)
649 {
650     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
651     nsAString name_str;
652     nsresult nsres;
653
654     TRACE("(%p)->(%s)\n", This, debugstr_w(v));
655
656     nsAString_Init(&name_str, v);
657     nsres = nsIDOMWindow_SetName(This->nswindow, &name_str);
658     nsAString_Finish(&name_str);
659     if(NS_FAILED(nsres))
660         ERR("SetName failed: %08x\n", nsres);
661
662     return S_OK;
663 }
664
665 static HRESULT WINAPI HTMLWindow2_get_name(IHTMLWindow2 *iface, BSTR *p)
666 {
667     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
668     nsAString name_str;
669     nsresult nsres;
670     HRESULT hres;
671
672     TRACE("(%p)->(%p)\n", This, p);
673
674     nsAString_Init(&name_str, NULL);
675     nsres = nsIDOMWindow_GetName(This->nswindow, &name_str);
676     if(NS_SUCCEEDED(nsres)) {
677         const PRUnichar *name;
678
679         nsAString_GetData(&name_str, &name);
680         if(*name) {
681             *p = SysAllocString(name);
682             hres = *p ? S_OK : E_OUTOFMEMORY;
683         }else {
684             *p = NULL;
685             hres = S_OK;
686         }
687     }else {
688         ERR("GetName failed: %08x\n", nsres);
689         hres = E_FAIL;
690     }
691     nsAString_Finish(&name_str);
692
693     return hres;
694 }
695
696 static HRESULT WINAPI HTMLWindow2_get_parent(IHTMLWindow2 *iface, IHTMLWindow2 **p)
697 {
698     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
699     TRACE("(%p)->(%p)\n", This, p);
700
701     if(This->parent) {
702         *p = HTMLWINDOW2(This->parent);
703         IHTMLWindow2_AddRef(*p);
704     }else
705         *p = NULL;
706
707     return S_OK;
708 }
709
710 static HRESULT WINAPI HTMLWindow2_open(IHTMLWindow2 *iface, BSTR url, BSTR name,
711          BSTR features, VARIANT_BOOL replace, IHTMLWindow2 **pomWindowResult)
712 {
713     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
714     FIXME("(%p)->(%s %s %s %x %p)\n", This, debugstr_w(url), debugstr_w(name),
715           debugstr_w(features), replace, pomWindowResult);
716     return E_NOTIMPL;
717 }
718
719 static HRESULT WINAPI HTMLWindow2_get_self(IHTMLWindow2 *iface, IHTMLWindow2 **p)
720 {
721     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
722
723     TRACE("(%p)->(%p)\n", This, p);
724
725     /* FIXME: We should return kind of proxy window here. */
726     IHTMLWindow2_AddRef(HTMLWINDOW2(This));
727     *p = HTMLWINDOW2(This);
728     return S_OK;
729 }
730
731 static HRESULT WINAPI HTMLWindow2_get_top(IHTMLWindow2 *iface, IHTMLWindow2 **p)
732 {
733     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
734     FIXME("(%p)->(%p)\n", This, p);
735     return E_NOTIMPL;
736 }
737
738 static HRESULT WINAPI HTMLWindow2_get_window(IHTMLWindow2 *iface, IHTMLWindow2 **p)
739 {
740     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
741
742     TRACE("(%p)->(%p)\n", This, p);
743
744     /* FIXME: We should return kind of proxy window here. */
745     IHTMLWindow2_AddRef(HTMLWINDOW2(This));
746     *p = HTMLWINDOW2(This);
747     return S_OK;
748 }
749
750 static HRESULT WINAPI HTMLWindow2_navigate(IHTMLWindow2 *iface, BSTR url)
751 {
752     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
753     FIXME("(%p)->(%s)\n", This, debugstr_w(url));
754     return E_NOTIMPL;
755 }
756
757 static HRESULT WINAPI HTMLWindow2_put_onfocus(IHTMLWindow2 *iface, VARIANT v)
758 {
759     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
760     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
761     return E_NOTIMPL;
762 }
763
764 static HRESULT WINAPI HTMLWindow2_get_onfocus(IHTMLWindow2 *iface, VARIANT *p)
765 {
766     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
767     FIXME("(%p)->(%p)\n", This, p);
768     return E_NOTIMPL;
769 }
770
771 static HRESULT WINAPI HTMLWindow2_put_onblur(IHTMLWindow2 *iface, VARIANT v)
772 {
773     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
774     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
775     return E_NOTIMPL;
776 }
777
778 static HRESULT WINAPI HTMLWindow2_get_onblur(IHTMLWindow2 *iface, VARIANT *p)
779 {
780     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
781     FIXME("(%p)->(%p)\n", This, p);
782     return E_NOTIMPL;
783 }
784
785 static HRESULT WINAPI HTMLWindow2_put_onload(IHTMLWindow2 *iface, VARIANT v)
786 {
787     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
788
789     TRACE("(%p)->(%s)\n", This, debugstr_variant(&v));
790
791     return set_window_event(This, EVENTID_LOAD, &v);
792 }
793
794 static HRESULT WINAPI HTMLWindow2_get_onload(IHTMLWindow2 *iface, VARIANT *p)
795 {
796     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
797
798     TRACE("(%p)->(%p)\n", This, p);
799
800     return get_window_event(This, EVENTID_LOAD, p);
801 }
802
803 static HRESULT WINAPI HTMLWindow2_put_onbeforeunload(IHTMLWindow2 *iface, VARIANT v)
804 {
805     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
806
807     TRACE("(%p)->(v(%d))\n", This, V_VT(&v));
808
809     return set_window_event(This, EVENTID_BEFOREUNLOAD, &v);
810 }
811
812 static HRESULT WINAPI HTMLWindow2_get_onbeforeunload(IHTMLWindow2 *iface, VARIANT *p)
813 {
814     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
815
816     TRACE("(%p)->(%p)\n", This, p);
817
818     return get_window_event(This, EVENTID_BEFOREUNLOAD, p);
819 }
820
821 static HRESULT WINAPI HTMLWindow2_put_onunload(IHTMLWindow2 *iface, VARIANT v)
822 {
823     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
824     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
825     return E_NOTIMPL;
826 }
827
828 static HRESULT WINAPI HTMLWindow2_get_onunload(IHTMLWindow2 *iface, VARIANT *p)
829 {
830     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
831     FIXME("(%p)->(%p)\n", This, p);
832     return E_NOTIMPL;
833 }
834
835 static HRESULT WINAPI HTMLWindow2_put_onhelp(IHTMLWindow2 *iface, VARIANT v)
836 {
837     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
838     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
839     return E_NOTIMPL;
840 }
841
842 static HRESULT WINAPI HTMLWindow2_get_onhelp(IHTMLWindow2 *iface, VARIANT *p)
843 {
844     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
845     FIXME("(%p)->(%p)\n", This, p);
846     return E_NOTIMPL;
847 }
848
849 static HRESULT WINAPI HTMLWindow2_put_onerror(IHTMLWindow2 *iface, VARIANT v)
850 {
851     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
852     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
853     return E_NOTIMPL;
854 }
855
856 static HRESULT WINAPI HTMLWindow2_get_onerror(IHTMLWindow2 *iface, VARIANT *p)
857 {
858     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
859     FIXME("(%p)->(%p)\n", This, p);
860     return E_NOTIMPL;
861 }
862
863 static HRESULT WINAPI HTMLWindow2_put_onresize(IHTMLWindow2 *iface, VARIANT v)
864 {
865     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
866     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
867     return E_NOTIMPL;
868 }
869
870 static HRESULT WINAPI HTMLWindow2_get_onresize(IHTMLWindow2 *iface, VARIANT *p)
871 {
872     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
873     FIXME("(%p)->(%p)\n", This, p);
874     return E_NOTIMPL;
875 }
876
877 static HRESULT WINAPI HTMLWindow2_put_onscroll(IHTMLWindow2 *iface, VARIANT v)
878 {
879     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
880     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
881     return E_NOTIMPL;
882 }
883
884 static HRESULT WINAPI HTMLWindow2_get_onscroll(IHTMLWindow2 *iface, VARIANT *p)
885 {
886     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
887     FIXME("(%p)->(%p)\n", This, p);
888     return E_NOTIMPL;
889 }
890
891 static HRESULT WINAPI HTMLWindow2_get_document(IHTMLWindow2 *iface, IHTMLDocument2 **p)
892 {
893     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
894
895     TRACE("(%p)->(%p)\n", This, p);
896
897     if(This->doc) {
898         /* FIXME: We should return a wrapper object here */
899         *p = HTMLDOC(&This->doc->basedoc);
900         IHTMLDocument2_AddRef(*p);
901     }else {
902         *p = NULL;
903     }
904
905     return S_OK;
906 }
907
908 static HRESULT WINAPI HTMLWindow2_get_event(IHTMLWindow2 *iface, IHTMLEventObj **p)
909 {
910     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
911
912     TRACE("(%p)->(%p)\n", This, p);
913
914     if(This->event)
915         IHTMLEventObj_AddRef(This->event);
916     *p = This->event;
917     return S_OK;
918 }
919
920 static HRESULT WINAPI HTMLWindow2_get__newEnum(IHTMLWindow2 *iface, IUnknown **p)
921 {
922     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
923     FIXME("(%p)->(%p)\n", This, p);
924     return E_NOTIMPL;
925 }
926
927 static HRESULT WINAPI HTMLWindow2_showModalDialog(IHTMLWindow2 *iface, BSTR dialog,
928         VARIANT *varArgIn, VARIANT *varOptions, VARIANT *varArgOut)
929 {
930     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
931     FIXME("(%p)->(%s %p %p %p)\n", This, debugstr_w(dialog), varArgIn, varOptions, varArgOut);
932     return E_NOTIMPL;
933 }
934
935 static HRESULT WINAPI HTMLWindow2_showHelp(IHTMLWindow2 *iface, BSTR helpURL, VARIANT helpArg,
936         BSTR features)
937 {
938     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
939     FIXME("(%p)->(%s v(%d) %s)\n", This, debugstr_w(helpURL), V_VT(&helpArg), debugstr_w(features));
940     return E_NOTIMPL;
941 }
942
943 static HRESULT WINAPI HTMLWindow2_get_screen(IHTMLWindow2 *iface, IHTMLScreen **p)
944 {
945     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
946     FIXME("(%p)->(%p)\n", This, p);
947     return E_NOTIMPL;
948 }
949
950 static HRESULT WINAPI HTMLWindow2_get_Option(IHTMLWindow2 *iface, IHTMLOptionElementFactory **p)
951 {
952     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
953
954     TRACE("(%p)->(%p)\n", This, p);
955
956     if(!This->option_factory)
957         This->option_factory = HTMLOptionElementFactory_Create(This);
958
959     *p = HTMLOPTFACTORY(This->option_factory);
960     IHTMLOptionElementFactory_AddRef(*p);
961
962     return S_OK;
963 }
964
965 static HRESULT WINAPI HTMLWindow2_focus(IHTMLWindow2 *iface)
966 {
967     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
968     FIXME("(%p)->()\n", This);
969     return E_NOTIMPL;
970 }
971
972 static HRESULT WINAPI HTMLWindow2_get_closed(IHTMLWindow2 *iface, VARIANT_BOOL *p)
973 {
974     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
975     FIXME("(%p)->(%p)\n", This, p);
976     return E_NOTIMPL;
977 }
978
979 static HRESULT WINAPI HTMLWindow2_blur(IHTMLWindow2 *iface)
980 {
981     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
982     FIXME("(%p)->()\n", This);
983     return E_NOTIMPL;
984 }
985
986 static HRESULT WINAPI HTMLWindow2_scroll(IHTMLWindow2 *iface, LONG x, LONG y)
987 {
988     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
989     FIXME("(%p)->(%d %d)\n", This, x, y);
990     return E_NOTIMPL;
991 }
992
993 static HRESULT WINAPI HTMLWindow2_get_clientInformation(IHTMLWindow2 *iface, IOmNavigator **p)
994 {
995     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
996     FIXME("(%p)->(%p)\n", This, p);
997     return E_NOTIMPL;
998 }
999
1000 static HRESULT WINAPI HTMLWindow2_setInterval(IHTMLWindow2 *iface, BSTR expression,
1001         LONG msec, VARIANT *language, LONG *timerID)
1002 {
1003     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1004     VARIANT expr;
1005
1006     TRACE("(%p)->(%s %d %p %p)\n", This, debugstr_w(expression), msec, language, timerID);
1007
1008     V_VT(&expr) = VT_BSTR;
1009     V_BSTR(&expr) = expression;
1010     return IHTMLWindow3_setInterval(HTMLWINDOW3(This), &expr, msec, language, timerID);
1011 }
1012
1013 static HRESULT WINAPI HTMLWindow2_clearInterval(IHTMLWindow2 *iface, LONG timerID)
1014 {
1015     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1016
1017     TRACE("(%p)->(%d)\n", This, timerID);
1018
1019     return clear_task_timer(&This->doc->basedoc, TRUE, timerID);
1020 }
1021
1022 static HRESULT WINAPI HTMLWindow2_put_offscreenBuffering(IHTMLWindow2 *iface, VARIANT v)
1023 {
1024     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1025     FIXME("(%p)->(v(%d))\n", This, V_VT(&v));
1026     return E_NOTIMPL;
1027 }
1028
1029 static HRESULT WINAPI HTMLWindow2_get_offscreenBuffering(IHTMLWindow2 *iface, VARIANT *p)
1030 {
1031     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1032     FIXME("(%p)->(%p)\n", This, p);
1033     return E_NOTIMPL;
1034 }
1035
1036 static HRESULT WINAPI HTMLWindow2_execScript(IHTMLWindow2 *iface, BSTR scode, BSTR language,
1037         VARIANT *pvarRet)
1038 {
1039     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1040     FIXME("(%p)->(%s %s %p)\n", This, debugstr_w(scode), debugstr_w(language), pvarRet);
1041     return E_NOTIMPL;
1042 }
1043
1044 static HRESULT WINAPI HTMLWindow2_toString(IHTMLWindow2 *iface, BSTR *String)
1045 {
1046     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1047
1048     static const WCHAR objectW[] = {'[','o','b','j','e','c','t',']',0};
1049
1050     TRACE("(%p)->(%p)\n", This, String);
1051
1052     if(!String)
1053         return E_INVALIDARG;
1054
1055     *String = SysAllocString(objectW);
1056     return *String ? S_OK : E_OUTOFMEMORY;
1057 }
1058
1059 static HRESULT WINAPI HTMLWindow2_scrollBy(IHTMLWindow2 *iface, LONG x, LONG y)
1060 {
1061     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1062     nsresult nsres;
1063
1064     TRACE("(%p)->(%d %d)\n", This, x, y);
1065
1066     nsres = nsIDOMWindow_ScrollBy(This->nswindow, x, y);
1067     if(NS_FAILED(nsres))
1068         ERR("ScrollBy failed: %08x\n", nsres);
1069
1070     return S_OK;
1071 }
1072
1073 static HRESULT WINAPI HTMLWindow2_scrollTo(IHTMLWindow2 *iface, LONG x, LONG y)
1074 {
1075     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1076     nsresult nsres;
1077
1078     TRACE("(%p)->(%d %d)\n", This, x, y);
1079
1080     nsres = nsIDOMWindow_ScrollTo(This->nswindow, x, y);
1081     if(NS_FAILED(nsres))
1082         ERR("ScrollTo failed: %08x\n", nsres);
1083
1084     return S_OK;
1085 }
1086
1087 static HRESULT WINAPI HTMLWindow2_moveTo(IHTMLWindow2 *iface, LONG x, LONG y)
1088 {
1089     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1090     FIXME("(%p)->(%d %d)\n", This, x, y);
1091     return E_NOTIMPL;
1092 }
1093
1094 static HRESULT WINAPI HTMLWindow2_moveBy(IHTMLWindow2 *iface, LONG x, LONG y)
1095 {
1096     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1097     FIXME("(%p)->(%d %d)\n", This, x, y);
1098     return E_NOTIMPL;
1099 }
1100
1101 static HRESULT WINAPI HTMLWindow2_resizeTo(IHTMLWindow2 *iface, LONG x, LONG y)
1102 {
1103     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1104     FIXME("(%p)->(%d %d)\n", This, x, y);
1105     return E_NOTIMPL;
1106 }
1107
1108 static HRESULT WINAPI HTMLWindow2_resizeBy(IHTMLWindow2 *iface, LONG x, LONG y)
1109 {
1110     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1111     FIXME("(%p)->(%d %d)\n", This, x, y);
1112     return E_NOTIMPL;
1113 }
1114
1115 static HRESULT WINAPI HTMLWindow2_get_external(IHTMLWindow2 *iface, IDispatch **p)
1116 {
1117     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1118
1119     TRACE("(%p)->(%p)\n", This, p);
1120
1121     *p = NULL;
1122
1123     if(!This->doc_obj->hostui)
1124         return S_OK;
1125
1126     return IDocHostUIHandler_GetExternal(This->doc_obj->hostui, p);
1127 }
1128
1129 static HRESULT HTMLWindow_invoke(IUnknown *iface, DISPID id, LCID lcid, WORD flags, DISPPARAMS *params,
1130         VARIANT *res, EXCEPINFO *ei, IServiceProvider *caller)
1131 {
1132     HTMLWindow *This = HTMLWINDOW2_THIS(iface);
1133     global_prop_t *prop;
1134     DWORD idx;
1135     HRESULT hres;
1136
1137     idx = id - MSHTML_DISPID_CUSTOM_MIN;
1138     if(idx >= This->global_prop_cnt)
1139         return DISP_E_MEMBERNOTFOUND;
1140
1141     prop = This->global_props+idx;
1142
1143     switch(prop->type) {
1144     case GLOBAL_SCRIPTVAR: {
1145         IDispatchEx *dispex;
1146         IDispatch *disp;
1147
1148         disp = get_script_disp(prop->script_host);
1149         if(!disp)
1150             return E_UNEXPECTED;
1151
1152         hres = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
1153         if(SUCCEEDED(hres)) {
1154             TRACE("%s >>>\n", debugstr_w(prop->name));
1155             hres = IDispatchEx_InvokeEx(dispex, prop->id, lcid, flags, params, res, ei, caller);
1156             if(hres == S_OK)
1157                 TRACE("%s <<<\n", debugstr_w(prop->name));
1158             else
1159                 WARN("%s <<< %08x\n", debugstr_w(prop->name), hres);
1160             IDispatchEx_Release(dispex);
1161         }else {
1162             FIXME("No IDispatchEx\n");
1163         }
1164         IDispatch_Release(disp);
1165         break;
1166     }
1167     case GLOBAL_ELEMENTVAR: {
1168         IHTMLElement *elem;
1169
1170         hres = IHTMLDocument3_getElementById(HTMLDOC3(&This->doc->basedoc), prop->name, &elem);
1171         if(FAILED(hres))
1172             return hres;
1173
1174         if(!elem)
1175             return DISP_E_MEMBERNOTFOUND;
1176
1177         V_VT(res) = VT_DISPATCH;
1178         V_DISPATCH(res) = (IDispatch*)elem;
1179         break;
1180     }
1181     default:
1182         ERR("invalid type %d\n", prop->type);
1183         hres = DISP_E_MEMBERNOTFOUND;
1184     }
1185
1186     return hres;
1187 }
1188
1189 #undef HTMLWINDOW2_THIS
1190
1191 static const IHTMLWindow2Vtbl HTMLWindow2Vtbl = {
1192     HTMLWindow2_QueryInterface,
1193     HTMLWindow2_AddRef,
1194     HTMLWindow2_Release,
1195     HTMLWindow2_GetTypeInfoCount,
1196     HTMLWindow2_GetTypeInfo,
1197     HTMLWindow2_GetIDsOfNames,
1198     HTMLWindow2_Invoke,
1199     HTMLWindow2_item,
1200     HTMLWindow2_get_length,
1201     HTMLWindow2_get_frames,
1202     HTMLWindow2_put_defaultStatus,
1203     HTMLWindow2_get_defaultStatus,
1204     HTMLWindow2_put_status,
1205     HTMLWindow2_get_status,
1206     HTMLWindow2_setTimeout,
1207     HTMLWindow2_clearTimeout,
1208     HTMLWindow2_alert,
1209     HTMLWindow2_confirm,
1210     HTMLWindow2_prompt,
1211     HTMLWindow2_get_Image,
1212     HTMLWindow2_get_location,
1213     HTMLWindow2_get_history,
1214     HTMLWindow2_close,
1215     HTMLWindow2_put_opener,
1216     HTMLWindow2_get_opener,
1217     HTMLWindow2_get_navigator,
1218     HTMLWindow2_put_name,
1219     HTMLWindow2_get_name,
1220     HTMLWindow2_get_parent,
1221     HTMLWindow2_open,
1222     HTMLWindow2_get_self,
1223     HTMLWindow2_get_top,
1224     HTMLWindow2_get_window,
1225     HTMLWindow2_navigate,
1226     HTMLWindow2_put_onfocus,
1227     HTMLWindow2_get_onfocus,
1228     HTMLWindow2_put_onblur,
1229     HTMLWindow2_get_onblur,
1230     HTMLWindow2_put_onload,
1231     HTMLWindow2_get_onload,
1232     HTMLWindow2_put_onbeforeunload,
1233     HTMLWindow2_get_onbeforeunload,
1234     HTMLWindow2_put_onunload,
1235     HTMLWindow2_get_onunload,
1236     HTMLWindow2_put_onhelp,
1237     HTMLWindow2_get_onhelp,
1238     HTMLWindow2_put_onerror,
1239     HTMLWindow2_get_onerror,
1240     HTMLWindow2_put_onresize,
1241     HTMLWindow2_get_onresize,
1242     HTMLWindow2_put_onscroll,
1243     HTMLWindow2_get_onscroll,
1244     HTMLWindow2_get_document,
1245     HTMLWindow2_get_event,
1246     HTMLWindow2_get__newEnum,
1247     HTMLWindow2_showModalDialog,
1248     HTMLWindow2_showHelp,
1249     HTMLWindow2_get_screen,
1250     HTMLWindow2_get_Option,
1251     HTMLWindow2_focus,
1252     HTMLWindow2_get_closed,
1253     HTMLWindow2_blur,
1254     HTMLWindow2_scroll,
1255     HTMLWindow2_get_clientInformation,
1256     HTMLWindow2_setInterval,
1257     HTMLWindow2_clearInterval,
1258     HTMLWindow2_put_offscreenBuffering,
1259     HTMLWindow2_get_offscreenBuffering,
1260     HTMLWindow2_execScript,
1261     HTMLWindow2_toString,
1262     HTMLWindow2_scrollBy,
1263     HTMLWindow2_scrollTo,
1264     HTMLWindow2_moveTo,
1265     HTMLWindow2_moveBy,
1266     HTMLWindow2_resizeTo,
1267     HTMLWindow2_resizeBy,
1268     HTMLWindow2_get_external
1269 };
1270
1271 #define HTMLWINDOW3_THIS(iface) DEFINE_THIS(HTMLWindow, HTMLWindow3, iface)
1272
1273 static HRESULT WINAPI HTMLWindow3_QueryInterface(IHTMLWindow3 *iface, REFIID riid, void **ppv)
1274 {
1275     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1276
1277     return IHTMLWindow2_QueryInterface(HTMLWINDOW2(This), riid, ppv);
1278 }
1279
1280 static ULONG WINAPI HTMLWindow3_AddRef(IHTMLWindow3 *iface)
1281 {
1282     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1283
1284     return IHTMLWindow2_AddRef(HTMLWINDOW2(This));
1285 }
1286
1287 static ULONG WINAPI HTMLWindow3_Release(IHTMLWindow3 *iface)
1288 {
1289     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1290
1291     return IHTMLWindow2_Release(HTMLWINDOW2(This));
1292 }
1293
1294 static HRESULT WINAPI HTMLWindow3_GetTypeInfoCount(IHTMLWindow3 *iface, UINT *pctinfo)
1295 {
1296     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1297
1298     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(This), pctinfo);
1299 }
1300
1301 static HRESULT WINAPI HTMLWindow3_GetTypeInfo(IHTMLWindow3 *iface, UINT iTInfo,
1302                                               LCID lcid, ITypeInfo **ppTInfo)
1303 {
1304     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1305
1306     return IDispatchEx_GetTypeInfo(DISPATCHEX(This), iTInfo, lcid, ppTInfo);
1307 }
1308
1309 static HRESULT WINAPI HTMLWindow3_GetIDsOfNames(IHTMLWindow3 *iface, REFIID riid,
1310                                                 LPOLESTR *rgszNames, UINT cNames,
1311                                                 LCID lcid, DISPID *rgDispId)
1312 {
1313     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1314
1315     return IDispatchEx_GetIDsOfNames(DISPATCHEX(This), riid, rgszNames, cNames, lcid, rgDispId);
1316 }
1317
1318 static HRESULT WINAPI HTMLWindow3_Invoke(IHTMLWindow3 *iface, DISPID dispIdMember,
1319                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
1320                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
1321 {
1322     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1323
1324     return IDispatchEx_Invoke(DISPATCHEX(This), dispIdMember, riid, lcid, wFlags, pDispParams,
1325             pVarResult, pExcepInfo, puArgErr);
1326 }
1327
1328 static HRESULT WINAPI HTMLWindow3_get_screenLeft(IHTMLWindow3 *iface, LONG *p)
1329 {
1330     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1331     FIXME("(%p)->(%p)\n", This, p);
1332     return E_NOTIMPL;
1333 }
1334
1335 static HRESULT WINAPI HTMLWindow3_get_screenTop(IHTMLWindow3 *iface, LONG *p)
1336 {
1337     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1338     FIXME("(%p)->(%p)\n", This, p);
1339     return E_NOTIMPL;
1340 }
1341
1342 static HRESULT WINAPI HTMLWindow3_attachEvent(IHTMLWindow3 *iface, BSTR event, IDispatch *pDisp, VARIANT_BOOL *pfResult)
1343 {
1344     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1345
1346     TRACE("(%p)->(%s %p %p)\n", This, debugstr_w(event), pDisp, pfResult);
1347
1348     return attach_event(&This->event_target, &This->doc_obj->basedoc, event, pDisp, pfResult);
1349 }
1350
1351 static HRESULT WINAPI HTMLWindow3_detachEvent(IHTMLWindow3 *iface, BSTR event, IDispatch *pDisp)
1352 {
1353     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1354     FIXME("(%p)->()\n", This);
1355     return E_NOTIMPL;
1356 }
1357
1358 static HRESULT window_set_timer(HTMLWindow *This, VARIANT *expr, LONG msec, VARIANT *language,
1359         BOOL interval, LONG *timer_id)
1360 {
1361     IDispatch *disp = NULL;
1362
1363     switch(V_VT(expr)) {
1364     case VT_DISPATCH:
1365         disp = V_DISPATCH(expr);
1366         IDispatch_AddRef(disp);
1367         break;
1368
1369     case VT_BSTR:
1370         disp = script_parse_event(This, V_BSTR(expr));
1371         break;
1372
1373     default:
1374         FIXME("unimplemented vt=%d\n", V_VT(expr));
1375         return E_NOTIMPL;
1376     }
1377
1378     if(!disp)
1379         return E_FAIL;
1380
1381     *timer_id = set_task_timer(&This->doc->basedoc, msec, interval, disp);
1382     IDispatch_Release(disp);
1383
1384     return S_OK;
1385 }
1386
1387 static HRESULT WINAPI HTMLWindow3_setTimeout(IHTMLWindow3 *iface, VARIANT *expression, LONG msec,
1388         VARIANT *language, LONG *timerID)
1389 {
1390     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1391
1392     TRACE("(%p)->(%p(%d) %d %p %p)\n", This, expression, V_VT(expression), msec, language, timerID);
1393
1394     return window_set_timer(This, expression, msec, language, FALSE, timerID);
1395 }
1396
1397 static HRESULT WINAPI HTMLWindow3_setInterval(IHTMLWindow3 *iface, VARIANT *expression, LONG msec,
1398         VARIANT *language, LONG *timerID)
1399 {
1400     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1401
1402     TRACE("(%p)->(%p %d %p %p)\n", This, expression, msec, language, timerID);
1403
1404     return window_set_timer(This, expression, msec, language, TRUE, timerID);
1405 }
1406
1407 static HRESULT WINAPI HTMLWindow3_print(IHTMLWindow3 *iface)
1408 {
1409     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1410     FIXME("(%p)\n", This);
1411     return E_NOTIMPL;
1412 }
1413
1414 static HRESULT WINAPI HTMLWindow3_put_onbeforeprint(IHTMLWindow3 *iface, VARIANT v)
1415 {
1416     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1417     FIXME("(%p)->()\n", This);
1418     return E_NOTIMPL;
1419 }
1420
1421 static HRESULT WINAPI HTMLWindow3_get_onbeforeprint(IHTMLWindow3 *iface, VARIANT *p)
1422 {
1423     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1424     FIXME("(%p)->(%p)\n", This, p);
1425     return E_NOTIMPL;
1426 }
1427
1428 static HRESULT WINAPI HTMLWindow3_put_onafterprint(IHTMLWindow3 *iface, VARIANT v)
1429 {
1430     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1431     FIXME("(%p)->()\n", This);
1432     return E_NOTIMPL;
1433 }
1434
1435 static HRESULT WINAPI HTMLWindow3_get_onafterprint(IHTMLWindow3 *iface, VARIANT *p)
1436 {
1437     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1438     FIXME("(%p)->(%p)\n", This, p);
1439     return E_NOTIMPL;
1440 }
1441
1442 static HRESULT WINAPI HTMLWindow3_get_clipboardData(IHTMLWindow3 *iface, IHTMLDataTransfer **p)
1443 {
1444     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1445     FIXME("(%p)->(%p)\n", This, p);
1446     return E_NOTIMPL;
1447 }
1448
1449 static HRESULT WINAPI HTMLWindow3_showModelessDialog(IHTMLWindow3 *iface, BSTR url,
1450         VARIANT *varArgIn, VARIANT *options, IHTMLWindow2 **pDialog)
1451 {
1452     HTMLWindow *This = HTMLWINDOW3_THIS(iface);
1453     FIXME("(%p)->(%s %p %p %p)\n", This, debugstr_w(url), varArgIn, options, pDialog);
1454     return E_NOTIMPL;
1455 }
1456
1457 #undef HTMLWINDOW3_THIS
1458
1459 static const IHTMLWindow3Vtbl HTMLWindow3Vtbl = {
1460     HTMLWindow3_QueryInterface,
1461     HTMLWindow3_AddRef,
1462     HTMLWindow3_Release,
1463     HTMLWindow3_GetTypeInfoCount,
1464     HTMLWindow3_GetTypeInfo,
1465     HTMLWindow3_GetIDsOfNames,
1466     HTMLWindow3_Invoke,
1467     HTMLWindow3_get_screenLeft,
1468     HTMLWindow3_get_screenTop,
1469     HTMLWindow3_attachEvent,
1470     HTMLWindow3_detachEvent,
1471     HTMLWindow3_setTimeout,
1472     HTMLWindow3_setInterval,
1473     HTMLWindow3_print,
1474     HTMLWindow3_put_onbeforeprint,
1475     HTMLWindow3_get_onbeforeprint,
1476     HTMLWindow3_put_onafterprint,
1477     HTMLWindow3_get_onafterprint,
1478     HTMLWindow3_get_clipboardData,
1479     HTMLWindow3_showModelessDialog
1480 };
1481
1482 #define HTMLWINDOW4_THIS(iface) DEFINE_THIS(HTMLWindow, HTMLWindow4, iface)
1483
1484 static HRESULT WINAPI HTMLWindow4_QueryInterface(IHTMLWindow4 *iface, REFIID riid, void **ppv)
1485 {
1486     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1487
1488     return IHTMLWindow2_QueryInterface(HTMLWINDOW2(This), riid, ppv);
1489 }
1490
1491 static ULONG WINAPI HTMLWindow4_AddRef(IHTMLWindow4 *iface)
1492 {
1493     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1494
1495     return IHTMLWindow2_AddRef(HTMLWINDOW2(This));
1496 }
1497
1498 static ULONG WINAPI HTMLWindow4_Release(IHTMLWindow4 *iface)
1499 {
1500     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1501
1502     return IHTMLWindow2_Release(HTMLWINDOW2(This));
1503 }
1504
1505 static HRESULT WINAPI HTMLWindow4_GetTypeInfoCount(IHTMLWindow4 *iface, UINT *pctinfo)
1506 {
1507     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1508
1509     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(This), pctinfo);
1510 }
1511
1512 static HRESULT WINAPI HTMLWindow4_GetTypeInfo(IHTMLWindow4 *iface, UINT iTInfo,
1513                                               LCID lcid, ITypeInfo **ppTInfo)
1514 {
1515     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1516
1517     return IDispatchEx_GetTypeInfo(DISPATCHEX(This), iTInfo, lcid, ppTInfo);
1518 }
1519
1520 static HRESULT WINAPI HTMLWindow4_GetIDsOfNames(IHTMLWindow4 *iface, REFIID riid,
1521                                                 LPOLESTR *rgszNames, UINT cNames,
1522                                                 LCID lcid, DISPID *rgDispId)
1523 {
1524     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1525
1526     return IDispatchEx_GetIDsOfNames(DISPATCHEX(This), riid, rgszNames, cNames, lcid, rgDispId);
1527 }
1528
1529 static HRESULT WINAPI HTMLWindow4_Invoke(IHTMLWindow4 *iface, DISPID dispIdMember,
1530                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
1531                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
1532 {
1533     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1534
1535     return IDispatchEx_Invoke(DISPATCHEX(This), dispIdMember, riid, lcid, wFlags, pDispParams,
1536             pVarResult, pExcepInfo, puArgErr);
1537 }
1538
1539 static HRESULT WINAPI HTMLWindow4_createPopup(IHTMLWindow4 *iface, VARIANT *varArgIn,
1540                             IDispatch **ppPopup)
1541 {
1542     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1543     FIXME("(%p)->(%p %p)\n", This, varArgIn, ppPopup);
1544     return E_NOTIMPL;
1545 }
1546
1547 static HRESULT WINAPI HTMLWindow4_get_frameElement(IHTMLWindow4 *iface, IHTMLFrameBase **p)
1548 {
1549     HTMLWindow *This = HTMLWINDOW4_THIS(iface);
1550     TRACE("(%p)->(%p)\n", This, p);
1551
1552     if(This->frame_element) {
1553         *p = HTMLFRAMEBASE(This->frame_element);
1554         IHTMLFrameBase_AddRef(*p);
1555     }else
1556         *p = NULL;
1557
1558     return S_OK;
1559 }
1560
1561 #undef HTMLWINDOW4_THIS
1562
1563 static const IHTMLWindow4Vtbl HTMLWindow4Vtbl = {
1564     HTMLWindow4_QueryInterface,
1565     HTMLWindow4_AddRef,
1566     HTMLWindow4_Release,
1567     HTMLWindow4_GetTypeInfoCount,
1568     HTMLWindow4_GetTypeInfo,
1569     HTMLWindow4_GetIDsOfNames,
1570     HTMLWindow4_Invoke,
1571     HTMLWindow4_createPopup,
1572     HTMLWindow4_get_frameElement
1573 };
1574
1575 #define DISPEX_THIS(iface) DEFINE_THIS(HTMLWindow, IDispatchEx, iface)
1576
1577 static HRESULT WINAPI WindowDispEx_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
1578 {
1579     HTMLWindow *This = DISPEX_THIS(iface);
1580
1581     return IHTMLWindow2_QueryInterface(HTMLWINDOW2(This), riid, ppv);
1582 }
1583
1584 static ULONG WINAPI WindowDispEx_AddRef(IDispatchEx *iface)
1585 {
1586     HTMLWindow *This = DISPEX_THIS(iface);
1587
1588     return IHTMLWindow2_AddRef(HTMLWINDOW2(This));
1589 }
1590
1591 static ULONG WINAPI WindowDispEx_Release(IDispatchEx *iface)
1592 {
1593     HTMLWindow *This = DISPEX_THIS(iface);
1594
1595     return IHTMLWindow2_Release(HTMLWINDOW2(This));
1596 }
1597
1598 static HRESULT WINAPI WindowDispEx_GetTypeInfoCount(IDispatchEx *iface, UINT *pctinfo)
1599 {
1600     HTMLWindow *This = DISPEX_THIS(iface);
1601
1602     TRACE("(%p)->(%p)\n", This, pctinfo);
1603
1604     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(&This->dispex), pctinfo);
1605 }
1606
1607 static HRESULT WINAPI WindowDispEx_GetTypeInfo(IDispatchEx *iface, UINT iTInfo,
1608                                                LCID lcid, ITypeInfo **ppTInfo)
1609 {
1610     HTMLWindow *This = DISPEX_THIS(iface);
1611
1612     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
1613
1614     return IDispatchEx_GetTypeInfo(DISPATCHEX(&This->dispex), iTInfo, lcid, ppTInfo);
1615 }
1616
1617 static HRESULT WINAPI WindowDispEx_GetIDsOfNames(IDispatchEx *iface, REFIID riid,
1618                                                  LPOLESTR *rgszNames, UINT cNames,
1619                                                  LCID lcid, DISPID *rgDispId)
1620 {
1621     HTMLWindow *This = DISPEX_THIS(iface);
1622
1623     TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
1624           lcid, rgDispId);
1625
1626     /* FIXME: Use script dispatch */
1627
1628     return IDispatchEx_GetIDsOfNames(DISPATCHEX(&This->dispex), riid, rgszNames, cNames, lcid, rgDispId);
1629 }
1630
1631 static HRESULT WINAPI WindowDispEx_Invoke(IDispatchEx *iface, DISPID dispIdMember,
1632                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
1633                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
1634 {
1635     HTMLWindow *This = DISPEX_THIS(iface);
1636
1637     TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
1638           lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
1639
1640     /* FIXME: Use script dispatch */
1641
1642     return IDispatchEx_Invoke(DISPATCHEX(&This->dispex), dispIdMember, riid, lcid, wFlags, pDispParams,
1643                               pVarResult, pExcepInfo, puArgErr);
1644 }
1645
1646 static global_prop_t *alloc_global_prop(HTMLWindow *This, global_prop_type_t type, BSTR name)
1647 {
1648     if(This->global_prop_cnt == This->global_prop_size) {
1649         global_prop_t *new_props;
1650         DWORD new_size;
1651
1652         if(This->global_props) {
1653             new_size = This->global_prop_size*2;
1654             new_props = heap_realloc(This->global_props, new_size*sizeof(global_prop_t));
1655         }else {
1656             new_size = 16;
1657             new_props = heap_alloc(new_size*sizeof(global_prop_t));
1658         }
1659         if(!new_props)
1660             return NULL;
1661         This->global_props = new_props;
1662         This->global_prop_size = new_size;
1663     }
1664
1665     This->global_props[This->global_prop_cnt].name = heap_strdupW(name);
1666     if(!This->global_props[This->global_prop_cnt].name)
1667         return NULL;
1668
1669     This->global_props[This->global_prop_cnt].type = type;
1670     return This->global_props + This->global_prop_cnt++;
1671 }
1672
1673 static inline DWORD prop_to_dispid(HTMLWindow *This, global_prop_t *prop)
1674 {
1675     return MSHTML_DISPID_CUSTOM_MIN + (prop-This->global_props);
1676 }
1677
1678 static HRESULT WINAPI WindowDispEx_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
1679 {
1680     HTMLWindow *This = DISPEX_THIS(iface);
1681     ScriptHost *script_host;
1682     DISPID id;
1683     DWORD i;
1684     HRESULT hres;
1685
1686     TRACE("(%p)->(%s %x %p)\n", This, debugstr_w(bstrName), grfdex, pid);
1687
1688     for(i=0; i < This->global_prop_cnt; i++) {
1689         /* FIXME: case sensitivity */
1690         if(!strcmpW(This->global_props[i].name, bstrName)) {
1691             *pid = MSHTML_DISPID_CUSTOM_MIN+i;
1692             return S_OK;
1693         }
1694     }
1695
1696     if(find_global_prop(This, bstrName, grfdex, &script_host, &id)) {
1697         global_prop_t *prop;
1698
1699         prop = alloc_global_prop(This, GLOBAL_SCRIPTVAR, bstrName);
1700         if(!prop)
1701             return E_OUTOFMEMORY;
1702
1703         prop->script_host = script_host;
1704         prop->id = id;
1705
1706         *pid = prop_to_dispid(This, prop);
1707         return S_OK;
1708     }
1709
1710     hres = IDispatchEx_GetDispID(DISPATCHEX(&This->dispex), bstrName, grfdex, pid);
1711     if(hres != DISP_E_UNKNOWNNAME)
1712         return hres;
1713
1714     if(This->doc) {
1715         global_prop_t *prop;
1716         IHTMLElement *elem;
1717
1718         hres = IHTMLDocument3_getElementById(HTMLDOC3(&This->doc->basedoc), bstrName, &elem);
1719         if(SUCCEEDED(hres) && elem) {
1720             IHTMLElement_Release(elem);
1721
1722             prop = alloc_global_prop(This, GLOBAL_ELEMENTVAR, bstrName);
1723             if(!prop)
1724                 return E_OUTOFMEMORY;
1725
1726             *pid = prop_to_dispid(This, prop);
1727             return S_OK;
1728         }
1729     }
1730
1731     return DISP_E_UNKNOWNNAME;
1732 }
1733
1734 static HRESULT WINAPI WindowDispEx_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
1735         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
1736 {
1737     HTMLWindow *This = DISPEX_THIS(iface);
1738
1739     TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This, id, lcid, wFlags, pdp, pvarRes, pei, pspCaller);
1740
1741     return IDispatchEx_InvokeEx(DISPATCHEX(&This->dispex), id, lcid, wFlags, pdp, pvarRes, pei, pspCaller);
1742 }
1743
1744 static HRESULT WINAPI WindowDispEx_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
1745 {
1746     HTMLWindow *This = DISPEX_THIS(iface);
1747
1748     TRACE("(%p)->(%s %x)\n", This, debugstr_w(bstrName), grfdex);
1749
1750     return IDispatchEx_DeleteMemberByName(DISPATCHEX(&This->dispex), bstrName, grfdex);
1751 }
1752
1753 static HRESULT WINAPI WindowDispEx_DeleteMemberByDispID(IDispatchEx *iface, DISPID id)
1754 {
1755     HTMLWindow *This = DISPEX_THIS(iface);
1756
1757     TRACE("(%p)->(%x)\n", This, id);
1758
1759     return IDispatchEx_DeleteMemberByDispID(DISPATCHEX(&This->dispex), id);
1760 }
1761
1762 static HRESULT WINAPI WindowDispEx_GetMemberProperties(IDispatchEx *iface, DISPID id, DWORD grfdexFetch, DWORD *pgrfdex)
1763 {
1764     HTMLWindow *This = DISPEX_THIS(iface);
1765
1766     TRACE("(%p)->(%x %x %p)\n", This, id, grfdexFetch, pgrfdex);
1767
1768     return IDispatchEx_GetMemberProperties(DISPATCHEX(&This->dispex), id, grfdexFetch, pgrfdex);
1769 }
1770
1771 static HRESULT WINAPI WindowDispEx_GetMemberName(IDispatchEx *iface, DISPID id, BSTR *pbstrName)
1772 {
1773     HTMLWindow *This = DISPEX_THIS(iface);
1774
1775     TRACE("(%p)->(%x %p)\n", This, id, pbstrName);
1776
1777     return IDispatchEx_GetMemberName(DISPATCHEX(&This->dispex), id, pbstrName);
1778 }
1779
1780 static HRESULT WINAPI WindowDispEx_GetNextDispID(IDispatchEx *iface, DWORD grfdex, DISPID id, DISPID *pid)
1781 {
1782     HTMLWindow *This = DISPEX_THIS(iface);
1783
1784     TRACE("(%p)->(%x %x %p)\n", This, grfdex, id, pid);
1785
1786     return IDispatchEx_GetNextDispID(DISPATCHEX(&This->dispex), grfdex, id, pid);
1787 }
1788
1789 static HRESULT WINAPI WindowDispEx_GetNameSpaceParent(IDispatchEx *iface, IUnknown **ppunk)
1790 {
1791     HTMLWindow *This = DISPEX_THIS(iface);
1792
1793     TRACE("(%p)->(%p)\n", This, ppunk);
1794
1795     *ppunk = NULL;
1796     return S_OK;
1797 }
1798
1799 #undef DISPEX_THIS
1800
1801 static const IDispatchExVtbl WindowDispExVtbl = {
1802     WindowDispEx_QueryInterface,
1803     WindowDispEx_AddRef,
1804     WindowDispEx_Release,
1805     WindowDispEx_GetTypeInfoCount,
1806     WindowDispEx_GetTypeInfo,
1807     WindowDispEx_GetIDsOfNames,
1808     WindowDispEx_Invoke,
1809     WindowDispEx_GetDispID,
1810     WindowDispEx_InvokeEx,
1811     WindowDispEx_DeleteMemberByName,
1812     WindowDispEx_DeleteMemberByDispID,
1813     WindowDispEx_GetMemberProperties,
1814     WindowDispEx_GetMemberName,
1815     WindowDispEx_GetNextDispID,
1816     WindowDispEx_GetNameSpaceParent
1817 };
1818
1819 static const tid_t HTMLWindow_iface_tids[] = {
1820     IHTMLWindow2_tid,
1821     IHTMLWindow3_tid,
1822     IHTMLWindow4_tid,
1823     0
1824 };
1825
1826 static const dispex_static_data_vtbl_t HTMLWindow_dispex_vtbl = {
1827     NULL,
1828     NULL,
1829     HTMLWindow_invoke
1830 };
1831
1832 static dispex_static_data_t HTMLWindow_dispex = {
1833     &HTMLWindow_dispex_vtbl,
1834     DispHTMLWindow2_tid,
1835     NULL,
1836     HTMLWindow_iface_tids
1837 };
1838
1839 HRESULT HTMLWindow_Create(HTMLDocumentObj *doc_obj, nsIDOMWindow *nswindow, HTMLWindow *parent, HTMLWindow **ret)
1840 {
1841     HTMLWindow *window;
1842
1843     window = heap_alloc_zero(sizeof(HTMLWindow));
1844     if(!window)
1845         return E_OUTOFMEMORY;
1846
1847     window->window_ref = heap_alloc(sizeof(windowref_t));
1848     if(!window->window_ref) {
1849         heap_free(window);
1850         return E_OUTOFMEMORY;
1851     }
1852
1853     window->lpHTMLWindow2Vtbl = &HTMLWindow2Vtbl;
1854     window->lpHTMLWindow3Vtbl = &HTMLWindow3Vtbl;
1855     window->lpHTMLWindow4Vtbl = &HTMLWindow4Vtbl;
1856     window->lpIDispatchExVtbl = &WindowDispExVtbl;
1857     window->ref = 1;
1858     window->doc_obj = doc_obj;
1859
1860     window->window_ref->window = window;
1861     window->window_ref->ref = 1;
1862
1863     init_dispex(&window->dispex, (IUnknown*)HTMLWINDOW2(window), &HTMLWindow_dispex);
1864
1865     if(nswindow) {
1866         nsIDOMWindow_AddRef(nswindow);
1867         window->nswindow = nswindow;
1868     }
1869
1870     window->scriptmode = parent ? parent->scriptmode : SCRIPTMODE_GECKO;
1871     window->readystate = READYSTATE_UNINITIALIZED;
1872     list_init(&window->script_hosts);
1873
1874     window->task_magic = get_task_target_magic();
1875     update_window_doc(window);
1876
1877     list_init(&window->children);
1878     list_add_head(&window_list, &window->entry);
1879
1880     if(parent) {
1881         IHTMLWindow2_AddRef(HTMLWINDOW2(window));
1882
1883         window->parent = parent;
1884         list_add_tail(&parent->children, &window->sibling_entry);
1885     }
1886
1887     *ret = window;
1888     return S_OK;
1889 }
1890
1891 void update_window_doc(HTMLWindow *window)
1892 {
1893     nsIDOMHTMLDocument *nshtmldoc;
1894     nsIDOMDocument *nsdoc;
1895     nsresult nsres;
1896
1897     nsres = nsIDOMWindow_GetDocument(window->nswindow, &nsdoc);
1898     if(NS_FAILED(nsres) || !nsdoc) {
1899         ERR("GetDocument failed: %08x\n", nsres);
1900         return;
1901     }
1902
1903     nsres = nsIDOMDocument_QueryInterface(nsdoc, &IID_nsIDOMHTMLDocument, (void**)&nshtmldoc);
1904     nsIDOMDocument_Release(nsdoc);
1905     if(NS_FAILED(nsres)) {
1906         ERR("Could not get nsIDOMHTMLDocument iface: %08x\n", nsres);
1907         return;
1908     }
1909
1910     if(!window->doc || window->doc->nsdoc != nshtmldoc) {
1911         HTMLDocumentNode *doc;
1912         HRESULT hres;
1913
1914         hres = create_doc_from_nsdoc(nshtmldoc, window->doc_obj, window, &doc);
1915         if(SUCCEEDED(hres)) {
1916             window_set_docnode(window, doc);
1917             htmldoc_release(&doc->basedoc);
1918         }else {
1919             ERR("create_doc_from_nsdoc failed: %08x\n", hres);
1920         }
1921     }
1922
1923     nsIDOMHTMLDocument_Release(nshtmldoc);
1924 }
1925
1926 HTMLWindow *nswindow_to_window(const nsIDOMWindow *nswindow)
1927 {
1928     HTMLWindow *iter;
1929
1930     LIST_FOR_EACH_ENTRY(iter, &window_list, HTMLWindow, entry) {
1931         if(iter->nswindow == nswindow)
1932             return iter;
1933     }
1934
1935     return NULL;
1936 }