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, 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"
26 #include "wx/msw/registry.h"
27 #include "wx/msw/missing.h"
28 #include "wx/filesys.h"
30 //We link to urlmon as it is required for CoInternetGetSession
31 #pragma comment(lib, "urlmon")
33 //Taken from wx/filesys.cpp
34 static wxString
EscapeFileNameCharsInURL(const char *in
)
38 for ( const unsigned char *p
= (const unsigned char*)in
; *p
; ++p
)
40 const unsigned char c
= *p
;
42 if ( c
== '/' || c
== '-' || c
== '.' || c
== '_' || c
== '~' ||
43 (c
>= '0' && c
<= '9') ||
44 (c
>= 'a' && c
<= 'z') ||
45 (c
>= 'A' && c
<= 'Z') )
51 s
<< wxString::Format("%%%02x", c
);
58 BEGIN_EVENT_TABLE(wxWebViewIE
, wxControl
)
59 EVT_ACTIVEX(wxID_ANY
, wxWebViewIE::onActiveXEvent
)
60 EVT_ERASE_BACKGROUND(wxWebViewIE::onEraseBg
)
63 bool wxWebViewIE::Create(wxWindow
* parent
,
71 if (!wxControl::Create(parent
, id
, pos
, size
, style
,
72 wxDefaultValidator
, name
))
79 m_historyLoadingFromList
= false;
80 m_historyEnabled
= true;
81 m_historyPosition
= -1;
82 m_zoomType
= wxWEB_VIEW_ZOOM_TYPE_TEXT
;
84 if (::CoCreateInstance(CLSID_WebBrowser
, NULL
,
85 CLSCTX_INPROC_SERVER
, // CLSCTX_INPROC,
86 IID_IWebBrowser2
, (void**)&m_webBrowser
) != 0)
88 wxLogError("Failed to initialize IE, CoCreateInstance returned an error");
92 m_ie
.SetDispatchPtr(m_webBrowser
); // wxAutomationObject will release itself
94 m_webBrowser
->put_RegisterAsBrowser(VARIANT_TRUE
);
95 m_webBrowser
->put_RegisterAsDropTarget(VARIANT_TRUE
);
96 //m_webBrowser->put_Silent(VARIANT_FALSE);
98 //We register a custom handler for the file protocol so we can handle
99 //Virtual file systems
100 ClassFactory
* cf
= new ClassFactory
;
101 IInternetSession
* session
;
102 if(CoInternetGetSession(0, &session
, 0) != S_OK
)
104 HRESULT hr
= session
->RegisterNameSpace(cf
, CLSID_FileProtocol
, L
"file", 0, NULL
, 0);
108 m_container
= new wxActiveXContainer(this, IID_IWebBrowser2
, m_webBrowser
);
110 SetBackgroundStyle(wxBG_STYLE_PAINT
);
111 SetDoubleBuffered(true);
117 void wxWebViewIE::LoadUrl(const wxString
& url
)
119 m_ie
.CallMethod("Navigate", (BSTR
) url
.wc_str(), NULL
, NULL
, NULL
, NULL
);
122 void wxWebViewIE::SetPage(const wxString
& html
, const wxString
& baseUrl
)
124 BSTR bstr
= SysAllocString(html
.wc_str());
126 // Creates a new one-dimensional array
127 SAFEARRAY
*psaStrings
= SafeArrayCreateVector(VT_VARIANT
, 0, 1);
128 if (psaStrings
!= NULL
)
132 HRESULT hr
= SafeArrayAccessData(psaStrings
, (LPVOID
*)¶m
);
134 param
->bstrVal
= bstr
;
135 hr
= SafeArrayUnaccessData(psaStrings
);
137 IHTMLDocument2
* document
= GetDocument();
138 document
->write(psaStrings
);
141 // SafeArrayDestroy calls SysFreeString for each BSTR
142 SafeArrayDestroy(psaStrings
);
144 //We send the events when we are done to mimic webkit
146 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED
,
147 GetId(), baseUrl
, "", false);
148 event
.SetEventObject(this);
149 HandleWindowEvent(event
);
151 //Document complete event
152 event
.SetEventType(wxEVT_COMMAND_WEB_VIEW_LOADED
);
153 event
.SetEventObject(this);
154 HandleWindowEvent(event
);
158 wxLogError("wxWebViewIE::SetPage() : psaStrings is NULL");
163 wxString
wxWebViewIE::GetPageSource()
165 IHTMLDocument2
* document
= GetDocument();
166 IHTMLElement
*bodyTag
= NULL
;
167 IHTMLElement
*htmlTag
= NULL
;
169 HRESULT hr
= document
->get_body(&bodyTag
);
172 hr
= bodyTag
->get_parentElement(&htmlTag
);
176 htmlTag
->get_outerHTML(&bstr
);
177 source
= wxString(bstr
);
187 wxWebViewZoom
wxWebViewIE::GetZoom()
189 if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
190 return GetIEOpticalZoom();
191 else if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_TEXT
)
192 return GetIETextZoom();
196 //Dummy return to stop compiler warnings
197 return wxWEB_VIEW_ZOOM_MEDIUM
;
201 void wxWebViewIE::SetZoom(wxWebViewZoom zoom
)
203 if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
204 SetIEOpticalZoom(zoom
);
205 else if(m_zoomType
== wxWEB_VIEW_ZOOM_TYPE_TEXT
)
211 void wxWebViewIE::SetIETextZoom(wxWebViewZoom level
)
213 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
214 //is 0 to 4 so the check is unnecessary, these match exactly with the
217 VariantInit (&zoomVariant
);
218 V_VT(&zoomVariant
) = VT_I4
;
219 V_I4(&zoomVariant
) = level
;
221 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
222 OLECMDEXECOPT_DONTPROMPTUSER
,
224 wxASSERT(result
== S_OK
);
227 wxWebViewZoom
wxWebViewIE::GetIETextZoom()
230 VariantInit (&zoomVariant
);
231 V_VT(&zoomVariant
) = VT_I4
;
233 HRESULT result
= m_webBrowser
->ExecWB(OLECMDID_ZOOM
,
234 OLECMDEXECOPT_DONTPROMPTUSER
,
236 wxASSERT(result
== S_OK
);
238 //We can safely cast here as we know that the range matches our enum
239 return static_cast<wxWebViewZoom
>(V_I4(&zoomVariant
));
242 void wxWebViewIE::SetIEOpticalZoom(wxWebViewZoom level
)
244 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
245 //is 10 to 1000 so the check is unnecessary
247 VariantInit (&zoomVariant
);
248 V_VT(&zoomVariant
) = VT_I4
;
250 //We make a somewhat arbitray map here, taken from values used by webkit
253 case wxWEB_VIEW_ZOOM_TINY
:
254 V_I4(&zoomVariant
) = 60;
256 case wxWEB_VIEW_ZOOM_SMALL
:
257 V_I4(&zoomVariant
) = 80;
259 case wxWEB_VIEW_ZOOM_MEDIUM
:
260 V_I4(&zoomVariant
) = 100;
262 case wxWEB_VIEW_ZOOM_LARGE
:
263 V_I4(&zoomVariant
) = 130;
265 case wxWEB_VIEW_ZOOM_LARGEST
:
266 V_I4(&zoomVariant
) = 160;
272 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)OLECMDID_OPTICAL_ZOOM
,
273 OLECMDEXECOPT_DODEFAULT
,
276 wxASSERT(result
== S_OK
);
279 wxWebViewZoom
wxWebViewIE::GetIEOpticalZoom()
282 VariantInit (&zoomVariant
);
283 V_VT(&zoomVariant
) = VT_I4
;
285 HRESULT result
= m_webBrowser
->ExecWB((OLECMDID
)OLECMDID_OPTICAL_ZOOM
,
286 OLECMDEXECOPT_DODEFAULT
, NULL
,
288 wxASSERT(result
== S_OK
);
290 const int zoom
= V_I4(&zoomVariant
);
292 //We make a somewhat arbitray map here, taken from values used by webkit
295 return wxWEB_VIEW_ZOOM_TINY
;
297 else if (zoom
> 65 && zoom
<= 90)
299 return wxWEB_VIEW_ZOOM_SMALL
;
301 else if (zoom
> 90 && zoom
<= 115)
303 return wxWEB_VIEW_ZOOM_MEDIUM
;
305 else if (zoom
> 115 && zoom
<= 145)
307 return wxWEB_VIEW_ZOOM_LARGE
;
309 else /*if (zoom > 145) */ //Using else removes a compiler warning
311 return wxWEB_VIEW_ZOOM_LARGEST
;
315 void wxWebViewIE::SetZoomType(wxWebViewZoomType type
)
320 wxWebViewZoomType
wxWebViewIE::GetZoomType() const
325 bool wxWebViewIE::CanSetZoomType(wxWebViewZoomType type
) const
327 //IE 6 and below only support text zoom, so check the registry to see what
328 //version we actually have
329 wxRegKey
key(wxRegKey::HKLM
, "Software\\Microsoft\\Internet Explorer");
331 key
.QueryValue("Version", value
);
333 long version
= wxAtoi(value
.Left(1));
334 if(version
<= 6 && type
== wxWEB_VIEW_ZOOM_TYPE_LAYOUT
)
340 void wxWebViewIE::Print()
342 m_webBrowser
->ExecWB(OLECMDID_PRINTPREVIEW
,
343 OLECMDEXECOPT_DODEFAULT
, NULL
, NULL
);
346 bool wxWebViewIE::CanGoBack()
349 return m_historyPosition
> 0;
354 bool wxWebViewIE::CanGoForward()
357 return m_historyPosition
!= static_cast<int>(m_historyList
.size()) - 1;
362 void wxWebViewIE::LoadHistoryItem(wxSharedPtr
<wxWebHistoryItem
> item
)
365 for(unsigned int i
= 0; i
< m_historyList
.size(); i
++)
367 //We compare the actual pointers to find the correct item
368 if(m_historyList
[i
].get() == item
.get())
371 wxASSERT_MSG(pos
!= static_cast<int>(m_historyList
.size()),
372 "invalid history item");
373 m_historyLoadingFromList
= true;
374 LoadUrl(item
->GetUrl());
375 m_historyPosition
= pos
;
378 wxVector
<wxSharedPtr
<wxWebHistoryItem
> > wxWebViewIE::GetBackwardHistory()
380 wxVector
<wxSharedPtr
<wxWebHistoryItem
> > backhist
;
381 //As we don't have std::copy or an iterator constructor in the wxwidgets
382 //native vector we construct it by hand
383 for(int i
= 0; i
< m_historyPosition
; i
++)
385 backhist
.push_back(m_historyList
[i
]);
390 wxVector
<wxSharedPtr
<wxWebHistoryItem
> > wxWebViewIE::GetForwardHistory()
392 wxVector
<wxSharedPtr
<wxWebHistoryItem
> > forwardhist
;
393 //As we don't have std::copy or an iterator constructor in the wxwidgets
394 //native vector we construct it by hand
395 for(int i
= m_historyPosition
+ 1; i
< static_cast<int>(m_historyList
.size()); i
++)
397 forwardhist
.push_back(m_historyList
[i
]);
402 void wxWebViewIE::GoBack()
404 LoadHistoryItem(m_historyList
[m_historyPosition
- 1]);
407 void wxWebViewIE::GoForward()
409 LoadHistoryItem(m_historyList
[m_historyPosition
+ 1]);
412 void wxWebViewIE::Stop()
414 m_ie
.CallMethod("Stop");
417 void wxWebViewIE::ClearHistory()
419 m_historyList
.clear();
420 m_historyPosition
= -1;
423 void wxWebViewIE::EnableHistory(bool enable
)
425 m_historyEnabled
= enable
;
426 m_historyList
.clear();
427 m_historyPosition
= -1;
430 void wxWebViewIE::Reload(wxWebViewReloadFlags flags
)
434 V_VT(&level
) = VT_I2
;
438 case wxWEB_VIEW_RELOAD_DEFAULT
:
439 V_I2(&level
) = REFRESH_NORMAL
;
441 case wxWEB_VIEW_RELOAD_NO_CACHE
:
442 V_I2(&level
) = REFRESH_COMPLETELY
;
445 wxFAIL_MSG("Unexpected reload type");
448 m_webBrowser
->Refresh2(&level
);
451 bool wxWebViewIE::IsOfflineMode()
453 wxVariant out
= m_ie
.GetProperty("Offline");
455 wxASSERT(out
.GetType() == "bool");
457 return out
.GetBool();
460 void wxWebViewIE::SetOfflineMode(bool offline
)
462 // FIXME: the wxWidgets docs do not really document what the return
463 // parameter of PutProperty is
464 const bool success
= m_ie
.PutProperty("Offline", (offline
?
470 bool wxWebViewIE::IsBusy()
472 if (m_isBusy
) return true;
474 wxVariant out
= m_ie
.GetProperty("Busy");
476 wxASSERT(out
.GetType() == "bool");
478 return out
.GetBool();
481 wxString
wxWebViewIE::GetCurrentURL()
483 wxVariant out
= m_ie
.GetProperty("LocationURL");
485 wxASSERT(out
.GetType() == "string");
486 return out
.GetString();
489 wxString
wxWebViewIE::GetCurrentTitle()
491 IHTMLDocument2
* document
= GetDocument();
494 document
->get_nameProp(&title
);
496 return wxString(title
);
499 bool wxWebViewIE::CanCut()
501 return CanExecCommand("Cut");
504 bool wxWebViewIE::CanCopy()
506 return CanExecCommand("Copy");
508 bool wxWebViewIE::CanPaste()
510 return CanExecCommand("Paste");
513 void wxWebViewIE::Cut()
518 void wxWebViewIE::Copy()
523 void wxWebViewIE::Paste()
525 ExecCommand("Paste");
528 bool wxWebViewIE::CanUndo()
530 return CanExecCommand("Undo");
532 bool wxWebViewIE::CanRedo()
534 return CanExecCommand("Redo");
537 void wxWebViewIE::Undo()
542 void wxWebViewIE::Redo()
547 void wxWebViewIE::SetEditable(bool enable
)
549 IHTMLDocument2
* document
= GetDocument();
551 document
->put_designMode(SysAllocString(L
"On"));
553 document
->put_designMode(SysAllocString(L
"Off"));
558 bool wxWebViewIE::IsEditable()
560 IHTMLDocument2
* document
= GetDocument();
562 document
->get_designMode(&mode
);
564 if(wxString(mode
) == "On")
570 void wxWebViewIE::SelectAll()
572 ExecCommand("SelectAll");
575 bool wxWebViewIE::HasSelection()
577 IHTMLDocument2
* document
= GetDocument();
578 IHTMLSelectionObject
* selection
;
580 HRESULT hr
= document
->get_selection(&selection
);
584 selection
->get_type(&type
);
585 sel
= wxString(type
);
586 selection
->Release();
589 return sel
!= "None";
592 void wxWebViewIE::DeleteSelection()
594 ExecCommand("Delete");
597 wxString
wxWebViewIE::GetSelectedText()
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_text(&text
);
615 selected
= wxString(text
);
620 selection
->Release();
626 wxString
wxWebViewIE::GetSelectedSource()
628 IHTMLDocument2
* document
= GetDocument();
629 IHTMLSelectionObject
* selection
;
631 HRESULT hr
= document
->get_selection(&selection
);
635 hr
= selection
->createRange(&disrange
);
638 IHTMLTxtRange
* range
;
639 hr
= disrange
->QueryInterface(IID_IHTMLTxtRange
, (void**)&range
);
643 range
->get_htmlText(&text
);
644 selected
= wxString(text
);
649 selection
->Release();
655 void wxWebViewIE::ClearSelection()
657 IHTMLDocument2
* document
= GetDocument();
658 IHTMLSelectionObject
* selection
;
660 HRESULT hr
= document
->get_selection(&selection
);
664 selection
->Release();
669 wxString
wxWebViewIE::GetPageText()
671 IHTMLDocument2
* document
= GetDocument();
674 HRESULT hr
= document
->get_body(&body
);
678 body
->get_innerText(&out
);
679 text
= wxString(out
);
686 void wxWebViewIE::RunScript(const wxString
& javascript
)
688 IHTMLDocument2
* document
= GetDocument();
689 IHTMLWindow2
* window
;
690 wxString language
= "javascript";
691 HRESULT hr
= document
->get_parentWindow(&window
);
696 V_VT(&level
) = VT_EMPTY
;
697 window
->execScript(SysAllocString(javascript
), SysAllocString(language
), &level
);
702 bool wxWebViewIE::CanExecCommand(wxString command
)
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()
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 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATING
,
744 GetId(), url
, target
, true);
745 event
.SetEventObject(this);
746 HandleWindowEvent(event
);
748 if (event
.IsVetoed())
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 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED
,
770 GetId(), url
, target
, false);
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 if(m_historyEnabled
&& !m_historyLoadingFromList
&& url
== GetCurrentURL())
797 //If we are not at the end of the list, then erase everything
798 //between us and the end before adding the new page
799 if(m_historyPosition
!= static_cast<int>(m_historyList
.size()) - 1)
801 m_historyList
.erase(m_historyList
.begin() + m_historyPosition
+ 1,
802 m_historyList
.end());
804 wxSharedPtr
<wxWebHistoryItem
> item(new wxWebHistoryItem(url
, GetCurrentTitle()));
805 m_historyList
.push_back(item
);
808 //Reset as we are done now
809 m_historyLoadingFromList
= false;
810 // TODO: set target parameter if possible
811 wxString target
= wxEmptyString
;
812 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_LOADED
, GetId(),
814 event
.SetEventObject(this);
815 HandleWindowEvent(event
);
819 case DISPID_STATUSTEXTCHANGE
:
824 case DISPID_TITLECHANGE
:
829 case DISPID_NAVIGATEERROR
:
831 wxWebNavigationError errorType
= wxWEB_NAV_ERR_OTHER
;
832 wxString errorCode
= "?";
833 switch (evt
[3].GetLong())
835 case INET_E_INVALID_URL
: // (0x800C0002L or -2146697214)
836 errorCode
= "INET_E_INVALID_URL";
837 errorType
= wxWEB_NAV_ERR_REQUEST
;
839 case INET_E_NO_SESSION
: // (0x800C0003L or -2146697213)
840 errorCode
= "INET_E_NO_SESSION";
841 errorType
= wxWEB_NAV_ERR_CONNECTION
;
843 case INET_E_CANNOT_CONNECT
: // (0x800C0004L or -2146697212)
844 errorCode
= "INET_E_CANNOT_CONNECT";
845 errorType
= wxWEB_NAV_ERR_CONNECTION
;
847 case INET_E_RESOURCE_NOT_FOUND
: // (0x800C0005L or -2146697211)
848 errorCode
= "INET_E_RESOURCE_NOT_FOUND";
849 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
851 case INET_E_OBJECT_NOT_FOUND
: // (0x800C0006L or -2146697210)
852 errorCode
= "INET_E_OBJECT_NOT_FOUND";
853 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
855 case INET_E_DATA_NOT_AVAILABLE
: // (0x800C0007L or -2146697209)
856 errorCode
= "INET_E_DATA_NOT_AVAILABLE";
857 errorType
= wxWEB_NAV_ERR_NOT_FOUND
;
859 case INET_E_DOWNLOAD_FAILURE
: // (0x800C0008L or -2146697208)
860 errorCode
= "INET_E_DOWNLOAD_FAILURE";
861 errorType
= wxWEB_NAV_ERR_CONNECTION
;
863 case INET_E_AUTHENTICATION_REQUIRED
: // (0x800C0009L or -2146697207)
864 errorCode
= "INET_E_AUTHENTICATION_REQUIRED";
865 errorType
= wxWEB_NAV_ERR_AUTH
;
867 case INET_E_NO_VALID_MEDIA
: // (0x800C000AL or -2146697206)
868 errorCode
= "INET_E_NO_VALID_MEDIA";
869 errorType
= wxWEB_NAV_ERR_REQUEST
;
871 case INET_E_CONNECTION_TIMEOUT
: // (0x800C000BL or -2146697205)
872 errorCode
= "INET_E_CONNECTION_TIMEOUT";
873 errorType
= wxWEB_NAV_ERR_CONNECTION
;
875 case INET_E_INVALID_REQUEST
: // (0x800C000CL or -2146697204)
876 errorCode
= "INET_E_INVALID_REQUEST";
877 errorType
= wxWEB_NAV_ERR_REQUEST
;
879 case INET_E_UNKNOWN_PROTOCOL
: // (0x800C000DL or -2146697203)
880 errorCode
= "INET_E_UNKNOWN_PROTOCOL";
881 errorType
= wxWEB_NAV_ERR_REQUEST
;
883 case INET_E_SECURITY_PROBLEM
: // (0x800C000EL or -2146697202)
884 errorCode
= "INET_E_SECURITY_PROBLEM";
885 errorType
= wxWEB_NAV_ERR_SECURITY
;
887 case INET_E_CANNOT_LOAD_DATA
: // (0x800C000FL or -2146697201)
888 errorCode
= "INET_E_CANNOT_LOAD_DATA";
889 errorType
= wxWEB_NAV_ERR_OTHER
;
891 case INET_E_CANNOT_INSTANTIATE_OBJECT
:
892 // CoCreateInstance will return an error code if this happens,
893 // we'll handle this above.
896 case INET_E_REDIRECT_FAILED
: // (0x800C0014L or -2146697196)
897 errorCode
= "INET_E_REDIRECT_FAILED";
898 errorType
= wxWEB_NAV_ERR_OTHER
;
900 case INET_E_REDIRECT_TO_DIR
: // (0x800C0015L or -2146697195)
901 errorCode
= "INET_E_REDIRECT_TO_DIR";
902 errorType
= wxWEB_NAV_ERR_REQUEST
;
904 case INET_E_CANNOT_LOCK_REQUEST
: // (0x800C0016L or -2146697194)
905 errorCode
= "INET_E_CANNOT_LOCK_REQUEST";
906 errorType
= wxWEB_NAV_ERR_OTHER
;
908 case INET_E_USE_EXTEND_BINDING
: // (0x800C0017L or -2146697193)
909 errorCode
= "INET_E_USE_EXTEND_BINDING";
910 errorType
= wxWEB_NAV_ERR_OTHER
;
912 case INET_E_TERMINATED_BIND
: // (0x800C0018L or -2146697192)
913 errorCode
= "INET_E_TERMINATED_BIND";
914 errorType
= wxWEB_NAV_ERR_OTHER
;
916 case INET_E_INVALID_CERTIFICATE
: // (0x800C0019L or -2146697191)
917 errorCode
= "INET_E_INVALID_CERTIFICATE";
918 errorType
= wxWEB_NAV_ERR_CERTIFICATE
;
920 case INET_E_CODE_DOWNLOAD_DECLINED
: // (0x800C0100L or -2146696960)
921 errorCode
= "INET_E_CODE_DOWNLOAD_DECLINED";
922 errorType
= wxWEB_NAV_ERR_USER_CANCELLED
;
924 case INET_E_RESULT_DISPATCHED
: // (0x800C0200L or -2146696704)
925 // cancel request cancelled...
926 errorCode
= "INET_E_RESULT_DISPATCHED";
927 errorType
= wxWEB_NAV_ERR_OTHER
;
929 case INET_E_CANNOT_REPLACE_SFP_FILE
: // (0x800C0300L or -2146696448)
930 errorCode
= "INET_E_CANNOT_REPLACE_SFP_FILE";
931 errorType
= wxWEB_NAV_ERR_SECURITY
;
933 case INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY
:
934 errorCode
= "INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY";
935 errorType
= wxWEB_NAV_ERR_SECURITY
;
937 case INET_E_CODE_INSTALL_SUPPRESSED
:
938 errorCode
= "INET_E_CODE_INSTALL_SUPPRESSED";
939 errorType
= wxWEB_NAV_ERR_SECURITY
;
943 wxString url
= evt
[1].GetString();
944 wxString target
= evt
[2].GetString();
945 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_ERROR
, GetId(),
947 event
.SetEventObject(this);
948 event
.SetInt(errorType
);
949 event
.SetString(errorCode
);
950 HandleWindowEvent(event
);
953 case DISPID_NEWWINDOW3
:
955 wxString url
= evt
[4].GetString();
957 wxWebNavigationEvent
event(wxEVT_COMMAND_WEB_VIEW_NEWWINDOW
,
958 GetId(), url
, wxEmptyString
, true);
959 event
.SetEventObject(this);
960 HandleWindowEvent(event
);
962 //We always cancel this event otherwise an Internet Exporer window
963 //is opened for the url
964 wxActiveXEventNativeMSW
* nativeParams
= evt
.GetNativeParameters();
965 *V_BOOLREF(&nativeParams
->pDispParams
->rgvarg
[3]) = VARIANT_TRUE
;
973 VirtualProtocol::VirtualProtocol()
977 m_fileSys
= new wxFileSystem
;
980 VirtualProtocol::~VirtualProtocol()
985 ULONG
VirtualProtocol::AddRef()
991 HRESULT
VirtualProtocol::QueryInterface(REFIID riid
, void **ppvObject
)
993 if ((riid
== IID_IUnknown
) || (riid
== IID_IInternetProtocol
)
994 || (riid
== IID_IInternetProtocolRoot
))
1007 ULONG
VirtualProtocol::Release()
1021 HRESULT
VirtualProtocol::Start(LPCWSTR szUrl
, IInternetProtocolSink
*pOIProtSink
,
1022 IInternetBindInfo
*pOIBindInfo
, DWORD grfPI
,
1023 HANDLE_PTR dwReserved
)
1025 m_protocolSink
= pOIProtSink
;
1026 //We have to clean up incoming paths from the webview control as they are
1027 //not properly escaped, see also the comment in filesys.cpp line 668
1028 wxString path
= wxString(szUrl
).BeforeFirst(':') + ":" +
1029 EscapeFileNameCharsInURL(wxString(szUrl
).AfterFirst(':'));
1030 path
.Replace("///", "/");
1031 m_file
= m_fileSys
->OpenFile(path
);
1034 return INET_E_RESOURCE_NOT_FOUND
;
1036 //We return the stream length for current and total size as we can always
1037 //read the whole file from the stream
1038 m_protocolSink
->ReportData(BSCF_FIRSTDATANOTIFICATION
|
1039 BSCF_DATAFULLYAVAILABLE
|
1040 BSCF_LASTDATANOTIFICATION
,
1041 m_file
->GetStream()->GetLength(),
1042 m_file
->GetStream()->GetLength());
1046 HRESULT
VirtualProtocol::Read(void *pv
, ULONG cb
, ULONG
*pcbRead
)
1048 //If the file is null we return false to indicte it is finished
1052 wxStreamError err
= m_file
->GetStream()->Read(pv
, cb
).GetLastError();
1053 *pcbRead
= m_file
->GetStream()->LastRead();
1055 if(err
== wxSTREAM_NO_ERROR
)
1060 m_protocolSink
->ReportResult(S_OK
, 0, NULL
);
1062 //As we are not eof there is more data
1065 else if(err
== wxSTREAM_EOF
)
1068 m_protocolSink
->ReportResult(S_OK
, 0, NULL
);
1069 //We are eof and so finished
1072 else if(err
== wxSTREAM_READ_ERROR
)
1075 return INET_E_DOWNLOAD_FAILURE
;
1079 HRESULT
ClassFactory::CreateInstance(IUnknown
* pUnkOuter
, REFIID riid
,
1083 return CLASS_E_NOAGGREGATION
;
1084 VirtualProtocol
* vp
= new VirtualProtocol
;
1086 HRESULT hr
= vp
->QueryInterface(riid
, ppvObject
);
1092 STDMETHODIMP
ClassFactory::LockServer(BOOL fLock
)
1098 ULONG
ClassFactory::AddRef(void)
1104 HRESULT
ClassFactory::QueryInterface(REFIID riid
, void **ppvObject
)
1106 if ((riid
== IID_IUnknown
) || (riid
== IID_IClassFactory
))
1120 ULONG
ClassFactory::Release(void)