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, 2011 Steven Lamerton
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 wxUSE_WEBVIEW && wxUSE_WEBVIEW_IE
26 #include "wx/msw/registry.h"
27 #include "wx/msw/missing.h"
28 #include "wx/filesys.h"
29 #include "wx/dynlib.h"
31 wxIMPLEMENT_DYNAMIC_CLASS(wxWebViewIE
, wxWebView
);
33 BEGIN_EVENT_TABLE(wxWebViewIE
, wxControl
)
34 EVT_ACTIVEX(wxID_ANY
, wxWebViewIE
::onActiveXEvent
)
35 EVT_ERASE_BACKGROUND(wxWebViewIE
::onEraseBg
)
38 bool wxWebViewIE
::Create(wxWindow
* parent
,
46 if (!wxControl
::Create(parent
, id
, pos
, size
, style
,
47 wxDefaultValidator
, name
))
54 m_historyLoadingFromList
= false;
55 m_historyEnabled
= true;
56 m_historyPosition
= -1;
57 m_zoomType
= wxWEB_VIEW_ZOOM_TYPE_TEXT
;
59 if (::CoCreateInstance(CLSID_WebBrowser
, NULL
,
60 CLSCTX_INPROC_SERVER
, // CLSCTX_INPROC,
61 IID_IWebBrowser2
, (void**)&m_webBrowser
) != 0)
63 wxLogError("Failed to initialize IE, CoCreateInstance returned an error");
67 m_ie
.SetDispatchPtr(m_webBrowser
); // wxAutomationObject will release itself
69 m_webBrowser
->put_RegisterAsBrowser(VARIANT_TRUE
);
70 m_webBrowser
->put_RegisterAsDropTarget(VARIANT_TRUE
);
72 m_container
= new wxActiveXContainer(this, IID_IWebBrowser2
, m_webBrowser
);
74 SetBackgroundStyle(wxBG_STYLE_PAINT
);
75 SetDoubleBuffered(true);
80 wxWebViewIE
::~wxWebViewIE()
82 for(unsigned int i
= 0; i
< m_factories
.size(); i
++)
84 m_factories
[i
]->Release();
88 void wxWebViewIE
::LoadURL(const wxString
& url
)
90 m_ie
.CallMethod("Navigate", (BSTR
) url
.wc_str());
93 void wxWebViewIE
::SetPage(const wxString
& html
, const wxString
& baseUrl
)
95 BSTR bstr
= SysAllocString(html
.wc_str());
97 // Creates a new one-dimensional array
98 SAFEARRAY
*psaStrings
= SafeArrayCreateVector(VT_VARIANT
, 0, 1);
99 if (psaStrings
!= NULL
)
103 HRESULT hr
= SafeArrayAccessData(psaStrings
, (LPVOID
*)¶m
);
105 param
->bstrVal
= bstr
;
106 hr
= SafeArrayUnaccessData(psaStrings
);
108 IHTMLDocument2
* document
= GetDocument();
109 document
->write(psaStrings
);
112 // SafeArrayDestroy calls SysFreeString for each BSTR
113 SafeArrayDestroy(psaStrings
);
115 //We send the events when we are done to mimic webkit
117 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED
,
118 GetId(), baseUrl
, "");
119 event
.SetEventObject(this);
120 HandleWindowEvent(event
);
122 //Document complete event
123 event
.SetEventType(wxEVT_COMMAND_WEB_VIEW_LOADED
);
124 event
.SetEventObject(this);
125 HandleWindowEvent(event
);
129 wxLogError("wxWebViewIE::SetPage() : psaStrings is NULL");
134 wxString wxWebViewIE
::GetPageSource() const
136 IHTMLDocument2
* document
= GetDocument();
137 IHTMLElement
*bodyTag
= NULL
;
138 IHTMLElement
*htmlTag
= NULL
;
140 HRESULT hr
= document
->get_body(&bodyTag
);
143 hr
= bodyTag
->get_parentElement(&htmlTag
);
147 htmlTag
->get_outerHTML(&bstr
);
148 source
= wxString(bstr
);
158 wxWebViewZoom wxWebViewIE
::GetZoom() const
160 if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
161 return GetIEOpticalZoom();
162 else if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_TEXT
)
163 return GetIETextZoom();
167 //Dummy return to stop compiler warnings
168 return wxWEB_VIEW_ZOOM_MEDIUM
;
172 void wxWebViewIE
::SetZoom(wxWebViewZoom zoom
)
174 if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
175 SetIEOpticalZoom(zoom
);
176 else if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_TEXT
)
182 void wxWebViewIE
::SetIETextZoom(wxWebViewZoom level
)
184 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
185 //is 0 to 4 so the check is unnecessary, these match exactly with the
188 VariantInit (&zoomVariant
);
189 V_VT(&zoomVariant
) = VT_I4
;
190 V_I4(&zoomVariant
) = level
;
192 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
193 OLECMDEXECOPT_DONTPROMPTUSER
,
195 wxASSERT(result
== S_OK
);
198 wxWebViewZoom wxWebViewIE
::GetIETextZoom() const
201 VariantInit (&zoomVariant
);
202 V_VT(&zoomVariant
) = VT_I4
;
204 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
205 OLECMDEXECOPT_DONTPROMPTUSER
,
207 wxASSERT(result
== S_OK
);
209 //We can safely cast here as we know that the range matches our enum
210 return static_cast<wxWebViewZoom
>(V_I4(&zoomVariant
));
213 void wxWebViewIE
::SetIEOpticalZoom(wxWebViewZoom level
)
215 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
216 //is 10 to 1000 so the check is unnecessary
218 VariantInit (&zoomVariant
);
219 V_VT(&zoomVariant
) = VT_I4
;
221 //We make a somewhat arbitray map here, taken from values used by webkit
224 case wxWEB_VIEW_ZOOM_TINY
:
225 V_I4(&zoomVariant
) = 60;
227 case wxWEB_VIEW_ZOOM_SMALL
:
228 V_I4(&zoomVariant
) = 80;
230 case wxWEB_VIEW_ZOOM_MEDIUM
:
231 V_I4(&zoomVariant
) = 100;
233 case wxWEB_VIEW_ZOOM_LARGE
:
234 V_I4(&zoomVariant
) = 130;
236 case wxWEB_VIEW_ZOOM_LARGEST
:
237 V_I4(&zoomVariant
) = 160;
243 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)63 /*OLECMDID_OPTICAL_ZOOM*/,
244 OLECMDEXECOPT_DODEFAULT
,
247 wxASSERT(result
== S_OK
);
250 wxWebViewZoom wxWebViewIE
::GetIEOpticalZoom() const
253 VariantInit (&zoomVariant
);
254 V_VT(&zoomVariant
) = VT_I4
;
256 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)63 /*OLECMDID_OPTICAL_ZOOM*/,
257 OLECMDEXECOPT_DODEFAULT
, NULL
,
259 wxASSERT(result
== S_OK
);
261 const int zoom
= V_I4(&zoomVariant
);
263 //We make a somewhat arbitray map here, taken from values used by webkit
266 return wxWEB_VIEW_ZOOM_TINY
;
268 else if (zoom
> 65 && zoom
<= 90)
270 return wxWEB_VIEW_ZOOM_SMALL
;
272 else if (zoom
> 90 && zoom
<= 115)
274 return wxWEB_VIEW_ZOOM_MEDIUM
;
276 else if (zoom
> 115 && zoom
<= 145)
278 return wxWEB_VIEW_ZOOM_LARGE
;
280 else /*if (zoom > 145) */ //Using else removes a compiler warning
282 return wxWEB_VIEW_ZOOM_LARGEST
;
286 void wxWebViewIE
::SetZoomType(wxWebViewZoomType type
)
291 wxWebViewZoomType wxWebViewIE
::GetZoomType() const
296 bool wxWebViewIE
::CanSetZoomType(wxWebViewZoomType type
) const
298 //IE 6 and below only support text zoom, so check the registry to see what
299 //version we actually have
300 wxRegKey
key(wxRegKey
::HKLM
, "Software\\Microsoft\\Internet Explorer");
302 key
.QueryValue("Version", value
);
304 long version
= wxAtoi(value
.Left(1));
305 if(version
<= 6 && type
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
311 void wxWebViewIE
::Print()
313 m_webBrowser
->ExecWB(OLECMDID_PRINTPREVIEW
,
314 OLECMDEXECOPT_DODEFAULT
, NULL
, NULL
);
317 bool wxWebViewIE
::CanGoBack() const
320 return m_historyPosition
> 0;
325 bool wxWebViewIE
::CanGoForward() const
328 return m_historyPosition
!= static_cast<int>(m_historyList
.size()) - 1;
333 void wxWebViewIE
::LoadHistoryItem(wxSharedPtr
<wxWebViewHistoryItem
> item
)
336 for(unsigned int i
= 0; i
< m_historyList
.size(); i
++)
338 //We compare the actual pointers to find the correct item
339 if(m_historyList
[i
].get() == item
.get())
342 wxASSERT_MSG(pos
!= static_cast<int>(m_historyList
.size()),
343 "invalid history item");
344 m_historyLoadingFromList
= true;
345 LoadURL(item
->GetUrl());
346 m_historyPosition
= pos
;
349 wxVector
<wxSharedPtr
<wxWebViewHistoryItem
> > wxWebViewIE
::GetBackwardHistory()
351 wxVector
<wxSharedPtr
<wxWebViewHistoryItem
> > backhist
;
352 //As we don't have std::copy or an iterator constructor in the wxwidgets
353 //native vector we construct it by hand
354 for(int i
= 0; i
< m_historyPosition
; i
++)
356 backhist
.push_back(m_historyList
[i
]);
361 wxVector
<wxSharedPtr
<wxWebViewHistoryItem
> > wxWebViewIE
::GetForwardHistory()
363 wxVector
<wxSharedPtr
<wxWebViewHistoryItem
> > forwardhist
;
364 //As we don't have std::copy or an iterator constructor in the wxwidgets
365 //native vector we construct it by hand
366 for(int i
= m_historyPosition
+ 1; i
< static_cast<int>(m_historyList
.size()); i
++)
368 forwardhist
.push_back(m_historyList
[i
]);
373 void wxWebViewIE
::GoBack()
375 LoadHistoryItem(m_historyList
[m_historyPosition
- 1]);
378 void wxWebViewIE
::GoForward()
380 LoadHistoryItem(m_historyList
[m_historyPosition
+ 1]);
383 void wxWebViewIE
::Stop()
385 m_ie
.CallMethod("Stop");
388 void wxWebViewIE
::ClearHistory()
390 m_historyList
.clear();
391 m_historyPosition
= -1;
394 void wxWebViewIE
::EnableHistory(bool enable
)
396 m_historyEnabled
= enable
;
397 m_historyList
.clear();
398 m_historyPosition
= -1;
401 void wxWebViewIE
::Reload(wxWebViewReloadFlags flags
)
405 V_VT(&level
) = VT_I2
;
409 case wxWEB_VIEW_RELOAD_DEFAULT
:
410 V_I2(&level
) = REFRESH_NORMAL
;
412 case wxWEB_VIEW_RELOAD_NO_CACHE
:
413 V_I2(&level
) = REFRESH_COMPLETELY
;
416 wxFAIL_MSG("Unexpected reload type");
419 m_webBrowser
->Refresh2(&level
);
422 bool wxWebViewIE
::IsOfflineMode()
424 wxVariant out
= m_ie
.GetProperty("Offline");
426 wxASSERT(out
.GetType() == "bool");
428 return out
.GetBool();
431 void wxWebViewIE
::SetOfflineMode(bool offline
)
433 // FIXME: the wxWidgets docs do not really document what the return
434 // parameter of PutProperty is
435 const bool success
= m_ie
.PutProperty("Offline", (offline ?
441 bool wxWebViewIE
::IsBusy() const
443 if (m_isBusy
) return true;
445 wxVariant out
= m_ie
.GetProperty("Busy");
447 wxASSERT(out
.GetType() == "bool");
449 return out
.GetBool();
452 wxString wxWebViewIE
::GetCurrentURL() const
454 wxVariant out
= m_ie
.GetProperty("LocationURL");
456 wxASSERT(out
.GetType() == "string");
457 return out
.GetString();
460 wxString wxWebViewIE
::GetCurrentTitle() const
462 IHTMLDocument2
* document
= GetDocument();
465 document
->get_nameProp(&title
);
467 return wxString(title
);
470 bool wxWebViewIE
::CanCut() const
472 return CanExecCommand("Cut");
475 bool wxWebViewIE
::CanCopy() const
477 return CanExecCommand("Copy");
479 bool wxWebViewIE
::CanPaste() const
481 return CanExecCommand("Paste");
484 void wxWebViewIE
::Cut()
489 void wxWebViewIE
::Copy()
494 void wxWebViewIE
::Paste()
496 ExecCommand("Paste");
499 bool wxWebViewIE
::CanUndo() const
501 return CanExecCommand("Undo");
503 bool wxWebViewIE
::CanRedo() const
505 return CanExecCommand("Redo");
508 void wxWebViewIE
::Undo()
513 void wxWebViewIE
::Redo()
518 void wxWebViewIE
::SetEditable(bool enable
)
520 IHTMLDocument2
* document
= GetDocument();
522 document
->put_designMode(SysAllocString(L
"On"));
524 document
->put_designMode(SysAllocString(L
"Off"));
529 bool wxWebViewIE
::IsEditable() const
531 IHTMLDocument2
* document
= GetDocument();
533 document
->get_designMode(&mode
);
535 if(wxString(mode
) == "On")
541 void wxWebViewIE
::SelectAll()
543 ExecCommand("SelectAll");
546 bool wxWebViewIE
::HasSelection() const
548 IHTMLDocument2
* document
= GetDocument();
549 IHTMLSelectionObject
* selection
;
551 HRESULT hr
= document
->get_selection(&selection
);
555 selection
->get_type(&type
);
556 sel
= wxString(type
);
557 selection
->Release();
560 return sel
!= "None";
563 void wxWebViewIE
::DeleteSelection()
565 ExecCommand("Delete");
568 wxString wxWebViewIE
::GetSelectedText() const
570 IHTMLDocument2
* document
= GetDocument();
571 IHTMLSelectionObject
* selection
;
573 HRESULT hr
= document
->get_selection(&selection
);
577 hr
= selection
->createRange(&disrange
);
580 IHTMLTxtRange
* range
;
581 hr
= disrange
->QueryInterface(IID_IHTMLTxtRange
, (void**)&range
);
585 range
->get_text(&text
);
586 selected
= wxString(text
);
591 selection
->Release();
597 wxString wxWebViewIE
::GetSelectedSource() const
599 IHTMLDocument2
* document
= GetDocument();
600 IHTMLSelectionObject
* selection
;
602 HRESULT hr
= document
->get_selection(&selection
);
606 hr
= selection
->createRange(&disrange
);
609 IHTMLTxtRange
* range
;
610 hr
= disrange
->QueryInterface(IID_IHTMLTxtRange
, (void**)&range
);
614 range
->get_htmlText(&text
);
615 selected
= wxString(text
);
620 selection
->Release();
626 void wxWebViewIE
::ClearSelection()
628 IHTMLDocument2
* document
= GetDocument();
629 IHTMLSelectionObject
* selection
;
631 HRESULT hr
= document
->get_selection(&selection
);
635 selection
->Release();
640 wxString wxWebViewIE
::GetPageText() const
642 IHTMLDocument2
* document
= GetDocument();
645 HRESULT hr
= document
->get_body(&body
);
649 body
->get_innerText(&out
);
650 text
= wxString(out
);
657 void wxWebViewIE
::RunScript(const wxString
& javascript
)
659 IHTMLDocument2
* document
= GetDocument();
660 IHTMLWindow2
* window
;
661 wxString language
= "javascript";
662 HRESULT hr
= document
->get_parentWindow(&window
);
667 V_VT(&level
) = VT_EMPTY
;
668 window
->execScript(SysAllocString(javascript
), SysAllocString(language
), &level
);
673 void wxWebViewIE
::RegisterHandler(wxSharedPtr
<wxWebViewHandler
> handler
)
675 wxDynamicLibrary
urlMon(wxT("urlmon.dll"));
676 if(urlMon
.HasSymbol(wxT("CoInternetGetSession")))
678 typedef HRESULT (WINAPI
*CoInternetGetSession_t
)(DWORD
, IInternetSession
**, DWORD
);
679 wxDYNLIB_FUNCTION(CoInternetGetSession_t
, CoInternetGetSession
, urlMon
);
681 ClassFactory
* cf
= new ClassFactory(handler
);
682 IInternetSession
* session
;
683 HRESULT res
= (*pfnCoInternetGetSession
)(0, &session
, 0);
686 wxFAIL_MSG("Could not retrive internet session");
689 HRESULT hr
= session
->RegisterNameSpace(cf
, CLSID_FileProtocol
, handler
->GetName(), 0, NULL
, 0);
692 wxFAIL_MSG("Could not register protocol");
694 m_factories
.push_back(cf
);
698 wxFAIL_MSG("urlmon does not contain CoInternetGetSession");
702 bool wxWebViewIE
::CanExecCommand(wxString command
) const
704 IHTMLDocument2
* document
= GetDocument();
705 VARIANT_BOOL enabled
;
707 document
->queryCommandEnabled(SysAllocString(command
.wc_str()), &enabled
);
710 return (enabled
== VARIANT_TRUE
);
713 void wxWebViewIE
::ExecCommand(wxString command
)
715 IHTMLDocument2
* document
= GetDocument();
716 document
->execCommand(SysAllocString(command
.wc_str()), VARIANT_FALSE
, VARIANT(), NULL
);
720 IHTMLDocument2
* wxWebViewIE
::GetDocument() const
722 wxVariant variant
= m_ie
.GetProperty("Document");
723 IHTMLDocument2
* document
= (IHTMLDocument2
*)variant
.GetVoidPtr();
730 void wxWebViewIE
::onActiveXEvent(wxActiveXEvent
& evt
)
732 if (m_webBrowser
== NULL
) return;
734 switch (evt
.GetDispatchId())
736 case DISPID_BEFORENAVIGATE2
:
740 wxString url
= evt
[1].GetString();
741 wxString target
= evt
[3].GetString();
743 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATING
,
744 GetId(), url
, target
);
745 event
.SetEventObject(this);
746 HandleWindowEvent(event
);
748 if (!event
.IsAllowed())
750 wxActiveXEventNativeMSW
* nativeParams
=
751 evt
.GetNativeParameters();
752 *V_BOOLREF(&nativeParams
->pDispParams
->rgvarg
[0]) = VARIANT_TRUE
;
755 // at this point, either the navigation event has been cancelled
756 // and we're not busy, either it was accepted and IWebBrowser2's
757 // Busy property will be true; so we don't need our override
764 case DISPID_NAVIGATECOMPLETE2
:
766 wxString url
= evt
[1].GetString();
767 // TODO: set target parameter if possible
768 wxString target
= wxEmptyString
;
769 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED
,
770 GetId(), url
, target
);
771 event
.SetEventObject(this);
772 HandleWindowEvent(event
);
776 case DISPID_PROGRESSCHANGE
:
782 case DISPID_DOCUMENTCOMPLETE
:
784 //Only send a complete even if we are actually finished, this brings
785 //the event in to line with webkit
787 m_webBrowser
->get_ReadyState( &rs
);
788 if(rs
!= READYSTATE_COMPLETE
)
791 wxString url
= evt
[1].GetString();
793 //As we are complete we also add to the history list, but not if the
794 //page is not the main page, ie it is a subframe
795 //We also have to check if we are loading a file:// url, if so we
796 //need to change the comparison as ie passes back a different style
798 if(m_historyEnabled
&& !m_historyLoadingFromList
&&
799 (url
== GetCurrentURL() ||
800 (GetCurrentURL().substr(0, 4) == "file" &&
801 wxFileSystem
::URLToFileName(GetCurrentURL()).GetFullPath() == url
)))
803 //If we are not at the end of the list, then erase everything
804 //between us and the end before adding the new page
805 if(m_historyPosition
!= static_cast<int>(m_historyList
.size()) - 1)
807 m_historyList
.erase(m_historyList
.begin() + m_historyPosition
+ 1,
808 m_historyList
.end());
810 wxSharedPtr
<wxWebViewHistoryItem
> item(new wxWebViewHistoryItem(url
, GetCurrentTitle()));
811 m_historyList
.push_back(item
);
814 //Reset as we are done now
815 m_historyLoadingFromList
= false;
816 // TODO: set target parameter if possible
817 wxString target
= wxEmptyString
;
818 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_LOADED
, GetId(),
820 event
.SetEventObject(this);
821 HandleWindowEvent(event
);
825 case DISPID_STATUSTEXTCHANGE
:
830 case DISPID_TITLECHANGE
:
832 wxString title
= evt
[0].GetString();
834 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_TITLE_CHANGED
,
835 GetId(), GetCurrentURL(), "");
836 event
.SetString(title
);
837 event
.SetEventObject(this);
838 HandleWindowEvent(event
);
842 case DISPID_NAVIGATEERROR
:
844 wxWebViewNavigationError errorType
= wxWEB_NAV_ERR_OTHER
;
845 wxString errorCode
= "?";
846 switch (evt
[3].GetLong())
848 case INET_E_INVALID_URL
: // (0x800C0002L or -2146697214)
849 errorCode
= "INET_E_INVALID_URL";
850 errorType
= wxWEB_NAV_ERR_REQUEST
;
852 case INET_E_NO_SESSION
: // (0x800C0003L or -2146697213)
853 errorCode
= "INET_E_NO_SESSION";
854 errorType
= wxWEB_NAV_ERR_CONNECTION
;
856 case INET_E_CANNOT_CONNECT
: // (0x800C0004L or -2146697212)
857 errorCode
= "INET_E_CANNOT_CONNECT";
858 errorType
= wxWEB_NAV_ERR_CONNECTION
;
860 case INET_E_RESOURCE_NOT_FOUND
: // (0x800C0005L or -2146697211)
861 errorCode
= "INET_E_RESOURCE_NOT_FOUND";
862 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
864 case INET_E_OBJECT_NOT_FOUND
: // (0x800C0006L or -2146697210)
865 errorCode
= "INET_E_OBJECT_NOT_FOUND";
866 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
868 case INET_E_DATA_NOT_AVAILABLE
: // (0x800C0007L or -2146697209)
869 errorCode
= "INET_E_DATA_NOT_AVAILABLE";
870 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
872 case INET_E_DOWNLOAD_FAILURE
: // (0x800C0008L or -2146697208)
873 errorCode
= "INET_E_DOWNLOAD_FAILURE";
874 errorType
= wxWEB_NAV_ERR_CONNECTION
;
876 case INET_E_AUTHENTICATION_REQUIRED
: // (0x800C0009L or -2146697207)
877 errorCode
= "INET_E_AUTHENTICATION_REQUIRED";
878 errorType
= wxWEB_NAV_ERR_AUTH
;
880 case INET_E_NO_VALID_MEDIA
: // (0x800C000AL or -2146697206)
881 errorCode
= "INET_E_NO_VALID_MEDIA";
882 errorType
= wxWEB_NAV_ERR_REQUEST
;
884 case INET_E_CONNECTION_TIMEOUT
: // (0x800C000BL or -2146697205)
885 errorCode
= "INET_E_CONNECTION_TIMEOUT";
886 errorType
= wxWEB_NAV_ERR_CONNECTION
;
888 case INET_E_INVALID_REQUEST
: // (0x800C000CL or -2146697204)
889 errorCode
= "INET_E_INVALID_REQUEST";
890 errorType
= wxWEB_NAV_ERR_REQUEST
;
892 case INET_E_UNKNOWN_PROTOCOL
: // (0x800C000DL or -2146697203)
893 errorCode
= "INET_E_UNKNOWN_PROTOCOL";
894 errorType
= wxWEB_NAV_ERR_REQUEST
;
896 case INET_E_SECURITY_PROBLEM
: // (0x800C000EL or -2146697202)
897 errorCode
= "INET_E_SECURITY_PROBLEM";
898 errorType
= wxWEB_NAV_ERR_SECURITY
;
900 case INET_E_CANNOT_LOAD_DATA
: // (0x800C000FL or -2146697201)
901 errorCode
= "INET_E_CANNOT_LOAD_DATA";
902 errorType
= wxWEB_NAV_ERR_OTHER
;
904 case INET_E_CANNOT_INSTANTIATE_OBJECT
:
905 // CoCreateInstance will return an error code if this happens,
906 // we'll handle this above.
909 case INET_E_REDIRECT_FAILED
: // (0x800C0014L or -2146697196)
910 errorCode
= "INET_E_REDIRECT_FAILED";
911 errorType
= wxWEB_NAV_ERR_OTHER
;
913 case INET_E_REDIRECT_TO_DIR
: // (0x800C0015L or -2146697195)
914 errorCode
= "INET_E_REDIRECT_TO_DIR";
915 errorType
= wxWEB_NAV_ERR_REQUEST
;
917 case INET_E_CANNOT_LOCK_REQUEST
: // (0x800C0016L or -2146697194)
918 errorCode
= "INET_E_CANNOT_LOCK_REQUEST";
919 errorType
= wxWEB_NAV_ERR_OTHER
;
921 case INET_E_USE_EXTEND_BINDING
: // (0x800C0017L or -2146697193)
922 errorCode
= "INET_E_USE_EXTEND_BINDING";
923 errorType
= wxWEB_NAV_ERR_OTHER
;
925 case INET_E_TERMINATED_BIND
: // (0x800C0018L or -2146697192)
926 errorCode
= "INET_E_TERMINATED_BIND";
927 errorType
= wxWEB_NAV_ERR_OTHER
;
929 case INET_E_INVALID_CERTIFICATE
: // (0x800C0019L or -2146697191)
930 errorCode
= "INET_E_INVALID_CERTIFICATE";
931 errorType
= wxWEB_NAV_ERR_CERTIFICATE
;
933 case INET_E_CODE_DOWNLOAD_DECLINED
: // (0x800C0100L or -2146696960)
934 errorCode
= "INET_E_CODE_DOWNLOAD_DECLINED";
935 errorType
= wxWEB_NAV_ERR_USER_CANCELLED
;
937 case INET_E_RESULT_DISPATCHED
: // (0x800C0200L or -2146696704)
938 // cancel request cancelled...
939 errorCode
= "INET_E_RESULT_DISPATCHED";
940 errorType
= wxWEB_NAV_ERR_OTHER
;
942 case INET_E_CANNOT_REPLACE_SFP_FILE
: // (0x800C0300L or -2146696448)
943 errorCode
= "INET_E_CANNOT_REPLACE_SFP_FILE";
944 errorType
= wxWEB_NAV_ERR_SECURITY
;
946 case INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY
:
947 errorCode
= "INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY";
948 errorType
= wxWEB_NAV_ERR_SECURITY
;
950 case INET_E_CODE_INSTALL_SUPPRESSED
:
951 errorCode
= "INET_E_CODE_INSTALL_SUPPRESSED";
952 errorType
= wxWEB_NAV_ERR_SECURITY
;
956 wxString url
= evt
[1].GetString();
957 wxString target
= evt
[2].GetString();
958 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_ERROR
, GetId(),
960 event
.SetEventObject(this);
961 event
.SetInt(errorType
);
962 event
.SetString(errorCode
);
963 HandleWindowEvent(event
);
966 case DISPID_NEWWINDOW3
:
968 wxString url
= evt
[4].GetString();
970 wxWebViewEvent
event(wxEVT_COMMAND_WEB_VIEW_NEWWINDOW
,
971 GetId(), url
, wxEmptyString
);
972 event
.SetEventObject(this);
973 HandleWindowEvent(event
);
975 //We always cancel this event otherwise an Internet Exporer window
976 //is opened for the url
977 wxActiveXEventNativeMSW
* nativeParams
= evt
.GetNativeParameters();
978 *V_BOOLREF(&nativeParams
->pDispParams
->rgvarg
[3]) = VARIANT_TRUE
;
986 VirtualProtocol
::VirtualProtocol(wxSharedPtr
<wxWebViewHandler
> handler
)
993 VirtualProtocol
::~VirtualProtocol()
997 ULONG VirtualProtocol
::AddRef()
1003 HRESULT VirtualProtocol
::QueryInterface(REFIID riid
, void **ppvObject
)
1005 if(riid
== IID_IUnknown
|| riid
== IID_IInternetProtocolRoot
||
1006 riid
== IID_IInternetProtocol
)
1008 *ppvObject
= (IInternetProtocol
*)this;
1019 ULONG VirtualProtocol
::Release()
1033 HRESULT VirtualProtocol
::Start(LPCWSTR szUrl
, IInternetProtocolSink
*pOIProtSink
,
1034 IInternetBindInfo
*pOIBindInfo
, DWORD grfPI
,
1035 HANDLE_PTR dwReserved
)
1038 wxUnusedVar(pOIBindInfo
);
1040 wxUnusedVar(dwReserved
);
1041 m_protocolSink
= pOIProtSink
;
1043 //We get the file itself from the protocol handler
1044 m_file
= m_handler
->GetFile(szUrl
);
1048 return INET_E_RESOURCE_NOT_FOUND
;
1050 //We return the stream length for current and total size as we can always
1051 //read the whole file from the stream
1052 wxFileOffset length
= m_file
->GetStream()->GetLength();
1053 m_protocolSink
->ReportData(BSCF_FIRSTDATANOTIFICATION
|
1054 BSCF_DATAFULLYAVAILABLE
|
1055 BSCF_LASTDATANOTIFICATION
,
1060 HRESULT VirtualProtocol
::Read(void *pv
, ULONG cb
, ULONG
*pcbRead
)
1062 //If the file is null we return false to indicte it is finished
1066 wxStreamError err
= m_file
->GetStream()->Read(pv
, cb
).GetLastError();
1067 *pcbRead
= m_file
->GetStream()->LastRead();
1069 if(err
== wxSTREAM_NO_ERROR
)
1074 m_protocolSink
->ReportResult(S_OK
, 0, NULL
);
1076 //As we are not eof there is more data
1079 else if(err
== wxSTREAM_EOF
)
1082 m_protocolSink
->ReportResult(S_OK
, 0, NULL
);
1083 //We are eof and so finished
1086 else if(err
== wxSTREAM_READ_ERROR
)
1089 return INET_E_DOWNLOAD_FAILURE
;
1093 //Dummy return to surpress a compiler warning
1095 return INET_E_DOWNLOAD_FAILURE
;
1099 HRESULT ClassFactory
::CreateInstance(IUnknown
* pUnkOuter
, REFIID riid
,
1103 return CLASS_E_NOAGGREGATION
;
1104 VirtualProtocol
* vp
= new VirtualProtocol(m_handler
);
1106 HRESULT hr
= vp
->QueryInterface(riid
, ppvObject
);
1112 STDMETHODIMP ClassFactory
::LockServer(BOOL fLock
)
1118 ULONG ClassFactory
::AddRef(void)
1124 HRESULT ClassFactory
::QueryInterface(REFIID riid
, void **ppvObject
)
1126 if ((riid
== IID_IUnknown
) || (riid
== IID_IClassFactory
))
1140 ULONG ClassFactory
::Release(void)
1155 #endif // wxUSE_WEBVIEW && wxUSE_WEBVIEW_IE