Implement undo and redo for the ie and gtk webkit backends. Extend the sample to...
[wxWidgets.git] / src / msw / webview_ie.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/webview_ie.cpp
3 // Purpose: wxMSW wxWebViewIE class implementation for web view component
4 // Author: Marianne Gagnon
5 // Id: $Id$
6 // Copyright: (c) 2010 Marianne Gagnon, Steven Lamerton
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
9
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
12
13 #if defined(__BORLANDC__)
14 #pragma hdrstop
15 #endif
16
17 #include "wx/msw/webview_ie.h"
18
19 #if wxUSE_WEBVIEW_IE
20
21 #include <olectl.h>
22 #include <oleidl.h>
23 #include <exdispid.h>
24 #include <exdisp.h>
25 #include <mshtml.h>
26
27 // Various definitions are missing from mingw
28 #ifdef __MINGW32__
29 typedef enum CommandStateChangeConstants {
30 CSC_UPDATECOMMANDS = (int) 0xFFFFFFFF,
31 CSC_NAVIGATEFORWARD = 0x1,
32 CSC_NAVIGATEBACK = 0x2
33 } CommandStateChangeConstants;
34
35 #define DISPID_COMMANDSTATECHANGE 105
36 #define DISPID_NAVIGATECOMPLETE2 252
37 #define DISPID_NAVIGATEERROR 271
38 #define DISPID_NEWWINDOW3 273
39 #define OLECMDID_OPTICAL_ZOOM 63
40 #define INET_E_ERROR_FIRST 0x800C0002L
41 #define INET_E_INVALID_URL 0x800C0002L
42 #define INET_E_NO_SESSION 0x800C0003L
43 #define INET_E_CANNOT_CONNECT 0x800C0004L
44 #define INET_E_RESOURCE_NOT_FOUND 0x800C0005L
45 #define INET_E_OBJECT_NOT_FOUND 0x800C0006L
46 #define INET_E_DATA_NOT_AVAILABLE 0x800C0007L
47 #define INET_E_DOWNLOAD_FAILURE 0x800C0008L
48 #define INET_E_AUTHENTICATION_REQUIRED 0x800C0009L
49 #define INET_E_NO_VALID_MEDIA 0x800C000AL
50 #define INET_E_CONNECTION_TIMEOUT 0x800C000BL
51 #define INET_E_INVALID_REQUEST 0x800C000CL
52 #define INET_E_UNKNOWN_PROTOCOL 0x800C000DL
53 #define INET_E_SECURITY_PROBLEM 0x800C000EL
54 #define INET_E_CANNOT_LOAD_DATA 0x800C000FL
55 #define INET_E_CANNOT_INSTANTIATE_OBJECT 0x800C0010L
56 #define INET_E_QUERYOPTION_UNKNOWN 0x800C0013L
57 #define INET_E_REDIRECT_FAILED 0x800C0014L
58 #define INET_E_REDIRECT_TO_DIR 0x800C0015L
59 #define INET_E_CANNOT_LOCK_REQUEST 0x800C0016L
60 #define INET_E_USE_EXTEND_BINDING 0x800C0017L
61 #define INET_E_TERMINATED_BIND 0x800C0018L
62 #define INET_E_INVALID_CERTIFICATE 0x800C0019L
63 #define INET_E_CODE_DOWNLOAD_DECLINED 0x800C0100L
64 #define INET_E_RESULT_DISPATCHED 0x800C0200L
65 #define INET_E_CANNOT_REPLACE_SFP_FILE 0x800C0300L
66 #define INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY 0x800C0500L
67 #define INET_E_CODE_INSTALL_SUPPRESSED 0x800C0400L
68
69 #define REFRESH_NORMAL 0
70 #define REFRESH_COMPLETELY 3
71 #endif
72
73 BEGIN_EVENT_TABLE(wxWebViewIE, wxControl)
74 EVT_ACTIVEX(wxID_ANY, wxWebViewIE::onActiveXEvent)
75 EVT_ERASE_BACKGROUND(wxWebViewIE::onEraseBg)
76 END_EVENT_TABLE()
77
78 bool wxWebViewIE::Create(wxWindow* parent,
79 wxWindowID id,
80 const wxString& url,
81 const wxPoint& pos,
82 const wxSize& size,
83 long style,
84 const wxString& name)
85 {
86 if (!wxControl::Create(parent, id, pos, size, style,
87 wxDefaultValidator, name))
88 {
89 return false;
90 }
91
92 m_webBrowser = NULL;
93 m_canNavigateBack = false;
94 m_canNavigateForward = false;
95 m_isBusy = false;
96 m_historyLoadingFromList = false;
97 m_historyEnabled = true;
98 m_historyPosition = -1;
99
100 if (::CoCreateInstance(CLSID_WebBrowser, NULL,
101 CLSCTX_INPROC_SERVER, // CLSCTX_INPROC,
102 IID_IWebBrowser2 , (void**)&m_webBrowser) != 0)
103 {
104 wxLogError("Failed to initialize IE, CoCreateInstance returned an error");
105 return false;
106 }
107
108 m_ie.SetDispatchPtr(m_webBrowser); // wxAutomationObject will release itself
109
110 m_webBrowser->put_RegisterAsBrowser(VARIANT_TRUE);
111 m_webBrowser->put_RegisterAsDropTarget(VARIANT_TRUE);
112 //m_webBrowser->put_Silent(VARIANT_FALSE);
113
114 m_container = new wxActiveXContainer(this, IID_IWebBrowser2, m_webBrowser);
115
116 SetBackgroundStyle(wxBG_STYLE_PAINT);
117 SetDoubleBuffered(true);
118 LoadUrl(url);
119 return true;
120 }
121
122
123 void wxWebViewIE::LoadUrl(const wxString& url)
124 {
125 wxVariant out = m_ie.CallMethod("Navigate", (BSTR) url.wc_str(),
126 NULL, NULL, NULL, NULL);
127
128 // FIXME: why is out value null??
129 //(HRESULT)(out.GetLong()) == S_OK;
130 }
131
132 void wxWebViewIE::SetPage(const wxString& html, const wxString& baseUrl)
133 {
134 LoadUrl("about:blank");
135
136 // Let the wx events generated for navigation events be processed, so
137 // that the underlying IE component completes its Document object.
138 // FIXME: calling wxYield is not elegant nor very reliable probably
139 wxYield();
140
141 wxVariant documentVariant = m_ie.GetProperty("Document");
142 void* documentPtr = documentVariant.GetVoidPtr();
143
144 wxASSERT (documentPtr != NULL);
145
146 // TODO: consider the "baseUrl" parameter if possible
147 // TODO: consider encoding
148 BSTR bstr = SysAllocString(html.wc_str());
149
150 // Creates a new one-dimensional array
151 SAFEARRAY *psaStrings = SafeArrayCreateVector(VT_VARIANT, 0, 1);
152 if (psaStrings != NULL)
153 {
154 VARIANT *param;
155 HRESULT hr = SafeArrayAccessData(psaStrings, (LPVOID*)&param);
156 param->vt = VT_BSTR;
157 param->bstrVal = bstr;
158
159 hr = SafeArrayUnaccessData(psaStrings);
160
161 IHTMLDocument2* document = (IHTMLDocument2*)documentPtr;
162 document->write(psaStrings);
163
164 // SafeArrayDestroy calls SysFreeString for each BSTR
165 SafeArrayDestroy(psaStrings);
166 }
167 else
168 {
169 wxLogError("wxWebViewIE::SetPage() : psaStrings is NULL");
170 }
171
172 }
173
174 wxString wxWebViewIE::GetPageSource()
175 {
176 wxVariant documentVariant = m_ie.GetProperty("Document");
177 void* documentPtr = documentVariant.GetVoidPtr();
178
179 if (documentPtr == NULL)
180 {
181 return wxEmptyString;
182 }
183
184 IHTMLDocument2* document = (IHTMLDocument2*)documentPtr;
185
186 IHTMLElement *bodyTag = NULL;
187 IHTMLElement *htmlTag = NULL;
188 document->get_body(&bodyTag);
189 wxASSERT(bodyTag != NULL);
190
191 document->Release();
192 bodyTag->get_parentElement(&htmlTag);
193 wxASSERT(htmlTag != NULL);
194
195 BSTR bstr;
196 htmlTag->get_outerHTML(&bstr);
197
198 bodyTag->Release();
199 htmlTag->Release();
200
201 //wxMessageBox(wxString(bstr));
202
203 // TODO: check encoding
204 return wxString(bstr);
205 }
206
207 // FIXME? retrieve OLECMDID_GETZOOMRANGE instead of hardcoding range 0-4
208 wxWebViewZoom wxWebViewIE::GetZoom()
209 {
210 const int zoom = GetIETextZoom();
211
212 switch (zoom)
213 {
214 case 0:
215 return wxWEB_VIEW_ZOOM_TINY;
216 break;
217 case 1:
218 return wxWEB_VIEW_ZOOM_SMALL;
219 break;
220 case 2:
221 return wxWEB_VIEW_ZOOM_MEDIUM;
222 break;
223 case 3:
224 return wxWEB_VIEW_ZOOM_LARGE;
225 break;
226 case 4:
227 return wxWEB_VIEW_ZOOM_LARGEST;
228 break;
229 default:
230 wxASSERT(false);
231 return wxWEB_VIEW_ZOOM_MEDIUM;
232 }
233 }
234 void wxWebViewIE::SetZoom(wxWebViewZoom zoom)
235 {
236 // I know I could cast from enum to int since wxWebViewZoom happens to
237 // match with IE's zoom levels, but I don't like doing that, what if enum
238 // values change...
239 switch (zoom)
240 {
241 case wxWEB_VIEW_ZOOM_TINY:
242 SetIETextZoom(0);
243 break;
244 case wxWEB_VIEW_ZOOM_SMALL:
245 SetIETextZoom(1);
246 break;
247 case wxWEB_VIEW_ZOOM_MEDIUM:
248 SetIETextZoom(2);
249 break;
250 case wxWEB_VIEW_ZOOM_LARGE:
251 SetIETextZoom(3);
252 break;
253 case wxWEB_VIEW_ZOOM_LARGEST:
254 SetIETextZoom(4);
255 break;
256 default:
257 wxASSERT(false);
258 }
259 }
260
261 void wxWebViewIE::SetIETextZoom(int level)
262 {
263 VARIANT zoomVariant;
264 VariantInit (&zoomVariant);
265 V_VT(&zoomVariant) = VT_I4;
266 V_I4(&zoomVariant) = level;
267
268 HRESULT result = m_webBrowser->ExecWB(OLECMDID_ZOOM,
269 OLECMDEXECOPT_DONTPROMPTUSER,
270 &zoomVariant, NULL);
271 wxASSERT (result == S_OK);
272
273 VariantClear (&zoomVariant);
274 }
275
276 int wxWebViewIE::GetIETextZoom()
277 {
278 VARIANT zoomVariant;
279 VariantInit (&zoomVariant);
280 V_VT(&zoomVariant) = VT_I4;
281 V_I4(&zoomVariant) = 4;
282
283 HRESULT result = m_webBrowser->ExecWB(OLECMDID_ZOOM,
284 OLECMDEXECOPT_DONTPROMPTUSER,
285 NULL, &zoomVariant);
286 wxASSERT (result == S_OK);
287
288 int zoom = V_I4(&zoomVariant);
289 // wxMessageBox(wxString::Format("Zoom : %i", zoom));
290 VariantClear (&zoomVariant);
291
292 return zoom;
293 }
294
295 void wxWebViewIE::SetIEOpticalZoom(float zoom)
296 {
297 // TODO: add support for optical zoom (IE7+ only)
298
299 // TODO: get range from OLECMDID_OPTICAL_GETZOOMRANGE instead of hardcoding?
300 wxASSERT(zoom >= 10.0f);
301 wxASSERT(zoom <= 1000.0f);
302
303 VARIANT zoomVariant;
304 VariantInit (&zoomVariant);
305 V_VT(&zoomVariant) = VT_I4;
306 V_I4(&zoomVariant) = (zoom * 100.0f);
307
308 HRESULT result = m_webBrowser->ExecWB((OLECMDID)OLECMDID_OPTICAL_ZOOM,
309 OLECMDEXECOPT_DODEFAULT,
310 &zoomVariant,
311 NULL);
312 wxASSERT (result == S_OK);
313 }
314
315 float wxWebViewIE::GetIEOpticalZoom()
316 {
317 // TODO: add support for optical zoom (IE7+ only)
318
319 VARIANT zoomVariant;
320 VariantInit (&zoomVariant);
321 V_VT(&zoomVariant) = VT_I4;
322 V_I4(&zoomVariant) = -1;
323
324 HRESULT result = m_webBrowser->ExecWB((OLECMDID)OLECMDID_OPTICAL_ZOOM,
325 OLECMDEXECOPT_DODEFAULT, NULL,
326 &zoomVariant);
327 wxASSERT (result == S_OK);
328
329 const int zoom = V_I4(&zoomVariant);
330 VariantClear (&zoomVariant);
331
332 return zoom / 100.0f;
333 }
334
335 void wxWebViewIE::SetZoomType(wxWebViewZoomType)
336 {
337 // TODO: add support for optical zoom (IE7+ only)
338 wxASSERT(false);
339 }
340
341 wxWebViewZoomType wxWebViewIE::GetZoomType() const
342 {
343 // TODO: add support for optical zoom (IE7+ only)
344 return wxWEB_VIEW_ZOOM_TYPE_TEXT;
345 }
346
347 bool wxWebViewIE::CanSetZoomType(wxWebViewZoomType) const
348 {
349 // both are supported
350 // TODO: IE6 only supports text zoom, check if it's IE6 first
351 return true;
352 }
353
354 void wxWebViewIE::Print()
355 {
356 m_webBrowser->ExecWB(OLECMDID_PRINTPREVIEW,
357 OLECMDEXECOPT_DODEFAULT, NULL, NULL);
358 }
359
360 bool wxWebViewIE::CanGoBack()
361 {
362 if(m_historyEnabled)
363 return m_historyPosition > 0;
364 else
365 return false;
366 }
367
368 bool wxWebViewIE::CanGoForward()
369 {
370 if(m_historyEnabled)
371 return m_historyPosition != m_historyList.size() - 1;
372 else
373 return false;
374 }
375
376 void wxWebViewIE::LoadHistoryItem(wxSharedPtr<wxWebHistoryItem> item)
377 {
378 int pos = -1;
379 for(unsigned int i = 0; i < m_historyList.size(); i++)
380 {
381 //We compare the actual pointers to find the correct item
382 if(m_historyList[i].get() == item.get())
383 pos = i;
384 }
385 wxASSERT_MSG(pos != m_historyList.size(), "invalid history item");
386 m_historyLoadingFromList = true;
387 LoadUrl(item->GetUrl());
388 m_historyPosition = pos;
389 }
390
391 wxVector<wxSharedPtr<wxWebHistoryItem> > wxWebViewIE::GetBackwardHistory()
392 {
393 wxVector<wxSharedPtr<wxWebHistoryItem> > backhist;
394 //As we don't have std::copy or an iterator constructor in the wxwidgets
395 //native vector we construct it by hand
396 for(int i = 0; i < m_historyPosition; i++)
397 {
398 backhist.push_back(m_historyList[i]);
399 }
400 return backhist;
401 }
402
403 wxVector<wxSharedPtr<wxWebHistoryItem> > wxWebViewIE::GetForwardHistory()
404 {
405 wxVector<wxSharedPtr<wxWebHistoryItem> > forwardhist;
406 //As we don't have std::copy or an iterator constructor in the wxwidgets
407 //native vector we construct it by hand
408 for(int i = m_historyPosition + 1; i < m_historyList.size(); i++)
409 {
410 forwardhist.push_back(m_historyList[i]);
411 }
412 return forwardhist;
413 }
414
415 void wxWebViewIE::GoBack()
416 {
417 LoadHistoryItem(m_historyList[m_historyPosition - 1]);
418 }
419
420 void wxWebViewIE::GoForward()
421 {
422 LoadHistoryItem(m_historyList[m_historyPosition + 1]);
423 }
424
425 void wxWebViewIE::Stop()
426 {
427 wxVariant out = m_ie.CallMethod("Stop");
428
429 // FIXME: why is out value null??
430 //return (HRESULT)(out.GetLong()) == S_OK;
431 }
432
433 void wxWebViewIE::ClearHistory()
434 {
435 m_historyList.clear();
436 m_historyPosition = -1;
437 }
438
439 void wxWebViewIE::EnableHistory(bool enable)
440 {
441 m_historyEnabled = enable;
442 m_historyList.clear();
443 m_historyPosition = -1;
444 }
445
446 void wxWebViewIE::Reload(wxWebViewReloadFlags flags)
447 {
448 VARIANTARG level;
449 VariantInit(&level);
450 V_VT(&level) = VT_I2;
451
452 switch(flags)
453 {
454 case wxWEB_VIEW_RELOAD_DEFAULT:
455 V_I2(&level) = REFRESH_NORMAL;
456 break;
457 case wxWEB_VIEW_RELOAD_NO_CACHE:
458 V_I2(&level) = REFRESH_COMPLETELY;
459 break;
460 default:
461 wxFAIL_MSG("Unexpected reload type");
462 }
463
464 m_webBrowser->Refresh2(&level);
465 }
466
467 bool wxWebViewIE::IsOfflineMode()
468 {
469 wxVariant out = m_ie.GetProperty("Offline");
470
471 wxASSERT(out.GetType() == "bool");
472
473 return out.GetBool();
474 }
475
476 void wxWebViewIE::SetOfflineMode(bool offline)
477 {
478 // FIXME: the wxWidgets docs do not really document what the return
479 // parameter of PutProperty is
480 const bool success = m_ie.PutProperty("Offline", (offline ?
481 VARIANT_TRUE :
482 VARIANT_FALSE));
483 wxASSERT(success);
484 }
485
486 bool wxWebViewIE::IsBusy()
487 {
488 if (m_isBusy) return true;
489
490 wxVariant out = m_ie.GetProperty("Busy");
491
492 wxASSERT(out.GetType() == "bool");
493
494 return out.GetBool();
495 }
496
497 wxString wxWebViewIE::GetCurrentURL()
498 {
499 wxVariant out = m_ie.GetProperty("LocationURL");
500
501 wxASSERT(out.GetType() == "string");
502 return out.GetString();
503 }
504
505 wxString wxWebViewIE::GetCurrentTitle()
506 {
507 wxVariant out = m_ie.GetProperty("LocationName");
508
509 wxASSERT(out.GetType() == "string");
510 return out.GetString();
511 }
512
513 bool wxWebViewIE::CanCut()
514 {
515 return CanExecCommand("Cut");
516 }
517
518 bool wxWebViewIE::CanCopy()
519 {
520 return CanExecCommand("Copy");
521 }
522 bool wxWebViewIE::CanPaste()
523 {
524 return CanExecCommand("Paste");
525 }
526
527 void wxWebViewIE::Cut()
528 {
529 ExecCommand("Cut");
530 }
531
532 void wxWebViewIE::Copy()
533 {
534 ExecCommand("Copy");
535 }
536
537 void wxWebViewIE::Paste()
538 {
539 ExecCommand("Paste");
540 }
541
542 bool wxWebViewIE::CanUndo()
543 {
544 return CanExecCommand("Undo");
545 }
546 bool wxWebViewIE::CanRedo()
547 {
548 return CanExecCommand("Redo");
549 }
550
551 void wxWebViewIE::Undo()
552 {
553 ExecCommand("Undo");
554 }
555
556 void wxWebViewIE::Redo()
557 {
558 ExecCommand("Redo");
559 }
560
561 bool wxWebViewIE::CanExecCommand(wxString command)
562 {
563 wxVariant documentVariant = m_ie.GetProperty("Document");
564 void* documentPtr = documentVariant.GetVoidPtr();
565 IHTMLDocument2* document = (IHTMLDocument2*)documentPtr;
566
567 wxASSERT(documentPtr && document);
568
569 VARIANT_BOOL enabled;
570 document->queryCommandEnabled(SysAllocString(command.wc_str()), &enabled);
571
572 return (enabled == VARIANT_TRUE);
573 }
574
575 void wxWebViewIE::ExecCommand(wxString command)
576 {
577 wxVariant documentVariant = m_ie.GetProperty("Document");
578 void* documentPtr = documentVariant.GetVoidPtr();
579 IHTMLDocument2* document = (IHTMLDocument2*)documentPtr;
580
581 wxASSERT(documentPtr && document);
582
583 document->execCommand(SysAllocString(command.wc_str()), VARIANT_FALSE, VARIANT(), NULL);
584 }
585
586 void wxWebViewIE::onActiveXEvent(wxActiveXEvent& evt)
587 {
588 if (m_webBrowser == NULL) return;
589
590 switch (evt.GetDispatchId())
591 {
592 case DISPID_BEFORENAVIGATE2:
593 {
594 m_isBusy = true;
595
596 wxString url = evt[1].GetString();
597 wxString target = evt[3].GetString();
598
599 wxWebNavigationEvent event(wxEVT_COMMAND_WEB_VIEW_NAVIGATING,
600 GetId(), url, target, true);
601 event.SetEventObject(this);
602 HandleWindowEvent(event);
603
604 if (event.IsVetoed())
605 {
606 wxActiveXEventNativeMSW* nativeParams =
607 evt.GetNativeParameters();
608 *V_BOOLREF(&nativeParams->pDispParams->rgvarg[0]) = VARIANT_TRUE;
609 }
610
611 // at this point, either the navigation event has been cancelled
612 // and we're not busy, either it was accepted and IWebBrowser2's
613 // Busy property will be true; so we don't need our override
614 // flag anymore.
615 m_isBusy = false;
616
617 break;
618 }
619
620 case DISPID_NAVIGATECOMPLETE2:
621 {
622 wxString url = evt[1].GetString();
623 // TODO: set target parameter if possible
624 wxString target = wxEmptyString;
625 wxWebNavigationEvent event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED,
626 GetId(), url, target, false);
627 event.SetEventObject(this);
628 HandleWindowEvent(event);
629 break;
630 }
631
632 case DISPID_PROGRESSCHANGE:
633 {
634 // download progress
635 break;
636 }
637
638 case DISPID_DOCUMENTCOMPLETE:
639 {
640 //Only send a complete even if we are actually finished, this brings
641 //the event in to line with webkit
642 READYSTATE rs;
643 m_webBrowser->get_ReadyState( &rs );
644 if(rs != READYSTATE_COMPLETE)
645 break;
646
647 wxString url = evt[1].GetString();
648 //As we are complete we also add to the history list
649 if(m_historyEnabled && !m_historyLoadingFromList)
650 {
651 //If we are not at the end of the list, then erase everything
652 //between us and the end before adding the new page
653 if(m_historyPosition != m_historyList.size() - 1)
654 {
655 m_historyList.erase(m_historyList.begin() + m_historyPosition + 1,
656 m_historyList.end());
657 }
658 wxSharedPtr<wxWebHistoryItem> item(new wxWebHistoryItem(url, GetCurrentTitle()));
659 m_historyList.push_back(item);
660 m_historyPosition++;
661 }
662 //Reset as we are done now
663 m_historyLoadingFromList = false;
664 // TODO: set target parameter if possible
665 wxString target = wxEmptyString;
666 wxWebNavigationEvent event(wxEVT_COMMAND_WEB_VIEW_LOADED, GetId(),
667 url, target, false);
668 event.SetEventObject(this);
669 HandleWindowEvent(event);
670 break;
671 }
672
673 case DISPID_STATUSTEXTCHANGE:
674 {
675 break;
676 }
677
678 case DISPID_TITLECHANGE:
679 {
680 break;
681 }
682
683 case DISPID_NAVIGATEERROR:
684 {
685 wxWebNavigationError errorType = wxWEB_NAV_ERR_OTHER;
686 wxString errorCode = "?";
687 switch (evt[3].GetLong())
688 {
689 case INET_E_INVALID_URL: // (0x800C0002L or -2146697214)
690 errorCode = "INET_E_INVALID_URL";
691 errorType = wxWEB_NAV_ERR_REQUEST;
692 break;
693 case INET_E_NO_SESSION: // (0x800C0003L or -2146697213)
694 errorCode = "INET_E_NO_SESSION";
695 errorType = wxWEB_NAV_ERR_CONNECTION;
696 break;
697 case INET_E_CANNOT_CONNECT: // (0x800C0004L or -2146697212)
698 errorCode = "INET_E_CANNOT_CONNECT";
699 errorType = wxWEB_NAV_ERR_CONNECTION;
700 break;
701 case INET_E_RESOURCE_NOT_FOUND: // (0x800C0005L or -2146697211)
702 errorCode = "INET_E_RESOURCE_NOT_FOUND";
703 errorType = wxWEB_NAV_ERR_NOT_FOUND;
704 break;
705 case INET_E_OBJECT_NOT_FOUND: // (0x800C0006L or -2146697210)
706 errorCode = "INET_E_OBJECT_NOT_FOUND";
707 errorType = wxWEB_NAV_ERR_NOT_FOUND;
708 break;
709 case INET_E_DATA_NOT_AVAILABLE: // (0x800C0007L or -2146697209)
710 errorCode = "INET_E_DATA_NOT_AVAILABLE";
711 errorType = wxWEB_NAV_ERR_NOT_FOUND;
712 break;
713 case INET_E_DOWNLOAD_FAILURE: // (0x800C0008L or -2146697208)
714 errorCode = "INET_E_DOWNLOAD_FAILURE";
715 errorType = wxWEB_NAV_ERR_CONNECTION;
716 break;
717 case INET_E_AUTHENTICATION_REQUIRED: // (0x800C0009L or -2146697207)
718 errorCode = "INET_E_AUTHENTICATION_REQUIRED";
719 errorType = wxWEB_NAV_ERR_AUTH;
720 break;
721 case INET_E_NO_VALID_MEDIA: // (0x800C000AL or -2146697206)
722 errorCode = "INET_E_NO_VALID_MEDIA";
723 errorType = wxWEB_NAV_ERR_REQUEST;
724 break;
725 case INET_E_CONNECTION_TIMEOUT: // (0x800C000BL or -2146697205)
726 errorCode = "INET_E_CONNECTION_TIMEOUT";
727 errorType = wxWEB_NAV_ERR_CONNECTION;
728 break;
729 case INET_E_INVALID_REQUEST: // (0x800C000CL or -2146697204)
730 errorCode = "INET_E_INVALID_REQUEST";
731 errorType = wxWEB_NAV_ERR_REQUEST;
732 break;
733 case INET_E_UNKNOWN_PROTOCOL: // (0x800C000DL or -2146697203)
734 errorCode = "INET_E_UNKNOWN_PROTOCOL";
735 errorType = wxWEB_NAV_ERR_REQUEST;
736 break;
737 case INET_E_SECURITY_PROBLEM: // (0x800C000EL or -2146697202)
738 errorCode = "INET_E_SECURITY_PROBLEM";
739 errorType = wxWEB_NAV_ERR_SECURITY;
740 break;
741 case INET_E_CANNOT_LOAD_DATA: // (0x800C000FL or -2146697201)
742 errorCode = "INET_E_CANNOT_LOAD_DATA";
743 errorType = wxWEB_NAV_ERR_OTHER;
744 break;
745 case INET_E_CANNOT_INSTANTIATE_OBJECT:
746 // CoCreateInstance will return an error code if this happens,
747 // we'll handle this above.
748 return;
749 break;
750 case INET_E_REDIRECT_FAILED: // (0x800C0014L or -2146697196)
751 errorCode = "INET_E_REDIRECT_FAILED";
752 errorType = wxWEB_NAV_ERR_OTHER;
753 break;
754 case INET_E_REDIRECT_TO_DIR: // (0x800C0015L or -2146697195)
755 errorCode = "INET_E_REDIRECT_TO_DIR";
756 errorType = wxWEB_NAV_ERR_REQUEST;
757 break;
758 case INET_E_CANNOT_LOCK_REQUEST: // (0x800C0016L or -2146697194)
759 errorCode = "INET_E_CANNOT_LOCK_REQUEST";
760 errorType = wxWEB_NAV_ERR_OTHER;
761 break;
762 case INET_E_USE_EXTEND_BINDING: // (0x800C0017L or -2146697193)
763 errorCode = "INET_E_USE_EXTEND_BINDING";
764 errorType = wxWEB_NAV_ERR_OTHER;
765 break;
766 case INET_E_TERMINATED_BIND: // (0x800C0018L or -2146697192)
767 errorCode = "INET_E_TERMINATED_BIND";
768 errorType = wxWEB_NAV_ERR_OTHER;
769 break;
770 case INET_E_INVALID_CERTIFICATE: // (0x800C0019L or -2146697191)
771 errorCode = "INET_E_INVALID_CERTIFICATE";
772 errorType = wxWEB_NAV_ERR_CERTIFICATE;
773 break;
774 case INET_E_CODE_DOWNLOAD_DECLINED: // (0x800C0100L or -2146696960)
775 errorCode = "INET_E_CODE_DOWNLOAD_DECLINED";
776 errorType = wxWEB_NAV_ERR_USER_CANCELLED;
777 break;
778 case INET_E_RESULT_DISPATCHED: // (0x800C0200L or -2146696704)
779 // cancel request cancelled...
780 errorCode = "INET_E_RESULT_DISPATCHED";
781 errorType = wxWEB_NAV_ERR_OTHER;
782 break;
783 case INET_E_CANNOT_REPLACE_SFP_FILE: // (0x800C0300L or -2146696448)
784 errorCode = "INET_E_CANNOT_REPLACE_SFP_FILE";
785 errorType = wxWEB_NAV_ERR_SECURITY;
786 break;
787 case INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY:
788 errorCode = "INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY";
789 errorType = wxWEB_NAV_ERR_SECURITY;
790 break;
791 case INET_E_CODE_INSTALL_SUPPRESSED:
792 errorCode = "INET_E_CODE_INSTALL_SUPPRESSED";
793 errorType = wxWEB_NAV_ERR_SECURITY;
794 break;
795 }
796
797 wxString url = evt[1].GetString();
798 wxString target = evt[2].GetString();
799 wxWebNavigationEvent event(wxEVT_COMMAND_WEB_VIEW_ERROR, GetId(),
800 url, target, false);
801 event.SetEventObject(this);
802 event.SetInt(errorType);
803 event.SetString(errorCode);
804 HandleWindowEvent(event);
805 break;
806 }
807
808 case DISPID_COMMANDSTATECHANGE:
809 {
810 long commandId = evt[0].GetLong();
811 bool enable = evt[1].GetBool();
812 if (commandId == CSC_NAVIGATEBACK)
813 {
814 m_canNavigateBack = enable;
815 }
816 else if (commandId == CSC_NAVIGATEFORWARD)
817 {
818 m_canNavigateForward = enable;
819 }
820 break;
821 }
822 case DISPID_NEWWINDOW3:
823 {
824 wxString url = evt[4].GetString();
825
826 wxWebNavigationEvent event(wxEVT_COMMAND_WEB_VIEW_NEWWINDOW,
827 GetId(), url, wxEmptyString, true);
828 event.SetEventObject(this);
829 HandleWindowEvent(event);
830
831 //If we veto the event then we cancel the new window
832 if (event.IsVetoed())
833 {
834 wxActiveXEventNativeMSW* nativeParams = evt.GetNativeParameters();
835 *V_BOOLREF(&nativeParams->pDispParams->rgvarg[3]) = VARIANT_TRUE;
836 }
837 break;
838 }
839 }
840
841 evt.Skip();
842 }
843
844 #endif