urlmon: Implement HlinkSimpleNavigateToMoniker.
[wine] / dlls / mshtml / htmldoc3.c
1 /*
2  * Copyright 2005 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 "config.h"
20
21 #include <stdarg.h>
22 #include <stdio.h>
23
24 #define COBJMACROS
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winuser.h"
29 #include "ole2.h"
30
31 #include "wine/debug.h"
32
33 #include "mshtml_private.h"
34 #include "htmlevent.h"
35
36 WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
37
38 #define HTMLDOC3_THIS(iface) DEFINE_THIS(HTMLDocument, HTMLDocument3, iface)
39
40 static HRESULT WINAPI HTMLDocument3_QueryInterface(IHTMLDocument3 *iface,
41                                                   REFIID riid, void **ppv)
42 {
43     HTMLDocument *This = HTMLDOC3_THIS(iface);
44     return IHTMLDocument2_QueryInterface(HTMLDOC(This), riid, ppv);
45 }
46
47 static ULONG WINAPI HTMLDocument3_AddRef(IHTMLDocument3 *iface)
48 {
49     HTMLDocument *This = HTMLDOC3_THIS(iface);
50     return IHTMLDocument2_AddRef(HTMLDOC(This));
51 }
52
53 static ULONG WINAPI HTMLDocument3_Release(IHTMLDocument3 *iface)
54 {
55     HTMLDocument *This = HTMLDOC3_THIS(iface);
56     return IHTMLDocument2_Release(HTMLDOC(This));
57 }
58
59 static HRESULT WINAPI HTMLDocument3_GetTypeInfoCount(IHTMLDocument3 *iface, UINT *pctinfo)
60 {
61     HTMLDocument *This = HTMLDOC3_THIS(iface);
62     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(This), pctinfo);
63 }
64
65 static HRESULT WINAPI HTMLDocument3_GetTypeInfo(IHTMLDocument3 *iface, UINT iTInfo,
66                                                 LCID lcid, ITypeInfo **ppTInfo)
67 {
68     HTMLDocument *This = HTMLDOC3_THIS(iface);
69     return IDispatchEx_GetTypeInfo(DISPATCHEX(This), iTInfo, lcid, ppTInfo);
70 }
71
72 static HRESULT WINAPI HTMLDocument3_GetIDsOfNames(IHTMLDocument3 *iface, REFIID riid,
73                                                 LPOLESTR *rgszNames, UINT cNames,
74                                                 LCID lcid, DISPID *rgDispId)
75 {
76     HTMLDocument *This = HTMLDOC3_THIS(iface);
77     return IDispatchEx_GetIDsOfNames(DISPATCHEX(This), riid, rgszNames, cNames, lcid, rgDispId);
78 }
79
80 static HRESULT WINAPI HTMLDocument3_Invoke(IHTMLDocument3 *iface, DISPID dispIdMember,
81                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
82                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
83 {
84     HTMLDocument *This = HTMLDOC3_THIS(iface);
85     return IDispatchEx_Invoke(DISPATCHEX(This), dispIdMember, riid, lcid, wFlags, pDispParams,
86             pVarResult, pExcepInfo, puArgErr);
87 }
88
89 static HRESULT WINAPI HTMLDocument3_releaseCapture(IHTMLDocument3 *iface)
90 {
91     HTMLDocument *This = HTMLDOC3_THIS(iface);
92     FIXME("(%p)\n", This);
93     return E_NOTIMPL;
94 }
95
96 static HRESULT WINAPI HTMLDocument3_recalc(IHTMLDocument3 *iface, VARIANT_BOOL fForce)
97 {
98     HTMLDocument *This = HTMLDOC3_THIS(iface);
99     FIXME("(%p)->(%x)\n", This, fForce);
100     return E_NOTIMPL;
101 }
102
103 static HRESULT WINAPI HTMLDocument3_createTextNode(IHTMLDocument3 *iface, BSTR text,
104                                                    IHTMLDOMNode **newTextNode)
105 {
106     HTMLDocument *This = HTMLDOC3_THIS(iface);
107     nsIDOMText *nstext;
108     HTMLDOMNode *node;
109     nsAString text_str;
110     nsresult nsres;
111
112     TRACE("(%p)->(%s %p)\n", This, debugstr_w(text), newTextNode);
113
114     if(!This->doc_node->nsdoc) {
115         WARN("NULL nsdoc\n");
116         return E_UNEXPECTED;
117     }
118
119     nsAString_InitDepend(&text_str, text);
120     nsres = nsIDOMHTMLDocument_CreateTextNode(This->doc_node->nsdoc, &text_str, &nstext);
121     nsAString_Finish(&text_str);
122     if(NS_FAILED(nsres)) {
123         ERR("CreateTextNode failed: %08x\n", nsres);
124         return E_FAIL;
125     }
126
127     node = HTMLDOMTextNode_Create(This->doc_node, (nsIDOMNode*)nstext);
128     nsIDOMElement_Release(nstext);
129
130     *newTextNode = HTMLDOMNODE(node);
131     IHTMLDOMNode_AddRef(HTMLDOMNODE(node));
132     return S_OK;
133 }
134
135 static HRESULT WINAPI HTMLDocument3_get_documentElement(IHTMLDocument3 *iface, IHTMLElement **p)
136 {
137     HTMLDocument *This = HTMLDOC3_THIS(iface);
138     nsIDOMElement *nselem = NULL;
139     HTMLDOMNode *node;
140     nsresult nsres;
141
142     TRACE("(%p)->(%p)\n", This, p);
143
144     if(This->window->readystate == READYSTATE_UNINITIALIZED) {
145         *p = NULL;
146         return S_OK;
147     }
148
149     if(!This->doc_node->nsdoc) {
150         WARN("NULL nsdoc\n");
151         return E_UNEXPECTED;
152     }
153
154     nsres = nsIDOMHTMLDocument_GetDocumentElement(This->doc_node->nsdoc, &nselem);
155     if(NS_FAILED(nsres)) {
156         ERR("GetDocumentElement failed: %08x\n", nsres);
157         return E_FAIL;
158     }
159
160     if(nselem) {
161         node = get_node(This->doc_node, (nsIDOMNode *)nselem, TRUE);
162         nsIDOMElement_Release(nselem);
163         IHTMLDOMNode_QueryInterface(HTMLDOMNODE(node), &IID_IHTMLElement, (void**)p);
164     }else {
165         *p = NULL;
166     }
167
168     return S_OK;
169 }
170
171 static HRESULT WINAPI HTMLDocument3_uniqueID(IHTMLDocument3 *iface, BSTR *p)
172 {
173     HTMLDocument *This = HTMLDOC3_THIS(iface);
174     FIXME("(%p)->(%p)\n", This, p);
175     return E_NOTIMPL;
176 }
177
178 static HRESULT WINAPI HTMLDocument3_attachEvent(IHTMLDocument3 *iface, BSTR event,
179                                                 IDispatch* pDisp, VARIANT_BOOL *pfResult)
180 {
181     HTMLDocument *This = HTMLDOC3_THIS(iface);
182
183     TRACE("(%p)->(%s %p %p)\n", This, debugstr_w(event), pDisp, pfResult);
184
185     return attach_event(&This->doc_node->node.event_target, This->doc_node->node.nsnode, This, event, pDisp, pfResult);
186 }
187
188 static HRESULT WINAPI HTMLDocument3_detachEvent(IHTMLDocument3 *iface, BSTR event,
189                                                 IDispatch *pDisp)
190 {
191     HTMLDocument *This = HTMLDOC3_THIS(iface);
192     FIXME("(%p)->(%s %p)\n", This, debugstr_w(event), pDisp);
193     return E_NOTIMPL;
194 }
195
196 static HRESULT WINAPI HTMLDocument3_put_onrowsdelete(IHTMLDocument3 *iface, VARIANT v)
197 {
198     HTMLDocument *This = HTMLDOC3_THIS(iface);
199     FIXME("(%p)->()\n", This);
200     return E_NOTIMPL;
201 }
202
203 static HRESULT WINAPI HTMLDocument3_get_onrowsdelete(IHTMLDocument3 *iface, VARIANT *p)
204 {
205     HTMLDocument *This = HTMLDOC3_THIS(iface);
206     FIXME("(%p)->(%p)\n", This, p);
207     return E_NOTIMPL;
208 }
209
210 static HRESULT WINAPI HTMLDocument3_put_onrowsinserted(IHTMLDocument3 *iface, VARIANT v)
211 {
212     HTMLDocument *This = HTMLDOC3_THIS(iface);
213     FIXME("(%p)->()\n", This);
214     return E_NOTIMPL;
215 }
216
217 static HRESULT WINAPI HTMLDocument3_get_onrowsinserted(IHTMLDocument3 *iface, VARIANT *p)
218 {
219     HTMLDocument *This = HTMLDOC3_THIS(iface);
220     FIXME("(%p)->(%p)\n", This, p);
221     return E_NOTIMPL;
222 }
223
224 static HRESULT WINAPI HTMLDocument3_put_oncellchange(IHTMLDocument3 *iface, VARIANT v)
225 {
226     HTMLDocument *This = HTMLDOC3_THIS(iface);
227     FIXME("(%p)->()\n", This);
228     return E_NOTIMPL;
229 }
230
231 static HRESULT WINAPI HTMLDocument3_get_oncellchange(IHTMLDocument3 *iface, VARIANT *p)
232 {
233     HTMLDocument *This = HTMLDOC3_THIS(iface);
234     FIXME("(%p)->(%p)\n", This, p);
235     return E_NOTIMPL;
236 }
237
238 static HRESULT WINAPI HTMLDocument3_put_ondatasetchanged(IHTMLDocument3 *iface, VARIANT v)
239 {
240     HTMLDocument *This = HTMLDOC3_THIS(iface);
241     FIXME("(%p)->()\n", This);
242     return E_NOTIMPL;
243 }
244
245 static HRESULT WINAPI HTMLDocument3_get_ondatasetchanged(IHTMLDocument3 *iface, VARIANT *p)
246 {
247     HTMLDocument *This = HTMLDOC3_THIS(iface);
248     FIXME("(%p)->(%p)\n", This, p);
249     return E_NOTIMPL;
250 }
251
252 static HRESULT WINAPI HTMLDocument3_put_ondataavailable(IHTMLDocument3 *iface, VARIANT v)
253 {
254     HTMLDocument *This = HTMLDOC3_THIS(iface);
255     FIXME("(%p)->()\n", This);
256     return E_NOTIMPL;
257 }
258
259 static HRESULT WINAPI HTMLDocument3_get_ondataavailable(IHTMLDocument3 *iface, VARIANT *p)
260 {
261     HTMLDocument *This = HTMLDOC3_THIS(iface);
262     FIXME("(%p)->(%p)\n", This, p);
263     return E_NOTIMPL;
264 }
265
266 static HRESULT WINAPI HTMLDocument3_put_ondatasetcomplete(IHTMLDocument3 *iface, VARIANT v)
267 {
268     HTMLDocument *This = HTMLDOC3_THIS(iface);
269     FIXME("(%p)->()\n", This);
270     return E_NOTIMPL;
271 }
272
273 static HRESULT WINAPI HTMLDocument3_get_ondatasetcomplete(IHTMLDocument3 *iface, VARIANT *p)
274 {
275     HTMLDocument *This = HTMLDOC3_THIS(iface);
276     FIXME("(%p)->(%p)\n", This, p);
277     return E_NOTIMPL;
278 }
279
280 static HRESULT WINAPI HTMLDocument3_put_onpropertychange(IHTMLDocument3 *iface, VARIANT v)
281 {
282     HTMLDocument *This = HTMLDOC3_THIS(iface);
283     FIXME("(%p)->()\n", This);
284     return E_NOTIMPL;
285 }
286
287 static HRESULT WINAPI HTMLDocument3_get_onpropertychange(IHTMLDocument3 *iface, VARIANT *p)
288 {
289     HTMLDocument *This = HTMLDOC3_THIS(iface);
290     FIXME("(%p)->(%p)\n", This, p);
291     return E_NOTIMPL;
292 }
293
294 static HRESULT WINAPI HTMLDocument3_put_dir(IHTMLDocument3 *iface, BSTR v)
295 {
296     HTMLDocument *This = HTMLDOC3_THIS(iface);
297     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
298     return E_NOTIMPL;
299 }
300
301 static HRESULT WINAPI HTMLDocument3_get_dir(IHTMLDocument3 *iface, BSTR *p)
302 {
303     HTMLDocument *This = HTMLDOC3_THIS(iface);
304     FIXME("(%p)->(%p)\n", This, p);
305     return E_NOTIMPL;
306 }
307
308 static HRESULT WINAPI HTMLDocument3_put_oncontextmenu(IHTMLDocument3 *iface, VARIANT v)
309 {
310     HTMLDocument *This = HTMLDOC3_THIS(iface);
311
312     TRACE("(%p)->()\n", This);
313
314     return set_doc_event(This, EVENTID_CONTEXTMENU, &v);
315 }
316
317 static HRESULT WINAPI HTMLDocument3_get_oncontextmenu(IHTMLDocument3 *iface, VARIANT *p)
318 {
319     HTMLDocument *This = HTMLDOC3_THIS(iface);
320
321     TRACE("(%p)->(%p)\n", This, p);
322
323     return get_doc_event(This, EVENTID_CONTEXTMENU, p);
324 }
325
326 static HRESULT WINAPI HTMLDocument3_put_onstop(IHTMLDocument3 *iface, VARIANT v)
327 {
328     HTMLDocument *This = HTMLDOC3_THIS(iface);
329     FIXME("(%p)->()\n", This);
330     return E_NOTIMPL;
331 }
332
333 static HRESULT WINAPI HTMLDocument3_get_onstop(IHTMLDocument3 *iface, VARIANT *p)
334 {
335     HTMLDocument *This = HTMLDOC3_THIS(iface);
336     FIXME("(%p)->(%p)\n", This, p);
337     return E_NOTIMPL;
338 }
339
340 static HRESULT WINAPI HTMLDocument3_createDocumentFragment(IHTMLDocument3 *iface,
341                                                            IHTMLDocument2 **ppNewDoc)
342 {
343     HTMLDocument *This = HTMLDOC3_THIS(iface);
344     FIXME("(%p)->(%p)\n", This, ppNewDoc);
345     return E_NOTIMPL;
346 }
347
348 static HRESULT WINAPI HTMLDocument3_get_parentDocument(IHTMLDocument3 *iface,
349                                                        IHTMLDocument2 **p)
350 {
351     HTMLDocument *This = HTMLDOC3_THIS(iface);
352     FIXME("(%p)->(%p)\n", This, p);
353     return E_NOTIMPL;
354 }
355
356 static HRESULT WINAPI HTMLDocument3_put_enableDownload(IHTMLDocument3 *iface,
357                                                        VARIANT_BOOL v)
358 {
359     HTMLDocument *This = HTMLDOC3_THIS(iface);
360     FIXME("(%p)->(%x)\n", This, v);
361     return E_NOTIMPL;
362 }
363
364 static HRESULT WINAPI HTMLDocument3_get_enableDownload(IHTMLDocument3 *iface,
365                                                        VARIANT_BOOL *p)
366 {
367     HTMLDocument *This = HTMLDOC3_THIS(iface);
368     FIXME("(%p)->(%p)\n", This, p);
369     return E_NOTIMPL;
370 }
371
372 static HRESULT WINAPI HTMLDocument3_put_baseUrl(IHTMLDocument3 *iface, BSTR v)
373 {
374     HTMLDocument *This = HTMLDOC3_THIS(iface);
375     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
376     return E_NOTIMPL;
377 }
378
379 static HRESULT WINAPI HTMLDocument3_get_baseUrl(IHTMLDocument3 *iface, BSTR *p)
380 {
381     HTMLDocument *This = HTMLDOC3_THIS(iface);
382     FIXME("(%p)->(%p)\n", This, p);
383     return E_NOTIMPL;
384 }
385
386 static HRESULT WINAPI HTMLDocument3_get_childNodes(IHTMLDocument3 *iface, IDispatch **p)
387 {
388     HTMLDocument *This = HTMLDOC3_THIS(iface);
389     FIXME("(%p)->(%p)\n", This, p);
390     return E_NOTIMPL;
391 }
392
393 static HRESULT WINAPI HTMLDocument3_put_inheritStyleSheets(IHTMLDocument3 *iface,
394                                                            VARIANT_BOOL v)
395 {
396     HTMLDocument *This = HTMLDOC3_THIS(iface);
397     FIXME("(%p)->()\n", This);
398     return E_NOTIMPL;
399 }
400
401 static HRESULT WINAPI HTMLDocument3_get_inheritStyleSheets(IHTMLDocument3 *iface,
402                                                            VARIANT_BOOL *p)
403 {
404     HTMLDocument *This = HTMLDOC3_THIS(iface);
405     FIXME("(%p)->(%p)\n", This, p);
406     return E_NOTIMPL;
407 }
408
409 static HRESULT WINAPI HTMLDocument3_put_onbeforeeditfocus(IHTMLDocument3 *iface, VARIANT v)
410 {
411     HTMLDocument *This = HTMLDOC3_THIS(iface);
412     FIXME("(%p)->()\n", This);
413     return E_NOTIMPL;
414 }
415
416 static HRESULT WINAPI HTMLDocument3_get_onbeforeeditfocus(IHTMLDocument3 *iface, VARIANT *p)
417 {
418     HTMLDocument *This = HTMLDOC3_THIS(iface);
419     FIXME("(%p)->(%p)\n", This, p);
420     return E_NOTIMPL;
421 }
422
423 static HRESULT WINAPI HTMLDocument3_getElementsByName(IHTMLDocument3 *iface, BSTR v,
424                                                       IHTMLElementCollection **ppelColl)
425 {
426     HTMLDocument *This = HTMLDOC3_THIS(iface);
427     FIXME("(%p)->(%s %p)\n", This, debugstr_w(v), ppelColl);
428     return E_NOTIMPL;
429 }
430
431
432 static HRESULT WINAPI HTMLDocument3_getElementById(IHTMLDocument3 *iface, BSTR v,
433                                                    IHTMLElement **pel)
434 {
435     HTMLDocument *This = HTMLDOC3_THIS(iface);
436     nsIDOMElement *nselem;
437     HTMLDOMNode *node;
438     nsIDOMNode *nsnode, *nsnode_by_id, *nsnode_by_name;
439     nsIDOMNodeList *nsnode_list;
440     nsAString id_str;
441     nsresult nsres;
442
443     TRACE("(%p)->(%s %p)\n", This, debugstr_w(v), pel);
444
445     if(!This->doc_node->nsdoc) {
446         WARN("NULL nsdoc\n");
447         return E_UNEXPECTED;
448     }
449
450     nsAString_InitDepend(&id_str, v);
451     /* get element by id attribute */
452     nsres = nsIDOMHTMLDocument_GetElementById(This->doc_node->nsdoc, &id_str, &nselem);
453     if(FAILED(nsres)) {
454         ERR("GetElementById failed: %08x\n", nsres);
455         nsAString_Finish(&id_str);
456         return E_FAIL;
457     }
458     nsnode_by_id = (nsIDOMNode*)nselem;
459
460     /* get first element by name attribute */
461     nsres = nsIDOMHTMLDocument_GetElementsByName(This->doc_node->nsdoc, &id_str, &nsnode_list);
462     nsAString_Finish(&id_str);
463     if(FAILED(nsres)) {
464         ERR("getElementsByName failed: %08x\n", nsres);
465         if(nsnode_by_id)
466             nsIDOMNode_Release(nsnode_by_id);
467         return E_FAIL;
468     }
469     nsIDOMNodeList_Item(nsnode_list, 0, &nsnode_by_name);
470     nsIDOMNodeList_Release(nsnode_list);
471
472
473     if(nsnode_by_name && nsnode_by_id) {
474         nsIDOM3Node *node3;
475         PRUint16 pos;
476
477         nsres = nsIDOMNode_QueryInterface(nsnode_by_name, &IID_nsIDOM3Node, (void**)&node3);
478         if(NS_FAILED(nsres)) {
479             FIXME("failed to get nsIDOM3Node interface: 0x%08x\n", nsres);
480             nsIDOMNode_Release(nsnode_by_name);
481             nsIDOMNode_Release(nsnode_by_id);
482             return E_FAIL;
483         }
484
485         nsres = nsIDOM3Node_CompareDocumentPosition(node3, nsnode_by_id, &pos);
486         nsIDOM3Node_Release(node3);
487         if(NS_FAILED(nsres)) {
488             FIXME("nsIDOM3Node_CompareDocumentPosition failed: 0x%08x\n", nsres);
489             nsIDOMNode_Release(nsnode_by_name);
490             nsIDOMNode_Release(nsnode_by_id);
491             return E_FAIL;
492         }
493
494         TRACE("CompareDocumentPosition gave: 0x%x\n", pos);
495         if(pos & PRECEDING || pos & CONTAINS) {
496             nsnode = nsnode_by_id;
497             nsIDOMNode_Release(nsnode_by_name);
498         }else {
499             nsnode = nsnode_by_name;
500             nsIDOMNode_Release(nsnode_by_id);
501         }
502     }else
503         nsnode = nsnode_by_name ? nsnode_by_name : nsnode_by_id;
504
505     if(nsnode) {
506         node = get_node(This->doc_node, nsnode, TRUE);
507         nsIDOMNode_Release(nsnode);
508
509         IHTMLDOMNode_QueryInterface(HTMLDOMNODE(node), &IID_IHTMLElement, (void**)pel);
510     }else {
511         *pel = NULL;
512     }
513
514     return S_OK;
515 }
516
517
518 static HRESULT WINAPI HTMLDocument3_getElementsByTagName(IHTMLDocument3 *iface, BSTR v,
519                                                          IHTMLElementCollection **pelColl)
520 {
521     HTMLDocument *This = HTMLDOC3_THIS(iface);
522     nsIDOMNodeList *nslist;
523     nsAString id_str, ns_str;
524     nsresult nsres;
525     static const WCHAR str[] = {'*',0};
526
527     TRACE("(%p)->(%s %p)\n", This, debugstr_w(v), pelColl);
528
529     if(!This->doc_node->nsdoc) {
530         WARN("NULL nsdoc\n");
531         return E_UNEXPECTED;
532     }
533
534     nsAString_InitDepend(&id_str, v);
535     nsAString_InitDepend(&ns_str, str);
536     nsres = nsIDOMHTMLDocument_GetElementsByTagNameNS(This->doc_node->nsdoc, &ns_str, &id_str, &nslist);
537     nsAString_Finish(&id_str);
538     nsAString_Finish(&ns_str);
539     if(FAILED(nsres)) {
540         ERR("GetElementByName failed: %08x\n", nsres);
541         return E_FAIL;
542     }
543
544     *pelColl = (IHTMLElementCollection*)create_collection_from_nodelist(This->doc_node, (IUnknown*)HTMLDOC3(This), nslist);
545     nsIDOMNodeList_Release(nslist);
546
547     return S_OK;
548 }
549
550 #undef HTMLDOC3_THIS
551
552 static const IHTMLDocument3Vtbl HTMLDocument3Vtbl = {
553     HTMLDocument3_QueryInterface,
554     HTMLDocument3_AddRef,
555     HTMLDocument3_Release,
556     HTMLDocument3_GetTypeInfoCount,
557     HTMLDocument3_GetTypeInfo,
558     HTMLDocument3_GetIDsOfNames,
559     HTMLDocument3_Invoke,
560     HTMLDocument3_releaseCapture,
561     HTMLDocument3_recalc,
562     HTMLDocument3_createTextNode,
563     HTMLDocument3_get_documentElement,
564     HTMLDocument3_uniqueID,
565     HTMLDocument3_attachEvent,
566     HTMLDocument3_detachEvent,
567     HTMLDocument3_put_onrowsdelete,
568     HTMLDocument3_get_onrowsdelete,
569     HTMLDocument3_put_onrowsinserted,
570     HTMLDocument3_get_onrowsinserted,
571     HTMLDocument3_put_oncellchange,
572     HTMLDocument3_get_oncellchange,
573     HTMLDocument3_put_ondatasetchanged,
574     HTMLDocument3_get_ondatasetchanged,
575     HTMLDocument3_put_ondataavailable,
576     HTMLDocument3_get_ondataavailable,
577     HTMLDocument3_put_ondatasetcomplete,
578     HTMLDocument3_get_ondatasetcomplete,
579     HTMLDocument3_put_onpropertychange,
580     HTMLDocument3_get_onpropertychange,
581     HTMLDocument3_put_dir,
582     HTMLDocument3_get_dir,
583     HTMLDocument3_put_oncontextmenu,
584     HTMLDocument3_get_oncontextmenu,
585     HTMLDocument3_put_onstop,
586     HTMLDocument3_get_onstop,
587     HTMLDocument3_createDocumentFragment,
588     HTMLDocument3_get_parentDocument,
589     HTMLDocument3_put_enableDownload,
590     HTMLDocument3_get_enableDownload,
591     HTMLDocument3_put_baseUrl,
592     HTMLDocument3_get_baseUrl,
593     HTMLDocument3_get_childNodes,
594     HTMLDocument3_put_inheritStyleSheets,
595     HTMLDocument3_get_inheritStyleSheets,
596     HTMLDocument3_put_onbeforeeditfocus,
597     HTMLDocument3_get_onbeforeeditfocus,
598     HTMLDocument3_getElementsByName,
599     HTMLDocument3_getElementById,
600     HTMLDocument3_getElementsByTagName
601 };
602
603 #define HTMLDOC4_THIS(iface) DEFINE_THIS(HTMLDocument, HTMLDocument4, iface)
604
605 static HRESULT WINAPI HTMLDocument4_QueryInterface(IHTMLDocument4 *iface,
606                                                    REFIID riid, void **ppv)
607 {
608     HTMLDocument *This = HTMLDOC4_THIS(iface);
609     return IHTMLDocument2_QueryInterface(HTMLDOC(This), riid, ppv);
610 }
611
612 static ULONG WINAPI HTMLDocument4_AddRef(IHTMLDocument4 *iface)
613 {
614     HTMLDocument *This = HTMLDOC4_THIS(iface);
615     return IHTMLDocument2_AddRef(HTMLDOC(This));
616 }
617
618 static ULONG WINAPI HTMLDocument4_Release(IHTMLDocument4 *iface)
619 {
620     HTMLDocument *This = HTMLDOC4_THIS(iface);
621     return IHTMLDocument2_Release(HTMLDOC(This));
622 }
623
624 static HRESULT WINAPI HTMLDocument4_GetTypeInfoCount(IHTMLDocument4 *iface, UINT *pctinfo)
625 {
626     HTMLDocument *This = HTMLDOC4_THIS(iface);
627     return IDispatchEx_GetTypeInfoCount(DISPATCHEX(This), pctinfo);
628 }
629
630 static HRESULT WINAPI HTMLDocument4_GetTypeInfo(IHTMLDocument4 *iface, UINT iTInfo,
631                                                 LCID lcid, ITypeInfo **ppTInfo)
632 {
633     HTMLDocument *This = HTMLDOC4_THIS(iface);
634     return IDispatchEx_GetTypeInfo(DISPATCHEX(This), iTInfo, lcid, ppTInfo);
635 }
636
637 static HRESULT WINAPI HTMLDocument4_GetIDsOfNames(IHTMLDocument4 *iface, REFIID riid,
638                                                 LPOLESTR *rgszNames, UINT cNames,
639                                                 LCID lcid, DISPID *rgDispId)
640 {
641     HTMLDocument *This = HTMLDOC4_THIS(iface);
642     return IDispatchEx_GetIDsOfNames(DISPATCHEX(This), riid, rgszNames, cNames, lcid, rgDispId);
643 }
644
645 static HRESULT WINAPI HTMLDocument4_Invoke(IHTMLDocument4 *iface, DISPID dispIdMember,
646                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
647                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
648 {
649     HTMLDocument *This = HTMLDOC4_THIS(iface);
650     return IDispatchEx_Invoke(DISPATCHEX(This), dispIdMember, riid, lcid, wFlags, pDispParams,
651             pVarResult, pExcepInfo, puArgErr);
652 }
653
654 static HRESULT WINAPI HTMLDocument4_focus(IHTMLDocument4 *iface)
655 {
656     HTMLDocument *This = HTMLDOC4_THIS(iface);
657     nsIDOMNSHTMLElement *nselem;
658     nsIDOMHTMLElement *nsbody;
659     nsresult nsres;
660
661     TRACE("(%p)->()\n", This);
662
663     nsres = nsIDOMHTMLDocument_GetBody(This->doc_node->nsdoc, &nsbody);
664     if(NS_FAILED(nsres) || !nsbody) {
665         ERR("GetBody failed: %08x\n", nsres);
666         return E_FAIL;
667     }
668
669     nsres = nsIDOMHTMLElement_QueryInterface(nsbody, &IID_nsIDOMNSHTMLElement, (void**)&nselem);
670     nsIDOMHTMLElement_Release(nsbody);
671     if(NS_FAILED(nsres)) {
672         ERR("Could not get nsIDOMNSHTMLElement: %08x\n", nsres);
673         return E_FAIL;
674     }
675
676     nsres = nsIDOMNSHTMLElement_Focus(nselem);
677     nsIDOMNSHTMLElement_Release(nselem);
678     if(NS_FAILED(nsres)) {
679         ERR("Focus failed: %08x\n", nsres);
680         return E_FAIL;
681     }
682
683     return S_OK;
684 }
685
686 static HRESULT WINAPI HTMLDocument4_hasFocus(IHTMLDocument4 *iface, VARIANT_BOOL *pfFocus)
687 {
688     HTMLDocument *This = HTMLDOC4_THIS(iface);
689     FIXME("(%p)->(%p)\n", This, pfFocus);
690     return E_NOTIMPL;
691 }
692
693 static HRESULT WINAPI HTMLDocument4_put_onselectionchange(IHTMLDocument4 *iface, VARIANT v)
694 {
695     HTMLDocument *This = HTMLDOC4_THIS(iface);
696     FIXME("(%p)->(v)\n", This);
697     return E_NOTIMPL;
698 }
699
700 static HRESULT WINAPI HTMLDocument4_get_onselectionchange(IHTMLDocument4 *iface, VARIANT *p)
701 {
702     HTMLDocument *This = HTMLDOC4_THIS(iface);
703     FIXME("(%p)->(%p)\n", This, p);
704     return E_NOTIMPL;
705 }
706
707 static HRESULT WINAPI HTMLDocument4_get_namespace(IHTMLDocument4 *iface, IDispatch **p)
708 {
709     HTMLDocument *This = HTMLDOC4_THIS(iface);
710     FIXME("(%p)->(%p)\n", This, p);
711     return E_NOTIMPL;
712 }
713
714 static HRESULT WINAPI HTMLDocument4_createDocumentFromUrl(IHTMLDocument4 *iface, BSTR bstrUrl,
715         BSTR bstrOptions, IHTMLDocument2 **newDoc)
716 {
717     HTMLDocument *This = HTMLDOC4_THIS(iface);
718     FIXME("(%p)->(%s %s %p)\n", This, debugstr_w(bstrUrl), debugstr_w(bstrOptions), newDoc);
719     return E_NOTIMPL;
720 }
721
722 static HRESULT WINAPI HTMLDocument4_put_media(IHTMLDocument4 *iface, BSTR v)
723 {
724     HTMLDocument *This = HTMLDOC4_THIS(iface);
725     FIXME("(%p)->(%s)\n", This, debugstr_w(v));
726     return E_NOTIMPL;
727 }
728
729 static HRESULT WINAPI HTMLDocument4_get_media(IHTMLDocument4 *iface, BSTR *p)
730 {
731     HTMLDocument *This = HTMLDOC4_THIS(iface);
732     FIXME("(%p)->(%p)\n", This, p);
733     return E_NOTIMPL;
734 }
735
736 static HRESULT WINAPI HTMLDocument4_createEventObject(IHTMLDocument4 *iface,
737         VARIANT *pvarEventObject, IHTMLEventObj **ppEventObj)
738 {
739     HTMLDocument *This = HTMLDOC4_THIS(iface);
740     FIXME("(%p)->(%p %p)\n", This, pvarEventObject, ppEventObj);
741     return E_NOTIMPL;
742 }
743
744 static HRESULT WINAPI HTMLDocument4_fireEvent(IHTMLDocument4 *iface, BSTR bstrEventName,
745         VARIANT *pvarEventObject, VARIANT_BOOL *pfCanceled)
746 {
747     HTMLDocument *This = HTMLDOC4_THIS(iface);
748     FIXME("(%p)->(%s %p %p)\n", This, debugstr_w(bstrEventName), pvarEventObject, pfCanceled);
749     return E_NOTIMPL;
750 }
751
752 static HRESULT WINAPI HTMLDocument4_createRenderStyle(IHTMLDocument4 *iface, BSTR v,
753         IHTMLRenderStyle **ppIHTMLRenderStyle)
754 {
755     HTMLDocument *This = HTMLDOC4_THIS(iface);
756     FIXME("(%p)->(%s %p)\n", This, debugstr_w(v), ppIHTMLRenderStyle);
757     return E_NOTIMPL;
758 }
759
760 static HRESULT WINAPI HTMLDocument4_put_oncontrolselect(IHTMLDocument4 *iface, VARIANT v)
761 {
762     HTMLDocument *This = HTMLDOC4_THIS(iface);
763     FIXME("(%p)->(v)\n", This);
764     return E_NOTIMPL;
765 }
766
767 static HRESULT WINAPI HTMLDocument4_get_oncontrolselect(IHTMLDocument4 *iface, VARIANT *p)
768 {
769     HTMLDocument *This = HTMLDOC4_THIS(iface);
770     FIXME("(%p)->(%p)\n", This, p);
771     return E_NOTIMPL;
772 }
773
774 static HRESULT WINAPI HTMLDocument4_get_URLEncoded(IHTMLDocument4 *iface, BSTR *p)
775 {
776     HTMLDocument *This = HTMLDOC4_THIS(iface);
777     FIXME("(%p)->(%p)\n", This, p);
778     return E_NOTIMPL;
779 }
780
781 #undef HTMLDOC4_THIS
782
783 static const IHTMLDocument4Vtbl HTMLDocument4Vtbl = {
784     HTMLDocument4_QueryInterface,
785     HTMLDocument4_AddRef,
786     HTMLDocument4_Release,
787     HTMLDocument4_GetTypeInfoCount,
788     HTMLDocument4_GetTypeInfo,
789     HTMLDocument4_GetIDsOfNames,
790     HTMLDocument4_Invoke,
791     HTMLDocument4_focus,
792     HTMLDocument4_hasFocus,
793     HTMLDocument4_put_onselectionchange,
794     HTMLDocument4_get_onselectionchange,
795     HTMLDocument4_get_namespace,
796     HTMLDocument4_createDocumentFromUrl,
797     HTMLDocument4_put_media,
798     HTMLDocument4_get_media,
799     HTMLDocument4_createEventObject,
800     HTMLDocument4_fireEvent,
801     HTMLDocument4_createRenderStyle,
802     HTMLDocument4_put_oncontrolselect,
803     HTMLDocument4_get_oncontrolselect,
804     HTMLDocument4_get_URLEncoded
805 };
806
807 void HTMLDocument_HTMLDocument3_Init(HTMLDocument *This)
808 {
809     This->lpHTMLDocument3Vtbl = &HTMLDocument3Vtbl;
810     This->lpHTMLDocument4Vtbl = &HTMLDocument4Vtbl;
811 }