1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/webview_ie.cpp
3 // Purpose: wxMSW wxWebViewIE class implementation for web view component
4 // Author: Marianne Gagnon
6 // Copyright: (c) 2010 Marianne Gagnon
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
13 #if defined(__BORLANDC__)
17 #include "wx/msw/webview_ie.h"
19 #if wxHAVE_WEB_BACKEND_IE
28 // FIXME: Seems like MINGW does not have these, how to handle cleanly?
29 #define DISPID_COMMANDSTATECHANGE 105
30 typedef enum CommandStateChangeConstants
{
31 CSC_UPDATECOMMANDS
= (int) 0xFFFFFFFF,
32 CSC_NAVIGATEFORWARD
= 0x1,
33 CSC_NAVIGATEBACK
= 0x2
34 } CommandStateChangeConstants
;
37 // FIXME: Seems like MINGW does not have these, how to handle cleanly?
38 #define DISPID_NAVIGATECOMPLETE2 252
39 #define DISPID_NAVIGATEERROR 271
40 #define OLECMDID_OPTICAL_ZOOM 63
41 #define INET_E_ERROR_FIRST 0x800C0002L
42 #define INET_E_INVALID_URL 0x800C0002L
43 #define INET_E_NO_SESSION 0x800C0003L
44 #define INET_E_CANNOT_CONNECT 0x800C0004L
45 #define INET_E_RESOURCE_NOT_FOUND 0x800C0005L
46 #define INET_E_OBJECT_NOT_FOUND 0x800C0006L
47 #define INET_E_DATA_NOT_AVAILABLE 0x800C0007L
48 #define INET_E_DOWNLOAD_FAILURE 0x800C0008L
49 #define INET_E_AUTHENTICATION_REQUIRED 0x800C0009L
50 #define INET_E_NO_VALID_MEDIA 0x800C000AL
51 #define INET_E_CONNECTION_TIMEOUT 0x800C000BL
52 #define INET_E_INVALID_REQUEST 0x800C000CL
53 #define INET_E_UNKNOWN_PROTOCOL 0x800C000DL
54 #define INET_E_SECURITY_PROBLEM 0x800C000EL
55 #define INET_E_CANNOT_LOAD_DATA 0x800C000FL
56 #define INET_E_CANNOT_INSTANTIATE_OBJECT 0x800C0010L
57 #define INET_E_QUERYOPTION_UNKNOWN 0x800C0013L
58 #define INET_E_REDIRECT_FAILED 0x800C0014L
59 #define INET_E_REDIRECT_TO_DIR 0x800C0015L
60 #define INET_E_CANNOT_LOCK_REQUEST 0x800C0016L
61 #define INET_E_USE_EXTEND_BINDING 0x800C0017L
62 #define INET_E_TERMINATED_BIND 0x800C0018L
63 #define INET_E_INVALID_CERTIFICATE 0x800C0019L
64 #define INET_E_CODE_DOWNLOAD_DECLINED 0x800C0100L
65 #define INET_E_RESULT_DISPATCHED 0x800C0200L
66 #define INET_E_CANNOT_REPLACE_SFP_FILE 0x800C0300L
67 #define INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY 0x800C0500L
68 #define INET_E_CODE_INSTALL_SUPPRESSED 0x800C0400L
70 #define REFRESH_COMPLETELY 3
73 BEGIN_EVENT_TABLE(wxWebViewIE
, wxControl
)
74 EVT_ACTIVEX(wxID_ANY
, wxWebViewIE::onActiveXEvent
)
75 EVT_ERASE_BACKGROUND(wxWebViewIE::onEraseBg
)
78 bool wxWebViewIE::Create(wxWindow
* parent
,
86 if (!wxControl::Create(parent
, id
, pos
, size
, style
,
87 wxDefaultValidator
, name
))
93 m_canNavigateBack
= false;
94 m_canNavigateForward
= false;
97 if (::CoCreateInstance(CLSID_WebBrowser
, NULL
,
98 CLSCTX_INPROC_SERVER
, // CLSCTX_INPROC,
99 IID_IWebBrowser2
, (void**)&m_webBrowser
) != 0)
101 wxLogError("Failed to initialize IE, CoCreateInstance returned an error");
105 m_ie
.SetDispatchPtr(m_webBrowser
); // wxAutomationObject will release itself
107 m_webBrowser
->put_RegisterAsBrowser(VARIANT_TRUE
);
108 m_webBrowser
->put_RegisterAsDropTarget(VARIANT_TRUE
);
109 //m_webBrowser->put_Silent(VARIANT_FALSE);
111 m_container
= new wxActiveXContainer(this, IID_IWebBrowser2
, m_webBrowser
);
113 SetBackgroundStyle(wxBG_STYLE_PAINT
);
114 SetDoubleBuffered(true);
119 void wxWebViewIE::LoadUrl(const wxString
& url
)
121 wxVariant out
= m_ie
.CallMethod("Navigate", (BSTR
) url
.wc_str(),
122 NULL
, NULL
, NULL
, NULL
);
124 // FIXME: why is out value null??
125 //(HRESULT)(out.GetLong()) == S_OK;
128 void wxWebViewIE::SetPage(const wxString
& html
, const wxString
& baseUrl
)
130 LoadUrl("about:blank");
132 // Let the wx events generated for navigation events be processed, so
133 // that the underlying IE component completes its Document object.
134 // FIXME: calling wxYield is not elegant nor very reliable probably
137 wxVariant documentVariant
= m_ie
.GetProperty("Document");
138 void* documentPtr
= documentVariant
.GetVoidPtr();
140 wxASSERT (documentPtr
!= NULL
);
142 // TODO: consider the "baseUrl" parameter if possible
143 // TODO: consider encoding
144 BSTR bstr
= SysAllocString(html
.wc_str());
146 // Creates a new one-dimensional array
147 SAFEARRAY
*psaStrings
= SafeArrayCreateVector(VT_VARIANT
, 0, 1);
148 if (psaStrings
!= NULL
)
151 HRESULT hr
= SafeArrayAccessData(psaStrings
, (LPVOID
*)¶m
);
153 param
->bstrVal
= bstr
;
155 hr
= SafeArrayUnaccessData(psaStrings
);
157 IHTMLDocument2
* document
= (IHTMLDocument2
*)documentPtr
;
158 document
->write(psaStrings
);
160 // SafeArrayDestroy calls SysFreeString for each BSTR
161 SafeArrayDestroy(psaStrings
);
165 wxLogError("wxWebViewIE::SetPage() : psaStrings is NULL");
170 wxString
wxWebViewIE::GetPageSource()
172 wxVariant documentVariant
= m_ie
.GetProperty("Document");
173 void* documentPtr
= documentVariant
.GetVoidPtr();
175 if (documentPtr
== NULL
)
177 return wxEmptyString
;
180 IHTMLDocument2
* document
= (IHTMLDocument2
*)documentPtr
;
182 IHTMLElement
*bodyTag
= NULL
;
183 IHTMLElement
*htmlTag
= NULL
;
184 document
->get_body(&bodyTag
);
185 wxASSERT(bodyTag
!= NULL
);
188 bodyTag
->get_parentElement(&htmlTag
);
189 wxASSERT(htmlTag
!= NULL
);
192 htmlTag
->get_outerHTML(&bstr
);
197 //wxMessageBox(wxString(bstr));
199 // TODO: check encoding
200 return wxString(bstr
);
203 // FIXME? retrieve OLECMDID_GETZOOMRANGE instead of hardcoding range 0-4
204 wxWebViewZoom
wxWebViewIE::GetZoom()
206 const int zoom
= GetIETextZoom();
211 return wxWEB_VIEW_ZOOM_TINY
;
214 return wxWEB_VIEW_ZOOM_SMALL
;
217 return wxWEB_VIEW_ZOOM_MEDIUM
;
220 return wxWEB_VIEW_ZOOM_LARGE
;
223 return wxWEB_VIEW_ZOOM_LARGEST
;
227 return wxWEB_VIEW_ZOOM_MEDIUM
;
230 void wxWebViewIE::SetZoom(wxWebViewZoom zoom
)
232 // I know I could cast from enum to int since wxWebViewZoom happens to
233 // match with IE's zoom levels, but I don't like doing that, what if enum
237 case wxWEB_VIEW_ZOOM_TINY
:
240 case wxWEB_VIEW_ZOOM_SMALL
:
243 case wxWEB_VIEW_ZOOM_MEDIUM
:
246 case wxWEB_VIEW_ZOOM_LARGE
:
249 case wxWEB_VIEW_ZOOM_LARGEST
:
257 void wxWebViewIE::SetIETextZoom(int level
)
260 VariantInit (&zoomVariant
);
261 V_VT(&zoomVariant
) = VT_I4
;
262 V_I4(&zoomVariant
) = level
;
264 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
265 OLECMDEXECOPT_DONTPROMPTUSER
,
267 wxASSERT (result
== S_OK
);
269 VariantClear (&zoomVariant
);
272 int wxWebViewIE::GetIETextZoom()
275 VariantInit (&zoomVariant
);
276 V_VT(&zoomVariant
) = VT_I4
;
277 V_I4(&zoomVariant
) = 4;
279 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
280 OLECMDEXECOPT_DONTPROMPTUSER
,
282 wxASSERT (result
== S_OK
);
284 int zoom
= V_I4(&zoomVariant
);
285 // wxMessageBox(wxString::Format("Zoom : %i", zoom));
286 VariantClear (&zoomVariant
);
291 void wxWebViewIE::SetIEOpticalZoom(float zoom
)
293 // TODO: add support for optical zoom (IE7+ only)
295 // TODO: get range from OLECMDID_OPTICAL_GETZOOMRANGE instead of hardcoding?
296 wxASSERT(zoom
>= 10.0f
);
297 wxASSERT(zoom
<= 1000.0f
);
300 VariantInit (&zoomVariant
);
301 V_VT(&zoomVariant
) = VT_I4
;
302 V_I4(&zoomVariant
) = (zoom
* 100.0f
);
304 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)OLECMDID_OPTICAL_ZOOM
,
305 OLECMDEXECOPT_DODEFAULT
,
308 wxASSERT (result
== S_OK
);
311 float wxWebViewIE::GetIEOpticalZoom()
313 // TODO: add support for optical zoom (IE7+ only)
316 VariantInit (&zoomVariant
);
317 V_VT(&zoomVariant
) = VT_I4
;
318 V_I4(&zoomVariant
) = -1;
320 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)OLECMDID_OPTICAL_ZOOM
,
321 OLECMDEXECOPT_DODEFAULT
, NULL
,
323 wxASSERT (result
== S_OK
);
325 const int zoom
= V_I4(&zoomVariant
);
326 VariantClear (&zoomVariant
);
328 return zoom
/ 100.0f
;
331 void wxWebViewIE::SetZoomType(wxWebViewZoomType
)
333 // TODO: add support for optical zoom (IE7+ only)
337 wxWebViewZoomType
wxWebViewIE::GetZoomType() const
339 // TODO: add support for optical zoom (IE7+ only)
340 return wxWEB_VIEW_ZOOM_TYPE_TEXT
;
343 bool wxWebViewIE::CanSetZoomType(wxWebViewZoomType
) const
345 // both are supported
346 // TODO: IE6 only supports text zoom, check if it's IE6 first
350 void wxWebViewIE::Print()
352 m_webBrowser
->ExecWB(OLECMDID_PRINTPREVIEW
,
353 OLECMDEXECOPT_DODEFAULT
, NULL
, NULL
);
356 void wxWebViewIE::GoBack()
358 wxVariant out
= m_ie
.CallMethod("GoBack");
360 // FIXME: why is out value null??
361 //return (HRESULT)(out.GetLong()) == S_OK;
364 void wxWebViewIE::GoForward()
366 wxVariant out
= m_ie
.CallMethod("GoForward");
368 // FIXME: why is out value null??
369 //return (HRESULT)(out.GetLong()) == S_OK;
372 void wxWebViewIE::Stop()
374 wxVariant out
= m_ie
.CallMethod("Stop");
376 // FIXME: why is out value null??
377 //return (HRESULT)(out.GetLong()) == S_OK;
381 void wxWebViewIE::Reload(wxWebViewReloadFlags flags
)
385 V_VT(&level
) = VT_I2
;
389 case wxWEB_VIEW_RELOAD_DEFAULT
:
390 V_I2(&level
) = REFRESH_NORMAL
;
392 case wxWEB_VIEW_RELOAD_NO_CACHE
:
393 V_I2(&level
) = REFRESH_COMPLETELY
;
396 wxFAIL_MSG("Unexpected reload type");
399 m_webBrowser
->Refresh2(&level
);
402 bool wxWebViewIE::IsOfflineMode()
404 wxVariant out
= m_ie
.GetProperty("Offline");
406 wxASSERT(out
.GetType() == "bool");
408 return out
.GetBool();
411 void wxWebViewIE::SetOfflineMode(bool offline
)
413 // FIXME: the wxWidgets docs do not really document what the return
414 // parameter of PutProperty is
415 const bool success
= m_ie
.PutProperty("Offline", (offline
?
421 bool wxWebViewIE::IsBusy()
423 if (m_isBusy
) return true;
425 wxVariant out
= m_ie
.GetProperty("Busy");
427 wxASSERT(out
.GetType() == "bool");
429 return out
.GetBool();
432 wxString
wxWebViewIE::GetCurrentURL()
434 wxVariant out
= m_ie
.GetProperty("LocationURL");
436 wxASSERT(out
.GetType() == "string");
437 return out
.GetString();
440 wxString
wxWebViewIE::GetCurrentTitle()
442 wxVariant out
= m_ie
.GetProperty("LocationName");
444 wxASSERT(out
.GetType() == "string");
445 return out
.GetString();
448 void wxWebViewIE::onActiveXEvent(wxActiveXEvent
& evt
)
450 if (m_webBrowser
== NULL
) return;
452 switch (evt
.GetDispatchId())
454 case DISPID_BEFORENAVIGATE2
:
458 wxString url
= evt
[1].GetString();
459 wxString target
= evt
[3].GetString();
461 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATING
,
462 GetId(), url
, target
, true);
463 event
.SetEventObject(this);
464 HandleWindowEvent(event
);
466 if (event
.IsVetoed())
468 wxActiveXEventNativeMSW
* nativeParams
=
469 evt
.GetNativeParameters();
470 *V_BOOLREF(&nativeParams
->pDispParams
->rgvarg
[0]) = VARIANT_TRUE
;
473 // at this point, either the navigation event has been cancelled
474 // and we're not busy, either it was accepted and IWebBrowser2's
475 // Busy property will be true; so we don't need our override
482 case DISPID_NAVIGATECOMPLETE2
:
484 wxString url
= evt
[1].GetString();
485 // TODO: set target parameter if possible
486 wxString target
= wxEmptyString
;
487 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED
,
488 GetId(), url
, target
, false);
489 event
.SetEventObject(this);
490 HandleWindowEvent(event
);
494 case DISPID_PROGRESSCHANGE
:
500 case DISPID_DOCUMENTCOMPLETE
:
502 wxString url
= evt
[1].GetString();
503 // TODO: set target parameter if possible
504 wxString target
= wxEmptyString
;
505 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_LOADED
, GetId(),
507 event
.SetEventObject(this);
508 HandleWindowEvent(event
);
512 case DISPID_STATUSTEXTCHANGE
:
517 case DISPID_TITLECHANGE
:
522 case DISPID_NAVIGATEERROR
:
524 wxWebNavigationError errorType
= wxWEB_NAV_ERR_OTHER
;
525 wxString errorCode
= "?";
526 switch (evt
[3].GetLong())
528 case INET_E_INVALID_URL
: // (0x800C0002L or -2146697214)
529 errorCode
= "INET_E_INVALID_URL";
530 errorType
= wxWEB_NAV_ERR_REQUEST
;
532 case INET_E_NO_SESSION
: // (0x800C0003L or -2146697213)
533 errorCode
= "INET_E_NO_SESSION";
534 errorType
= wxWEB_NAV_ERR_CONNECTION
;
536 case INET_E_CANNOT_CONNECT
: // (0x800C0004L or -2146697212)
537 errorCode
= "INET_E_CANNOT_CONNECT";
538 errorType
= wxWEB_NAV_ERR_CONNECTION
;
540 case INET_E_RESOURCE_NOT_FOUND
: // (0x800C0005L or -2146697211)
541 errorCode
= "INET_E_RESOURCE_NOT_FOUND";
542 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
544 case INET_E_OBJECT_NOT_FOUND
: // (0x800C0006L or -2146697210)
545 errorCode
= "INET_E_OBJECT_NOT_FOUND";
546 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
548 case INET_E_DATA_NOT_AVAILABLE
: // (0x800C0007L or -2146697209)
549 errorCode
= "INET_E_DATA_NOT_AVAILABLE";
550 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
552 case INET_E_DOWNLOAD_FAILURE
: // (0x800C0008L or -2146697208)
553 errorCode
= "INET_E_DOWNLOAD_FAILURE";
554 errorType
= wxWEB_NAV_ERR_CONNECTION
;
556 case INET_E_AUTHENTICATION_REQUIRED
: // (0x800C0009L or -2146697207)
557 errorCode
= "INET_E_AUTHENTICATION_REQUIRED";
558 errorType
= wxWEB_NAV_ERR_AUTH
;
560 case INET_E_NO_VALID_MEDIA
: // (0x800C000AL or -2146697206)
561 errorCode
= "INET_E_NO_VALID_MEDIA";
562 errorType
= wxWEB_NAV_ERR_REQUEST
;
564 case INET_E_CONNECTION_TIMEOUT
: // (0x800C000BL or -2146697205)
565 errorCode
= "INET_E_CONNECTION_TIMEOUT";
566 errorType
= wxWEB_NAV_ERR_CONNECTION
;
568 case INET_E_INVALID_REQUEST
: // (0x800C000CL or -2146697204)
569 errorCode
= "INET_E_INVALID_REQUEST";
570 errorType
= wxWEB_NAV_ERR_REQUEST
;
572 case INET_E_UNKNOWN_PROTOCOL
: // (0x800C000DL or -2146697203)
573 errorCode
= "INET_E_UNKNOWN_PROTOCOL";
574 errorType
= wxWEB_NAV_ERR_REQUEST
;
576 case INET_E_SECURITY_PROBLEM
: // (0x800C000EL or -2146697202)
577 errorCode
= "INET_E_SECURITY_PROBLEM";
578 errorType
= wxWEB_NAV_ERR_SECURITY
;
580 case INET_E_CANNOT_LOAD_DATA
: // (0x800C000FL or -2146697201)
581 errorCode
= "INET_E_CANNOT_LOAD_DATA";
582 errorType
= wxWEB_NAV_ERR_OTHER
;
584 case INET_E_CANNOT_INSTANTIATE_OBJECT
:
585 // CoCreateInstance will return an error code if this happens,
586 // we'll handle this above.
589 case INET_E_REDIRECT_FAILED
: // (0x800C0014L or -2146697196)
590 errorCode
= "INET_E_REDIRECT_FAILED";
591 errorType
= wxWEB_NAV_ERR_OTHER
;
593 case INET_E_REDIRECT_TO_DIR
: // (0x800C0015L or -2146697195)
594 errorCode
= "INET_E_REDIRECT_TO_DIR";
595 errorType
= wxWEB_NAV_ERR_REQUEST
;
597 case INET_E_CANNOT_LOCK_REQUEST
: // (0x800C0016L or -2146697194)
598 errorCode
= "INET_E_CANNOT_LOCK_REQUEST";
599 errorType
= wxWEB_NAV_ERR_OTHER
;
601 case INET_E_USE_EXTEND_BINDING
: // (0x800C0017L or -2146697193)
602 errorCode
= "INET_E_USE_EXTEND_BINDING";
603 errorType
= wxWEB_NAV_ERR_OTHER
;
605 case INET_E_TERMINATED_BIND
: // (0x800C0018L or -2146697192)
606 errorCode
= "INET_E_TERMINATED_BIND";
607 errorType
= wxWEB_NAV_ERR_OTHER
;
609 case INET_E_INVALID_CERTIFICATE
: // (0x800C0019L or -2146697191)
610 errorCode
= "INET_E_INVALID_CERTIFICATE";
611 errorType
= wxWEB_NAV_ERR_CERTIFICATE
;
613 case INET_E_CODE_DOWNLOAD_DECLINED
: // (0x800C0100L or -2146696960)
614 errorCode
= "INET_E_CODE_DOWNLOAD_DECLINED";
615 errorType
= wxWEB_NAV_ERR_USER_CANCELLED
;
617 case INET_E_RESULT_DISPATCHED
: // (0x800C0200L or -2146696704)
618 // cancel request cancelled...
619 errorCode
= "INET_E_RESULT_DISPATCHED";
620 errorType
= wxWEB_NAV_ERR_OTHER
;
622 case INET_E_CANNOT_REPLACE_SFP_FILE
: // (0x800C0300L or -2146696448)
623 errorCode
= "INET_E_CANNOT_REPLACE_SFP_FILE";
624 errorType
= wxWEB_NAV_ERR_SECURITY
;
626 case INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY
:
627 errorCode
= "INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY";
628 errorType
= wxWEB_NAV_ERR_SECURITY
;
630 case INET_E_CODE_INSTALL_SUPPRESSED
:
631 errorCode
= "INET_E_CODE_INSTALL_SUPPRESSED";
632 errorType
= wxWEB_NAV_ERR_SECURITY
;
636 wxString url
= evt
[1].GetString();
637 wxString target
= evt
[2].GetString();
638 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_ERROR
, GetId(),
640 event
.SetEventObject(this);
641 event
.SetInt(errorType
);
642 event
.SetString(errorCode
);
643 HandleWindowEvent(event
);
647 case DISPID_COMMANDSTATECHANGE
:
649 long commandId
= evt
[0].GetLong();
650 bool enable
= evt
[1].GetBool();
651 if (commandId
== CSC_NAVIGATEBACK
)
653 m_canNavigateBack
= enable
;
655 else if (commandId
== CSC_NAVIGATEFORWARD
)
657 m_canNavigateForward
= enable
;
661 case DISPID_NEWWINDOW2
:
663 wxActiveXEventNativeMSW
* nativeParams
= evt
.GetNativeParameters();
664 // Cancel the attempt to open a new window
665 *V_BOOLREF(&nativeParams
->pDispParams
->rgvarg
[0]) = VARIANT_TRUE
;